require 'win/library' module Win module GUI # Contains constants and Win32API functions related to end user input # module Input include Win::Library # Internal constants: # Key event delay KEY_DELAY = 0.00001 # Windows keyboard-related Constants: # Key down keyboard event (the key is being depressed) KEYEVENTF_KEYDOWN = 0 # Key up keyboard event (the key is being released) KEYEVENTF_KEYUP = 2 # Extended kb event. If specified, the scan code was preceded by a prefix byte having the value 0xE0 (224). KEYEVENTF_EXTENDEDKEY = 1 # Virtual key codes: # Control-break processing VK_CANCEL = 0x03 # Backspace? key VK_BACK = 0x08 # Tab key VK_TAB = 0x09 # Shift key VK_SHIFT = 0x10 # Ctrl key VK_CONTROL = 0x11 # ENTER key VK_RETURN = 0x0D # ALT key VK_ALT = 0x12 # ALT key alias VK_MENU = 0x12 # PAUSE key VK_PAUSE = 0x13 # CAPS LOCK key VK_CAPITAL = 0x14 # ESC key VK_ESCAPE = 0x1B # SPACEBAR VK_SPACE = 0x20 # PAGE UP key VK_PRIOR = 0x21 # PAGE DOWN key VK_NEXT = 0x22 # END key VK_END = 0x23 # HOME key VK_HOME = 0x24 # LEFT ARROW key VK_LEFT = 0x25 # UP ARROW key VK_UP = 0x26 # RIGHT ARROW key VK_RIGHT = 0x27 # DOWN ARROW key VK_DOWN = 0x28 # SELECT key VK_SELECT = 0x29 # PRINT key VK_PRINT = 0x2A # EXECUTE key VK_EXECUTE = 0x2B # PRINT SCREEN key VK_SNAPSHOT = 0x2C # INS key VK_INSERT = 0x2D # DEL key VK_DELETE = 0x2E # HELP key VK_HELP = 0x2F # Public Type MOUSEINPUT # dx As Long # dy As Long # mouseData As Long # dwFlags As Long # time As Long # dwExtraInfo As Long # End Type # # Public Type INPUT_TYPE # dwType As Long # xi(0 To 23) As Byte # End Type # dwFlags: # Specifies that the dx and dy parameters contain normalized absolute coordinates. If not set, those parameters # contain relative data: the change in position since the last reported position. This flag can be set, or not # set, regardless of what kind of mouse or mouse-like device, if any, is connected to the system. For further # information about relative mouse motion, see mouse_event Remarks section. MOUSEEVENTF_ABSOLUTE = 0x8000 #Specifies that movement occurred. MOUSEEVENTF_MOVE = 0x01 #Specifies that the left button is down. MOUSEEVENTF_LEFTDOWN = 0x02 #Specifies that the left button is up. MOUSEEVENTF_LEFTUP = 0x04 #Specifies that the right button is down. MOUSEEVENTF_RIGHTDOWN = 0x08 #Specifies that the right button is up. MOUSEEVENTF_RIGHTUP = 0x010 #Specifies that the middle button is down. MOUSEEVENTF_MIDDLEDOWN = 0x20 #Specifies that the middle button is up. MOUSEEVENTF_MIDDLEUP = 0x040 #Windows NT/2000/XP: Specifies that the wheel has been moved, if the mouse has a wheel. The amount of movement #is specified in dwData MOUSEEVENTF_WHEEL = 0x80 #Windows 2000/XP: Specifies that an X button was pressed. MOUSEEVENTF_XDOWN = 0x100 #Windows 2000/XP: Specifies that an X button was released. MOUSEEVENTF_XUP = 0x200 # dwData: # One wheel click is defined as WHEEL_DELTA, which is 120. WHEEL_DELTA = 120 # Set if the first X button was pressed or released. XBUTTON1 = 1 # Set if the second X button was pressed or released. XBUTTON2 = 2 # Indicates NO data if dwFlags are NOT any of MOUSEEVENTF_WHEEL, MOUSEEVENTF_XDOWN, or MOUSEEVENTF_XUP INPUT_MOUSE = 0 ## # The keybd_event function synthesizes a keystroke. The system can use such a synthesized keystroke to generate # a WM_KEYUP or WM_KEYDOWN message. The keyboard driver's interrupt handler calls the keybd_event function. # # !! Windows NT/2000/XP/Vista:This function has been superseded. Use SendInput instead. # # [*Syntax*] VOID keybd_event( BYTE bVk, BYTE bScan, DWORD dwFlags, PTR dwExtraInfo); # # bVk:: [in] Specifies a virtual-key code. The code must be a value in the range 1 to 254. # For a complete list, see Virtual-Key Codes. # bScan:: [in] Specifies a hardware scan code for the key. # dwFlags:: [in] Specifies various aspects of function operation. This parameter can be # one or more of the following values: # KEYEVENTF_EXTENDEDKEY, KEYEVENTF_KEYUP, KEYEVENTF_KEYDOWN # dwExtraInfo:: [in] Specifies an additional value associated with the key stroke. # # NO Return Value # --- # *Remarks*: # - An application can simulate a press of the PRINTSCRN key in order to obtain a screen snapshot # and save it to the clipboard. To do this, call keybd_event with the bVk parameter set to VK_SNAPSHOT. # # - Windows NT/2000/XP: The keybd_event function can toggle the NUM LOCK, CAPS LOCK, and SCROLL LOCK keys. # - Windows 95/98/Me: The keybd_event function can toggle only the CAPS LOCK and SCROLL LOCK keys. # # :call-seq: # keybd_event( virtual_key, scan_code, flags, extra_info ) # function :keybd_event, [:char, :char, :ulong, :ulong], :void ## # The mouse_event function synthesizes mouse motion and button clicks. # # !! Windows NT/2000/XP/Vista:This function has been superseded. Use SendInput instead. # # [*Syntax*] VOID mouse_event( DWORD dwFlags, DWORD dx, DWORD dy, DWORD dwData, ULONG_PTR dwExtraInfo ); # # dwFlags:: [in] Specifies various aspects of mouse motion and button clicking. This parameter can be certain # combinations of the following values. The values that specify mouse button status are set to indicate # changes in status, not ongoing conditions. For example, if the left mouse button is pressed and # held down, MOUSEEVENTF_LEFTDOWN is set when the left button is first pressed, but not for subsequent # motions. Similarly, MOUSEEVENTF_LEFTUP is set only when the button is first released. You cannot # specify both MOUSEEVENTF_WHEEL and either MOUSEEVENTF_XDOWN or MOUSEEVENTF_XUP simultaneously, # because they both require use of the dwData field: # MOUSEEVENTF_ABSOLUTE, MOUSEEVENTF_MOVE, MOUSEEVENTF_LEFTDOWN, MOUSEEVENTF_LEFTUP, # MOUSEEVENTF_RIGHTDOWN, MOUSEEVENTF_RIGHTUP, MOUSEEVENTF_MIDDLEDOWN, MOUSEEVENTF_MIDDLEUP, # MOUSEEVENTF_WHEEL, MOUSEEVENTF_XDOWN, MOUSEEVENTF_XUP # dx:: [in] Specifies the mouse's absolute position along the x-axis or its amount of motion since the # last mouse event was generated, depending on the setting of MOUSEEVENTF_ABSOLUTE. Absolute data is # specified as the mouse's actual x-coordinate; relative data is specified as the number of mickeys moved. # A mickey is the amount that a mouse has to move for it to report that it has moved. # dy:: [in] Specifies the mouse's absolute position along the y-axis or its amount of motion since the # last mouse event was generated, depending on the setting of MOUSEEVENTF_ABSOLUTE. Absolute data is # specified as the mouse's actual y-coordinate; relative data is specified as the number of mickeys moved. # dwData:: [in] # - If dwFlags contains MOUSEEVENTF_WHEEL, then data specifies the amount of wheel movement. # A positive value indicates that the wheel was rotated forward, away from the user; a negative value # indicates that the wheel was rotated backward, toward the user. One wheel click is defined as # WHEEL_DELTA, which is 120. # - If dwFlags contains MOUSEEVENTF_WHHEEL, then data specifies the amount of # wheel movement. A positive value indicates that the wheel was rotated to the right; a negative value # indicates that the wheel was rotated to the left. One wheel click is defined as WHEEL_DELTA (= 120). # - Windows 2000/XP: If flags contains MOUSEEVENTF_XDOWN or MOUSEEVENTF_XUP, then data specifies which # X buttons were pressed or released. This value may be any combination of the following flags. # - If flags is not MOUSEEVENTF_WHEEL, MOUSEEVENTF_XDOWN, or MOUSEEVENTF_XUP, then data should be zero. # XBUTTON1 - Set if the first X button was pressed or released. # XBUTTON2 - Set if the second X button was pressed or released. # dwExtraInfo:: [in] Specifies an additional value associated with the mouse event. An application # calls GetMessageExtraInfo to obtain this extra information. # NO Return Value # --- # *Remarks*: # - If the mouse has moved, indicated by MOUSEEVENTF_MOVE being set, dx and dy hold information about # that motion. The information is specified as absolute or relative integer values. # - If MOUSEEVENTF_ABSOLUTE value is specified, dx and dy contain normalized absolute coordinates between # 0 and 65,535. The event procedure maps these coordinates onto the display surface. Coordinate (0,0) maps # onto the upper-left corner of the display surface, (65535,65535) maps onto the lower-right corner. # - If the MOUSEEVENTF_ABSOLUTE value is not specified, dx and dy specify relative motions from when the # last mouse event was generated (the last reported position). Positive values mean the mouse moved right # (or down); negative values mean the mouse moved left (or up). Relative mouse motion is subject to the # settings for mouse speed and acceleration level. An end user sets these values using the Mouse application # in Control Panel. An application obtains and sets these values with the SystemParametersInfo function. # - The system applies two tests to the specified relative mouse motion when applying acceleration. If the # specified distance along either the x or y axis is greater than the first mouse threshold value, and the # mouse acceleration level is not zero, the operating system doubles the distance. If the specified distance # along either the x- or y-axis is greater than the second mouse threshold value, and the mouse acceleration # level is equal to two, the operating system doubles the distance that resulted from applying the first # threshold test. It is thus possible for the operating system to multiply relatively-specified mouse motion # along the x- or y-axis by up to four times. # - Once acceleration has been applied, the system scales the resultant value by the desired mouse speed. # Mouse speed can range from 1 (slowest) to 20 (fastest) and represents how much the pointer moves based # on the distance the mouse moves. The default value is 10, which results in no additional modification # to the mouse motion. # - The mouse_event function is used to synthesize mouse events by applications that need to do so. It is also # used by applications that need to obtain more information from the mouse than its position and button state. # For example, if a tablet manufacturer wants to pass pen-based information to its own applications, it can # write a DLL that communicates directly to the tablet hardware, obtains the extra information, and saves it # in a queue. The DLL then calls mouse_event with the standard button and x/y position data, along with, # in the dwExtraInfo parameter, some pointer or index to the queued extra information. When the application # needs the extra information, it calls the DLL with the pointer or index stored in dwExtraInfo, and the DLL # returns the extra information. # # :call-seq: # mouse_event( flags, dx, dy, data, extra_info ) # function :mouse_event, [:ulong, :ulong, :ulong, :ulong, :ulong, ], :void ## # SetCursorPos Function moves the cursor to the specified screen coordinates. If the new coordinates are not # within the screen rectangle set by the most recent ClipCursor function call, the system automatically adjusts # the coordinates so that the cursor stays within the rectangle. # # [*Syntax*] BOOL SetCursorPos( int X, int Y ); # # X:: [in] Specifies the new x-coordinate of the cursor, in screen coordinates. # Y:: [in] Specifies the new y-coordinate of the cursor, in screen coordinates. # # *Returns*:: Nonzero(*true*) if successful or zero(*false*) otherwise. To get extended error information, # call GetLastError. Enhanced to return true/false instead of nonzero/zero # --- # *Remarks*: The cursor is a shared resource. A window should move the cursor only when the cursor is in the # window's client area. The calling process must have WINSTA_WRITEATTRIBUTES access to the window station. # The input desktop must be the current desktop when you call SetCursorPos. Call OpenInputDesktop to determine # whether the current desktop is the input desktop. If it is not, call SetThreadDesktop with the HDESK returned # by OpenInputDesktop to switch to that desktop. # # :call-seq: # success = set_cursor_pos(x,y) # function :SetCursorPos, [:int, :int], :int, boolean: true ## # GetCursorPos Function retrieves the cursor's position, in screen coordinates. # # [*Syntax*] BOOL GetCursorPos( LPPOINT lpPoint ); # # lpPoint:: [out] Pointer to a POINT structure that receives the screen coordinates of the cursor. # # *Returns*:: Returns nonzero if successful or zero otherwise. To get extended error information, call # GetLastError. # --- # *Remarks*: # The cursor position is always specified in screen coordinates and is not affected by the mapping mode # of the window that contains the cursor. # The calling process must have WINSTA_READATTRIBUTES access to the window station. # The input desktop must be the current desktop when you call GetCursorPos. Call OpenInputDesktop to # determine whether the current desktop is the input desktop. If it is not, call SetThreadDesktop with # the HDESK returned by OpenInputDesktop to switch to that desktop. # --- # Enhanced (snake_case) API: accepts no args, returns a pair (x, y) of cursor coordinates # # :call-seq: # x, y = get_cursor_pos() # function :GetCursorPos, [:pointer], :int8, &->(api) { point = FFI::MemoryPointer.new(:long, 2) res = api.call point res == 0 ? [nil, nil] : point.read_array_of_long(2) } # weird lambda literal instead of normal block is needed because current version of RDoc # goes crazy if block is attached to meta-definition # Convenience methods ## # Emulates combinations of (any amount of) keys pressed one after another (Ctrl+Alt+P) and then released # *keys should be a sequence of a virtual-key codes. The codes must be a value in the range 1 to 254. # For a complete list, see msdn:Virtual Key Codes. def keystroke(*keys) return if keys.empty? keybd_event keys.first, 0, KEYEVENTF_KEYDOWN, 0 sleep KEY_DELAY keystroke *keys[1..-1] sleep KEY_DELAY keybd_event keys.first, 0, KEYEVENTF_KEYUP, 0 end # types text message into window holding the focus def type_in(message) message.scan(/./m) do |char| keystroke(*char.to_vkeys) end end end end end