|
@@ -19,55 +19,66 @@ import java.net.Socket;
|
|
|
import java.net.SocketTimeoutException;
|
|
|
import java.util.ArrayList;
|
|
|
|
|
|
-import javax.swing.*;
|
|
|
+import javax.swing.DefaultListModel;
|
|
|
+import javax.swing.JButton;
|
|
|
+import javax.swing.JFrame;
|
|
|
+import javax.swing.JLabel;
|
|
|
+import javax.swing.JList;
|
|
|
+import javax.swing.JPanel;
|
|
|
+import javax.swing.JScrollPane;
|
|
|
+import javax.swing.JTextField;
|
|
|
+import javax.swing.ListSelectionModel;
|
|
|
+import javax.swing.SwingConstants;
|
|
|
+import javax.swing.WindowConstants;
|
|
|
import javax.swing.border.EmptyBorder;
|
|
|
|
|
|
import common.Message;
|
|
|
import common.Message.MessageType;
|
|
|
|
|
|
@SuppressWarnings("serial")
|
|
|
-public class ChatWindow extends JFrame implements ActionListener, Runnable, KeyListener{
|
|
|
+public class ChatWindow extends JFrame implements ActionListener, Runnable,
|
|
|
+ KeyListener {
|
|
|
Thread getMessages;
|
|
|
static File confFile = new File("client.properties");
|
|
|
-
|
|
|
+
|
|
|
String adress, username;
|
|
|
ArrayList<String> lastMess = new ArrayList<String>();
|
|
|
int port, messIndex = 0;
|
|
|
-
|
|
|
+
|
|
|
Socket so = new Socket();
|
|
|
ObjectInputStream objIn;
|
|
|
PrintWriter out;
|
|
|
-
|
|
|
+
|
|
|
GridBagLayout g = new GridBagLayout();
|
|
|
GridBagConstraints con = new GridBagConstraints();
|
|
|
-
|
|
|
+
|
|
|
JPanel right = new JPanel();
|
|
|
JLabel lblUsersOnline = new JLabel("Users online:");
|
|
|
DefaultListModel<String> model = new DefaultListModel<String>();
|
|
|
JList<String> userList = new JList<String>(model);
|
|
|
JButton reconnect = new JButton("Reconnect");
|
|
|
-
|
|
|
+
|
|
|
Console chat = new Console();
|
|
|
JScrollPane scroll = new JScrollPane(chat);
|
|
|
JTextField write = new JTextField();
|
|
|
-
|
|
|
+
|
|
|
public ChatWindow(String adress, int port, String username) {
|
|
|
this.adress = adress;
|
|
|
this.port = port;
|
|
|
this.username = username;
|
|
|
-
|
|
|
- //List config
|
|
|
+
|
|
|
+ // List config
|
|
|
userList.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
|
|
|
userList.setLayoutOrientation(JList.VERTICAL);
|
|
|
- //Label config
|
|
|
- lblUsersOnline.setHorizontalAlignment(JLabel.CENTER);
|
|
|
+ // Label config
|
|
|
+ lblUsersOnline.setHorizontalAlignment(SwingConstants.CENTER);
|
|
|
lblUsersOnline.setBorder(new EmptyBorder(5, 5, 5, 5));
|
|
|
- //Layout config
|
|
|
+ // Layout config
|
|
|
right.setLayout(g);
|
|
|
con.fill = GridBagConstraints.HORIZONTAL;
|
|
|
con.weightx = 1;
|
|
|
con.gridx = 0;
|
|
|
-
|
|
|
+
|
|
|
right.add(lblUsersOnline, con);
|
|
|
|
|
|
con.weighty = 1;
|
|
@@ -77,75 +88,81 @@ public class ChatWindow extends JFrame implements ActionListener, Runnable, KeyL
|
|
|
con.weighty = 0;
|
|
|
con.fill = GridBagConstraints.HORIZONTAL;
|
|
|
right.add(reconnect, con);
|
|
|
-
|
|
|
+
|
|
|
setLayout(new BorderLayout());
|
|
|
- add(chat, BorderLayout.NORTH); add(write, BorderLayout.SOUTH); add(right, BorderLayout.EAST);
|
|
|
-
|
|
|
- //Scrollbar config
|
|
|
+ add(chat, BorderLayout.NORTH);
|
|
|
+ add(write, BorderLayout.SOUTH);
|
|
|
+ add(right, BorderLayout.EAST);
|
|
|
+
|
|
|
+ // Scrollbar config
|
|
|
add(scroll);
|
|
|
scroll.setMinimumSize(new Dimension(100, 100));
|
|
|
scroll.setViewportView(chat);
|
|
|
scroll.setSize(500, 130);
|
|
|
-
|
|
|
- //Listener config
|
|
|
+
|
|
|
+ // Listener config
|
|
|
reconnect.addActionListener(this);
|
|
|
write.addKeyListener(this);
|
|
|
-
|
|
|
- //Window config
|
|
|
+
|
|
|
+ // Window config
|
|
|
this.setLocation(new Point(100, 100));
|
|
|
setSize(600, 600);
|
|
|
setVisible(true);
|
|
|
setTitle("Chat on " + adress + " | Username: " + username);
|
|
|
setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
|
|
|
-
|
|
|
+
|
|
|
connect(adress, port, username);
|
|
|
}
|
|
|
-
|
|
|
+
|
|
|
public void send(String text) {
|
|
|
if (so.isConnected() && !so.isClosed())
|
|
|
out.println(text);
|
|
|
else {
|
|
|
- chat.log(new Message("Not connected to server!", MessageType.WARNING, false));
|
|
|
+ chat.log(new Message("Not connected to server!",
|
|
|
+ MessageType.WARNING, false));
|
|
|
write.setEnabled(false);
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
+
|
|
|
void connect(String address, int port, String username) {
|
|
|
- chat.log(new Message("Connecting to " + address + " on port " + port + ".", MessageType.INFO, false));
|
|
|
+ chat.log(new Message("Connecting to " + address + " on port " + port
|
|
|
+ + ".", MessageType.INFO, false));
|
|
|
if (getMessages != null)
|
|
|
getMessages.interrupt();
|
|
|
-
|
|
|
+
|
|
|
try {
|
|
|
so.close();
|
|
|
objIn.close();
|
|
|
out.close();
|
|
|
} catch (NullPointerException ex) {
|
|
|
- //Nothing
|
|
|
+ // Nothing
|
|
|
} catch (IOException ex) {
|
|
|
chat.log(new Message(ex.toString(), MessageType.ERROR, false));
|
|
|
}
|
|
|
-
|
|
|
+
|
|
|
try {
|
|
|
so = new Socket();
|
|
|
so.connect(new InetSocketAddress(address, port), 1000);
|
|
|
- objIn = new ObjectInputStream(so.getInputStream());
|
|
|
- out = new PrintWriter(so.getOutputStream(), true);
|
|
|
+ objIn = new ObjectInputStream(so.getInputStream());
|
|
|
+ out = new PrintWriter(so.getOutputStream(), true);
|
|
|
} catch (SocketTimeoutException ex) {
|
|
|
- chat.log(new Message("Could not connect to server. (Connection timed out!)", MessageType.ERROR, false));
|
|
|
+ chat.log(new Message(
|
|
|
+ "Could not connect to server. (Connection timed out!)",
|
|
|
+ MessageType.ERROR, false));
|
|
|
return;
|
|
|
} catch (IOException e) {
|
|
|
chat.log(new Message(e.toString(), MessageType.ERROR, false));
|
|
|
return;
|
|
|
}
|
|
|
-
|
|
|
- send(username); //First packet sent to server sets username
|
|
|
-
|
|
|
+
|
|
|
+ send(username); // First packet sent to server sets username
|
|
|
+
|
|
|
getMessages = new Thread(this);
|
|
|
getMessages.start();
|
|
|
-
|
|
|
+
|
|
|
write.setEnabled(true);
|
|
|
}
|
|
|
-
|
|
|
+
|
|
|
@Override
|
|
|
public void actionPerformed(ActionEvent e) {
|
|
|
if (e.getSource() == reconnect)
|
|
@@ -157,31 +174,35 @@ public class ChatWindow extends JFrame implements ActionListener, Runnable, KeyL
|
|
|
try {
|
|
|
getMessages();
|
|
|
} catch (EOFException eof) {
|
|
|
- chat.log(new Message(eof.toString() + " Disconnected from host.", MessageType.ERROR, false));
|
|
|
+ chat.log(new Message(eof.toString() + " Disconnected from host.",
|
|
|
+ MessageType.ERROR, false));
|
|
|
} catch (ClassNotFoundException cnf) {
|
|
|
- chat.log(new Message("The message recieved from the server could not be understood. Are you using the right version?", MessageType.ERROR, false));
|
|
|
+ chat.log(new Message(
|
|
|
+ "The message recieved from the server could not be understood. Are you using the right version?",
|
|
|
+ MessageType.ERROR, false));
|
|
|
} catch (IOException e) {
|
|
|
chat.log(new Message(e.toString(), MessageType.ERROR, false));
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
+
|
|
|
public void getMessages() throws IOException, ClassNotFoundException {
|
|
|
- while(!getMessages.isInterrupted()) {
|
|
|
+ while (!getMessages.isInterrupted()) {
|
|
|
Object fromServer = objIn.readObject();
|
|
|
if (fromServer instanceof Message) {
|
|
|
- Message mess = ((Message)fromServer);
|
|
|
+ Message mess = ((Message) fromServer);
|
|
|
chat.log(mess);
|
|
|
-
|
|
|
+
|
|
|
if (mess.usersOnline == null)
|
|
|
continue;
|
|
|
-
|
|
|
+
|
|
|
model = new DefaultListModel<String>();
|
|
|
- for (String user: mess.usersOnline)
|
|
|
+ for (String user : mess.usersOnline)
|
|
|
model.addElement(user);
|
|
|
-
|
|
|
+
|
|
|
userList.setModel(model);
|
|
|
} else if (fromServer instanceof String) {
|
|
|
- chat.log(new Message((String)fromServer, MessageType.NORMAL, false));
|
|
|
+ chat.log(new Message((String) fromServer, MessageType.NORMAL,
|
|
|
+ false));
|
|
|
} else
|
|
|
throw new ClassNotFoundException();
|
|
|
}
|
|
@@ -190,37 +211,39 @@ public class ChatWindow extends JFrame implements ActionListener, Runnable, KeyL
|
|
|
@Override
|
|
|
public void keyPressed(KeyEvent eKey) {
|
|
|
int keyCode = eKey.getKeyCode();
|
|
|
- switch( keyCode ) {
|
|
|
- case KeyEvent.VK_UP:
|
|
|
- if (messIndex > 0)
|
|
|
- messIndex--;
|
|
|
- if (!lastMess.isEmpty())
|
|
|
- write.setText(lastMess.get(messIndex));
|
|
|
- break;
|
|
|
- case KeyEvent.VK_DOWN:
|
|
|
- if (messIndex <= lastMess.size())
|
|
|
- messIndex++;
|
|
|
- if (messIndex >= lastMess.size()) {
|
|
|
- messIndex = lastMess.size();
|
|
|
- write.setText("");
|
|
|
- } else
|
|
|
- write.setText(lastMess.get(messIndex));
|
|
|
- break;
|
|
|
- case KeyEvent.VK_ENTER:
|
|
|
- String text = write.getText().trim();
|
|
|
- if (!text.equals("")) {
|
|
|
- send(text);
|
|
|
- lastMess.add(text);
|
|
|
- messIndex = lastMess.size();
|
|
|
- write.setText("");
|
|
|
- }
|
|
|
- break;
|
|
|
- }
|
|
|
+ switch (keyCode) {
|
|
|
+ case KeyEvent.VK_UP:
|
|
|
+ if (messIndex > 0)
|
|
|
+ messIndex--;
|
|
|
+ if (!lastMess.isEmpty())
|
|
|
+ write.setText(lastMess.get(messIndex));
|
|
|
+ break;
|
|
|
+ case KeyEvent.VK_DOWN:
|
|
|
+ if (messIndex <= lastMess.size())
|
|
|
+ messIndex++;
|
|
|
+ if (messIndex >= lastMess.size()) {
|
|
|
+ messIndex = lastMess.size();
|
|
|
+ write.setText("");
|
|
|
+ } else
|
|
|
+ write.setText(lastMess.get(messIndex));
|
|
|
+ break;
|
|
|
+ case KeyEvent.VK_ENTER:
|
|
|
+ String text = write.getText().trim();
|
|
|
+ if (!text.equals("")) {
|
|
|
+ send(text);
|
|
|
+ lastMess.add(text);
|
|
|
+ messIndex = lastMess.size();
|
|
|
+ write.setText("");
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
- public void keyReleased(KeyEvent arg0) {}
|
|
|
+ public void keyReleased(KeyEvent arg0) {
|
|
|
+ }
|
|
|
|
|
|
@Override
|
|
|
- public void keyTyped(KeyEvent arg0) {}
|
|
|
+ public void keyTyped(KeyEvent arg0) {
|
|
|
+ }
|
|
|
}
|