mouse_emu.ino 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  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. int8_t tilt;
  77. if (utilt >= 0x7E) {
  78. tilt = 126;
  79. }else{
  80. tilt = (int8_t)utilt;
  81. }
  82. if (direction == 0x01) {
  83. //Down
  84. tilt = tilt * -1;
  85. }
  86. #ifdef ENABLE_MOUSE_DEBUG
  87. Serial0_write(resp_start_of_info_msg);
  88. Serial0_write(tilt);
  89. Serial0_write(resp_end_of_msg);
  90. #endif
  91. Mouse_scroll(tilt);
  92. delay(MIN_KEY_EVENTS_DELAY);
  93. Mouse_scroll(0);
  94. return resp_ok;
  95. }
  96. uint8_t mouse_emulation(uint8_t subtype, uint8_t value) {
  97. switch (subtype) {
  98. case SUBTYPE_MOUSE_CLICK:
  99. Mouse_move(0,0);
  100. if (value == PAYLOAD_MOUSE_BTN_LEFT) {
  101. Mouse_click(MOUSE_LEFT);
  102. } else if (value == PAYLOAD_MOUSE_BTN_RIGHT) {
  103. Mouse_click(MOUSE_RIGHT);
  104. } else if (value == PAYLOAD_MOUSE_BTN_MID) {
  105. Mouse_click(MOUSE_MIDDLE);
  106. } else {
  107. return resp_invalid_key_value;
  108. }
  109. return resp_ok;
  110. case SUBTYPE_MOUSE_PRESS:
  111. Mouse_move(0,0);
  112. if (value == PAYLOAD_MOUSE_BTN_LEFT) {
  113. Mouse_press(MOUSE_LEFT);
  114. } else if (value == PAYLOAD_MOUSE_BTN_RIGHT) {
  115. Mouse_press(MOUSE_RIGHT);
  116. } else if (value == PAYLOAD_MOUSE_BTN_MID) {
  117. Mouse_press(MOUSE_MIDDLE);
  118. } else {
  119. return resp_invalid_key_value;
  120. }
  121. return resp_ok;
  122. case SUBTYPE_MOUSE_RELEASE:
  123. if (value == PAYLOAD_MOUSE_BTN_LEFT) {
  124. Mouse_release(MOUSE_LEFT);
  125. } else if (value == PAYLOAD_MOUSE_BTN_RIGHT) {
  126. Mouse_release(MOUSE_RIGHT);
  127. } else if (value == PAYLOAD_MOUSE_BTN_MID) {
  128. Mouse_release(MOUSE_MIDDLE);
  129. } else {
  130. return resp_invalid_key_value;
  131. }
  132. return resp_ok;
  133. case SUBTYPE_MOUSE_SETPOS:
  134. if (value == 0x00) {
  135. reset_cursor_to_home();
  136. } else if (value == 0x01) {
  137. reset_cursor_to_home();
  138. move_cursor_to_pos(0, 127);
  139. } else if (value == 0x02) {
  140. reset_cursor_to_home();
  141. move_cursor_to_pos(127, 0);
  142. } else if (value == 0x03) {
  143. reset_cursor_to_home();
  144. move_cursor_to_pos(127, 127);
  145. } else {
  146. return resp_invalid_key_value;
  147. }
  148. Mouse_move(0,0);
  149. return resp_ok;
  150. case SUBTYPE_MOUSE_RESET:
  151. Mouse_release(MOUSE_LEFT);
  152. Mouse_release(MOUSE_RIGHT);
  153. Mouse_release(MOUSE_MIDDLE);
  154. delay(MIN_KEY_EVENTS_DELAY);
  155. return resp_ok;
  156. default:
  157. return resp_invalid_opr_type;
  158. }
  159. return resp_invalid_opr_type;
  160. }