java鼠标钩子的运用

      去年寒假回家闲着无事,就在家中敲敲代码。主要是用java语言写一些小的运用:放大镜,屏幕截图

简单画图工具、取色器什么的,,最后整合成一个小型的软件。不过在这个过程中,发现java中实现屏幕

截图、取色的时候对于鼠标的操作还是显得有些不够,上面的功能一般是事先得到整个屏幕,之后再

在该图片上操作。这使得界面软件必须处于激活状态,很不方便。

      其实,采用上面的那种操作实现功能也并不见得有多不好,不过兴趣使然,想看看java借助一些第

三方jar能否实现与windows底层的交互。因此,上网查了一下,总结如下:

      通过查找,了解到了可以通过"鼠标钩子"实现,何为"钩子",下面为"扒来的":

           钩子(Hook),是Windows消息处理机制的一个平台,应用程序可以在上面设置子程以监视指定窗
      口的某种消息,而且所监视的窗口可以是其他进程所创建的。当消息到达后,在目标窗口处理函数之
      前处理它。钩子机制允许应用程序截获处理window消息或特定事件。钩子实际上是一个处理消息的程
      序段,通过系统调用,把它挂入系统。每当特定的消息发出,在没有到达       目的窗口前,钩子程序就
       先捕获该消息,亦即钩子函数先得到控制权。这时钩子函数即可以加工处理(改变)该消息,也可以
       不作处理而继续传递该消息,还可以强制结束消息的传递。
             Hook API是指Windows开放给程序员的编程接口,使得在用户级别下可以对操作系统进行控制,
       也就是一般的应用程序都需要调用API来完成某些功能,Hook API的意思就是在这些应用程序调用
       真正的系统API前可以先被截获,从而进行一些处理再调用真正的API来完成功能。
                 1) 键盘钩子和低级键盘钩子可以监视各种键盘消息。 
              (2) 鼠标钩子和低级鼠标钩子可以监视各种鼠标消息。 
              (3) 外壳钩子可以监视各种Shell事件消息。比如启动和关闭应用程序。 
              (4) 日志钩子可以记录从系统消息队列中取出的各种事件消息。 
              (5) 窗口过程钩子监视所有从系统消息队列发往目标窗口的消息
        说了这么多概念性的东西,但是我们怎样用java实现鼠标钩子呢?
            这就需要借助jna来操作了
                 JNA(Java Native Access )提供一组Java工具类用于在运行期动态访问系统本地库(native library:如Window的dll)而不需要编写任何Native/JNI代码。开发人员只要在一个java接口中描述目标native library的函数与结构,JNA将自动实现Java接口到native function的映射。
 既然知道了实现的方法,那就动手吧,下面是我实现的代码,当然也有借鉴网上这方面的资料.
 MouseLLHook.java
