subghz_scene_receiver.c 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. #include "../subghz_i.h"
  2. #include "../views/receiver.h"
  3. #include <dolphin/dolphin.h>
  4. static const NotificationSequence subghs_sequence_rx = {
  5. &message_green_255,
  6. &message_vibro_on,
  7. &message_note_c6,
  8. &message_delay_50,
  9. &message_sound_off,
  10. &message_vibro_off,
  11. &message_delay_50,
  12. NULL,
  13. };
  14. static const NotificationSequence subghs_sequence_rx_locked = {
  15. &message_green_255,
  16. &message_display_backlight_on,
  17. &message_vibro_on,
  18. &message_note_c6,
  19. &message_delay_50,
  20. &message_sound_off,
  21. &message_vibro_off,
  22. &message_delay_500,
  23. &message_display_backlight_off,
  24. NULL,
  25. };
  26. static void subghz_scene_receiver_update_statusbar(void* context) {
  27. SubGhz* subghz = context;
  28. FuriString* history_stat_str;
  29. history_stat_str = furi_string_alloc();
  30. if(!subghz_history_get_text_space_left(subghz->txrx->history, history_stat_str)) {
  31. FuriString* frequency_str;
  32. FuriString* modulation_str;
  33. frequency_str = furi_string_alloc();
  34. modulation_str = furi_string_alloc();
  35. subghz_get_frequency_modulation(subghz, frequency_str, modulation_str);
  36. subghz_view_receiver_add_data_statusbar(
  37. subghz->subghz_receiver,
  38. furi_string_get_cstr(frequency_str),
  39. furi_string_get_cstr(modulation_str),
  40. furi_string_get_cstr(history_stat_str));
  41. furi_string_free(frequency_str);
  42. furi_string_free(modulation_str);
  43. } else {
  44. subghz_view_receiver_add_data_statusbar(
  45. subghz->subghz_receiver, furi_string_get_cstr(history_stat_str), "", "");
  46. subghz->state_notifications = SubGhzNotificationStateIDLE;
  47. }
  48. furi_string_free(history_stat_str);
  49. }
  50. void subghz_scene_receiver_callback(SubGhzCustomEvent event, void* context) {
  51. furi_assert(context);
  52. SubGhz* subghz = context;
  53. view_dispatcher_send_custom_event(subghz->view_dispatcher, event);
  54. }
  55. static void subghz_scene_add_to_history_callback(
  56. SubGhzReceiver* receiver,
  57. SubGhzProtocolDecoderBase* decoder_base,
  58. void* context) {
  59. furi_assert(context);
  60. SubGhz* subghz = context;
  61. FuriString* str_buff;
  62. str_buff = furi_string_alloc();
  63. if(subghz_history_add_to_history(subghz->txrx->history, decoder_base, subghz->txrx->preset)) {
  64. furi_string_reset(str_buff);
  65. subghz->state_notifications = SubGhzNotificationStateRxDone;
  66. subghz_history_get_text_item_menu(
  67. subghz->txrx->history, str_buff, subghz_history_get_item(subghz->txrx->history) - 1);
  68. subghz_view_receiver_add_item_to_menu(
  69. subghz->subghz_receiver,
  70. furi_string_get_cstr(str_buff),
  71. subghz_history_get_type_protocol(
  72. subghz->txrx->history, subghz_history_get_item(subghz->txrx->history) - 1));
  73. subghz_scene_receiver_update_statusbar(subghz);
  74. }
  75. subghz_receiver_reset(receiver);
  76. furi_string_free(str_buff);
  77. subghz->txrx->rx_key_state = SubGhzRxKeyStateAddKey;
  78. }
  79. void subghz_scene_receiver_on_enter(void* context) {
  80. SubGhz* subghz = context;
  81. FuriString* str_buff;
  82. str_buff = furi_string_alloc();
  83. if(subghz->txrx->rx_key_state == SubGhzRxKeyStateIDLE) {
  84. subghz_preset_init(
  85. subghz, "AM650", subghz_setting_get_default_frequency(subghz->setting), NULL, 0);
  86. subghz_history_reset(subghz->txrx->history);
  87. subghz->txrx->rx_key_state = SubGhzRxKeyStateStart;
  88. }
  89. subghz_view_receiver_set_lock(subghz->subghz_receiver, subghz->lock);
  90. //Load history to receiver
  91. subghz_view_receiver_exit(subghz->subghz_receiver);
  92. for(uint8_t i = 0; i < subghz_history_get_item(subghz->txrx->history); i++) {
  93. furi_string_reset(str_buff);
  94. subghz_history_get_text_item_menu(subghz->txrx->history, str_buff, i);
  95. subghz_view_receiver_add_item_to_menu(
  96. subghz->subghz_receiver,
  97. furi_string_get_cstr(str_buff),
  98. subghz_history_get_type_protocol(subghz->txrx->history, i));
  99. subghz->txrx->rx_key_state = SubGhzRxKeyStateAddKey;
  100. }
  101. furi_string_free(str_buff);
  102. subghz_scene_receiver_update_statusbar(subghz);
  103. subghz_view_receiver_set_callback(
  104. subghz->subghz_receiver, subghz_scene_receiver_callback, subghz);
  105. subghz_receiver_set_rx_callback(
  106. subghz->txrx->receiver, subghz_scene_add_to_history_callback, subghz);
  107. subghz->state_notifications = SubGhzNotificationStateRx;
  108. if(subghz->txrx->txrx_state == SubGhzTxRxStateRx) {
  109. subghz_rx_end(subghz);
  110. };
  111. if((subghz->txrx->txrx_state == SubGhzTxRxStateIDLE) ||
  112. (subghz->txrx->txrx_state == SubGhzTxRxStateSleep)) {
  113. subghz_begin(
  114. subghz,
  115. subghz_setting_get_preset_data_by_name(
  116. subghz->setting, furi_string_get_cstr(subghz->txrx->preset->name)));
  117. subghz_rx(subghz, subghz->txrx->preset->frequency);
  118. }
  119. subghz_view_receiver_set_idx_menu(subghz->subghz_receiver, subghz->txrx->idx_menu_chosen);
  120. view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewIdReceiver);
  121. }
  122. bool subghz_scene_receiver_on_event(void* context, SceneManagerEvent event) {
  123. SubGhz* subghz = context;
  124. bool consumed = false;
  125. if(event.type == SceneManagerEventTypeCustom) {
  126. switch(event.event) {
  127. case SubGhzCustomEventViewReceiverBack:
  128. // Stop CC1101 Rx
  129. subghz->state_notifications = SubGhzNotificationStateIDLE;
  130. if(subghz->txrx->txrx_state == SubGhzTxRxStateRx) {
  131. subghz_rx_end(subghz);
  132. subghz_sleep(subghz);
  133. };
  134. subghz->txrx->hopper_state = SubGhzHopperStateOFF;
  135. subghz->txrx->idx_menu_chosen = 0;
  136. subghz_receiver_set_rx_callback(subghz->txrx->receiver, NULL, subghz);
  137. if(subghz->txrx->rx_key_state == SubGhzRxKeyStateAddKey) {
  138. subghz->txrx->rx_key_state = SubGhzRxKeyStateExit;
  139. scene_manager_next_scene(subghz->scene_manager, SubGhzSceneNeedSaving);
  140. } else {
  141. subghz->txrx->rx_key_state = SubGhzRxKeyStateIDLE;
  142. subghz_preset_init(
  143. subghz,
  144. "AM650",
  145. subghz_setting_get_default_frequency(subghz->setting),
  146. NULL,
  147. 0);
  148. scene_manager_search_and_switch_to_previous_scene(
  149. subghz->scene_manager, SubGhzSceneStart);
  150. }
  151. consumed = true;
  152. break;
  153. case SubGhzCustomEventViewReceiverOK:
  154. subghz->txrx->idx_menu_chosen =
  155. subghz_view_receiver_get_idx_menu(subghz->subghz_receiver);
  156. scene_manager_next_scene(subghz->scene_manager, SubGhzSceneReceiverInfo);
  157. DOLPHIN_DEED(DolphinDeedSubGhzReceiverInfo);
  158. consumed = true;
  159. break;
  160. case SubGhzCustomEventViewReceiverConfig:
  161. subghz->state_notifications = SubGhzNotificationStateIDLE;
  162. subghz->txrx->idx_menu_chosen =
  163. subghz_view_receiver_get_idx_menu(subghz->subghz_receiver);
  164. scene_manager_next_scene(subghz->scene_manager, SubGhzSceneReceiverConfig);
  165. consumed = true;
  166. break;
  167. case SubGhzCustomEventViewReceiverOffDisplay:
  168. notification_message(subghz->notifications, &sequence_display_backlight_off);
  169. consumed = true;
  170. break;
  171. case SubGhzCustomEventViewReceiverUnlock:
  172. subghz->lock = SubGhzLockOff;
  173. consumed = true;
  174. break;
  175. default:
  176. break;
  177. }
  178. } else if(event.type == SceneManagerEventTypeTick) {
  179. if(subghz->txrx->hopper_state != SubGhzHopperStateOFF) {
  180. subghz_hopper_update(subghz);
  181. subghz_scene_receiver_update_statusbar(subghz);
  182. }
  183. switch(subghz->state_notifications) {
  184. case SubGhzNotificationStateRx:
  185. notification_message(subghz->notifications, &sequence_blink_cyan_10);
  186. break;
  187. case SubGhzNotificationStateRxDone:
  188. if(subghz->lock != SubGhzLockOn) {
  189. notification_message(subghz->notifications, &subghs_sequence_rx);
  190. } else {
  191. notification_message(subghz->notifications, &subghs_sequence_rx_locked);
  192. }
  193. subghz->state_notifications = SubGhzNotificationStateRx;
  194. break;
  195. default:
  196. break;
  197. }
  198. }
  199. return consumed;
  200. }
  201. void subghz_scene_receiver_on_exit(void* context) {
  202. UNUSED(context);
  203. }