`

Socket多人聊天(文字+图片+多文件发送和接收)

阅读更多

主要实现:

1.群聊

2.私聊

3.发送文字(可选择字体,颜色)

4.发送图片

5.发送文件,支持多个文件同时发送/接收。

 

消息对象:

 

package com.socket.tcp.basechat;

import java.io.File;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;

/**
 * 包装发送的消息对象。
 * @author lucky star.
 *
 */
public class MsgInfo implements Serializable{
	/**
	 * serialVersionUID
	 */
	private static final long serialVersionUID = 2817564515497626133L;
	private String clientId; // 标示每个客户端
	// 私聊人
	private String user;
	// 是否私聊,设置USER时自动设置为TRUE,否则为FALSE。
	private boolean isSL = false;
	// 文件大小
	private int fileLen = -1;

	public int getFileLen() {
		return fileLen;
	}

	public void setFileLen(int fileLen) {
		this.fileLen = fileLen;
	}

	// 文件列表
	private String[] files = {};
	
	public String[] getFiles() {
		return files;
	}

	public void setFiles(String[] files) {
		this.files = files;
	}

	public boolean isSL() {
		return isSL;
	}

	public String getUser() {
		return user;
	}
	public void setUser(String user) {
		this.user = user;
		isSL = true;
	}

	// 文本消息
	private String msgContent;
	// 图片消息
	private List<File> imgs = new ArrayList<File>();
	// 消息发送时间
	private Date sendTime;
	// 发送人
	private String sender;
	// online list
	private List<String> onlines = new ArrayList<String>();
	
	public List<String> getOnlines() {
		return onlines;
	}
	public void setOnlines(List<String> onlines) {
		this.onlines = onlines;
	}
	public String getClientId() {
		return clientId;
	}
	public void setClientId(String clientId) {
		this.clientId = clientId;
	}
	// 一个客户端下线后通知其他客户端时,new一个空的消息对象。
	// 仅携带onlines,通知其他客户端更新任意列表
	public MsgInfo() {
		super();
	}
	public MsgInfo(String msgContent, List<File> imgs, List<File> attaches,
			Date sendTime, String sender) {
		super();
		this.msgContent = msgContent;
		this.imgs = imgs;
		this.sendTime = sendTime;
		this.sender = sender;
	}
	// 私聊时用到这个构造方法。
	public MsgInfo(String msgContent, List<File> imgs, List<File> attaches,
			Date sendTime, String sender,String user) {
		this(msgContent,imgs,attaches,sendTime,sender);
		this.user = user;
	}
	public MsgInfo(String msgContent) {
		this.msgContent = msgContent;
	}
	
	public String getMsgContent() {
		return msgContent;
	}
	public void setMsgContent(String msgContent) {
		this.msgContent = msgContent;
	}
	public List<File> getImgs() {
		return imgs;
	}
	public void setImgs(List<File> imgs) {
		this.imgs = imgs;
	}
	public Date getSendTime() {
		return sendTime;
	}
	public void setSendTime(Date sendTime) {
		this.sendTime = sendTime;
	}
	public String getSender() {
		return sender;
	}
	public void setSender(String sender) {
		this.sender = sender;
	}
	
	public boolean isEmpty() {
		boolean a = msgContent == null || msgContent.equals("");
		boolean b = imgs == null || imgs.size() == 0;
		boolean d = files == null || files.length == 0;
		if (a && b && d) return true;
		return false;
	}
	
	public boolean isMsg() {
		if (isEmpty()) {
			if (sender != null && !"".equals(sender)) {
				return false;
			}
		}
		return true;
	}
}


服务端:

 

 

package com.socket.tcp.basechat;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.net.BindException;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;


/**
 * Server端。
 * @author lucky star
 *
 */
public class Server {
	private ServerSocket ss = null;
	private boolean isStarted = false;
	private Map<String,RecvThread> clients = new HashMap<String,Server.RecvThread>();

	public void start() {
		try {
			ss = new ServerSocket(8888);
			isStarted = true;
			System.out.println("Tcp Server started.");
		} catch (BindException e) {
			System.out.println("端口被占用,请关闭相关程序。");
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

		while (isStarted) {
			try {
				Socket s = ss.accept();
				RecvThread rt = new RecvThread(s);
				new Thread(rt).start();
			} catch (IOException e) {
				System.out.println(e.getMessage());
//				e.printStackTrace();
			}
		}
	}

	private class RecvThread implements Runnable {
		private Socket s = null;
		private ObjectInputStream ois = null;
		private ObjectOutputStream oos = null;
		private boolean isConnected = false;
		private String user;

		public RecvThread(Socket s) {
			this.s = s;
			try {
				ois = new ObjectInputStream(s.getInputStream());
				oos = new ObjectOutputStream(s.getOutputStream());
//				System.out.println("oos:" + oos);
			} catch (IOException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}

			isConnected = true;
		}

		// 发送对象
		public void sendMsg(MsgInfo mi) {
			try {
				oos.writeObject(mi);
				oos.flush();
			} catch (IOException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}

		// 转发消息给各客户端
		public void zfMsg(MsgInfo mi) {
			if (mi.isSL()) { // 私聊
				// 发给对方。
				// 根据用户名得到对方的RecvThread
				RecvThread rt = clients.get(mi.getUser());
				rt.sendMsg(mi);
				System.out.println("sl.");

			} else { // 群聊
				System.out.println("群聊。" + clients.size());
				List<String> onlines = new ArrayList<String>();
				
				Set<String> keys = clients.keySet();
				Iterator<String> itKeys = keys.iterator();
				while (itKeys.hasNext()) {
					String key = itKeys.next();
					onlines.add(key);
				}
				
				mi.setOnlines(onlines);
				mi.setClientId(this.s.toString());
				
				// 转发消息给各个客户端
				itKeys = keys.iterator();
				while (itKeys.hasNext()) {
					String key = itKeys.next();
					RecvThread rt = clients.get(key);
					if (mi.isMsg()) { // 发送消息时不转发给自己
						if (rt != this) {
							rt.sendMsg(mi);
						}
					}
					else { // 获取在线用户列表。
						rt.sendMsg(mi);
					}
					System.out.println("转发消息给 " + rt.user);
				}
			}

		}

		public void run() {
			while (isConnected) {
				try {

					// 读取信息
					MsgInfo mi = (MsgInfo) ois.readObject();
					// 激活连接
					if (!mi.isMsg()) { // 建立连接后发送一条空的信息,将用户名带到服务器。
						clients.put(mi.getSender(), this);
						user = mi.getSender();
						System.out.println(mi.getSender() + " Join.");
					}
					// 转发给客户端
					zfMsg(mi);

				} catch (IOException e) {
					// disconnect();
					disConnect();
					// e.printStackTrace();
				} catch (ClassNotFoundException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
		}

		// 发送对象信息使用
		public void disConnect() {
			try {
				System.out.println("Client " + user
						+ " exit.");
				isConnected = false;
				if (ois != null)
					ois.close();
				if (oos != null)
					oos.close();
				if (s != null)
					s.close();
				clients.remove(this.user);
				// 一个客户端退出后,通知其他客户端
				MsgInfo mi = new MsgInfo();
				zfMsg(mi);
			} catch (IOException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}

	}

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		new Server().start();
	}

}

 

 

客户端:

package com.socket.tcp.basechat;

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.FlowLayout;
import java.awt.Font;
import java.awt.GraphicsEnvironment;
import java.awt.Image;
import java.awt.MenuItem;
import java.awt.Panel;
import java.awt.PopupMenu;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.awt.dnd.DnDConstants;
import java.awt.dnd.DropTarget;
import java.awt.dnd.DropTargetDragEvent;
import java.awt.dnd.DropTargetDropEvent;
import java.awt.dnd.DropTargetEvent;
import java.awt.dnd.DropTargetListener;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.io.BufferedInputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.RandomAccessFile;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.ArrayList;
import java.util.List;

import javax.imageio.ImageIO;
import javax.swing.DefaultComboBoxModel;
import javax.swing.ImageIcon;
import javax.swing.JButton;
import javax.swing.JCheckBox;
import javax.swing.JColorChooser;
import javax.swing.JComboBox;
import javax.swing.JDialog;
import javax.swing.JFileChooser;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JList;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.JTextArea;
import javax.swing.JTextPane;
import javax.swing.ListSelectionModel;
import javax.swing.border.EmptyBorder;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import javax.swing.filechooser.FileFilter;
import javax.swing.table.TableModel;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.Document;
import javax.swing.text.SimpleAttributeSet;
import javax.swing.text.StyleConstants;

/**
 * 实现群、私聊、发送文字、图片、文件。
 * 文件支持单次同时发送7个,多线程接收。
 * @author lucky star
 *
 */
public class Main extends JFrame {

	private JPanel contentPane;
	private JTextPane sendPane = null;
	private JTextPane recvPane = null;
	private JList onlineList = null;
	private JButton sendFileBtn = null, recvFileBtn = null;
	// 发送和接收的进度条
	// private JProgressBar sendProgressBar = null, recvProgressBar = null;
	// 要发送的文件
	private List<String> sendFiles = new ArrayList<String>();
	// 要接收的文件
	private List<String> recvFiles = new ArrayList<String>();
	// 文件发送进度
	private int fileLen = 0;
	// chat use TCP protocol
	private Socket s = null;
	private ObjectOutputStream oos = null;
	private Document doc = null;
	private Document slDoc = null;
	private String user = null;
	// 是否勾选私聊
	private boolean isSL = false;
	// 与哪个私聊
	private String slUser = null;
	// 私聊面板
	private JTextPane sltextPane = null;
	private Font f = null; // 字体对话框返回的字体。
	// 发送和接受消息面板的样式
	private SimpleAttributeSet msgAttrSet = new SimpleAttributeSet();
	// 显示发送人和时间的样式
	private SimpleAttributeSet tipAttrSet = null;
	private JTable table;

	public Font getF() {
		return f;
	}

	public void setF(Font f) {
		this.f = f;
	}

	/**
	 * Launch the application.
	 */
	/*
	 * public static void main(String[] args) { EventQueue.invokeLater(new
	 * Runnable() { public void run() { try { Main frame = new Main();
	 * frame.setVisible(true); } catch (Exception e) { e.printStackTrace(); } }
	 * }); }
	 */

	// connect to server
	public void connect() {
		try {
			s = new Socket(InetAddress.getLocalHost(), 8888);
			oos = new ObjectOutputStream(s.getOutputStream());
		} catch (UnknownHostException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

	// send msg to server
	public void sendMsg(MsgInfo mi) {
		try {
			oos.writeObject(mi);
			oos.flush();
		} catch (IOException e) {
			System.out.println("send msg failed.");
			e.printStackTrace();
		}
	}

	// disconnect to server when exit.
	public void disconnect() {
		try {
			if (oos != null)
				oos.close();
			if (s != null)
				s.close();
		} catch (IOException e) {
			System.out.println("exit chat.");
			e.printStackTrace();
		}
	}

	public void insertString(String str, AttributeSet attributeset) {
		try {
			doc.insertString(doc.getLength(), str, attributeset);
		} catch (BadLocationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

	public void insertImg(Image img) {
		recvPane.insertIcon(new ImageIcon(img));
	}

	/**
	 * Create the frame.
	 */
	public Main(String user) {
		this.user = user;
		setTitle("User " + user);

		FontAttr fa = new FontAttr(new Font("楷体", Font.PLAIN, 12), Color.BLUE,
				Color.WHITE);
		tipAttrSet = fa.getAttributeSet();
		setF(new Font("楷体", Font.PLAIN, 12));

		setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
		setBounds(100, 100, 773, 651);
		setLocationRelativeTo(null);
		contentPane = new JPanel();
		contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
		setContentPane(contentPane);
		contentPane.setLayout(null);

		JScrollPane jscrollPane1 = new JScrollPane();
		jscrollPane1.setBounds(0, 28, 432, 199);
		contentPane.add(jscrollPane1);

		recvPane = new JTextPane();
		jscrollPane1.setViewportView(recvPane);
		recvPane.setEditable(false);
		doc = recvPane.getStyledDocument();

		JButton imgBtn = new JButton("\u56FE\u7247");
		imgBtn.addActionListener(new ActionListener() {
			public void actionPerformed(ActionEvent arg0) {
				// 选择图片
				JFileChooser jfc = new JFileChooser();
				jfc.setDialogTitle("选择一张图片");
				jfc.setFileFilter(new FileFilter() {
					private String accepts[] = { "jpg", "jpeg", "gif", "png",
							"bmp" };

					@Override
					public String getDescription() {
						// TODO Auto-generated method stub
						return null;
					}

					@Override
					public boolean accept(File arg0) {
						boolean b = false;
						for (String s : accepts) {
							if (arg0.getName().endsWith(s)) {
								b = true;
								break;
							}
						}
						return b;
					}
				});

				int r = jfc.showOpenDialog(Main.this);
				if (r == JFileChooser.APPROVE_OPTION) { // 确定
					File f = jfc.getSelectedFile();
					List<File> imgs = new ArrayList<File>();
					imgs.add(f);

					MsgInfo mi = new MsgInfo();
					mi.setSender(Main.this.user);
					mi.setImgs(imgs);
					if (isSL) { // 私聊
						mi.setUser(slUser);
						// 将图片添加到私聊窗口
						try {
							try {
								slDoc.insertString(
										slDoc.getLength(),
										Main.this.user + "\t"
												+ TimeUtil.getSysTime() + "\n",
										tipAttrSet);
								sltextPane.setCaretPosition(slDoc.getLength());
								sltextPane.insertIcon(new ImageIcon(ImageIO
										.read(f)));
								slDoc.insertString(slDoc.getLength(), "\n",
										tipAttrSet);
							} catch (BadLocationException e) {
								// TODO Auto-generated catch block
								e.printStackTrace();
							}
						} catch (IOException e) {
							// TODO Auto-generated catch block
							e.printStackTrace();
						}
					} else {
						// 将图片添加到群聊窗口
						try {
							insertString(
									Main.this.user + "\t"
											+ TimeUtil.getSysTime() + "\n",
									tipAttrSet);
							recvPane.setCaretPosition(doc.getLength());
							insertImg(ImageIO.read(f));
							insertString("\n", tipAttrSet);
						} catch (IOException e) {
							// TODO Auto-generated catch block
							e.printStackTrace();
						}
					}
					sendMsg(mi);
				}
			}
		});
		imgBtn.setBounds(264, 453, 65, 23);
		contentPane.add(imgBtn);

		JButton fileBtn = new JButton("\u6587\u4EF6");
		fileBtn.addActionListener(new ActionListener() {
			public void actionPerformed(ActionEvent arg0) { // 发送文件
				JFileChooser jfc = new JFileChooser("D:/");
				jfc.setDialogTitle("选择文件或目录");
				jfc.setDialogType(JFileChooser.OPEN_DIALOG);
				int r = jfc.showOpenDialog(Main.this);
				if (r == JFileChooser.APPROVE_OPTION) { // 单击确定按钮
					File file = jfc.getSelectedFile();
					table.getModel().setValueAt(file.getName(), 0, 0);
					sendFiles.clear();
					sendFiles.add(file.getPath());
					sendFileBtn.setEnabled(true);
					//new SendFileTrd(10000, 0, file.getPath()).start();
				}
			}
		});
		fileBtn.setBounds(339, 453, 72, 23);
		contentPane.add(fileBtn);

		JScrollPane scrollPane_1 = new JScrollPane();
		scrollPane_1.setBounds(0, 487, 431, 86);
		contentPane.add(scrollPane_1);

		sendPane = new JTextPane();
		scrollPane_1.setViewportView(sendPane);

		JButton sendMsgBtn = new JButton("\u53D1\u9001");
		sendMsgBtn.addActionListener(new ActionListener() {
		sendMsgBtn.setBounds(366, 583, 66, 23);
		contentPane.add(sendMsgBtn);

		JLabel label_3 = new JLabel("\u5728\u7EBF\u5217\u8868");
		label_3.setBounds(450, 10, 54, 15);
		contentPane.add(label_3);

		JButton button = new JButton("\u5B57\u4F53");
		button.addActionListener(new ActionListener() { // 字体
		button.setBounds(0, 456, 72, 23);
		contentPane.add(button);

		JButton button_1 = new JButton("\u5B57\u4F53\u989C\u8272");
		button_1.addActionListener(new ActionListener() {
		button_1.setBounds(78, 455, 94, 23);
		contentPane.add(button_1);

		JButton button_2 = new JButton("\u80CC\u666F");
		button_2.addActionListener(new ActionListener() {
		button_2.setBounds(182, 453, 72, 23);
		contentPane.add(button_2);

		addWindowListener(new WindowAdapter() {
			@Override
			public void windowClosing(WindowEvent windowevent) {
				disconnect();
				System.exit(0);
			}
		});

		JScrollPane scrollPane = new JScrollPane();
		scrollPane.setBounds(0, 272, 432, 171);
		contentPane.add(scrollPane);

		sltextPane = new JTextPane();
		slDoc = sltextPane.getStyledDocument();
		scrollPane.setViewportView(sltextPane);

		JLabel label_1 = new JLabel("\u7FA4\u804A\uFF1A");
		label_1.setBounds(0, 3, 54, 15);
		contentPane.add(label_1);

		onlineList = new JList();
		onlineList.setBounds(442, 28, 313, 314);
		contentPane.add(onlineList);

		JLabel label_2 = new JLabel("\u79C1\u804A\uFF1A");
		label_2.setBounds(0, 245, 54, 15);
		contentPane.add(label_2);

		final JCheckBox checkBox = new JCheckBox("\u79C1\u804A");
		checkBox.setBounds(239, 583, 103, 23);
		contentPane.add(checkBox);

		JScrollPane scrollPane_2 = new JScrollPane();
		scrollPane_2.setBounds(442, 387, 313, 192);
		contentPane.add(scrollPane_2);
		String[][] data = new String[7][2];
		for (int i = 0; i < 7; i++) {
			for (int j = 0; j < 2; j++) {
				data[i][j] = "";
			}
		}
		String[] headers = { "文件", "进度" };
		PopupMenu popupmenu = new PopupMenu();
		popupmenu.add(new MenuItem("移除"));
		table = new JTable(data, headers);
		scrollPane_2.setViewportView(table);
		table.setAutoResizeMode(JTable.AUTO_RESIZE_ALL_COLUMNS);
		table.setRowHeight(25);
		table.add(popupmenu);
		table.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);

		JLabel label = new JLabel("\u6587\u4EF6\u53D1\u9001/\u63A5\u6536\uFF1A");
		label.setBounds(450, 352, 138, 25);
		contentPane.add(label);
		onlineList.addMouseListener(new MouseAdapter() {
			@Override
			public void mousePressed(MouseEvent mouseevent) {
				// 私聊
				slUser = (String) onlineList.getSelectedValue();
			}
		});

		// 私聊checkbox
		checkBox.addChangeListener(new ChangeListener() {

			public void stateChanged(ChangeEvent arg0) {
				isSL = checkBox.isSelected();
				boolean a = false,b =false;
				if (isSL && (slUser == null || slUser.equals(""))) {
					a = true;
				}
				else if (slUser != null && slUser.equals(Main.this.user)) {
					b = true;
				}
				
				if (a || b) {
					String tipTxt = a?"请先选择要私聊的人":"跟自己聊天很有意思吗?";
					JOptionPane.showMessageDialog(Main.this, tipTxt, "提示", JOptionPane.WARNING_MESSAGE);
					checkBox.setSelected(false);
					isSL = false;
				}
			}
		});

		// table.add
		// 文件拖拽支持
		DropFile df = new DropFile();
		DropTarget dt = new DropTarget(contentPane,
				DnDConstants.ACTION_REFERENCE, df, true);
		contentPane.setDropTarget(dt);

		sendFileBtn = new JButton("\u53D1\u9001\u6587\u4EF6");
		sendFileBtn.setEnabled(false);
		sendFileBtn.addActionListener(new ActionListener() {
			public void actionPerformed(ActionEvent actionevent) { // 发送文件
				sendFileBtn.setEnabled(false);
				int port = 9999; // 发送文件的端口号,10000起。
				// 启动发送文件的线程。
				for (String filename : Main.this.sendFiles) {
					port++;
					// 对应的行和列
					int rowIdx = 0;
					TableModel tm = table.getModel();
					int rows = tm.getRowCount();
					for (int i = 0; i < rows; i++) {
						String column0 = (String) tm.getValueAt(i, 0);
						if (filename.endsWith(column0)) {
							// 第二列是文件发送的进度。
							rowIdx = i;
							System.out.println(column0 + ":" + rowIdx);
							break;
						}
					}
					if (isSL) {
						try {
							slDoc.insertString(
									slDoc.getLength(),
											"您\t"
											+ TimeUtil.getSysTime()
											+ "\n给"
											+ slUser
											+ "发送文件:"
											+ filename.substring(filename
													.lastIndexOf("\\")) + "\n",
									tipAttrSet);
						} catch (BadLocationException e) {
							// TODO Auto-generated catch block
							e.printStackTrace();
						}
					} else {
						insertString(
								"您"
										+ TimeUtil.getSysTime()
										+ "\n"
										+ Main.this.user
										+ "发送文件:"
										+ filename.substring(filename
												.lastIndexOf("\\")) + "给所有人\n",
								tipAttrSet);
					}
					System.out.println("启动发送文件" + filename + "线程。");
					new SendFileTrd(port, rowIdx, filename).start();
				}
			}
		});
		sendFileBtn.setBounds(555, 583, 81, 23);
		contentPane.add(sendFileBtn);

		recvFileBtn = new JButton("\u63A5\u6536\u6587\u4EF6");
		recvFileBtn.setEnabled(false);
		recvFileBtn.addActionListener(new ActionListener() {
			public void actionPerformed(ActionEvent actionevent) { // 接收文件
				recvFileBtn.setEnabled(false);
				// 弹出文件保存的位置
				String dir = null;
				int rowIdx = 0;
				JFileChooser jfc = new JFileChooser();
				jfc.setDialogTitle("选择文件保存的位置");
				jfc.setSelectedFile(new File(
						"D:\\"));
				jfc.setFileSelectionMode(JFileChooser.DIRECTORIES_ONLY); // 选择文件夹。
				int r = jfc.showOpenDialog(Main.this);
				if (r == JFileChooser.APPROVE_OPTION) {
					File path = jfc.getSelectedFile();
					dir = path.getPath();
				}
				System.out.println("共"+Main.this.recvFiles.size()+"个文件。");
				int port = 9999; // 接收文件的端口号,10000起。
				for (int i = 0; i < Main.this.recvFiles.size(); i++) {
					port++;
					String filename = Main.this.recvFiles.get(i);
					int rows = table.getModel().getRowCount();
					for (int j=0;j<rows;j++) {
						if (table.getModel().getValueAt(j, 0).equals(filename)) {
							rowIdx = j;
							break;
						}
					}
					new RecvFileTrd(port, rowIdx, dir, filename).start();
					System.out.println("开始接收文件" + filename);
				}
			}
		});
		recvFileBtn.setBounds(646, 583, 81, 23);
		contentPane.add(recvFileBtn);

		connect();
		new Thread(new RecvTrd(this)).start();
		// send empty msg to get online user list
		MsgInfo mi = new MsgInfo();
		mi.setSender(user);
		sendMsg(mi);
	}

	public SimpleAttributeSet getMsgAttrSet() {
		return msgAttrSet;
	}

	public void setMsgAttrSet(SimpleAttributeSet msgAttrSet) {
		this.msgAttrSet = msgAttrSet;
	}

	/**
	 * 接收消息、图片、文件的线程。
	 * 
	 * @author lucky star
	 * 
	 */
	private class RecvTrd implements Runnable {
		private Main mcf = null;
		private boolean isConnected = false;
		private ObjectInputStream ois = null;

		public RecvTrd(Main mcf) {
			this.mcf = mcf;
			try {
				ois = new ObjectInputStream(mcf.s.getInputStream());
				isConnected = true;
			} catch (IOException e) {
				disconnect();
				System.out.println("连接服务器异常:" + e.getMessage());
			}
		}

		public void run() {
			while (isConnected) {
				try {
					// read data from InputStream.
					MsgInfo mi = (MsgInfo) ois.readObject();

					if (!mi.isSL()) { // 群聊信息显示到群聊窗口。
						// if data is not null,then append it to the receive
						// panel.
						if (mi.getMsgContent() != null
								&& !"".equals(mi.getMsgContent())) {
							insertString(
									mi.getSender() + "\t"
											+ TimeUtil.getSysTime() + "\n",
									tipAttrSet);
							insertString(mi.getMsgContent() + "\n", msgAttrSet);

							// put the frame to front,let the use to know that
							// there is msg coming.
							mcf.toFront();
						}

						if (mi.getImgs() != null && mi.getImgs().size() > 0) { // 发送图片
							List<File> imgs = mi.getImgs();
							insertString(
									mi.getSender() + "\t"
											+ TimeUtil.getSysTime() + "\n",
									tipAttrSet);
							recvPane.setCaretPosition(doc.getLength());
							for (int i = 0; i < imgs.size(); i++) {
								File f = imgs.get(i);
								insertImg(ImageIO.read(f));
							}
							insertString("\n", msgAttrSet);
							mcf.toFront();
						}

						// get online user list
						if (mi.getOnlines() != null
								&& mi.getOnlines().size() > 0) {
							// put the current online user to onlineList
							mcf.onlineList.setListData(mi.getOnlines()
									.toArray());
						}
					} else { // 信息显示到私聊窗口。
						System.out.println("私聊。");
						if (mi.getMsgContent() != null
								&& !mi.getMsgContent().equals("")) {
							try {
								slDoc.insertString(
										slDoc.getLength(),
										mi.getSender() + "\t"
												+ TimeUtil.getSysTime() + "\n",
										tipAttrSet);
								slDoc.insertString(slDoc.getLength(),
										mi.getMsgContent() + "\n", msgAttrSet);
							} catch (BadLocationException e) {
								// TODO Auto-generated catch block
								e.printStackTrace();
							}
							mcf.toFront();
						}
						if (mi.getImgs() != null && mi.getImgs().size() > 0) { // 发送图片
							List<File> imgs = mi.getImgs();
							try {
								slDoc.insertString(
										slDoc.getLength(),
										mi.getSender() + "\t"
												+ TimeUtil.getSysTime()
												+ "\n\n", tipAttrSet);
							} catch (BadLocationException e) {
								// TODO Auto-generated catch block
								e.printStackTrace();
							}
							
							sltextPane.setCaretPosition(slDoc.getLength());
							for (int i = 0; i < imgs.size(); i++) {
								File f = imgs.get(i);
								sltextPane.insertIcon(new ImageIcon(ImageIO
										.read(f)));
							}
							insertString("\n", tipAttrSet);
							mcf.toFront();
						}
						
					}

					if (mi.getFiles() != null && mi.getFiles().length > 0) { // 发送过来文件
						System.out.println("接收文件");
						if (mi.isSL()) {
							Main.this.isSL = true;
							Main.this.slUser = mi.getSender();
						}
						String[] files = mi.getFiles();
						recvFiles.clear();
						System.out.println("发送文件"+files.length+"个");
						for (int i = 0; i < files.length; i++) {
							String f = files[i];
							// d:\a\b\c.txt
							table.getModel().setValueAt(
									f.substring(f.lastIndexOf("\\") + 1), i, 0);
							recvFiles.add(f.substring(f.lastIndexOf("\\")+1));
						}
						// 通知对方有文件接收
						recvFileBtn.setEnabled(true);
						mcf.toFront();
					}
					if (mi.getFileLen() != -1) { // 接收文件
						// recvProgressBar.setMaximum(mi.getFileLen());
						fileLen = mi.getFileLen();
						System.out.println("需发送数据包 " + fileLen + "次...");
					}
				} catch (IOException e) {
					System.out.println("read msg failed.");
					isConnected = false;
					try {
						if (ois != null)
							ois.close();
						if (s != null)
							s.close();
					} catch (IOException e1) {
						// TODO Auto-generated catch block
						e1.printStackTrace();
					}
					// e.printStackTrace();
				} catch (ClassNotFoundException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
		}

	}

	/**
	 * 字体属性集合。
	 * 
	 * @author lucky star
	 * 
	 */
	private class FontAttr {
		@Override
		public String toString() {
			return "FontAttr [fontName=" + fontName + ", fontStyle="
					+ fontStyle + ", fontSize=" + fontSize + ", foreColor="
					+ foreColor + ", backColor=" + backColor + "]";
		}

		private String fontName;
		private int fontStyle;
		private int fontSize;
		private Color foreColor = null, backColor = null;
		private SimpleAttributeSet attrSet = new SimpleAttributeSet();

		public FontAttr(SimpleAttributeSet sas, Font f) {
			this(f);
			this.attrSet = sas;
		}

		public FontAttr(Font f, Color foreColor, Color backColor) {
			this(f);
			this.foreColor = foreColor;
			this.backColor = backColor;
		}

		public FontAttr(Font f) {
			this.fontName = f.getName();
			this.fontSize = f.getSize();
			this.fontStyle = f.getStyle();
		}

		public SimpleAttributeSet getAttributeSet() {
			if (fontName != null && !"".equals(fontName)) {
				StyleConstants.setFontFamily(attrSet, fontName);
			}
			if (fontStyle == 0) { // 常规
				StyleConstants.setBold(attrSet, false);
				StyleConstants.setItalic(attrSet, false);
			} else if (fontStyle == 1) { // 粗体
				StyleConstants.setBold(attrSet, true);
				StyleConstants.setItalic(attrSet, false);
			} else if (fontStyle == 2) { // 斜体
				StyleConstants.setBold(attrSet, false);
				StyleConstants.setItalic(attrSet, true);
			} else if (fontStyle == 3) { // 粗体 斜体
				StyleConstants.setBold(attrSet, true);
				StyleConstants.setItalic(attrSet, true);
			}
			if (foreColor != null) { // 背景色
				StyleConstants.setForeground(attrSet, foreColor);
			}
			if (backColor != null) { // 前景色
				StyleConstants.setBackground(attrSet, backColor);
			}
			if (fontSize != -1) {
				StyleConstants.setFontSize(attrSet, fontSize);
			}

			return attrSet;
		}
	}

	/**
	 * 字体选择对话框。
	 * 
	 * @author lucky star
	 * 
	 */
	private class FontDialog extends JDialog {

		private final JPanel contentPanel = new JPanel();
		private JComboBox fontNameBox = null;
		private JComboBox fontStyleBox = null;
		private JComboBox fontSizeBox = null;
		private JTextArea txtrHereIs = null;

		private String fontName;
		private String fontStyle;
		private String fontSize;
		private int fontSty;
		private int fontSiz;

		/**
		 * Create the dialog.
		 */
		public FontDialog(final Main main) {
			this.setModal(true);
			setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
			setLocationRelativeTo(main);
			setTitle("\u5B57\u4F53");
			setBounds(100, 100, 483, 234);
			getContentPane().setLayout(new BorderLayout());
			contentPanel.setBorder(new EmptyBorder(5, 5, 5, 5));
			getContentPane().add(contentPanel, BorderLayout.CENTER);
			contentPanel.setLayout(null);
			{
				JLabel lblf = new JLabel("\u5B57\u4F53(F):");
				lblf.setBounds(0, 10, 54, 15);
				contentPanel.add(lblf);
			}
			{
				JLabel lbly = new JLabel("\u5B57\u5F62(Y):");
				lbly.setBounds(182, 10, 54, 15);
				contentPanel.add(lbly);
			}
			{
				JLabel lbls = new JLabel("\u5927\u5C0F(S):");
				lbls.setBounds(315, 10, 54, 15);
				contentPanel.add(lbls);
			}
			{
				JLabel label = new JLabel("\u663E\u793A\u6548\u679C:");
				label.setBounds(126, 82, 64, 15);
				contentPanel.add(label);
			}

			Panel panel = new Panel();
			panel.setBounds(196, 40, 228, 113);
			contentPanel.add(panel);
			panel.setLayout(null);
			{
				txtrHereIs = new JTextArea();
				txtrHereIs.setBounds(39, 38, 177, 44);
				txtrHereIs
						.setText("\u8FD9\u91CC\u663E\u793A\u9884\u89C8\r\nHere is the preview");
				panel.add(txtrHereIs);
			}
			{
				fontNameBox = new JComboBox();
				fontNameBox.setBounds(49, 7, 123, 21);
				contentPanel.add(fontNameBox);
				fontNameBox.addItemListener(new ItemListener() {
					public void itemStateChanged(ItemEvent itemevent) {
						fontName = (String) itemevent.getItem();
						System.out.println(fontName);

						// change preview
						Font f = new Font(fontName, fontSty, fontSiz);
						txtrHereIs.setFont(f);
					}
				});
			}
			{
				fontStyleBox = new JComboBox();
				fontStyleBox.setBounds(232, 7, 73, 21);
				contentPanel.add(fontStyleBox);
				fontStyleBox.addItemListener(new ItemListener() {
					public void itemStateChanged(ItemEvent itemevent) {
						fontStyle = (String) itemevent.getItem();
						fontSty = getFontStyleByCnName(fontStyle);
						// change preview
						Font f = new Font(fontName, fontSty, fontSiz);
						txtrHereIs.setFont(f);
					}
				});
			}
			{
				fontSizeBox = new JComboBox();
				fontSizeBox.setBounds(379, 7, 78, 21);
				contentPanel.add(fontSizeBox);
				fontSizeBox.addItemListener(new ItemListener() {
					public void itemStateChanged(ItemEvent itemevent) {
						fontSize = (String) itemevent.getItem();
						fontSiz = Integer.parseInt(fontSize);

						// change preview
						Font f = new Font(fontName, fontSty, fontSiz);
						txtrHereIs.setFont(f);
					}
				});
			}
			{
				JPanel buttonPane = new JPanel();
				buttonPane.setLayout(new FlowLayout(FlowLayout.RIGHT));
				getContentPane().add(buttonPane, BorderLayout.SOUTH);
				{
					JButton okButton = new JButton("\u786E\u5B9A");
					okButton.addActionListener(new ActionListener() {
						public void actionPerformed(ActionEvent actionevent) {
							int fontSty = getFontStyleByCnName(fontStyle);
							int fontSiz = Integer.parseInt(fontSize);
							/*
							 * JOptionPane.showMessageDialog(FontDialog.this,
							 * "你选择的字体名称:" + fontName + ",字体样式:" + fontStyle +
							 * ",字体大小:" + fontSiz, "提示",
							 * JOptionPane.CLOSED_OPTION);
							 */
							main.setF(getChoice());
							Font f = main.getF();
							System.out.println("fontName:" + f.getName()
									+ ",fontStyle:" + f.getStyle()
									+ ",fontSize:" + f.getSize());
							FontDialog.this.dispose();
						}
					});
					okButton.setActionCommand("OK");
					buttonPane.add(okButton);
					getRootPane().setDefaultButton(okButton);
				}
				{
					JButton cancelButton = new JButton("\u53D6\u6D88");
					cancelButton.setActionCommand("Cancel");
					buttonPane.add(cancelButton);
					cancelButton.addActionListener(new ActionListener() {
						public void actionPerformed(ActionEvent actionevent) {
							FontDialog.this.dispose();
						}
					});
				}
			}

			// 初始化字体名称
			GraphicsEnvironment ge = GraphicsEnvironment
					.getLocalGraphicsEnvironment();
			String[] fontNames = ge.getAvailableFontFamilyNames();
			fontNameBox.setModel(new DefaultComboBoxModel(fontNames));
			// 初始化字体样式
			String[] fontStyles = { "常规", "斜体", "粗体", "粗斜体" };
			fontStyleBox.setModel(new DefaultComboBoxModel(fontStyles));
			// 初始化字体大小
			String[] fontSizes = { "8", "9", "10", "11", "12", "14", "16",
					"18", "20", "22", "24", "26", "28", "36", "48", "72" };
			fontSizeBox.setModel(new DefaultComboBoxModel(fontSizes));
			System.out.println("finish.");

			// 根据聊天窗口的字体设置来设置

			fontNameBox.setSelectedItem(main.getF().getName());
			fontSizeBox.setSelectedItem(main.getF().getSize() + "");
			fontStyleBox.setSelectedIndex(main.getF().getStyle());
			fontStyle = (String) fontStyleBox.getSelectedItem();
			fontSize = (String) fontSizeBox.getSelectedItem();
			fontSty = getFontStyleByCnName(fontStyle);
			fontSiz = Integer.parseInt(fontSize);
		}

		public Font getChoice() {
			return new Font(fontName, fontSty, fontSiz);
		}

		public int getFontStyleByCnName(String fontStyle) {
			if (fontStyle.equals("常规")) {
				return Font.PLAIN;
			}
			if (fontStyle.equals("斜体")) {
				return Font.ITALIC;
			}
			if (fontStyle.equals("粗体")) {
				return Font.BOLD;
			}
			if (fontStyle.equals("粗斜体")) {
				return Font.BOLD + Font.ITALIC;
			}
			return -1;
		}
	}

	/**
	 * 拖拽文件。
	 * 
	 * @author lucky star
	 * 
	 */
	private class DropFile implements DropTargetListener {

		public void dragEnter(DropTargetDragEvent droptargetdragevent) {
			// TODO Auto-generated method stub

		}

		public void dragOver(DropTargetDragEvent droptargetdragevent) {
			// TODO Auto-generated method stub

		}

		public void dropActionChanged(DropTargetDragEvent droptargetdragevent) {
			// TODO Auto-generated method stub

		}

		public void dragExit(DropTargetEvent droptargetevent) {
			// TODO Auto-generated method stub

		}

		public void drop(DropTargetDropEvent droptargetdropevent) {
			droptargetdropevent.acceptDrop(DnDConstants.ACTION_REFERENCE);
			if (droptargetdropevent
					.isDataFlavorSupported(DataFlavor.javaFileListFlavor)) {
				Transferable trans = droptargetdropevent.getTransferable();
				try {

					@SuppressWarnings("unchecked")
					List<File> files = (List<File>) trans
							.getTransferData(DataFlavor.javaFileListFlavor);
					
					if (files.size() > 7) { // 主要是显示发送/接收的table事写死的7行,没有添加动态添加行。
						JOptionPane.showMessageDialog(Main.this, "一次最多只能发送7个文件", "提示", JOptionPane.WARNING_MESSAGE);
						return; 
					}

					for (int i = 0; i < files.size(); i++) {
						File f = files.get(i);
						table.getModel().setValueAt(f.getName(), i, 0);
						sendFiles.add(f.getPath());
					}

					if (files.size() > 0) {
						sendFileBtn.setEnabled(true);
					}

				} catch (UnsupportedFlavorException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				} catch (IOException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
		}

	}

	/**
	 * 发送文件的线程,一次发送一个文件。多个文件时多个发送线程同时启动。
	 * 
	 * @author lucky star
	 * 
	 */
	private class SendFileTrd extends Thread {
		private int port = 0;
		private int rowIdx = 0;
		private String filename = null;
		private int tmpfileLen = 0;

		public SendFileTrd(int port, int rowIdx, String filename) {
			this.port = port;
			this.rowIdx = rowIdx;
			this.filename = filename;
		}

		public void run() {

			// 获取要发送的文件
			File f = new File(filename);
			int db = 0;
			if ((f.length() % (1024 * 100) != 0)) {
				db = (int) (f.length() / (1024 * 100)) + 1;
			}
			if (db < 1) {
				db = 1;
			}

			System.out.println("[SendFileTrd]:将要发送数据包" + db + "次");

			// 先发信息通知对方接收文件,并将文件的总大小发过去。
			MsgInfo mi = new MsgInfo();
			mi.setSender(user);
			// 接收方只能看到文件名,不能看到发送方的文件完整路径。
			String[] sfiles = new String[sendFiles.size()];
			for (int i = 0; i < sendFiles.size(); i++) {
				sfiles[i] = sendFiles.get(i).substring(
						sendFiles.get(i).lastIndexOf("\\") + 1);
				System.out.println("通知对方接收文件:" + sfiles[i]);
			}
			mi.setFiles(sfiles);
			if (isSL) {
				mi.setUser(slUser);				
			}
			mi.setFileLen(db); // 设置对方接收进度条的最大值。
			sendMsg(mi);
			System.out.println("已经通知对方。。。");

			ServerSocket ss = null;
			Socket s = null;
			DataOutputStream dos = null;
			FileInputStream fis = null;

			// 创建文件发送 服务器,对方点接收文件按钮后创建一个Socket,连接发送服务器。
			try {
				ss = new ServerSocket(port);
				s = ss.accept(); // 阻塞直到对方单击接收文件按钮,选择保存路径,这样就激活连接了。
				// 获取网络输出流
				dos = new DataOutputStream(s.getOutputStream());
				// sendProgressBar.setMaximum(db);
				// sendProgressBar.setMinimum(0);
				fis = new FileInputStream(f);
				byte[] buff = new byte[1024 * 100]; // 一次发送100Kb
				int len = -1;

				while ((len = fis.read(buff)) != -1) {
					tmpfileLen++;
					// sendProgressBar.setValue(tmpfileLen);
					// sendProgressBar
					// .setString("已发送"
					// + (100 * tmpfileLen / db)
					// + "%");
					table.getModel().setValueAt(
							"已发送" + (100 * tmpfileLen / db) + "%", rowIdx, 1);

					// 写数据到网络输出流
					dos.write(buff, 0, len);
					dos.flush();
				}

				try {
					if (isSL) {
						slDoc.insertString(slDoc.getLength(), "系统消息:\t"
								+ TimeUtil.getSysTime() + "\n文件【" + filename
								+ "】发送完毕\n", tipAttrSet);
					} else {
						insertString("系统消息:\t" + TimeUtil.getSysTime()
								+ "\n文件【" + filename + "】发送完毕\n", tipAttrSet);
					}
				} catch (BadLocationException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}

				// sendProgressBar.setValue(0);
				// sendProgressBar.setString("");
				sendFiles.remove(filename);

				// 一个文件发送完毕,从表格中清除。
				table.getModel().setValueAt("", rowIdx, 0);
				table.getModel().setValueAt("", rowIdx, 1);

				System.out.println("send rowIdx:" + rowIdx);

				if (sendFiles.size() == 0) {
					if (isSL) {
						try {
							slDoc.insertString(slDoc.getLength(), "系统消息:\t"
									+ TimeUtil.getSysTime() + "\n所有文件都已发送完毕\n",
									tipAttrSet);
						} catch (BadLocationException e) {
							// TODO Auto-generated catch block
							e.printStackTrace();
						}
					} else {
						insertString("系统消息:\t" + TimeUtil.getSysTime()
								+ "\n所有文件都已发送完毕\n", tipAttrSet);
					}
				}
			} catch (IOException e) {
				try {
					if (isSL) {
						slDoc.insertString(slDoc.getLength(),
								"系统消息:\t" + TimeUtil.getSysTime() + "\n文件发送异常:"
										+ e.getMessage() + "\n", tipAttrSet);
					} else {
						insertString("系统消息:\t" + TimeUtil.getSysTime()
								+ "\n文件发送异常:" + e.getMessage() + "\n",
								tipAttrSet);
					}
				} catch (BadLocationException ex) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			} finally {
				try {
					if (fis != null)
						fis.close();
					if (dos != null)
						dos.close();
					if (s != null)
						s.close();
					if (ss != null)
						ss.close();
				} catch (IOException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}

		}
	}

	/**
	 * 接收文件的线程。一个线程负责接收一个文件。多个文件时多个接收线程同时启动。
	 * 
	 * @author lucky star
	 * 
	 */
	private class RecvFileTrd extends Thread {
		private Socket s = null;
		// table的row index,用于在接收某个文件完毕后,从表格中删除。
		private int rowIdx = 0;
		// 文件保存的位置
		private String filePath = null;
		// 接收的文件名
		private String recvName = null;
		// 端口号
		private int port = 0;
		private int tmpFileLen = 0;
		private int total = 0; // 记录已经读取的文件字节数。

		public RecvFileTrd(int port, int rowIdx, String filePath,
				String recvName) {
			this.port = port;
			this.rowIdx = rowIdx;
			this.filePath = filePath;
			this.recvName = recvName;
		}

		public void run() {
			DataInputStream dis = null;
			RandomAccessFile raf = null;
			try {
				// 连接文件发送服务器。
				s = new Socket(InetAddress.getLocalHost(), port);
				// 获取网络输入流。
				dis = new DataInputStream(new BufferedInputStream(
						s.getInputStream()));
				// 文件要保存的位置。
				File path = new File(filePath +File.separator+ recvName);
				int num = 0;
				while (path.exists()) { // 文件存在时自动改名。
					num++;
					recvName = recvName.substring(0,recvName.lastIndexOf(".")) + "("+num +")"+ recvName.substring(recvName.lastIndexOf("."));
					path = new File(filePath +File.separator+ recvName);
				}
				path.createNewFile();

				raf = new RandomAccessFile(path, "rw");
				// recvProgressBar.setMinimum(0);
				// recvProgressBar.setMaximum(fileLen);

				System.out.println("接收数据包" + fileLen + "次");

				byte[] buff = new byte[1024 * 100]; // 一次接收100Kb
				int len = -1;

				// 不能使用一次读取100K计数加1的方式计算读取的进度。
				// 因为可能没有读取到100K。这样计算的进度就有误。
				// 通过total叠加每次读取的字节数,然后除以100K,计算进度。
				// fileLen是文件发送服务器计算的文件长度。它是通过文件总大小除以100K计算来的。
				while ((len = dis.read(buff)) != -1) {
					total += len;
					tmpFileLen = total / (1024*100);
					// recvProgressBar.setValue(tmpFileLen);
					// recvProgressBar.setString("已接收"
					// + (100 * tmpFileLen / fileLen) + "%");
					// System.out.println("recv file progress:" + (100 *
					// tmpFileLen / fileLen) + "%");
					table.getModel().setValueAt(
							"已接收" + (100 * tmpFileLen / fileLen) + "%", rowIdx,
							1);
					System.out.println("tmpFileLen:" + tmpFileLen + ",fileLen:" + fileLen + ",len:" + len);
					raf.write(buff, 0, len);
				}
				
				System.out.println("tmpFileLen :" + tmpFileLen + ",fileLen:" + fileLen);

				try {
					if (isSL) {
						slDoc.insertString(slDoc.getLength(), "系统消息:\t"
								+ TimeUtil.getSysTime() + "\n文件【" + recvName
								+ "】接收完毕。\n文件保存在" + filePath + "\n", tipAttrSet);
					} else {
						insertString("系统消息:\t" + TimeUtil.getSysTime()
								+ "\n文件【" + recvName + "】接收完毕。\n文件保存在"
								+ filePath + "\n", tipAttrSet);
					}
				} catch (BadLocationException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}

				// recvProgressBar.setValue(0);
				// recvProgressBar.setString("");
				recvFiles.remove(recvName);
				// 一个文件接收完毕,从表格中删除。
				table.getModel().setValueAt("", rowIdx, 0);
				table.getModel().setValueAt("", rowIdx, 1);
				System.out.println("recv rowIdx:" + rowIdx);
				if (recvFiles.size() == 0) {
					if (isSL) {
						try {
							slDoc.insertString(slDoc.getLength(),
									"系统消息:\t" + TimeUtil.getSysTime()
											+ "\n所有文件都已接收完毕。\n", tipAttrSet);
						} catch (BadLocationException e) {
							// TODO Auto-generated catch block
							e.printStackTrace();
						}
					} else {
						insertString("系统消息:\t" + TimeUtil.getSysTime()
								+ "\n所有文件都已接收完毕。\n", tipAttrSet);
					}
				}
			} catch (UnknownHostException e) {
				try {
					if (isSL) {
						slDoc.insertString(slDoc.getLength(), "系统消息:\t"
								+ TimeUtil.getSysTime() + "\n无法连接文件发送服务器\n",
								tipAttrSet);
					} else {
						insertString("系统消息:\t" + TimeUtil.getSysTime()
								+ "\n无法连接文件发送服务器\n", tipAttrSet);
					}
				} catch (BadLocationException ex) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			} catch (IOException e) {
				try {
					if (isSL) {
						slDoc.insertString(slDoc.getLength(),
								"系统消息:\t" + TimeUtil.getSysTime() + "\n接收文件异常:"
										+ e.getMessage() + "\n", tipAttrSet);
					} else {
						insertString("系统消息:\t" + TimeUtil.getSysTime()
								+ "\n接收文件异常:" + e.getMessage() + "\n",
								tipAttrSet);
					}
				} catch (BadLocationException ex) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
				System.out.println(e.getMessage());
			} finally {
				try {
					if (dis != null)
						dis.close();
					if (raf != null)
						raf.close();
					if (s != null)
						s.close();
				} catch (IOException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
		}
	}
}


辅助类:

 

package com.socket.tcp.basechat;

import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JDialog;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.border.EmptyBorder;

/**
 * 登陆窗口。
 * @author lucky star
 *
 */
public class LoginChat extends JDialog {

	private final JPanel contentPanel = new JPanel();
	private JTextField userNameField;

	/**
	 * Launch the application.b
	 */
	public static void main(String[] args) {
		try {
			LoginChat dialog = new LoginChat();
			dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
			dialog.setVisible(true);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	/**
	 * Create the dialog.
	 */
	public LoginChat() {
		setLocationByPlatform(true);
		setBounds(100, 100, 450, 114);
		getContentPane().setLayout(new BorderLayout());
		contentPanel.setBorder(new EmptyBorder(5, 5, 5, 5));
		getContentPane().add(contentPanel, BorderLayout.CENTER);
		contentPanel.setLayout(null);
		{
			JLabel label = new JLabel("\u7528\u6237\u540D\uFF1A");
			label.setBounds(10, 10, 54, 15);
			contentPanel.add(label);
		}

		userNameField = new JTextField();
		userNameField.setBounds(75, 7, 349, 21);
		contentPanel.add(userNameField);
		userNameField.setColumns(10);
		{
			JPanel buttonPane = new JPanel();
			buttonPane.setLayout(new FlowLayout(FlowLayout.RIGHT));
			getContentPane().add(buttonPane, BorderLayout.SOUTH);
			{
				JButton okButton = new JButton("OK");
				okButton.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent actionevent) {
						String txt = userNameField.getText();
						if (txt == null || "".equals(txt)) {
							JOptionPane.showMessageDialog(LoginChat.this, "请输入用户名","提示",JOptionPane.WARNING_MESSAGE);
							userNameField.requestFocus();
							return;
						}
						EventQueue.invokeLater(new Runnable() {
							public void run() {
								try {
									Main frame = new Main(userNameField
											.getText());
									frame.setVisible(true);
								} catch (Exception e) {
									e.printStackTrace();
								}
							}
						});
						LoginChat.this.dispose();
					}
				});
				okButton.setActionCommand("OK");
				buttonPane.add(okButton);
				getRootPane().setDefaultButton(okButton);
			}
			{
				JButton cancelButton = new JButton("Cancel");
				cancelButton.addActionListener(new ActionListener() {
					public void actionPerformed(ActionEvent actionevent) {
						LoginChat.this.dispose();
					}
				});
				cancelButton.setActionCommand("Cancel");
				buttonPane.add(cancelButton);
			}
		}
	}
}

 

package com.socket.tcp.basechat;

import java.text.SimpleDateFormat;
import java.util.Calendar;

public final class TimeUtil {

	// get the current system date and time.
	public static String getSysTime() {
		return new SimpleDateFormat("yyyy/MM/dd HH:mm:ss").format(Calendar.getInstance().getTime());
	}
}


服务器端其实很简单:

 

1.启动服务器,监听某端口

2.阻塞等待客户端连接

3.客户端连接上了,启动一个线程处理客户端信息的转发(转发给所有人,或某个人)。

根据登录名区分各个客户端。——主要功能不在这,只是提供输入用户名并没有做注册和登陆验证。

 

客户端:

包含一个群聊面板,一个私聊面板,可选择字体,颜色,可发送图片,可选择发送单个文件,一个支持多个文件同时发送和接收的表格。

为每个发送的文件启动一个文件发送线程处理,在线程中先发送信息告诉接收方要接收的文件以及要发送的文件大小(除以100*1024后的结果,用于计算发送和接收文件的进度)。每次发送或接收100KB。在一个文件发送完毕后从表格中清除。

接收方单击接收按钮后提示选择文件保存的位置,然后启动为每个文件启动接收各个文件的接收文件线程。某个文件接收完毕后,从表格中删除。

 

如转载请注明出处,谢谢。

分享到:
评论

相关推荐

    java socket多人聊天(文字+图片+文件)

    在"java socket多人聊天(文字+图片+文件)"这个项目中,开发者创建了一个支持多用户同时在线聊天的系统,不仅可以发送文字消息,还能交换图片和文件。以下是关于这个项目的一些关键知识点和实现细节: 1. **Socket...

    java socket多人聊天(文字+图片+文件传输)

    本项目实现了基于Java Socket的多人聊天功能,包括群聊、私聊以及文本、图片和文件的发送与接收,提供了丰富的交互体验。 首先,我们要理解Socket的工作原理。Socket是网络通信的一种端点,它允许两个应用程序通过...

    Java编写的多人聊天+用户在线即时聊天系统源代码

    该源代码是一个基于Java技术实现的多人在线即时聊天系统,旨在提供一个实时交流的平台,让用户可以方便地进行文字、语音甚至文件的交互。在这个系统中,我们可以学习到多个Java编程相关的知识点,包括网络编程、多...

    java 多人聊天 socket

    Java多人聊天系统是基于Socket网络编程技术实现的,它允许多个用户通过客户端与服务器进行实时的文字对话。在Java中,Socket通信是实现网络应用程序间通信的基础,它提供了两台计算机间建立TCP连接的方法,使得数据...

    Qt实现聊天小程序,发送文字以及图片并显示

    在本文中,我们将深入探讨如何使用Qt框架来实现一个简单的聊天小程序,该程序支持发送文字消息和图片,并具有用户友好的界面。Qt是一个强大的跨平台应用程序开发框架,它提供了丰富的库和工具,使得构建GUI应用变得...

    java swing 多人聊天室

    Java Swing 多人聊天室是一个基于Java编程语言的桌面应用程序,设计用于实现多个用户之间的实时通信。这个系统利用了Swing库来构建图形用户界面(GUI),提供了丰富的组件和功能,使得用户可以方便地进行文字聊天。...

    局域网多人聊天,语音聊天,文件传输

    本项目“局域网多人聊天,语音聊天,文件传输”是基于C++编程语言实现的,它允许在同一局域网内的多台计算机之间进行实时的文字聊天、语音交流以及文件共享。下面我们将详细探讨这一系统的核心技术和实现方法。 ...

    java多人聊天室(服务器端和客户端)

    Java多人聊天室是一个基于TCP协议和Java IO技术实现的网络通信项目,主要目的是提供一个平台,使得多个用户可以通过网络进行实时的文字交流。这个项目不仅涵盖了基础的编程概念,还涉及了网络编程的核心原理,是学习...

    C# QQ多人聊天.zip

    总的来说,这个项目提供了C# Winform环境下使用Socket进行网络通信的实例,适合初学者了解和学习多人聊天室的基本原理和实现。通过扩展这个项目,可以增加更多功能,如文件传输、语音聊天、表情支持等,进一步增强...

    python socket局域网聊天与文件传输.zip

    "实现多线程"意味着程序能够同时处理多个任务,比如在接收和发送消息或文件时,不会阻塞其他操作,提高了程序的效率和响应性。最后,"跨平台,windows和linux下运行都可"指出该程序可以在多种操作系统环境下运行,如...

    socket TCP网络聊天室编程

    它需要有登录、发送文字消息、发送文件、接收消息和文件等功能。客户端通过TCP Socket与服务器建立连接,发送用户的输入,并接收服务器返回的信息。 5. **服务器(server)**:服务器端负责监听端口,等待客户端...

    多人聊天室

    8. **用户界面设计**:聊天室的用户界面需要友好且易于使用,包括显示在线用户列表、接收和发送消息的功能,以及可能的文件传输、表情支持等功能。Java GUI库如Swing或JavaFX可以用来创建这样的界面。 9. **数据库...

    python实现简单多人聊天室

    Python实现的简单多人聊天室是一种基于网络通信的应用,它允许多个用户通过客户端连接到服务器,进行实时的文字交流。在这个系统中,服务器端负责接收新用户的连接、管理在线用户列表以及广播用户发送的消息,而...

    多人在线聊天(Java版)

    在这个Java版的多人聊天系统中,主要包含客户端和服务器端两个核心部分,它们各自承担不同的功能。 客户端模块是用户与聊天系统的交互界面。在Java中,可以使用Swing或JavaFX等库来构建图形用户界面(GUI),提供...

    java swing语音聊天室,文字的有单聊,群聊,强制下线等

    文字聊天功能的实现依赖于Swing中的JTextArea和JTextPane组件,它们用于显示和输入文字信息。同时,事件监听器(ActionListener)用于捕捉用户点击发送按钮的动作,将输入的文字发送到服务器,并在接收方的界面上...

    基于Socket的as3的小型聊天室

    在互联网编程中,Socket是实现客户端与服务器之间实时通信的基础技术,它允许应用程序通过网络发送和接收数据。 【描述】提到的应用是用AS3实现的聊天室客户端,这意味着它依赖于Adobe Flash Player或Adobe AIR运行...

    socket通信

    一旦连接建立成功,客户端就可以通过Socket向服务端发送数据,如文字消息、图片等。同样,客户端也可以接收服务端发送过来的数据,并显示在用户界面上。 在这个“多人聊天”环境中,服务端需要维护一个客户端连接...

    C#(Socket&Tcp)实现的局域网聊天工具源码

    本项目“C#(Socket&Tcp)实现的局域网聊天工具源码”就是这样一个实例,它利用C#的Socket库和TCP协议实现了局域网内的通信功能,提供了丰富的交互方式,如多人聊天、一对多及一对一私聊,并且具备聊天记录的保存和...

    vb视频聊天源码程序

    这些控件能够访问摄像头和麦克风,捕获并编码音视频数据,然后通过网络发送到接收端。在接收端,同样需要解码并播放这些数据。 2. **网络通信协议**:视频聊天需要一个可靠的网络通信协议来传输数据。TCP(传输控制...

Global site tag (gtag.js) - Google Analytics