#include "../subghz_i.h" #include "../helpers/subghz_custom_event.h" #include #include #include void subghz_scene_receiver_info_callback(GuiButtonType result, InputType type, void* context) { furi_assert(context); SubGhz* subghz = context; if((result == GuiButtonTypeCenter) && (type == InputTypePress)) { view_dispatcher_send_custom_event( subghz->view_dispatcher, SubGhzCustomEventSceneReceiverInfoTxStart); } else if((result == GuiButtonTypeCenter) && (type == InputTypeRelease)) { view_dispatcher_send_custom_event( subghz->view_dispatcher, SubGhzCustomEventSceneReceiverInfoTxStop); } else if((result == GuiButtonTypeRight) && (type == InputTypeShort)) { view_dispatcher_send_custom_event( subghz->view_dispatcher, SubGhzCustomEventSceneReceiverInfoSave); } } static bool subghz_scene_receiver_info_update_parser(void* context) { SubGhz* subghz = context; if(subghz_txrx_load_decoder_by_name_protocol( subghz->txrx, subghz_history_get_protocol_name(subghz->history, subghz->idx_menu_chosen))) { //todo we are trying to deserialize without checking for errors, since it is assumed that we just received this signal subghz_protocol_decoder_base_deserialize( subghz_txrx_get_decoder(subghz->txrx), subghz_history_get_raw_data(subghz->history, subghz->idx_menu_chosen)); SubGhzRadioPreset* preset = subghz_history_get_radio_preset(subghz->history, subghz->idx_menu_chosen); subghz_txrx_set_preset( subghz->txrx, furi_string_get_cstr(preset->name), preset->frequency, preset->data, preset->data_size); return true; } return false; } void subghz_scene_receiver_info_draw_widget(SubGhz* subghz) { if(subghz_scene_receiver_info_update_parser(subghz)) { FuriString* frequency_str = furi_string_alloc(); FuriString* modulation_str = furi_string_alloc(); FuriString* text = furi_string_alloc(); subghz_txrx_get_frequency_and_modulation( subghz->txrx, frequency_str, modulation_str, false); widget_add_string_element( subghz->widget, 78, 0, AlignLeft, AlignTop, FontSecondary, furi_string_get_cstr(frequency_str)); widget_add_string_element( subghz->widget, 113, 0, AlignLeft, AlignTop, FontSecondary, furi_string_get_cstr(modulation_str)); subghz_protocol_decoder_base_get_string(subghz_txrx_get_decoder(subghz->txrx), text); widget_add_string_multiline_element( subghz->widget, 0, 0, AlignLeft, AlignTop, FontSecondary, furi_string_get_cstr(text)); furi_string_free(frequency_str); furi_string_free(modulation_str); furi_string_free(text); if(subghz_txrx_protocol_is_serializable(subghz->txrx)) { widget_add_button_element( subghz->widget, GuiButtonTypeRight, "Save", subghz_scene_receiver_info_callback, subghz); } // Removed static check if(subghz_txrx_protocol_is_transmittable(subghz->txrx, false)) { widget_add_button_element( subghz->widget, GuiButtonTypeCenter, "Send", subghz_scene_receiver_info_callback, subghz); } } else { widget_add_icon_element(subghz->widget, 37, 15, &I_DolphinCommon_56x48); widget_add_string_element( subghz->widget, 13, 8, AlignLeft, AlignBottom, FontSecondary, "Error history parse."); } view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewIdWidget); } void subghz_scene_receiver_info_on_enter(void* context) { SubGhz* subghz = context; keeloq_reset_original_btn(); subghz_custom_btns_reset(); subghz_scene_receiver_info_draw_widget(subghz); } bool subghz_scene_receiver_info_on_event(void* context, SceneManagerEvent event) { SubGhz* subghz = context; if(event.type == SceneManagerEventTypeCustom) { if(event.event == SubGhzCustomEventSceneReceiverInfoTxStart) { if(!subghz_scene_receiver_info_update_parser(subghz)) { return false; } //CC1101 Stop RX -> Start TX subghz_txrx_hopper_pause(subghz->txrx); if(!subghz_tx_start( subghz, subghz_history_get_raw_data(subghz->history, subghz->idx_menu_chosen))) { subghz_txrx_rx_start(subghz->txrx); subghz_txrx_hopper_unpause(subghz->txrx); subghz->state_notifications = SubGhzNotificationStateRx; } else { subghz->state_notifications = SubGhzNotificationStateTx; } return true; } else if(event.event == SubGhzCustomEventSceneReceiverInfoTxStop) { //CC1101 Stop Tx -> Start RX subghz->state_notifications = SubGhzNotificationStateIDLE; widget_reset(subghz->widget); subghz_scene_receiver_info_draw_widget(subghz); subghz_txrx_stop(subghz->txrx); if(!subghz->in_decoder_scene) { subghz_txrx_rx_start(subghz->txrx); subghz_txrx_hopper_unpause(subghz->txrx); subghz->state_notifications = SubGhzNotificationStateRx; } return true; } else if(event.event == SubGhzCustomEventSceneReceiverInfoSave) { //CC1101 Stop RX -> Save subghz->state_notifications = SubGhzNotificationStateIDLE; subghz_txrx_hopper_set_state(subghz->txrx, SubGhzHopperStateOFF); subghz_txrx_stop(subghz->txrx); if(!subghz_scene_receiver_info_update_parser(subghz)) { return false; } if(subghz_txrx_protocol_is_serializable(subghz->txrx)) { subghz_file_name_clear(subghz); if(subghz->in_decoder_scene) { subghz->in_decoder_scene_skip = true; } scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveName); } return true; } } else if(event.type == SceneManagerEventTypeTick) { if(subghz_txrx_hopper_get_state(subghz->txrx) != SubGhzHopperStateOFF) { subghz_txrx_hopper_update(subghz->txrx); } switch(subghz->state_notifications) { case SubGhzNotificationStateTx: notification_message(subghz->notifications, &sequence_blink_magenta_10); break; case SubGhzNotificationStateRx: notification_message(subghz->notifications, &sequence_blink_cyan_10); break; case SubGhzNotificationStateRxDone: notification_message(subghz->notifications, &sequence_blink_green_100); subghz->state_notifications = SubGhzNotificationStateRx; break; default: break; } } return false; } void subghz_scene_receiver_info_on_exit(void* context) { SubGhz* subghz = context; if(subghz->in_decoder_scene && !subghz->in_decoder_scene_skip) { subghz->in_decoder_scene = false; } widget_reset(subghz->widget); keeloq_reset_mfname(); keeloq_reset_kl_type(); keeloq_reset_original_btn(); subghz_custom_btns_reset(); star_line_reset_mfname(); star_line_reset_kl_type(); }