mouse_emu.ino 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /*
  2. mouse_emu.ino
  3. Author: tobychui
  4. This code file handle keyboard emulation related functionality
  5. When opr_type is set to 0x02, the sub-handler will process the
  6. request here.
  7. -- Mouse Write (opr_type = 0x02) --
  8. 0x00 = Reserved
  9. 0x01 = Mouse Click
  10. 0x02 = Mouse Press
  11. 0x03 = Mouse Release
  12. 0x00 = Left Button
  13. 0x01 = Right Button
  14. 0x02 = Middle button
  15. 0x04 = Mouse Position Presets
  16. 0x00 = [0,0] (top left)
  17. 0x01 = [0, max] (bottom left)
  18. 0x02 = [max, 0] (top right)
  19. 0x03 = [max, max] (bottom right)
  20. 0x05 = Release All Mouse Buttons
  21. -- Mouse Move (opr_type = 0x03) --
  22. This operation is a special case that
  23. takes an additional 4 payloads
  24. byte[0] opr_type (0x03)
  25. byte[1] val_x (max 0x7E)
  26. byte[2] val_y (max 0x7E)
  27. byte[3] signed_x (0x00 = positive, 0x01 = negative)
  28. byte[4] signed_y (0x00 = positive, 0x01 = negative)
  29. -- Mouse Scroll (opr_type = 0x04) --
  30. This operation is another special case
  31. that use opr_subtype field as direction and
  32. opr_payload as tilt value
  33. Note: the value is directly written to the USB_HID report
  34. the resp is dependent to the OS
  35. opr_subtype
  36. 0x00 = positive
  37. 0x01 = negative
  38. opr_value = tilt (max 0x7E)
  39. */
  40. #include "usbkvm_fw.h"
  41. //#define ENABLE_MOUSE_DEBUG
  42. //Move cursor back to [0,0] position
  43. void reset_cursor_to_home() {
  44. for (int i = 0; i < 16; i++) {
  45. //16 loops should be able to home a mouse to [0,0] from a 4k display
  46. Mouse_move(-1 * (int8_t)127, -1 * (int8_t)127);
  47. }
  48. }
  49. //Move the cursor to target position, with multiplier of 10x
  50. void move_cursor_to_pos(uint8_t x, uint8_t y) {
  51. for (int i = 0; i < 10; i++) {
  52. Mouse_move((int8_t)x, (int8_t)y);
  53. }
  54. }
  55. //Handle mouse move, x, y, and direction of x, y (positive: 0x00, negative 0x01)
  56. uint8_t mouse_move(uint8_t ux, uint8_t uy, uint8_t dx, uint8_t dy) {
  57. if (ux > 0x7E) ux = 0x7E;
  58. if (uy > 0x7E) uy = 0x7E;
  59. int8_t x = ux;
  60. int8_t y = uy;
  61. if (dx == 0x01)
  62. x = -x;
  63. if (dy == 0x01)
  64. y = -y;
  65. #ifdef ENABLE_MOUSE_DEBUG
  66. Serial0_write(resp_start_of_info_msg);
  67. Serial0_write(ux);
  68. Serial0_write(uy);
  69. Serial0_write(resp_end_of_msg);
  70. #endif
  71. Mouse_move(x, y);
  72. return resp_ok;
  73. }
  74. //Handle mouse move, direction accept 0x00 (down / right) or 0x01 (up / left)
  75. uint8_t mouse_wheel(uint8_t direction, uint8_t utilt) {
  76. if (utilt >= 0x7E) {
  77. utilt = 0x7E;
  78. }
  79. int8_t tilt = utilt;
  80. if (direction == 0x01) {
  81. //Down
  82. tilt = -tilt;
  83. }
  84. #ifdef ENABLE_MOUSE_DEBUG
  85. Serial0_write(resp_start_of_info_msg);
  86. Serial0_write(tilt);
  87. Serial0_write(resp_end_of_msg);
  88. #endif
  89. Mouse_scroll(tilt);
  90. Mouse_scroll(0);
  91. return resp_ok;
  92. }
  93. uint8_t mouse_emulation(uint8_t subtype, uint8_t value) {
  94. switch (subtype) {
  95. case SUBTYPE_MOUSE_CLICK:
  96. if (value == PAYLOAD_MOUSE_BTN_LEFT) {
  97. Mouse_click(MOUSE_LEFT);
  98. } else if (value == PAYLOAD_MOUSE_BTN_RIGHT) {
  99. Mouse_click(MOUSE_RIGHT);
  100. } else if (value == PAYLOAD_MOUSE_BTN_MID) {
  101. Mouse_click(MOUSE_MIDDLE);
  102. } else {
  103. return resp_invalid_key_value;
  104. }
  105. return resp_ok;
  106. case SUBTYPE_MOUSE_PRESS:
  107. if (value == PAYLOAD_MOUSE_BTN_LEFT) {
  108. Mouse_press(MOUSE_LEFT);
  109. } else if (value == PAYLOAD_MOUSE_BTN_RIGHT) {
  110. Mouse_press(MOUSE_RIGHT);
  111. } else if (value == PAYLOAD_MOUSE_BTN_MID) {
  112. Mouse_press(MOUSE_MIDDLE);
  113. } else {
  114. return resp_invalid_key_value;
  115. }
  116. return resp_ok;
  117. case SUBTYPE_MOUSE_RELEASE:
  118. if (value == PAYLOAD_MOUSE_BTN_LEFT) {
  119. Mouse_release(MOUSE_LEFT);
  120. } else if (value == PAYLOAD_MOUSE_BTN_RIGHT) {
  121. Mouse_release(MOUSE_RIGHT);
  122. } else if (value == PAYLOAD_MOUSE_BTN_MID) {
  123. Mouse_release(MOUSE_MIDDLE);
  124. } else {
  125. return resp_invalid_key_value;
  126. }
  127. return resp_ok;
  128. case SUBTYPE_MOUSE_SETPOS:
  129. if (value == 0x00) {
  130. reset_cursor_to_home();
  131. } else if (value == 0x01) {
  132. reset_cursor_to_home();
  133. move_cursor_to_pos(0, 127);
  134. } else if (value == 0x02) {
  135. reset_cursor_to_home();
  136. move_cursor_to_pos(127, 0);
  137. } else if (value == 0x03) {
  138. reset_cursor_to_home();
  139. move_cursor_to_pos(127, 127);
  140. } else {
  141. return resp_invalid_key_value;
  142. }
  143. return resp_ok;
  144. case SUBTYPE_MOUSE_RESET:
  145. Mouse_release(MOUSE_LEFT);
  146. Mouse_release(MOUSE_RIGHT);
  147. Mouse_release(MOUSE_MIDDLE);
  148. Mouse_scroll(0);
  149. delay(MIN_KEY_EVENTS_DELAY);
  150. return resp_ok;
  151. default:
  152. return resp_invalid_opr_type;
  153. }
  154. return resp_invalid_opr_type;
  155. }