rpc_gui.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. #include "flipper.pb.h"
  2. #include "rpc_i.h"
  3. #include "gui.pb.h"
  4. #include <gui/gui_i.h>
  5. typedef struct {
  6. Rpc* rpc;
  7. Gui* gui;
  8. } RpcGuiSystem;
  9. void rpc_system_gui_screen_frame_callback(uint8_t* data, size_t size, void* context) {
  10. furi_assert(data);
  11. furi_assert(size == 1024);
  12. furi_assert(context);
  13. RpcGuiSystem* rpc_gui = context;
  14. PB_Main* frame = furi_alloc(sizeof(PB_Main));
  15. frame->which_content = PB_Main_gui_screen_stream_frame_tag;
  16. frame->command_status = PB_CommandStatus_OK;
  17. frame->content.gui_screen_stream_frame.data = furi_alloc(PB_BYTES_ARRAY_T_ALLOCSIZE(size));
  18. uint8_t* buffer = frame->content.gui_screen_stream_frame.data->bytes;
  19. uint16_t* frame_size_msg = &frame->content.gui_screen_stream_frame.data->size;
  20. *frame_size_msg = size;
  21. memcpy(buffer, data, size);
  22. rpc_send_and_release(rpc_gui->rpc, frame);
  23. free(frame);
  24. }
  25. void rpc_system_gui_start_screen_stream_process(const PB_Main* request, void* context) {
  26. furi_assert(request);
  27. furi_assert(context);
  28. RpcGuiSystem* rpc_gui = context;
  29. rpc_send_and_release_empty(rpc_gui->rpc, request->command_id, PB_CommandStatus_OK);
  30. gui_set_framebuffer_callback(rpc_gui->gui, rpc_system_gui_screen_frame_callback, context);
  31. }
  32. void rpc_system_gui_stop_screen_stream_process(const PB_Main* request, void* context) {
  33. furi_assert(request);
  34. furi_assert(context);
  35. RpcGuiSystem* rpc_gui = context;
  36. rpc_send_and_release_empty(rpc_gui->rpc, request->command_id, PB_CommandStatus_OK);
  37. gui_set_framebuffer_callback(rpc_gui->gui, NULL, NULL);
  38. }
  39. void rpc_system_gui_send_input_event_request_process(const PB_Main* request, void* context) {
  40. furi_assert(request);
  41. furi_assert(request->which_content == PB_Main_gui_send_input_event_request_tag);
  42. furi_assert(context);
  43. RpcGuiSystem* rpc_gui = context;
  44. InputEvent event;
  45. bool invalid = false;
  46. switch(request->content.gui_send_input_event_request.key) {
  47. case PB_Gui_InputKey_UP:
  48. event.key = InputKeyUp;
  49. break;
  50. case PB_Gui_InputKey_DOWN:
  51. event.key = InputKeyDown;
  52. break;
  53. case PB_Gui_InputKey_RIGHT:
  54. event.key = InputKeyRight;
  55. break;
  56. case PB_Gui_InputKey_LEFT:
  57. event.key = InputKeyLeft;
  58. break;
  59. case PB_Gui_InputKey_OK:
  60. event.key = InputKeyOk;
  61. break;
  62. case PB_Gui_InputKey_BACK:
  63. event.key = InputKeyBack;
  64. break;
  65. default:
  66. // Invalid key
  67. invalid = true;
  68. break;
  69. }
  70. switch(request->content.gui_send_input_event_request.type) {
  71. case PB_Gui_InputType_PRESS:
  72. event.type = InputTypePress;
  73. break;
  74. case PB_Gui_InputType_RELEASE:
  75. event.type = InputTypeRelease;
  76. break;
  77. case PB_Gui_InputType_SHORT:
  78. event.type = InputTypeShort;
  79. break;
  80. case PB_Gui_InputType_LONG:
  81. event.type = InputTypeLong;
  82. break;
  83. case PB_Gui_InputType_REPEAT:
  84. event.type = InputTypeRepeat;
  85. break;
  86. default:
  87. // Invalid type
  88. invalid = true;
  89. break;
  90. }
  91. if(invalid) {
  92. rpc_send_and_release_empty(
  93. rpc_gui->rpc, request->command_id, PB_CommandStatus_ERROR_INVALID_PARAMETERS);
  94. return;
  95. }
  96. FuriPubSub* input_events = furi_record_open("input_events");
  97. furi_check(input_events);
  98. furi_pubsub_publish(input_events, &event);
  99. furi_record_close("input_events");
  100. rpc_send_and_release_empty(rpc_gui->rpc, request->command_id, PB_CommandStatus_OK);
  101. }
  102. void* rpc_system_gui_alloc(Rpc* rpc) {
  103. furi_assert(rpc);
  104. RpcGuiSystem* rpc_gui = furi_alloc(sizeof(RpcGuiSystem));
  105. rpc_gui->gui = furi_record_open("gui");
  106. rpc_gui->rpc = rpc;
  107. RpcHandler rpc_handler = {
  108. .message_handler = NULL,
  109. .decode_submessage = NULL,
  110. .context = rpc_gui,
  111. };
  112. rpc_handler.message_handler = rpc_system_gui_start_screen_stream_process;
  113. rpc_add_handler(rpc, PB_Main_gui_start_screen_stream_request_tag, &rpc_handler);
  114. rpc_handler.message_handler = rpc_system_gui_stop_screen_stream_process;
  115. rpc_add_handler(rpc, PB_Main_gui_stop_screen_stream_request_tag, &rpc_handler);
  116. rpc_handler.message_handler = rpc_system_gui_send_input_event_request_process;
  117. rpc_add_handler(rpc, PB_Main_gui_send_input_event_request_tag, &rpc_handler);
  118. return rpc_gui;
  119. }
  120. void rpc_system_gui_free(void* ctx) {
  121. furi_assert(ctx);
  122. RpcGuiSystem* rpc_gui = ctx;
  123. furi_assert(rpc_gui->gui);
  124. gui_set_framebuffer_callback(rpc_gui->gui, NULL, NULL);
  125. furi_record_close("gui");
  126. free(rpc_gui);
  127. }