zoukankan      html  css  js  c++  java
  • java通过控制鼠标实现屏幕广播

    java实现屏幕共享的小程序中提到截取屏幕时是没鼠标,为了看到教师端界面上的鼠标,可以在截取屏幕的时候,把鼠标绘制到每一张截图上去,但是由于截图的时候是一张张截取的,所以看到的鼠标难免会有点卡,之前写了java鼠标操控小程序,可以通过这种方式来看到鼠标的演示。

    实现的方式也挺简单的,前面两篇文章分别实现了鼠标控制和不带鼠标的屏幕分享功能,把这两个结合一下就ok了,下面简单分析下。

    服务端,将SendScreenImg和SendMouseMessage看作两个工具类,分别监听不同的端口,他们两个都实现了Thread类,我们用线程池ExecutorService类控制他们。

    使用了两个端口,因为暂时还不知道该如何吧鼠标信息和图片的信息一起发送,或许可以把图片转换成字节数组的形式,把鼠标的坐标放在数组前面,不过这样的话鼠标可能也会不连贯,因为传送鼠标坐标的速度会比传图片的要快一些,嗯,有空再试试。

    客户端类比上面就是了。

    下面是代码:

    服务端:

    主程序:

    /*
     * 屏幕广播类,调用了两个工具类:发送截屏信息的类和发送鼠标的信息类,利用了线程池。
     */
    public class BroderCast {
    	public  static void main(String[] args)
    	{
    		new BroderCast();
    		System.out.println("开始");
    	}
    	public BroderCast()
    	{
    		ExecutorService exector = Executors.newFixedThreadPool(2);
    		exector.execute(new SendScreenImg());
    		exector.execute(new SendMouseMessage());
    	}
    }

    发送截图:

    import java.awt.Dimension;
    import java.awt.Rectangle;
    import java.awt.Robot;
    import java.awt.Toolkit;
    import java.awt.image.BufferedImage;
    import java.io.DataOutputStream;
    import java.io.IOException;
    import java.net.ServerSocket;
    import java.net.Socket;
    import java.util.zip.ZipEntry;
    import java.util.zip.ZipOutputStream;
    import javax.imageio.ImageIO;
    
    /*
     * 工具类:发送教师端截屏信息给学生端,没有鼠标信息,使用了8002号端口
     * 		可以在发送的图片上面组件绘制鼠标的信息,从而实现在学生端界面上见到鼠标信息,暂未实现该功能
     * 				
     */
    public	class SendScreenImg extends Thread
    {
    	public int serverPort=8002;
    	private ServerSocket serverSocket;
    	private Robot robot;
    	public  Dimension screen;
    	public Rectangle rect ;
    	private Socket socket; 
    	public static void main(String args[])
    	{
    		new SendScreenImg().start();
    	}
    	
    	public void changeServerPort(int serverPort)
    	{
    		if(this.serverPort == serverPort) return ;
    		try{
    			this.serverSocket.close();    //有必要先关闭当前端口
    			this.serverPort = serverPort;
    			serverSocket = new ServerSocket(this.serverPort);
    			serverSocket.setSoTimeout(8000000);
    		}catch(Exception e){}
    		
    	}
    	
    	//构造方法  开启套接字连接      机器人robot   获取屏幕大小
    	public SendScreenImg()
    	{
    		try {
    			serverSocket = new ServerSocket(getServerPort());
    			serverSocket.setSoTimeout(864000000);
    			robot = new Robot();
    		} catch (Exception e) {
    			e.printStackTrace();
    		}
    		screen = Toolkit.getDefaultToolkit().getScreenSize();  //获取主屏幕的大小
    		rect = new Rectangle(screen);                          //构造相应大小的矩形
    		
    	}
    	@Override
    	public void run() 
    	{
    		//实时等待接收截屏消息
    		while(true){
    			try {
    				socket = serverSocket.accept();
    				ZipOutputStream zip = new ZipOutputStream(new DataOutputStream(socket.getOutputStream()));
    				zip.setLevel(9);     //设置压缩级别
    				try{
    					BufferedImage img = robot.createScreenCapture(rect);
    					zip.putNextEntry(new ZipEntry("test.jpg"));
                        ImageIO.write(img, "jpg", zip);
                        if(zip!=null)zip.close();
                        System.out.println("学生端口已经连接");
                    } catch (IOException ioe) {
                        System.out.println("被控端:disconnect");
                    }
                } catch (IOException ioe) {
               	 System.out.println("连接出错");
                } finally {
                    if (socket != null) {
                        try {
                            socket.close();
                        } catch (IOException e) {
                        }
                    }
                }
            }
    	}
    }

    发送鼠标信息:

    /*
     * 工具类:获取鼠标的信息并且发送给学生端
     */
    public class SendMouseMessage extends Thread{
    	private int OPERATE_PORT = 8001;
    	private ServerSocket server;
    	private Socket socket;
    	private String operateStr;
    	public static void main(String[] args)
    	{
    		new SendMouseMessage().start();
    	}
    	public SendMouseMessage(){
    		try {
    			server = new ServerSocket(OPERATE_PORT);
    			//JOptionPane.showMessageDialog(null, "已经开始监听");
    		} catch (IOException e1) {
    			e1.printStackTrace();
    		}		
    	}
    	//多线程  在无线的循环中监听客户端的
    	public void run()
    	{
    		while(true){
    			Point point = MouseInfo.getPointerInfo().getLocation();  //
    			operateStr ="Movemouse,"+point.x+","+point.y;
    			try {
    				socket = server.accept();
    				socket.setSoTimeout(1000000);
    				DataOutputStream output =new DataOutputStream(socket.getOutputStream());
    				output.write(operateStr.getBytes());
    				output.flush();   //刷行输出流,并且使所有缓冲的输出字节写出
    				output.close();   //关闭输出流且释放资源
    	
    				System.out.println("INFO:  "+operateStr);
    			} catch (IOException e) {
    				System.out.println("已经停止连接");
    				break;   //断开连接的时候就停止无线循环
    			}
    		}
    	}
    }

    客户端:

    主程序:

    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
    
    import com.Tool.OperateMouse;
    import com.Tool.ReceiveImages;
    
    public class ReceiveBroderCast {
    	public ExecutorService exector;
    	public static String IP="202.216.60.9";
    	
    	public static void main(String[] args)
    	{
    		new ReceiveBroderCast(IP);
    	}
    	
    	public ReceiveBroderCast(String IP) {
    		exector = Executors.newFixedThreadPool(2);
    		exector.execute(new ReceiveImages(IP));
    		exector.execute(new OperateMouse(IP));
    	}
    }

    接收截图:

    /*
     * ly  2014-11-20
     * 该类用于接收教师端的屏幕信息,不包括鼠标
     * 使用socket()
     */
    public  class ReceiveImages extends  Thread{
    	public BorderInit frame ;
    	public Socket socket;
    	public String IP;
    	
    	public static void main(String[] args){
    		new ReceiveImages("202.216.60.7").start();
    	}
    	public ReceiveImages(String IP)
    	{
    		frame=new BorderInit();
    		this.IP=IP;
    		
    	}
    	public void run() {
    		while(frame.getFlag()){
    			System.out.println("已经连接"+(System.currentTimeMillis()/1000)%24%60+"秒");
    			try {
    				socket = new Socket(IP,8002);
    				DataInputStream ImgInput = new DataInputStream(socket.getInputStream());
    				ZipInputStream imgZip = new ZipInputStream(ImgInput);
    				Image img = null;
    				try{
    					imgZip.getNextEntry();   //到Zip文件流的开始处
    					img = ImageIO.read(imgZip); //按照字节读取Zip图片流里面的图片
    					frame.jlbImg.setIcon(new ImageIcon(img));
    					frame.validate();
    				}catch (IOException e) {e.printStackTrace();}
    				
    				try{
    					imgZip.close();
    				} catch (IOException e) {
    					System.out.println("连接断开");
    				}
    				try {
    					TimeUnit.MILLISECONDS.sleep(50);// 接收图片间隔时间
    				} catch (InterruptedException ie) {
    					ie.printStackTrace();
    				}		
    			} catch (IOException e) {
    				e.printStackTrace();
    			}finally{
    				try {
    					socket.close();
    				} catch (IOException e) {}	
    			}		
    		}	
    	}
    }
    class BorderInit extends JFrame
    {
    	private static final long serialVersionUID = 1L;
    	public JLabel jlbImg;
    	private boolean flag;
    	public boolean getFlag(){
    		return this.flag;
    	}
    	public BorderInit()
    	{
    		this.flag=true;
    		this.jlbImg = new JLabel();
    		this.setTitle("远程监控--IP:"  + "--主题:" );
    		this.setSize(400, 400);
    		//this.setUndecorated(true);
    		//this.setAlwaysOnTop(true);  //始终在最前面
    		this.add(jlbImg);
    		this.setLocationRelativeTo(null);
    		this.setExtendedState(Frame.MAXIMIZED_BOTH);
    		this.setDefaultCloseOperation(DISPOSE_ON_CLOSE);
    		this.setVisible(true);
    		this.validate();
    	
    		//窗口关闭事件
    		this.addWindowListener(new WindowAdapter() {
    			public void windowClosing(WindowEvent e) {
    				flag=false;
    				BorderInit.this.dispose();
    				System.out.println("窗体关闭");
    				System.gc();    //垃圾回收
    			}
    		});
    	}
    }

    接收鼠标信息并控制鼠标移动:

    import java.awt.AWTException;
    import java.awt.Robot;
    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.DataInputStream;
    import java.io.IOException;
    import java.io.InputStream;
    import java.net.Socket;
    
    import javax.swing.JOptionPane;
    
    /*
     * 学生端   控制鼠标和教师端一致
     * 该类  负责接收鼠标的信息  并且用robot.mouseMove()函数控制鼠标移动  
     */
    public class OperateMouse extends Thread{
    	public static void main(String[] args)
    	{
    		new OperateMouse("202.116.60.7").start();
    	}
    	private Socket socket;
    	public String IP;
    	private int OPERATE_PORT = 8001;
    	private Robot robot;
    	
    	public OperateMouse(String IP)
    	{
    		this.IP = IP;
    	}
    	public void run() {
    		while(true){
    			try {
    				socket = new Socket(IP,OPERATE_PORT);
    				robot = new Robot();
    				//获取鼠标移动的信息
    				DataInputStream dataIn = new DataInputStream(socket.getInputStream());        
    				String info="";
    				int r;
    				while((r=dataIn.read()) != -1){
    					info +=""+(char)r;   //把字节数组中所有元素都变为字符型
    				}
    				dataIn.close();
    				System.out.println("数据流断开"+info);
    			 	if(info!=null){
    			 			String s[] = info.trim().split(","); 
    						if("Movemouse".equals(s[0].trim()));
    						{
    							if (s.length == 3) {
    								int x = Integer.parseInt(s[1].trim());
    								int y = Integer.parseInt(s[2].trim());
    								System.out.println("输出鼠标的信息"+x+"  "+ y);
    								robot.mouseMove(x, y);
    							}
    						}
    					}
    			} catch (IOException e) {
    				System.out.println("已断开连接");
    				break;
    			} catch (AWTException e) {
    				e.printStackTrace();
    			}
    		}
    	}
    }

    本博客出自博客园兰幽

  • 相关阅读:
    RIA Animation test.
    深入浅出REST
    HTTP header中的 Cachecontrol
    Silverlight操纵html元素
    Comparing Azure storage and SQL Data Services
    SharePoint 2010 集成Window Live 认证遇到的问题 part 2
    SharePoint2010 此工作簿未存储在 Excel Services 应用程序 中受信任的位置,因此无法打开。
    SharePoint2010 使用Designer开发工作流 如何引用其他列表的查阅项
    SharePoint 2010 item级的权限控制
    Infopath 2010 如何设计虚线底框应用于合同中的输入框
  • 原文地址:https://www.cnblogs.com/LZYY/p/4164321.html
Copyright © 2011-2022 走看看