Radio app improvements (#2680)

This commit is contained in:
RocketGod
2025-06-05 12:25:09 -07:00
committed by GitHub
parent 1070d951e6
commit b50d18eafc
2 changed files with 197 additions and 68 deletions

View File

@@ -1,5 +1,6 @@
/* /*
* Copyright (C) 2024 HTotoo * Copyright (C) 2024 HT Otto
* Copyright (C) 2025 RocketGod - Added modes from my Flipper Zero RF Jammer App - https://betaskynet.com
* *
* This file is part of PortaPack. * This file is part of PortaPack.
* *
@@ -26,6 +27,7 @@
#include "baseband_api.hpp" #include "baseband_api.hpp"
#include "string_format.hpp" #include "string_format.hpp"
#include "portapack_persistent_memory.hpp" #include "portapack_persistent_memory.hpp"
#include "oversample.hpp"
using namespace portapack; using namespace portapack;
using namespace modems; using namespace modems;
@@ -37,16 +39,89 @@ void FmRadioView::focus() {
field_frequency.focus(); field_frequency.focus();
} }
void FmRadioView::change_mode(int32_t mod) {
field_bw.on_change = [this](size_t n, OptionsField::value_t) { (void)n; };
audio::output::stop();
receiver_model.disable();
baseband::shutdown();
audio_spectrum_update = false; // Reset spectrum update flag
std::fill(audio_spectrum, audio_spectrum + 128, 0); // Clear spectrum buffer
ReceiverModel::Mode receiver_mode = static_cast<ReceiverModel::Mode>(mod);
bool is_ssb = (mod == static_cast<int32_t>(ReceiverModel::Mode::AMAudio) &&
(field_modulation.selected_index() == 3 || field_modulation.selected_index() == 4));
switch (mod) {
case static_cast<int32_t>(ReceiverModel::Mode::AMAudio):
audio_sampling_rate = audio::Rate::Hz_24000; // Increased to 24 kHz for better AM/SSB audio
freqman_set_bandwidth_option(0, field_bw); // AM_MODULATION
baseband::run_image(portapack::spi_flash::image_tag_am_audio);
receiver_mode = ReceiverModel::Mode::AMAudio;
field_bw.set_by_value(0); // DSB default
receiver_model.set_modulation(receiver_mode);
if (is_ssb) {
receiver_model.set_am_configuration(field_modulation.selected_index() == 3 ? 1 : 2); // 1=USB, 2=LSB
} else {
receiver_model.set_am_configuration(0); // DSB
}
field_bw.on_change = [this](size_t index, OptionsField::value_t n) {
radio_bw = index;
receiver_model.set_am_configuration(n);
};
break;
case static_cast<int32_t>(ReceiverModel::Mode::NarrowbandFMAudio):
audio_sampling_rate = audio::Rate::Hz_24000;
freqman_set_bandwidth_option(1, field_bw); // NFM_MODULATION
baseband::run_image(portapack::spi_flash::image_tag_nfm_audio);
receiver_mode = ReceiverModel::Mode::NarrowbandFMAudio;
field_bw.set_by_value(2); // 16k default
receiver_model.set_nbfm_configuration(field_bw.selected_index_value());
field_bw.on_change = [this](size_t index, OptionsField::value_t n) {
radio_bw = index;
receiver_model.set_nbfm_configuration(n);
};
break;
case static_cast<int32_t>(ReceiverModel::Mode::WidebandFMAudio):
audio_sampling_rate = audio::Rate::Hz_48000;
freqman_set_bandwidth_option(2, field_bw); // WFM_MODULATION
baseband::run_image(portapack::spi_flash::image_tag_wfm_audio);
receiver_mode = ReceiverModel::Mode::WidebandFMAudio;
field_bw.set_by_value(0); // 200k default
receiver_model.set_wfm_configuration(field_bw.selected_index_value());
field_bw.on_change = [this](size_t index, OptionsField::value_t n) {
radio_bw = index;
receiver_model.set_wfm_configuration(n);
};
break;
default:
break;
}
receiver_model.set_modulation(receiver_mode);
receiver_model.set_sampling_rate(3072000);
receiver_model.set_baseband_bandwidth(1750000);
audio::set_rate(audio_sampling_rate);
audio::output::start();
receiver_model.set_headphone_volume(receiver_model.headphone_volume()); // WM8731 hack
receiver_model.enable();
}
FmRadioView::FmRadioView(NavigationView& nav) FmRadioView::FmRadioView(NavigationView& nav)
: nav_{nav} { : nav_{nav} {
baseband::run_image(portapack::spi_flash::image_tag_wfm_audio); baseband::run_image(portapack::spi_flash::image_tag_wfm_audio);
add_children({&rssi, add_children({&field_rf_amp,
&field_rf_amp,
&field_lna, &field_lna,
&field_vga, &field_vga,
&field_volume, &field_volume,
&field_frequency, &field_frequency,
&field_bw,
&text_mode_label,
&field_modulation,
&btn_fav_save, &btn_fav_save,
&txt_save_help, &txt_save_help,
&btn_fav_0, &btn_fav_0,
@@ -60,12 +135,14 @@ FmRadioView::FmRadioView(NavigationView& nav)
&btn_fav_8, &btn_fav_8,
&btn_fav_9, &btn_fav_9,
&audio, &audio,
&waveform}); &waveform,
&rssi});
txt_save_help.visible(false); txt_save_help.visible(false);
for (uint8_t i = 0; i < 12; ++i) { for (uint8_t i = 0; i < 12; ++i) {
if (freq_fav_list[i] == 0) { if (freq_fav_list[i].frequency == 0) {
freq_fav_list[i] = 87000000; freq_fav_list[i].frequency = 87000000;
freq_fav_list[i].modulation = static_cast<int32_t>(ReceiverModel::Mode::WidebandFMAudio);
} }
} }
@@ -73,42 +150,20 @@ FmRadioView::FmRadioView(NavigationView& nav)
field_frequency.set_value(87000000); field_frequency.set_value(87000000);
} }
receiver_model.set_modulation(ReceiverModel::Mode::WidebandFMAudio);
field_frequency.set_step(25000); field_frequency.set_step(25000);
receiver_model.enable(); change_mode(static_cast<int32_t>(ReceiverModel::Mode::WidebandFMAudio));
audio::output::start(); field_modulation.set_by_value(static_cast<int32_t>(ReceiverModel::Mode::WidebandFMAudio));
btn_fav_0.on_select = [this](Button&) { btn_fav_0.on_select = [this](Button&) { on_btn_clicked(0); };
on_btn_clicked(0); btn_fav_1.on_select = [this](Button&) { on_btn_clicked(1); };
}; btn_fav_2.on_select = [this](Button&) { on_btn_clicked(2); };
btn_fav_1.on_select = [this](Button&) { btn_fav_3.on_select = [this](Button&) { on_btn_clicked(3); };
on_btn_clicked(1); btn_fav_4.on_select = [this](Button&) { on_btn_clicked(4); };
}; btn_fav_5.on_select = [this](Button&) { on_btn_clicked(5); };
btn_fav_2.on_select = [this](Button&) { btn_fav_6.on_select = [this](Button&) { on_btn_clicked(6); };
on_btn_clicked(2); btn_fav_7.on_select = [this](Button&) { on_btn_clicked(7); };
}; btn_fav_8.on_select = [this](Button&) { on_btn_clicked(8); };
btn_fav_3.on_select = [this](Button&) { btn_fav_9.on_select = [this](Button&) { on_btn_clicked(9); };
on_btn_clicked(3);
};
btn_fav_4.on_select = [this](Button&) {
on_btn_clicked(4);
};
btn_fav_5.on_select = [this](Button&) {
on_btn_clicked(5);
};
btn_fav_6.on_select = [this](Button&) {
on_btn_clicked(6);
};
btn_fav_7.on_select = [this](Button&) {
on_btn_clicked(7);
};
btn_fav_8.on_select = [this](Button&) {
on_btn_clicked(8);
};
btn_fav_9.on_select = [this](Button&) {
on_btn_clicked(9);
};
btn_fav_save.on_select = [this](Button&) { btn_fav_save.on_select = [this](Button&) {
save_fav = !save_fav; save_fav = !save_fav;
@@ -117,20 +172,31 @@ FmRadioView::FmRadioView(NavigationView& nav)
txt_save_help.set_dirty(); txt_save_help.set_dirty();
}; };
field_modulation.on_change = [this](size_t index, int32_t mod) {
change_mode(mod);
if (index == 3 || index == 4) { // USB or LSB
receiver_model.set_am_configuration(index == 3 ? 1 : 2); // 1=USB, 2=LSB
}
};
update_fav_btn_texts(); update_fav_btn_texts();
} }
void FmRadioView::on_btn_clicked(uint8_t i) { void FmRadioView::on_btn_clicked(uint8_t i) {
if (save_fav) { if (save_fav) {
save_fav = false; save_fav = false;
freq_fav_list[i] = field_frequency.value(); freq_fav_list[i].frequency = field_frequency.value();
freq_fav_list[i].modulation = field_modulation.selected_index_value();
freq_fav_list[i].bandwidth = radio_bw;
update_fav_btn_texts(); update_fav_btn_texts();
txt_save_help.visible(save_fav); txt_save_help.visible(save_fav);
txt_save_help.set_text(""); txt_save_help.set_text("");
txt_save_help.set_dirty(); txt_save_help.set_dirty();
return; return;
} }
field_frequency.set_value(freq_fav_list[i]); field_frequency.set_value(freq_fav_list[i].frequency);
field_modulation.set_by_value(freq_fav_list[i].modulation);
change_mode(freq_fav_list[i].modulation);
} }
std::string FmRadioView::to_nice_freq(rf::Frequency freq) { std::string FmRadioView::to_nice_freq(rf::Frequency freq) {
@@ -141,16 +207,16 @@ std::string FmRadioView::to_nice_freq(rf::Frequency freq) {
} }
void FmRadioView::update_fav_btn_texts() { void FmRadioView::update_fav_btn_texts() {
btn_fav_0.set_text(to_nice_freq(freq_fav_list[0])); btn_fav_0.set_text(to_nice_freq(freq_fav_list[0].frequency));
btn_fav_1.set_text(to_nice_freq(freq_fav_list[1])); btn_fav_1.set_text(to_nice_freq(freq_fav_list[1].frequency));
btn_fav_2.set_text(to_nice_freq(freq_fav_list[2])); btn_fav_2.set_text(to_nice_freq(freq_fav_list[2].frequency));
btn_fav_3.set_text(to_nice_freq(freq_fav_list[3])); btn_fav_3.set_text(to_nice_freq(freq_fav_list[3].frequency));
btn_fav_4.set_text(to_nice_freq(freq_fav_list[4])); btn_fav_4.set_text(to_nice_freq(freq_fav_list[4].frequency));
btn_fav_5.set_text(to_nice_freq(freq_fav_list[5])); btn_fav_5.set_text(to_nice_freq(freq_fav_list[5].frequency));
btn_fav_6.set_text(to_nice_freq(freq_fav_list[6])); btn_fav_6.set_text(to_nice_freq(freq_fav_list[6].frequency));
btn_fav_7.set_text(to_nice_freq(freq_fav_list[7])); btn_fav_7.set_text(to_nice_freq(freq_fav_list[7].frequency));
btn_fav_8.set_text(to_nice_freq(freq_fav_list[8])); btn_fav_8.set_text(to_nice_freq(freq_fav_list[8].frequency));
btn_fav_9.set_text(to_nice_freq(freq_fav_list[9])); btn_fav_9.set_text(to_nice_freq(freq_fav_list[9].frequency));
} }
FmRadioView::~FmRadioView() { FmRadioView::~FmRadioView() {
@@ -160,9 +226,16 @@ FmRadioView::~FmRadioView() {
} }
void FmRadioView::on_audio_spectrum() { void FmRadioView::on_audio_spectrum() {
for (size_t i = 0; i < audio_spectrum_data->db.size(); i++) if (audio_spectrum_data && audio_spectrum_data->db.size() <= 128) {
for (size_t i = 0; i < audio_spectrum_data->db.size(); i++) {
audio_spectrum[i] = ((int16_t)audio_spectrum_data->db[i] - 127) * 256; audio_spectrum[i] = ((int16_t)audio_spectrum_data->db[i] - 127) * 256;
}
waveform.set_dirty(); waveform.set_dirty();
} else {
// Fallback: Clear waveform if no valid data
std::fill(audio_spectrum, audio_spectrum + 128, 0);
waveform.set_dirty();
}
} }
} // namespace ui::external_app::fmradio } // namespace ui::external_app::fmradio

View File

@@ -1,5 +1,6 @@
/* /*
* Copyright (C) 2024 HTotoo * Copyright (C) 2024 HTotoo
* Copyright (C) 2025 RocketGod - Added modes from my Flipper Zero RF Jammer App - https://betaskynet.com
* *
* This file is part of PortaPack. * This file is part of PortaPack.
* *
@@ -38,6 +39,9 @@
#include "radio_state.hpp" #include "radio_state.hpp"
#include "log_file.hpp" #include "log_file.hpp"
#include "utility.hpp" #include "utility.hpp"
#include "audio.hpp"
#include "freqman_db.hpp"
#include "ui_freqman.hpp"
using namespace ui; using namespace ui;
@@ -62,23 +66,55 @@ class FmRadioView : public View {
int16_t audio_spectrum[128]{0}; int16_t audio_spectrum[128]{0};
bool audio_spectrum_update = false; bool audio_spectrum_update = false;
AudioSpectrum* audio_spectrum_data{nullptr}; AudioSpectrum* audio_spectrum_data{nullptr};
rf::Frequency freq_fav_list[12] = {0}; struct Favorite {
rf::Frequency frequency = 0;
int32_t modulation = static_cast<int32_t>(ReceiverModel::Mode::WidebandFMAudio);
uint8_t bandwidth = 0;
};
Favorite freq_fav_list[12];
audio::Rate audio_sampling_rate = audio::Rate::Hz_48000;
uint8_t radio_bw = 0;
app_settings::SettingsManager settings_{ app_settings::SettingsManager settings_{
"rx_fmradio", "rx_fmradio",
app_settings::Mode::RX, app_settings::Mode::RX,
{{"favlist0"sv, &freq_fav_list[0]}, {{"favlist0_freq"sv, &freq_fav_list[0].frequency},
{"favlist1"sv, &freq_fav_list[1]}, {"favlist1_freq"sv, &freq_fav_list[1].frequency},
{"favlist2"sv, &freq_fav_list[2]}, {"favlist2_freq"sv, &freq_fav_list[2].frequency},
{"favlist3"sv, &freq_fav_list[3]}, {"favlist3_freq"sv, &freq_fav_list[3].frequency},
{"favlist4"sv, &freq_fav_list[4]}, {"favlist4_freq"sv, &freq_fav_list[4].frequency},
{"favlist5"sv, &freq_fav_list[5]}, {"favlist5_freq"sv, &freq_fav_list[5].frequency},
{"favlist6"sv, &freq_fav_list[6]}, {"favlist6_freq"sv, &freq_fav_list[6].frequency},
{"favlist7"sv, &freq_fav_list[7]}, {"favlist7_freq"sv, &freq_fav_list[7].frequency},
{"favlist8"sv, &freq_fav_list[8]}, {"favlist8_freq"sv, &freq_fav_list[8].frequency},
{"favlist9"sv, &freq_fav_list[9]}, {"favlist9_freq"sv, &freq_fav_list[9].frequency},
{"favlist10"sv, &freq_fav_list[10]}, {"favlist10_freq"sv, &freq_fav_list[10].frequency},
{"favlist11"sv, &freq_fav_list[11]}}}; {"favlist11_freq"sv, &freq_fav_list[11].frequency},
{"favlist0_mod"sv, &freq_fav_list[0].modulation},
{"favlist1_mod"sv, &freq_fav_list[1].modulation},
{"favlist2_mod"sv, &freq_fav_list[2].modulation},
{"favlist3_mod"sv, &freq_fav_list[3].modulation},
{"favlist4_mod"sv, &freq_fav_list[4].modulation},
{"favlist5_mod"sv, &freq_fav_list[5].modulation},
{"favlist6_mod"sv, &freq_fav_list[6].modulation},
{"favlist7_mod"sv, &freq_fav_list[7].modulation},
{"favlist8_mod"sv, &freq_fav_list[8].modulation},
{"favlist9_mod"sv, &freq_fav_list[9].modulation},
{"favlist10_mod"sv, &freq_fav_list[10].modulation},
{"favlist11_mod"sv, &freq_fav_list[11].modulation},
{"favlist0_bw"sv, &freq_fav_list[0].bandwidth},
{"favlist1_bw"sv, &freq_fav_list[1].bandwidth},
{"favlist2_bw"sv, &freq_fav_list[2].bandwidth},
{"favlist3_bw"sv, &freq_fav_list[3].bandwidth},
{"favlist4_bw"sv, &freq_fav_list[4].bandwidth},
{"favlist5_bw"sv, &freq_fav_list[5].bandwidth},
{"favlist6_bw"sv, &freq_fav_list[6].bandwidth},
{"favlist7_bw"sv, &freq_fav_list[7].bandwidth},
{"favlist8_bw"sv, &freq_fav_list[8].bandwidth},
{"favlist9_bw"sv, &freq_fav_list[9].bandwidth},
{"favlist10_bw"sv, &freq_fav_list[10].bandwidth},
{"favlist11_bw"sv, &freq_fav_list[11].bandwidth},
{"radio_bw"sv, &radio_bw}}};
RFAmpField field_rf_amp{ RFAmpField field_rf_amp{
{13 * 8, 0 * 16}}; {13 * 8, 0 * 16}};
@@ -95,6 +131,24 @@ class FmRadioView : public View {
{0 * 8, 0 * 16}, {0 * 8, 0 * 16},
nav_}; nav_};
OptionsField field_bw{
{10 * 8, FMR_BTNGRID_TOP + 6 * 34},
6,
{}};
Text text_mode_label{
{20 * 8, FMR_BTNGRID_TOP + 6 * 34, 5 * 8, 1 * 28},
"MODE:"};
OptionsField field_modulation{
{26 * 8, FMR_BTNGRID_TOP + 6 * 34},
4,
{{"AM", static_cast<int32_t>(ReceiverModel::Mode::AMAudio)},
{"NFM", static_cast<int32_t>(ReceiverModel::Mode::NarrowbandFMAudio)},
{"WFM", static_cast<int32_t>(ReceiverModel::Mode::WidebandFMAudio)},
{"USB", static_cast<int32_t>(ReceiverModel::Mode::AMAudio)},
{"LSB", static_cast<int32_t>(ReceiverModel::Mode::AMAudio)}}};
TextField txt_save_help{ TextField txt_save_help{
{2, FMR_BTNGRID_TOP + 6 * 34 - 20, 12 * 8, 16}, {2, FMR_BTNGRID_TOP + 6 * 34 - 20, 12 * 8, 16},
" "}; " "};
@@ -124,10 +178,12 @@ class FmRadioView : public View {
Button btn_fav_save{{2, FMR_BTNGRID_TOP + 6 * 34, 7 * 8, 1 * 28}, "Save"}; Button btn_fav_save{{2, FMR_BTNGRID_TOP + 6 * 34, 7 * 8, 1 * 28}, "Save"};
bool save_fav = false; bool save_fav = false;
void on_btn_clicked(uint8_t i); void on_btn_clicked(uint8_t i);
void update_fav_btn_texts(); void update_fav_btn_texts();
std::string to_nice_freq(rf::Frequency freq); std::string to_nice_freq(rf::Frequency freq);
void on_audio_spectrum(); void on_audio_spectrum();
void change_mode(int32_t mod);
MessageHandlerRegistration message_handler_audio_spectrum{ MessageHandlerRegistration message_handler_audio_spectrum{
Message::ID::AudioSpectrum, Message::ID::AudioSpectrum,