Reputation: 430
I'm trying to learn how to do MVC and DAO but my implementation is failing right now and I can't get around it.
Here is my view, or the relevant parts of it:
package gui;
import javax.swing.JFrame;
public class LoginFrame{
private JPanel contentPane;
private final JLabel credentialsLabel = new JLabel("Credentials");
private JTextField usernameField;
private JPasswordField passwordField;
private JButton btnLogin;
/**
* Create the frame.
*/
public LoginFrame() {
createAndShowGUI();
}
private void createAndShowGUI()
{
EventQueue.invokeLater(new Runnable() {
public void run() {
try {
JFrame frame = new JFrame();
frame.setTitle("Login");
frame.setContentPane(createContentPane());
frame.setBounds(100, 100, 480, 237);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setVisible(true);
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
//A function that sets up the basic structure of the GUI.
private JPanel createContentPane() {
contentPane = new JPanel();
contentPane.setForeground(Color.LIGHT_GRAY);
contentPane.setBackground(Color.DARK_GRAY);
contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
GridBagLayout gbl_contentPane = new GridBagLayout();
gbl_contentPane.columnWidths = new int[]{106, 0, 0, 153, 0};
gbl_contentPane.rowHeights = new int[]{65, 27, 39, 36, 0, 0};
gbl_contentPane.columnWeights = new double[]{0.0, 0.0, 0.0, 1.0, Double.MIN_VALUE};
gbl_contentPane.rowWeights = new double[]{0.0, 0.0, 0.0, 0.0, 0.0, Double.MIN_VALUE};
contentPane.setLayout(gbl_contentPane);
//login button
btnLogin = new JButton("login");
GridBagConstraints gbc_btnLogin = new GridBagConstraints();
gbc_btnLogin.anchor = GridBagConstraints.WEST;
gbc_btnLogin.insets = new Insets(0, 0, 5, 5);
gbc_btnLogin.gridx = 2;
gbc_btnLogin.gridy = 3;
contentPane.add(btnLogin, gbc_btnLogin);
//register button
JButton btnRegister = new JButton("register");
btnRegister.setFont(new Font("Helvetica", Font.PLAIN, 13));
GridBagConstraints gbc_btnRegister = new GridBagConstraints();
gbc_btnRegister.anchor = GridBagConstraints.WEST;
gbc_btnRegister.insets = new Insets(0, 0, 5, 0);
gbc_btnRegister.gridx = 3;
gbc_btnRegister.gridy = 3;
contentPane.add(btnRegister, gbc_btnRegister);
return contentPane;
}
//the Action Listener for the Login Button passed by controller
public void buttonActionListeners(ActionListener al) {
btnLogin.setActionCommand("login");
btnLogin.addActionListener(al);
}
}
My Model:
package functions;
import database.LoginDAO;
public class LoginModel {
LoginDAO logindao;
public LoginModel(LoginDAO logindao){
this.logindao = logindao;
}
public void attemptLogin(String username, char[] password) {
System.out.println("Testing login attempt");
logindao.attemptLogin(username, password);
}
}
My controller:
package controller;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import functions.LoginModel;
import gui.LoginFrame;
public class LoginControl implements ActionListener {
LoginModel model;
LoginFrame view;
public LoginControl(LoginModel model, LoginFrame view){
this.model = model;
this.view = view;
//add action listener from control to view.
view.buttonActionListeners(this);
}
//action performed by view
public void actionPerformed(ActionEvent ae)
{
System.out.println("Testing Action Performed");
String action = ae.getActionCommand();
if(action.equals("login")){
System.out.println("Testing Action Performed");
model.attemptLogin(view.getUsername(),view.getPassword());
}
}
}
Main:
import functions.LoginModel;
import java.io.IOException;
import java.sql.SQLException;
import controller.LoginControl;
import database.LoginDAO;
import gui.LoginFrame;
public class Main {
public static void main(String[] args) throws IOException, SQLException {
LoginFrame frame = new LoginFrame();
LoginDAO loginDao = new LoginDAO();
LoginModel model = new LoginModel(loginDao);
LoginControl controller = new LoginControl(model, frame);
}
}
The Error:
Exception in thread "main" java.lang.NullPointerException
at gui.LoginFrame.buttonActionListeners(LoginFrame.java:160)
at controller.LoginControl.<init>(LoginControl.java:16)
at Main.main(Main.java:14)
The lines that the error references:
btnLogin.setActionCommand("login"); //in View
view.buttonActionListeners(this); //in Control
LoginControl controller = new LoginControl(model, frame); //in main.
So from what I've tried to understand: view.buttonActionListeners(this); somewhere in here I'm passing a null, so either this is null or view is null? how can view be null if we initiated LoginFrame view;
Thanks for helping me solve this bug.
Upvotes: 0
Views: 590
Reputation: 1294
Ok... got it. your btnLogin property is null when you make the call to
view.buttonActionListeners(this);
It happens because it is initialized on a Runnable, that is asynchronous, and probably didn't run yet. From this part of the code I don't see the need to it be called asynchronously. So I would suggest to set the properties synchronously. So it should be:
private void createAndShowGUI()
{
JFrame frame = new JFrame();
frame.setTitle("Login");
frame.setContentPane(createContentPane());
frame.setBounds(100, 100, 480, 237);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setVisible(true);
}
Upvotes: 1