subghz_scene_receiver.c 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. #include "../subghz_i.h"
  2. #include "../views/receiver.h"
  3. #include <dolphin/dolphin.h>
  4. #include <lib/subghz/protocols/bin_raw.h>
  5. static const NotificationSequence subghs_sequence_rx = {
  6. &message_green_255,
  7. &message_vibro_on,
  8. &message_note_c6,
  9. &message_delay_50,
  10. &message_sound_off,
  11. &message_vibro_off,
  12. &message_delay_50,
  13. NULL,
  14. };
  15. static const NotificationSequence subghs_sequence_rx_locked = {
  16. &message_green_255,
  17. &message_display_backlight_on,
  18. &message_vibro_on,
  19. &message_note_c6,
  20. &message_delay_50,
  21. &message_sound_off,
  22. &message_vibro_off,
  23. &message_delay_500,
  24. &message_display_backlight_off,
  25. NULL,
  26. };
  27. static void subghz_scene_receiver_update_statusbar(void* context) {
  28. SubGhz* subghz = context;
  29. FuriString* history_stat_str = furi_string_alloc();
  30. if(!subghz_history_get_text_space_left(subghz->history, history_stat_str)) {
  31. FuriString* frequency_str = furi_string_alloc();
  32. FuriString* modulation_str = furi_string_alloc();
  33. subghz_txrx_get_frequency_and_modulation(subghz->txrx, frequency_str, modulation_str);
  34. subghz_view_receiver_add_data_statusbar(
  35. subghz->subghz_receiver,
  36. furi_string_get_cstr(frequency_str),
  37. furi_string_get_cstr(modulation_str),
  38. furi_string_get_cstr(history_stat_str));
  39. furi_string_free(frequency_str);
  40. furi_string_free(modulation_str);
  41. } else {
  42. subghz_view_receiver_add_data_statusbar(
  43. subghz->subghz_receiver, furi_string_get_cstr(history_stat_str), "", "");
  44. subghz->state_notifications = SubGhzNotificationStateIDLE;
  45. }
  46. furi_string_free(history_stat_str);
  47. }
  48. void subghz_scene_receiver_callback(SubGhzCustomEvent event, void* context) {
  49. furi_assert(context);
  50. SubGhz* subghz = context;
  51. view_dispatcher_send_custom_event(subghz->view_dispatcher, event);
  52. }
  53. static void subghz_scene_add_to_history_callback(
  54. SubGhzReceiver* receiver,
  55. SubGhzProtocolDecoderBase* decoder_base,
  56. void* context) {
  57. furi_assert(context);
  58. SubGhz* subghz = context;
  59. SubGhzHistory* history = subghz->history;
  60. FuriString* str_buff = furi_string_alloc();
  61. SubGhzRadioPreset preset = subghz_txrx_get_preset(subghz->txrx);
  62. if(subghz_history_add_to_history(history, decoder_base, &preset)) {
  63. furi_string_reset(str_buff);
  64. subghz->state_notifications = SubGhzNotificationStateRxDone;
  65. uint16_t item_history = subghz_history_get_item(history);
  66. subghz_history_get_text_item_menu(history, str_buff, item_history - 1);
  67. subghz_view_receiver_add_item_to_menu(
  68. subghz->subghz_receiver,
  69. furi_string_get_cstr(str_buff),
  70. subghz_history_get_type_protocol(history, item_history - 1));
  71. subghz_scene_receiver_update_statusbar(subghz);
  72. }
  73. subghz_receiver_reset(receiver);
  74. furi_string_free(str_buff);
  75. subghz_rx_key_state_set(subghz, SubGhzRxKeyStateAddKey);
  76. }
  77. void subghz_scene_receiver_on_enter(void* context) {
  78. SubGhz* subghz = context;
  79. SubGhzHistory* history = subghz->history;
  80. FuriString* str_buff;
  81. str_buff = furi_string_alloc();
  82. if(subghz_rx_key_state_get(subghz) == SubGhzRxKeyStateIDLE) {
  83. subghz_set_default_preset(subghz);
  84. subghz_history_reset(history);
  85. subghz_rx_key_state_set(subghz, SubGhzRxKeyStateStart);
  86. }
  87. subghz_view_receiver_set_lock(subghz->subghz_receiver, subghz_is_locked(subghz));
  88. //Load history to receiver
  89. subghz_view_receiver_exit(subghz->subghz_receiver);
  90. for(uint8_t i = 0; i < subghz_history_get_item(history); i++) {
  91. furi_string_reset(str_buff);
  92. subghz_history_get_text_item_menu(history, str_buff, i);
  93. subghz_view_receiver_add_item_to_menu(
  94. subghz->subghz_receiver,
  95. furi_string_get_cstr(str_buff),
  96. subghz_history_get_type_protocol(history, i));
  97. subghz_rx_key_state_set(subghz, SubGhzRxKeyStateAddKey);
  98. }
  99. furi_string_free(str_buff);
  100. subghz_scene_receiver_update_statusbar(subghz);
  101. subghz_view_receiver_set_callback(
  102. subghz->subghz_receiver, subghz_scene_receiver_callback, subghz);
  103. subghz_txrx_set_rx_calback(subghz->txrx, subghz_scene_add_to_history_callback, subghz);
  104. subghz->state_notifications = SubGhzNotificationStateRx;
  105. subghz_txrx_rx_start(subghz->txrx);
  106. subghz_view_receiver_set_idx_menu(subghz->subghz_receiver, subghz->idx_menu_chosen);
  107. //to use a universal decoder, we are looking for a link to it
  108. furi_check(
  109. subghz_txrx_load_decoder_by_name_protocol(subghz->txrx, SUBGHZ_PROTOCOL_BIN_RAW_NAME));
  110. view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewIdReceiver);
  111. }
  112. bool subghz_scene_receiver_on_event(void* context, SceneManagerEvent event) {
  113. SubGhz* subghz = context;
  114. bool consumed = false;
  115. if(event.type == SceneManagerEventTypeCustom) {
  116. switch(event.event) {
  117. case SubGhzCustomEventViewReceiverBack:
  118. // Stop CC1101 Rx
  119. subghz->state_notifications = SubGhzNotificationStateIDLE;
  120. subghz_txrx_stop(subghz->txrx);
  121. subghz_txrx_hopper_set_state(subghz->txrx, SubGhzHopperStateOFF);
  122. subghz->idx_menu_chosen = 0;
  123. subghz_txrx_set_rx_calback(subghz->txrx, NULL, subghz);
  124. if(subghz_rx_key_state_get(subghz) == SubGhzRxKeyStateAddKey) {
  125. subghz_rx_key_state_set(subghz, SubGhzRxKeyStateExit);
  126. scene_manager_next_scene(subghz->scene_manager, SubGhzSceneNeedSaving);
  127. } else {
  128. subghz_rx_key_state_set(subghz, SubGhzRxKeyStateIDLE);
  129. subghz_set_default_preset(subghz);
  130. scene_manager_search_and_switch_to_previous_scene(
  131. subghz->scene_manager, SubGhzSceneStart);
  132. }
  133. consumed = true;
  134. break;
  135. case SubGhzCustomEventViewReceiverOK:
  136. subghz->idx_menu_chosen = subghz_view_receiver_get_idx_menu(subghz->subghz_receiver);
  137. scene_manager_next_scene(subghz->scene_manager, SubGhzSceneReceiverInfo);
  138. DOLPHIN_DEED(DolphinDeedSubGhzReceiverInfo);
  139. consumed = true;
  140. break;
  141. case SubGhzCustomEventViewReceiverConfig:
  142. subghz->state_notifications = SubGhzNotificationStateIDLE;
  143. subghz->idx_menu_chosen = subghz_view_receiver_get_idx_menu(subghz->subghz_receiver);
  144. scene_manager_next_scene(subghz->scene_manager, SubGhzSceneReceiverConfig);
  145. consumed = true;
  146. break;
  147. case SubGhzCustomEventViewReceiverOffDisplay:
  148. notification_message(subghz->notifications, &sequence_display_backlight_off);
  149. consumed = true;
  150. break;
  151. case SubGhzCustomEventViewReceiverUnlock:
  152. subghz_unlock(subghz);
  153. consumed = true;
  154. break;
  155. default:
  156. break;
  157. }
  158. } else if(event.type == SceneManagerEventTypeTick) {
  159. if(subghz_txrx_hopper_get_state(subghz->txrx) != SubGhzHopperStateOFF) {
  160. subghz_txrx_hopper_update(subghz->txrx);
  161. subghz_scene_receiver_update_statusbar(subghz);
  162. }
  163. SubGhzThresholdRssiData ret_rssi = subghz_threshold_get_rssi_data(subghz->threshold_rssi);
  164. subghz_receiver_rssi(subghz->subghz_receiver, ret_rssi.rssi);
  165. subghz_protocol_decoder_bin_raw_data_input_rssi(
  166. (SubGhzProtocolDecoderBinRAW*)subghz_txrx_get_decoder(subghz->txrx), ret_rssi.rssi);
  167. switch(subghz->state_notifications) {
  168. case SubGhzNotificationStateRx:
  169. notification_message(subghz->notifications, &sequence_blink_cyan_10);
  170. break;
  171. case SubGhzNotificationStateRxDone:
  172. if(!subghz_is_locked(subghz)) {
  173. notification_message(subghz->notifications, &subghs_sequence_rx);
  174. } else {
  175. notification_message(subghz->notifications, &subghs_sequence_rx_locked);
  176. }
  177. subghz->state_notifications = SubGhzNotificationStateRx;
  178. break;
  179. default:
  180. break;
  181. }
  182. }
  183. return consumed;
  184. }
  185. void subghz_scene_receiver_on_exit(void* context) {
  186. UNUSED(context);
  187. }