subghz_scene_receiver.c 9.0 KB

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