package com.kiritor;  import java.awt.AWTException; import java.awt.Color; import java.awt.Cursor; import java.awt.MouseInfo; import java.awt.Point; import java.awt.Robot;  import com.sun.jna.Pointer; import com.sun.jna.platform.win32.*; import com.sun.jna.platform.win32.WinUser.*; import com.sun.jna.platform.win32.WinDef.HMODULE; import com.sun.jna.platform.win32.WinDef.LRESULT; import com.sun.jna.platform.win32.WinDef.WPARAM; import com.sun.jna.platform.win32.WinUser.HHOOK; import com.sun.jna.platform.win32.WinUser.KBDLLHOOKSTRUCT; import com.sun.jna.platform.win32.WinUser.LowLevelKeyboardProc;  public class MouseLLHook {  	 	private boolean flag; 	public ColorFrame getColorFrame() { 		return colorFrame; 	}   	public void setColorFrame(ColorFrame colorFrame) { 		this.colorFrame = colorFrame; 	}  	private static ColorFrame colorFrame = null; 	// 鼠标钩子函数里判断按键类型的常数 	public static Point mousepoint; 	public static MouseInfo mouseInfo = null; 	static Color pixel = new Color(0, 0, 0); 	static Integer R = 0, G = 0, B = 0; 	static Integer X = 0, Y = 0; 	public static final int WM_LBUTTONUP = 514; 	public static final int WM_LBUTTONDOWN = 513; 	public static final int WM_RBUTTONUP = 517; 	public static final int WM_RBUTTONDOWN = 516; 	public static final int WM_MOUSEHWHEEL = 526; 	public static final int WM_MOUSEWHEEL = 522; 	public static final int WM_MOUSEMOVE = 512;  	static HHOOK mouseHHK, keyboardHHK;// 鼠标、键盘钩子的句柄 	static LowLevelMouseProc mouseHook;// 鼠标钩子函数 	static LowLevelKeyboardProc keyboardHook;// 键盘钩子函数     public MouseLLHook(ColorFrame colorFrame)     { 	   this.colorFrame=colorFrame;     } 	 	 	// 安装钩子 	static void setHook() { 		HMODULE hMod = Kernel32.INSTANCE.GetModuleHandle(null); 		mouseHHK = User32.INSTANCE.SetWindowsHookEx(WinUser.WH_MOUSE_LL, 				mouseHook, hMod, 0); 		mouseHHK.nativeType(); 		keyboardHHK = User32.INSTANCE.SetWindowsHookEx(WinUser.WH_KEYBOARD_LL, 				keyboardHook, hMod, 0); 	}  	// 卸载钩子 	static void unhook() { 		User32.INSTANCE.UnhookWindowsHookEx(keyboardHHK); 		User32.INSTANCE.UnhookWindowsHookEx(mouseHHK); 	}  	public  void getColor(String[] args) {  		keyboardHook = new LowLevelKeyboardProc() {  			@Override 			// 该函数参数的意思参考:http://msdn.microsoft.com/en-us/library/windows/desktop/ms644985(v=vs.85).aspx 			public LRESULT callback(int nCode, WPARAM wParam, 					KBDLLHOOKSTRUCT lParam) { 				int w = wParam.intValue(); 				// 按下alt键时w=.WM_SYSKEYDOWN; 按下其他大部分键时w=WinUser.WM_KEYDOWN 				if (w == WinUser.WM_KEYDOWN || w == WinUser.WM_SYSKEYDOWN) 					System.out.println("key down: vkCode = " + lParam.vkCode); 				else if (w == WinUser.WM_KEYUP || w == WinUser.WM_SYSKEYUP) 					System.out.println("key up: vkCode = " + lParam.vkCode);  				// 如果按下'q'退出程序,'q'的vkCode是81 				if (lParam.vkCode == 81) { 					unhook(); 					System.err.println("program terminated."); 					System.exit(0); 				} 				return User32.INSTANCE.CallNextHookEx(keyboardHHK, nCode, 						wParam, lParam.getPointer()); 			} 		};  		mouseHook = new LowLevelMouseProc() {  			@Override 			// 该函数参数的意思参考:http://msdn.microsoft.com/en-us/library/windows/desktop/ms644986(v=vs.85).aspx 			public LRESULT callback(int nCode, WPARAM wParam, 					MOUSEHOOKSTRUCT lParam) { 				switch (wParam.intValue()) { 				case WM_MOUSEMOVE: 					//System.out.print("mouse moved:"); 					break; 				case WM_LBUTTONDOWN: { 					try { 						 						Robot robot = new Robot(); 						mousepoint = mouseInfo.getPointerInfo().getLocation(); 						// System.out.println(mousepoint); 						pixel = robot.getPixelColor(mousepoint.x, mousepoint.y); 						X = mousepoint.x; 						Y = mousepoint.y; 						R = pixel.getRed(); 						G = pixel.getGreen(); 						B = pixel.getBlue(); 				 						String s16 = "#" + Integer.toHexString(R) 								+ Integer.toHexString(G) 								+ Integer.toHexString(B);// 得到颜色的十六进制表示。 					 						Color col = new Color(R, G, B); 						 						if(colorFrame.isFlag()==false) 						{ 							colorFrame.getRedText().setText(R.toString()); 							colorFrame.getGreenText().setText(G.toString()); 							colorFrame.getBlueText().setText(B.toString()); 							colorFrame.getColor16Text().setText(s16); 							colorFrame.getNewColor().setBackground(col); 						    colorFrame.setFlag(true); 						} 						 						colorFrame.setVisible(true); 					} catch (AWTException ex) { 						ex.printStackTrace(); 					} 				} 					break; 				case WM_LBUTTONUP: 					//System.out.print("mouse left button up"); 					break; 				case WM_RBUTTONUP: 					//System.out.print("mouse right button up:"); 					break; 				case WM_RBUTTONDOWN: 					//System.out.print("mouse right button down:"); 					break; 				case WM_MOUSEWHEEL: 					//System.out.print("mouse wheel rotated:"); 					break; 				} 				//System.out.println("(" + lParam.pt.x + "," + lParam.pt.y + ")"); 				return User32.INSTANCE.CallNextHookEx(mouseHHK, nCode, wParam, 						lParam.getPointer()); 			} 		};  		setHook();  		int result; 		MSG msg = new MSG(); 		// 消息循环 		// 实际上while循环一次都不执行,这些代码的作用我理解是让程序在GetMessage函数这里阻塞,不然程序就结束了。 		while ((result = User32.INSTANCE.GetMessage(msg, null, 0, 0)) != 0) { 			if (result == -1) { 				System.err.println("error in GetMessage"); 				unhook(); 				break; 			} else { 				User32.INSTANCE.TranslateMessage(msg); 				User32.INSTANCE.DispatchMessage(msg); 			} 		} 		unhook(); 	}  }

package com.kiritor; import com.sun.jna.Structure; import com.sun.jna.platform.win32.BaseTSD.ULONG_PTR; import com.sun.jna.platform.win32.WinDef.HWND; import com.sun.jna.platform.win32.WinDef.LRESULT; import com.sun.jna.platform.win32.WinDef.WPARAM; import com.sun.jna.platform.win32.WinUser.HOOKPROC; import com.sun.jna.platform.win32.WinUser.POINT;  interface LowLevelMouseProc extends HOOKPROC {     LRESULT callback(int nCode, WPARAM wParam, MOUSEHOOKSTRUCT lParam); }  public class MOUSEHOOKSTRUCT extends Structure {     public class ByReference extends MOUSEHOOKSTRUCT implements             Structure.ByReference {     };      public POINT pt;     public HWND hwnd;     public int wHitTestCode;     public ULONG_PTR dwExtraInfo; }  
ColorFrame.java
package com.kiritor;  import java.awt.AWTException; import java.awt.BorderLayout; import java.awt.Color; import java.awt.Cursor; import java.awt.Dimension; import java.awt.Font; import java.awt.Graphics; import java.awt.GraphicsDevice; import java.awt.GraphicsEnvironment; import java.awt.Image; import java.awt.Menu; import java.awt.MenuItem; import java.awt.PopupMenu; import java.awt.Rectangle; import java.awt.Robot; import java.awt.SystemTray; import java.awt.Toolkit; import java.awt.TrayIcon; import java.awt.datatransfer.StringSelection; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.MouseAdapter; import java.awt.event.MouseEvent; import java.awt.event.MouseMotionAdapter;  import javax.swing.ImageIcon; import javax.swing.JButton; import javax.swing.JFrame; import javax.swing.JLabel; import javax.swing.JPanel; import javax.swing.JWindow;  public class ColorFrame extends JWindow { 	 	private boolean flag= false; 	public boolean isFlag() { 		return flag; 	}  	public void setFlag(boolean flag) { 		this.flag = flag; 	}  	private JLabel color16 = null; 	private JLabel color16Text = null; 	private JLabel red = null; 	private JLabel redText = null; 	private JLabel green = null; 	private JLabel greenText = null; 	private JLabel blue = null; 	private JLabel blueText = null; 	private JPanel newColor = null; 	private JLabel newLabel = null; 	private int xx; 	private int yy; 	private boolean isDraging; 	private TrayIcon trayIcon; 	private JPanel bgJPanel = null; 	private ImageIcon bg = new ImageIcon(getClass().getClassLoader() 			.getResource("images/bg.png"));  	/** 	 * 剪贴板工作区 	 */ 	// 复制到剪贴板的按钮 	private JLabel jiantieban = null; 	private ImageIcon jiantiebanIcon = new ImageIcon(getClass() 			.getClassLoader().getResource("images/jiantieban.png"));  	private ImageIcon jiantiebanIcon_entered = new ImageIcon(getClass() 			.getClassLoader().getResource("images/jiantieban_entered.png")); 	// 重新取色的按钮  	private JLabel getAgain = null; 	private ImageIcon getAgainIcon = new ImageIcon(getClass().getClassLoader() 			.getResource("images/getagain.png")); 	private ImageIcon getAgainIcon_entered = new ImageIcon(getClass() 			.getClassLoader().getResource("images/getagain_entered.png"));  	// 取消操作的按钮  	private JLabel cancel = null;  	private ImageIcon cancelIcon = new ImageIcon(getClass().getClassLoader() 			.getResource("images/cancel.png")); 	private ImageIcon cancel_entered = new ImageIcon(getClass() 			.getClassLoader().getResource("images/cancel_entered.png"));  	public ColorFrame(){} 	 	public ColorFrame(int x,int y) { 		setLocation(x, y); 	}  	public void showUI() { 		init();  		setSize(150, 150); 		setLayout(null); 		setAlwaysOnTop(true); 		addMouseListener(new MouseAdapter() { 			public void mousePressed(MouseEvent e) { 				requestFocus(); 				isDraging = true; 				xx = e.getX(); 				yy = e.getY(); 			}  			public void mouseReleased(MouseEvent e) { 				isDraging = false; 			} 			 			@Override 			public void mouseEntered(MouseEvent e) { 				// TODO Auto-generated method stub 				setFlag(true);//窗体内部暂时设置为不能取色 			} 		}); 		addMouseMotionListener(new MouseMotionAdapter() { 			public void mouseDragged(MouseEvent e) { 				if (isDraging) { 					int left = getLocation().x; 					int top = getLocation().y; 					setLocation(left + e.getX() - xx, top + e.getY() - yy); 				} 			} 		});  		setVisible(true); 		// 		// TestEvent t = new TestEvent(this); 		// t.setLayout(null); 		// t.add(bgJPanel); 		// this.getContentPane().add( t); 		add(bgJPanel); 		//setLocationRelativeTo(null);  	}  	public void init() {  		bgJPanel = new JPanel() { 			public void paintComponent(Graphics g) { 				super.paintComponent(g); 				g.drawImage(bg.getImage(), 0, 0, null); 				this.setOpaque(false); 			} 		}; 		bgJPanel.setOpaque(false); 		bgJPanel.setLayout(null); 		bgJPanel.setBounds(0, 0, 150, 150);  		newColor = new JPanel(); 		newLabel = new JLabel("颜色"); 		newLabel.setFont(new Font("华文楷体", 0, 12)); 		newLabel.setBounds(30, 10, 30, 20); 		newColor.setBounds(20, 35, 50, 50); 		newColor.setBackground(Color.gray);  		red = new JLabel("Red:"); 		red.setFont(new Font("华文楷体", 0, 12)); 		red.setBounds(80, 35, 30, 10);  		redText = new JLabel("sdf"); 		redText.setFont(new Font("华文楷体", 0, 12)); 		redText.setBounds(115, 35, 30, 10); 		green = new JLabel("Green:"); 		green.setFont(new Font("华文楷体", 0, 12)); 		green.setBounds(80, 53, 40, 10); 		greenText = new JLabel("sdf"); 		greenText.setFont(new Font("华文楷体", 0, 12)); 		greenText.setBounds(115, 53, 30, 10); 		blue = new JLabel("Blue:"); 		blue.setFont(new Font("华文楷体", 0, 12)); 		blue.setBounds(80, 71, 30, 10); 		blueText = new JLabel("sdf"); 		blueText.setFont(new Font("华文楷体", 0, 12)); 		blueText.setBounds(115, 71, 30, 10);  		color16 = new JLabel("十六进制:"); 		color16.setBounds(20, 95, 45, 10); 		color16.setFont(new Font("华文楷体", 0, 10)); 		color16Text = new JLabel("#aaaa"); 		color16Text.setBounds(70, 95, 60, 10); 		color16Text.setFont(new Font("华文楷体", 0, 12)); 		color16Text.setForeground(Color.red);  		/** 工作区按钮初始化的地方 */         //剪贴板操作 		jiantieban = new JLabel(jiantiebanIcon_entered); 		jiantieban.addMouseListener(new MouseAdapter() { 			@Override 			public void mouseClicked(MouseEvent e) { 				// TODO Auto-generated method stub 				 				setClipboard(color16Text.getText()); 			    			} 		 			@Override 			public void mouseEntered(MouseEvent e) { 				// TODO Auto-generated method stub 				jiantieban.setIcon(jiantiebanIcon); 				setFlag(true); 			} 		 			@Override 			public void mouseExited(MouseEvent e) { 				// TODO Auto-generated method stub 				jiantieban.setIcon(jiantiebanIcon_entered); 			   			} 		}); 		jiantieban.setBounds(10, 115, 40, 20);  		//重新拾取操作 		getAgain = new JLabel(getAgainIcon_entered); 		getAgain.addMouseListener(new MouseAdapter() { 			@Override 			public void mouseClicked(MouseEvent e) { 				// TODO Auto-generated method stub 				setFlag(false);  	            setVisible(false); 			}  			@Override 			public void mouseEntered(MouseEvent e) { 				// TODO Auto-generated method stub 				getAgain.setIcon(getAgainIcon); 				 			}  			@Override 			public void mouseExited(MouseEvent e) { 				// TODO Auto-generated method stub 				getAgain.setIcon(getAgainIcon_entered); 			} 		}); 		getAgain.setBounds(55, 115, 40, 20); 		 		//取消操作按钮的操作 		cancel = new JLabel(cancel_entered); 		cancel.addMouseListener(new MouseAdapter() { 			@Override 			public void mouseClicked(MouseEvent e) { 				setVisible(false); 			}  			@Override 			public void mouseEntered(MouseEvent e) { 				// TODO Auto-generated method stub 				cancel.setIcon(cancelIcon); 				 			}  			@Override 			public void mouseExited(MouseEvent e) { 				// TODO Auto-generated method stub 				cancel.setIcon(cancel_entered); 			} 		}); 		cancel.setBounds(100, 115, 40, 20); 		 		 		bgJPanel.add(newLabel); 		bgJPanel.add(newColor); 		bgJPanel.add(red); 		bgJPanel.add(redText); 		bgJPanel.add(green); 		bgJPanel.add(greenText); 		bgJPanel.add(blue); 		bgJPanel.add(blueText); 		bgJPanel.add(color16); 		bgJPanel.add(color16Text); 		bgJPanel.add(jiantieban); 		bgJPanel.add(getAgain); 		bgJPanel.add(cancel); 		 	}  	public JPanel getNewColor() { 		return newColor; 	}  	public void setNewColor(JPanel newColor) { 		this.newColor = newColor; 	}  	public JLabel getColor16Text() { 		return color16Text; 	}  	public void setColor16Text(JLabel color16Text) { 		this.color16Text = color16Text; 	}  	public JLabel getRedText() { 		return redText; 	}  	public void setRedText(JLabel redText) { 		this.redText = redText; 	}  	public JLabel getGreenText() { 		return greenText; 	}  	public void setGreenText(JLabel greenText) { 		this.greenText = greenText; 	}  	public JLabel getBlueText() { 		return blueText; 	}  	public void setBlueText(JLabel blueText) { 		this.blueText = blueText; 	}  	public  void getColor(String[] args) { 		 		 		new MouseLLHook(this).getColor(null); 	} 	public static void main(String[] args) { 		ColorFrame colorFrame = new ColorFrame(); 		colorFrame.showUI(); 		MouseLLHook hook = new MouseLLHook(colorFrame); 		hook.getColor(null); 	} 	/*复制到剪贴板中去*/ 	public  void setClipboard(String str)  	{     	StringSelection ss = new StringSelection(str);     	Toolkit.getDefaultToolkit().getSystemClipboard().setContents(ss, null);     }  } 
由于一些图片文件丢失,界面效果图就不贴了!
该程序要用到的jar包
 http://download.csdn.net/detail/kiritor/5115277
 

你可能感兴趣的:(java鼠标钩子的运用)