2015-07-08 08:39:24 -07:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Jared Boone, ShareBrained Technology, Inc.
|
2016-07-30 05:27:28 +02:00
|
|
|
* Copyright (C) 2016 Furrtek
|
2024-02-08 18:28:49 +01:00
|
|
|
* Copyright (C) 2024 u-foka
|
|
|
|
* Copyleft (ɔ) 2024 zxkmm under GPL license
|
2015-07-08 08:39:24 -07:00
|
|
|
*
|
|
|
|
* This file is part of PortaPack.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, Inc., 51 Franklin Street,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ui_navigation.hpp"
|
2016-04-28 14:59:14 +02:00
|
|
|
|
2016-08-23 08:45:33 +02:00
|
|
|
#include "bmp_modal_warning.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "bmp_splash.hpp"
|
|
|
|
#include "event_m0.hpp"
|
2017-06-11 02:53:06 +01:00
|
|
|
#include "portapack_persistent_memory.hpp"
|
2023-04-23 21:48:45 +02:00
|
|
|
#include "portapack_shared_memory.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "portapack.hpp"
|
2015-07-08 08:39:24 -07:00
|
|
|
|
2020-08-08 23:06:53 +02:00
|
|
|
#include "ui_about_simple.hpp"
|
2017-07-14 10:02:21 +01:00
|
|
|
#include "ui_adsb_rx.hpp"
|
2021-03-07 16:05:23 -06:00
|
|
|
#include "ui_aprs_rx.hpp"
|
2017-10-14 16:30:49 +01:00
|
|
|
#include "ui_aprs_tx.hpp"
|
2017-01-16 08:40:17 +00:00
|
|
|
#include "ui_bht_tx.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_btle_rx.hpp"
|
2020-04-21 14:22:09 +02:00
|
|
|
#include "ui_debug.hpp"
|
2016-12-09 18:21:47 +01:00
|
|
|
#include "ui_encoders.hpp"
|
2017-09-19 20:14:56 +01:00
|
|
|
#include "ui_fileman.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_flash_utility.hpp"
|
2023-06-07 08:33:32 -07:00
|
|
|
#include "ui_font_fixed_8x16.hpp"
|
2016-12-05 12:56:41 +01:00
|
|
|
#include "ui_freqman.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_iq_trim.hpp"
|
|
|
|
#include "ui_level.hpp"
|
|
|
|
#include "ui_looking_glass_app.hpp"
|
2017-03-22 03:21:06 +00:00
|
|
|
#include "ui_mictx.hpp"
|
2024-10-15 22:14:33 +02:00
|
|
|
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_playlist.hpp"
|
2017-02-07 17:48:17 +00:00
|
|
|
#include "ui_pocsag_tx.hpp"
|
2016-07-27 03:03:40 +02:00
|
|
|
#include "ui_rds.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_recon.hpp"
|
2017-05-18 11:06:11 +01:00
|
|
|
#include "ui_scanner.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_sd_over_usb.hpp"
|
2017-10-14 16:30:49 +01:00
|
|
|
#include "ui_sd_wipe.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_search.hpp"
|
2018-05-16 09:45:13 +01:00
|
|
|
#include "ui_settings.hpp"
|
2017-10-14 16:30:49 +01:00
|
|
|
#include "ui_siggen.hpp"
|
|
|
|
#include "ui_sonde.hpp"
|
2023-09-21 08:43:10 -07:00
|
|
|
#include "ui_ss_viewer.hpp"
|
2023-05-23 12:39:22 +08:00
|
|
|
// #include "ui_test.hpp"
|
2023-05-19 13:39:35 -07:00
|
|
|
#include "ui_text_editor.hpp"
|
2017-09-23 12:02:32 +01:00
|
|
|
#include "ui_touchtunes.hpp"
|
2017-10-15 15:53:40 +01:00
|
|
|
#include "ui_view_wav.hpp"
|
2023-11-28 21:11:30 +01:00
|
|
|
#include "ui_weatherstation.hpp"
|
Subghz decoder (#1646)
* Initial commit - wip
* Half part of the transition of baseband processor.
* More SGD
* WIP, Weather refactor, UI improv
* Rename
* Added 4msps, and fixes
* Fixes
* princeton working
* Renamed proc_weather, bc now multifunctional
* Proto: bett
* FPS_CAME = 4,
FPS_PRASTEL = 5,
FPS_AIRFORCE = 6,
* Came Atomo, fixes
* Separate weather and sgd, bc of baseband size limit
* Fix display
* Save space
* More protos
* Dooya proto added
* More protos
* add protos
* More protos
* Move weather to ext app
* nw
* Revert "Move weather to ext app"
This reverts commit 8a84aac2f59274b72de7c7803deb137a21838076.
* revert
* Fix merge
* Better naming
* More protos
* More protos
* Add protos
* Fix warning
* Add NeroRadio
* more protos
* more protos
* More protos
* Shrink a bit
* fixes
* More protos
* Nicer code
* Fix naming
* Fix format
* Remove unused
* Fix some protos, that needs a LOOOONG part with the same lo/high
* Modify key calculation
2023-12-16 23:37:51 +01:00
|
|
|
#include "ui_subghzd.hpp"
|
2016-12-09 18:21:47 +01:00
|
|
|
#include "ui_whipcalc.hpp"
|
2024-04-29 17:38:27 +02:00
|
|
|
#include "ui_battinfo.hpp"
|
2023-10-02 20:19:22 +02:00
|
|
|
#include "ui_external_items_menu_loader.hpp"
|
2016-12-09 18:21:47 +01:00
|
|
|
|
2016-01-31 09:34:24 +01:00
|
|
|
#include "ais_app.hpp"
|
2018-06-10 10:15:43 +01:00
|
|
|
#include "analog_audio_app.hpp"
|
2024-09-20 21:59:17 +12:00
|
|
|
// #include "ble_comm_app.hpp"
|
2023-11-03 03:08:49 -04:00
|
|
|
#include "ble_rx_app.hpp"
|
2023-11-01 06:46:41 -04:00
|
|
|
#include "ble_tx_app.hpp"
|
2018-06-10 10:15:43 +01:00
|
|
|
#include "capture_app.hpp"
|
2016-01-31 09:34:24 +01:00
|
|
|
#include "ert_app.hpp"
|
2016-08-23 08:45:33 +02:00
|
|
|
#include "pocsag_app.hpp"
|
2018-12-18 16:25:21 +00:00
|
|
|
#include "soundboard_app.hpp"
|
2016-01-31 09:34:24 +01:00
|
|
|
|
2016-02-16 11:09:00 -08:00
|
|
|
#include "core_control.hpp"
|
2016-04-20 10:22:05 -07:00
|
|
|
#include "file.hpp"
|
2023-09-27 12:03:02 -07:00
|
|
|
#include "file_reader.hpp"
|
2016-02-19 15:34:02 -08:00
|
|
|
#include "png_writer.hpp"
|
2024-03-25 02:44:49 -05:00
|
|
|
#include "file_path.hpp"
|
2024-04-03 16:27:13 +02:00
|
|
|
#include "ff.h"
|
|
|
|
|
|
|
|
#include <locale>
|
|
|
|
#include <codecvt>
|
2015-07-08 08:39:24 -07:00
|
|
|
|
2018-01-07 23:13:08 +00:00
|
|
|
using portapack::receiver_model;
|
2018-02-22 07:04:19 +00:00
|
|
|
using portapack::transmitter_model;
|
2023-06-23 16:13:39 -07:00
|
|
|
namespace pmem = portapack::persistent_memory;
|
2018-01-07 23:13:08 +00:00
|
|
|
|
2015-07-08 08:39:24 -07:00
|
|
|
namespace ui {
|
|
|
|
|
2024-01-20 23:35:39 +01:00
|
|
|
bool CstrCmp::operator()(const char* a, const char* b) const {
|
|
|
|
return strcmp(a, b) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NavigationView::AppMap generate_app_map(const NavigationView::AppList& appList) {
|
|
|
|
NavigationView::AppMap out;
|
|
|
|
|
|
|
|
for (auto& app : appList) {
|
|
|
|
if (app.id == nullptr) {
|
|
|
|
// Skip items with no id
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto res = out.emplace(app.id, app);
|
|
|
|
if (!res.second) {
|
|
|
|
chDbgPanic("Application cannot be added, ID not unique!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(u-foka): Check consistency of command names (where we add rx/tx postfix)
|
|
|
|
const NavigationView::AppList NavigationView::appList = {
|
|
|
|
/* HOME ******************************************************************/
|
|
|
|
{nullptr, "Receive", HOME, Color::cyan(), &bitmap_icon_receivers, new ViewFactory<ReceiversMenuView>()},
|
|
|
|
{nullptr, "Transmit", HOME, Color::cyan(), &bitmap_icon_transmit, new ViewFactory<TransmittersMenuView>()},
|
|
|
|
{"capture", "Capture", HOME, Color::red(), &bitmap_icon_capture, new ViewFactory<CaptureAppView>()},
|
|
|
|
{"replay", "Replay", HOME, Color::green(), &bitmap_icon_replay, new ViewFactory<PlaylistView>()},
|
|
|
|
{"scanner", "Scanner", HOME, Color::green(), &bitmap_icon_scanner, new ViewFactory<ScannerView>()},
|
|
|
|
{"microphone", "Microphone", HOME, Color::green(), &bitmap_icon_microphone, new ViewFactory<MicTXView>()},
|
|
|
|
{"lookingglass", "Looking Glass", HOME, Color::green(), &bitmap_icon_looking, new ViewFactory<GlassView>()},
|
|
|
|
{nullptr, "Utilities", HOME, Color::cyan(), &bitmap_icon_utilities, new ViewFactory<UtilitiesMenuView>()},
|
|
|
|
{nullptr, "Settings", HOME, Color::cyan(), &bitmap_icon_setup, new ViewFactory<SettingsMenuView>()},
|
|
|
|
{nullptr, "Debug", HOME, Color::light_grey(), &bitmap_icon_debug, new ViewFactory<DebugMenuView>()},
|
|
|
|
/* RX ********************************************************************/
|
|
|
|
{"adsbrx", "ADS-B", RX, Color::green(), &bitmap_icon_adsb, new ViewFactory<ADSBRxView>()},
|
|
|
|
{"ais", "AIS Boats", RX, Color::green(), &bitmap_icon_ais, new ViewFactory<AISAppView>()},
|
|
|
|
{"aprsrx", "APRS", RX, Color::green(), &bitmap_icon_aprs, new ViewFactory<APRSRXView>()},
|
|
|
|
{"audio", "Audio", RX, Color::green(), &bitmap_icon_speaker, new ViewFactory<AnalogAudioView>()},
|
|
|
|
//{"blecomm", "BLE Comm", RX, ui::Color::orange(), &bitmap_icon_btle, new ViewFactory<BLECommView>()},
|
|
|
|
{"blerx", "BLE Rx", RX, Color::green(), &bitmap_icon_btle, new ViewFactory<BLERxView>()},
|
|
|
|
{"ert", "ERT Meter", RX, Color::green(), &bitmap_icon_ert, new ViewFactory<ERTAppView>()},
|
|
|
|
{"level", "Level", RX, Color::green(), &bitmap_icon_options_radio, new ViewFactory<LevelView>()},
|
|
|
|
{"pocsag", "POCSAG", RX, Color::green(), &bitmap_icon_pocsag, new ViewFactory<POCSAGAppView>()},
|
|
|
|
{"radiosonde", "Radiosnde", RX, Color::green(), &bitmap_icon_sonde, new ViewFactory<SondeView>()},
|
|
|
|
{"recon", "Recon", RX, Color::green(), &bitmap_icon_scanner, new ViewFactory<ReconView>()},
|
|
|
|
{"search", "Search", RX, Color::yellow(), &bitmap_icon_search, new ViewFactory<SearchView>()},
|
|
|
|
{"subghzd", "SubGhzD", RX, Color::yellow(), &bitmap_icon_remote, new ViewFactory<SubGhzDView>()},
|
2024-04-29 13:30:21 +02:00
|
|
|
{"weather", "Weather", RX, Color::green(), &bitmap_icon_thermometer, new ViewFactory<WeatherView>()},
|
2024-10-20 00:03:47 +02:00
|
|
|
//{"fskrx", "FSK RX", RX, Color::yellow(), &bitmap_icon_remote, new ViewFactory<FskxRxMainView>()}, //for JT
|
2024-01-20 23:35:39 +01:00
|
|
|
//{"dmr", "DMR", RX, Color::dark_grey(), &bitmap_icon_dmr, new ViewFactory<NotImplementedView>()},
|
|
|
|
//{"sigfox", "SIGFOX", RX, Color::dark_grey(), &bitmap_icon_fox, new ViewFactory<NotImplementedView>()},
|
|
|
|
//{"lora", "LoRa", RX, Color::dark_grey(), &bitmap_icon_lora, new ViewFactory<NotImplementedView>()},
|
|
|
|
//{"sstv", "SSTV", RX, Color::dark_grey(), &bitmap_icon_sstv, new ViewFactory<NotImplementedView>()},
|
|
|
|
//{"tetra", "TETRA", RX, Color::dark_grey(), &bitmap_icon_tetra, new ViewFactory<NotImplementedView>()},
|
|
|
|
/* TX ********************************************************************/
|
2024-08-09 23:47:08 +02:00
|
|
|
//{"adsbtx", "ADS-B TX", TX, ui::Color::green(), &bitmap_icon_adsb, new ViewFactory<ADSBTxView>()},
|
2024-01-20 23:35:39 +01:00
|
|
|
{"aprstx", "APRS TX", TX, ui::Color::green(), &bitmap_icon_aprs, new ViewFactory<APRSTXView>()},
|
|
|
|
{"bht", "BHT Xy/EP", TX, ui::Color::green(), &bitmap_icon_bht, new ViewFactory<BHTView>()},
|
|
|
|
{"bletx", "BLE Tx", TX, ui::Color::green(), &bitmap_icon_btle, new ViewFactory<BLETxView>()},
|
|
|
|
{"ooktx", "OOK", TX, ui::Color::yellow(), &bitmap_icon_remote, new ViewFactory<EncodersView>()},
|
|
|
|
{"pocsagtx", "POCSAG TX", TX, ui::Color::green(), &bitmap_icon_pocsag, new ViewFactory<POCSAGTXView>()},
|
|
|
|
{"rdstx", "RDS", TX, ui::Color::green(), &bitmap_icon_rds, new ViewFactory<RDSView>()},
|
|
|
|
{"soundbrd", "Soundbrd", TX, ui::Color::green(), &bitmap_icon_soundboard, new ViewFactory<SoundBoardView>()},
|
|
|
|
{"touchtune", "TouchTune", TX, ui::Color::green(), &bitmap_icon_touchtunes, new ViewFactory<TouchTunesView>()},
|
|
|
|
/* UTILITIES *************************************************************/
|
|
|
|
{"antennalength", "Antenna Length", UTILITIES, Color::green(), &bitmap_icon_tools_antenna, new ViewFactory<WhipCalcView>()},
|
|
|
|
{"filemanager", "File Manager", UTILITIES, Color::green(), &bitmap_icon_dir, new ViewFactory<FileManagerView>()},
|
|
|
|
{"freqman", "Freq. Manager", UTILITIES, Color::green(), &bitmap_icon_freqman, new ViewFactory<FrequencyManagerView>()},
|
|
|
|
{"notepad", "Notepad", UTILITIES, Color::dark_cyan(), &bitmap_icon_notepad, new ViewFactory<TextEditorView>()},
|
|
|
|
{"iqtrim", "IQ Trim", UTILITIES, Color::orange(), &bitmap_icon_trim, new ViewFactory<IQTrimView>()},
|
|
|
|
{nullptr, "SD Over USB", UTILITIES, Color::yellow(), &bitmap_icon_hackrf, new ViewFactory<SdOverUsbView>()},
|
|
|
|
{"signalgen", "Signal Gen", UTILITIES, Color::green(), &bitmap_icon_cwgen, new ViewFactory<SigGenView>()},
|
|
|
|
//{"testapp", "Test App", UTILITIES, Color::dark_grey(), nullptr, new ViewFactory<TestView>()},
|
2024-10-15 22:14:33 +02:00
|
|
|
|
2024-01-20 23:35:39 +01:00
|
|
|
{"wavview", "Wav View", UTILITIES, Color::yellow(), &bitmap_icon_soundboard, new ViewFactory<ViewWavView>()},
|
|
|
|
// Dangerous apps.
|
|
|
|
{nullptr, "Flash Utility", UTILITIES, Color::red(), &bitmap_icon_temperature, new ViewFactory<FlashUtilityView>()},
|
|
|
|
{nullptr, "Wipe SD card", UTILITIES, Color::red(), &bitmap_icon_tools_wipesd, new ViewFactory<WipeSDView>()},
|
|
|
|
};
|
|
|
|
|
|
|
|
const NavigationView::AppMap NavigationView::appMap = generate_app_map(NavigationView::appList);
|
2024-01-07 18:56:09 +01:00
|
|
|
|
|
|
|
bool NavigationView::StartAppByName(const char* name) {
|
2024-01-12 17:33:41 +01:00
|
|
|
home(false);
|
2024-01-20 23:35:39 +01:00
|
|
|
|
|
|
|
auto it = appMap.find(name);
|
|
|
|
if (it != appMap.end()) {
|
|
|
|
push_view(std::unique_ptr<View>(it->second.viewFactory->produce(*this)));
|
2024-01-07 18:56:09 +01:00
|
|
|
return true;
|
|
|
|
}
|
2024-01-20 23:35:39 +01:00
|
|
|
|
2024-01-07 18:56:09 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
/* StatusTray ************************************************************/
|
|
|
|
|
|
|
|
StatusTray::StatusTray(Point pos)
|
|
|
|
: View{{pos, {0, height}}},
|
|
|
|
pos_(pos) {
|
|
|
|
set_focusable(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void StatusTray::add(Widget* child) {
|
|
|
|
width_ += child->parent_rect().width();
|
|
|
|
add_child(child);
|
|
|
|
}
|
|
|
|
|
|
|
|
void StatusTray::update_layout() {
|
|
|
|
// Widen the tray's parent rect.
|
|
|
|
auto rect = parent_rect();
|
|
|
|
set_parent_rect({{rect.left() - width_, rect.top()}, {rect.right() + width_, height}});
|
|
|
|
|
|
|
|
// Update the children.
|
|
|
|
auto x = 0;
|
|
|
|
for (auto child : children()) {
|
|
|
|
auto size = child->parent_rect().size();
|
|
|
|
child->set_parent_rect({{x, 0}, size});
|
|
|
|
x += size.width();
|
|
|
|
}
|
|
|
|
set_dirty();
|
|
|
|
}
|
|
|
|
|
|
|
|
void StatusTray::clear() {
|
|
|
|
// More efficient than 'remove_children'.
|
|
|
|
for (auto child : children())
|
|
|
|
child->set_parent(nullptr);
|
|
|
|
children_.clear();
|
|
|
|
width_ = 0;
|
|
|
|
set_parent_rect({pos_, {width_, height}});
|
|
|
|
set_dirty();
|
|
|
|
}
|
|
|
|
|
|
|
|
void StatusTray::paint(Painter&) {
|
|
|
|
}
|
|
|
|
|
2015-07-08 08:39:24 -07:00
|
|
|
/* SystemStatusView ******************************************************/
|
|
|
|
|
2018-01-08 03:47:37 +00:00
|
|
|
SystemStatusView::SystemStatusView(
|
2023-05-19 08:16:05 +12:00
|
|
|
NavigationView& nav)
|
|
|
|
: nav_(nav) {
|
|
|
|
add_children({
|
|
|
|
&backdrop,
|
|
|
|
&button_back,
|
|
|
|
&title,
|
|
|
|
&button_title,
|
2023-06-23 16:13:39 -07:00
|
|
|
&status_icons,
|
2023-05-19 08:16:05 +12:00
|
|
|
});
|
|
|
|
|
2023-10-09 05:04:46 -05:00
|
|
|
rtc_battery_workaround();
|
|
|
|
|
2023-10-15 18:04:26 -05:00
|
|
|
ui::load_blacklist();
|
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
if (pmem::should_use_sdcard_for_pmem()) {
|
|
|
|
pmem::load_persistent_settings_from_file();
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
|
|
|
|
2023-12-24 01:54:56 -06:00
|
|
|
// configure CLKOUT per pmem setting
|
|
|
|
portapack::clock_manager.enable_clock_output(pmem::clkout_enabled());
|
|
|
|
|
2023-11-19 15:08:29 +01:00
|
|
|
// force apply of selected sdcard speed override at UI startup
|
|
|
|
pmem::set_config_sdcard_high_speed_io(pmem::config_sdcard_high_speed_io(), false);
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
button_back.id = -1; // Special ID used by FocusManager
|
2024-05-27 21:02:52 +02:00
|
|
|
title.set_style(Theme::getInstance()->bg_dark);
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
button_back.on_select = [this](ImageButton&) {
|
2023-06-23 16:13:39 -07:00
|
|
|
if (pmem::should_use_sdcard_for_pmem()) {
|
|
|
|
pmem::save_persistent_settings_to_file();
|
2023-06-04 21:25:25 +02:00
|
|
|
}
|
2023-05-19 08:16:05 +12:00
|
|
|
if (this->on_back)
|
|
|
|
this->on_back();
|
|
|
|
};
|
|
|
|
|
|
|
|
button_title.on_select = [this](ImageButton&) {
|
|
|
|
this->on_title();
|
|
|
|
};
|
|
|
|
|
|
|
|
button_converter.on_select = [this](ImageButton&) {
|
|
|
|
this->on_converter();
|
|
|
|
};
|
|
|
|
|
2023-07-05 02:12:40 -05:00
|
|
|
toggle_speaker.on_change = [this](bool v) {
|
|
|
|
pmem::set_config_speaker_disable(v);
|
|
|
|
audio::output::update_audio_mute();
|
|
|
|
refresh();
|
2023-05-19 08:16:05 +12:00
|
|
|
};
|
|
|
|
|
2023-07-05 02:12:40 -05:00
|
|
|
toggle_mute.on_change = [this](bool v) {
|
|
|
|
pmem::set_config_audio_mute(v);
|
|
|
|
audio::output::update_audio_mute();
|
|
|
|
refresh();
|
2023-06-26 01:08:09 -05:00
|
|
|
};
|
|
|
|
|
2024-01-30 13:58:40 +08:00
|
|
|
toggle_stealth.on_change = [this, &nav](bool v) {
|
2023-07-05 02:12:40 -05:00
|
|
|
pmem::set_stealth_mode(v);
|
2024-01-30 13:58:40 +08:00
|
|
|
if (nav.is_valid() && v) {
|
|
|
|
nav.display_modal(
|
|
|
|
"Stealth",
|
|
|
|
"You just enabled stealth mode.\n"
|
|
|
|
"When you transmit,\n"
|
|
|
|
"screen will turn off;\n");
|
|
|
|
}
|
2023-07-05 02:12:40 -05:00
|
|
|
refresh();
|
2023-05-19 08:16:05 +12:00
|
|
|
};
|
|
|
|
|
2024-04-29 17:38:27 +02:00
|
|
|
battery_icon.on_select = [this]() { on_battery_details(); };
|
|
|
|
battery_text.on_select = [this]() { on_battery_details(); };
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
button_bias_tee.on_select = [this](ImageButton&) {
|
|
|
|
this->on_bias_tee();
|
|
|
|
};
|
|
|
|
|
|
|
|
button_camera.on_select = [this](ImageButton&) {
|
|
|
|
this->on_camera();
|
|
|
|
};
|
|
|
|
|
|
|
|
button_sleep.on_select = [this](ImageButton&) {
|
|
|
|
DisplaySleepMessage message;
|
|
|
|
EventDispatcher::send_message(message);
|
|
|
|
};
|
|
|
|
|
|
|
|
button_clock_status.on_select = [this](ImageButton&) {
|
|
|
|
this->on_clk();
|
|
|
|
};
|
2023-06-19 14:06:07 -05:00
|
|
|
|
2023-07-05 02:12:40 -05:00
|
|
|
// Initialize toggle buttons
|
|
|
|
toggle_speaker.set_value(pmem::config_speaker_disable());
|
|
|
|
toggle_mute.set_value(pmem::config_audio_mute());
|
|
|
|
toggle_stealth.set_value(pmem::stealth_mode());
|
|
|
|
|
2023-08-20 04:13:04 -05:00
|
|
|
audio::output::stop();
|
2023-06-19 14:06:07 -05:00
|
|
|
audio::output::update_audio_mute();
|
2023-06-23 16:13:39 -07:00
|
|
|
refresh();
|
2016-01-31 09:34:24 +01:00
|
|
|
}
|
|
|
|
|
2024-04-29 17:38:27 +02:00
|
|
|
// when battery icon / text is clicked
|
|
|
|
void SystemStatusView::on_battery_details() {
|
|
|
|
if (!nav_.is_valid()) return;
|
|
|
|
if (batt_info_up) return;
|
|
|
|
batt_info_up = true;
|
|
|
|
nav_.push<BattinfoView>();
|
|
|
|
nav_.set_on_pop([this]() {
|
|
|
|
batt_info_up = false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void SystemStatusView::on_battery_data(const BatteryStateMessage* msg) {
|
2024-07-17 11:18:57 +02:00
|
|
|
if (!batt_was_inited) {
|
|
|
|
batt_was_inited = true;
|
|
|
|
refresh();
|
|
|
|
}
|
2024-04-29 17:38:27 +02:00
|
|
|
if (!pmem::ui_hide_numeric_battery()) {
|
2024-07-17 11:18:57 +02:00
|
|
|
battery_text.set_battery(msg->valid_mask, msg->percent, msg->on_charger);
|
2024-04-29 17:38:27 +02:00
|
|
|
}
|
|
|
|
if (!pmem::ui_hide_battery_icon()) {
|
2024-07-17 11:18:57 +02:00
|
|
|
battery_icon.set_battery(msg->valid_mask, msg->percent, msg->on_charger);
|
2024-04-29 17:38:27 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-01-07 23:13:08 +00:00
|
|
|
void SystemStatusView::refresh() {
|
2023-06-23 16:13:39 -07:00
|
|
|
// NB: Order of insertion is the display order Left->Right.
|
|
|
|
// TODO: Might be better to support hide and only add once.
|
|
|
|
status_icons.clear();
|
2023-06-27 21:34:25 -05:00
|
|
|
if (!pmem::ui_hide_camera()) status_icons.add(&button_camera);
|
|
|
|
if (!pmem::ui_hide_sleep()) status_icons.add(&button_sleep);
|
2023-07-05 02:12:40 -05:00
|
|
|
if (!pmem::ui_hide_stealth()) status_icons.add(&toggle_stealth);
|
2023-06-27 21:34:25 -05:00
|
|
|
if (!pmem::ui_hide_converter()) status_icons.add(&button_converter);
|
|
|
|
if (!pmem::ui_hide_bias_tee()) status_icons.add(&button_bias_tee);
|
|
|
|
if (!pmem::ui_hide_clock()) status_icons.add(&button_clock_status);
|
2023-07-05 02:12:40 -05:00
|
|
|
if (!pmem::ui_hide_mute()) status_icons.add(&toggle_mute);
|
2023-06-26 01:08:09 -05:00
|
|
|
|
|
|
|
// Display "Disable speaker" icon only if AK4951 Codec which has separate speaker/headphone control
|
2023-07-05 02:12:40 -05:00
|
|
|
if (audio::speaker_disable_supported() && !pmem::ui_hide_speaker()) status_icons.add(&toggle_speaker);
|
2023-06-26 01:08:09 -05:00
|
|
|
|
2024-04-29 17:38:27 +02:00
|
|
|
if (battery::BatteryManagement::isDetected()) {
|
2024-07-17 11:18:57 +02:00
|
|
|
batt_was_inited = true;
|
2024-04-29 17:38:27 +02:00
|
|
|
if (!pmem::ui_hide_battery_icon()) {
|
|
|
|
status_icons.add(&battery_icon);
|
|
|
|
};
|
|
|
|
if (!pmem::ui_hide_numeric_battery()) {
|
|
|
|
status_icons.add(&battery_text);
|
|
|
|
}
|
|
|
|
}
|
2024-02-07 16:07:30 +08:00
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
if (!pmem::ui_hide_sd_card()) status_icons.add(&sd_card_status_view);
|
2024-04-29 17:38:27 +02:00
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
status_icons.update_layout();
|
|
|
|
|
|
|
|
// Clock status
|
|
|
|
bool external_clk = portapack::clock_manager.get_reference().source == ClockManager::ReferenceSource::External;
|
|
|
|
button_clock_status.set_bitmap(external_clk ? &bitmap_icon_clk_ext : &bitmap_icon_clk_int);
|
2024-05-27 21:02:52 +02:00
|
|
|
button_clock_status.set_foreground(pmem::clkout_enabled() ? *Theme::getInstance()->status_active : Theme::getInstance()->fg_light->foreground);
|
2023-06-23 16:13:39 -07:00
|
|
|
|
|
|
|
// Antenna DC Bias
|
2023-05-19 08:16:05 +12:00
|
|
|
if (portapack::get_antenna_bias()) {
|
|
|
|
button_bias_tee.set_bitmap(&bitmap_icon_biast_on);
|
2024-05-27 21:02:52 +02:00
|
|
|
button_bias_tee.set_foreground(Theme::getInstance()->warning_dark->foreground);
|
2023-05-19 08:16:05 +12:00
|
|
|
} else {
|
|
|
|
button_bias_tee.set_bitmap(&bitmap_icon_biast_off);
|
2024-05-27 21:02:52 +02:00
|
|
|
button_bias_tee.set_foreground(Theme::getInstance()->fg_light->foreground);
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
// Converter
|
2023-07-05 02:12:40 -05:00
|
|
|
button_converter.set_bitmap(pmem::config_updown_converter() ? &bitmap_icon_downconvert : &bitmap_icon_upconvert);
|
2024-05-27 21:02:52 +02:00
|
|
|
button_converter.set_foreground(pmem::config_converter() ? Theme::getInstance()->fg_red->foreground : Theme::getInstance()->fg_light->foreground);
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
set_dirty();
|
2018-01-07 23:13:08 +00:00
|
|
|
}
|
|
|
|
|
2016-02-07 10:33:15 -08:00
|
|
|
void SystemStatusView::set_back_enabled(bool new_value) {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (new_value) {
|
|
|
|
add_child(&button_back);
|
|
|
|
} else {
|
|
|
|
remove_child(&button_back);
|
|
|
|
}
|
2020-08-01 21:55:09 -04:00
|
|
|
}
|
|
|
|
|
2024-04-21 01:34:55 -05:00
|
|
|
void SystemStatusView::set_back_hidden(bool new_value) {
|
|
|
|
button_back.hidden(new_value);
|
|
|
|
}
|
|
|
|
|
2020-08-01 21:55:09 -04:00
|
|
|
void SystemStatusView::set_title_image_enabled(bool new_value) {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (new_value) {
|
|
|
|
add_child(&button_title);
|
|
|
|
} else {
|
|
|
|
remove_child(&button_title);
|
|
|
|
}
|
2016-01-31 09:34:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void SystemStatusView::set_title(const std::string new_value) {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (new_value.empty()) {
|
|
|
|
title.set(default_title);
|
|
|
|
} else {
|
2023-06-27 03:15:58 -05:00
|
|
|
// Limit length of title string to prevent partial characters if too many StatusView icons
|
2023-06-27 21:34:25 -05:00
|
|
|
size_t max_len = (status_icons.parent_rect().left() - title.parent_rect().left()) / 8;
|
2023-06-27 03:15:58 -05:00
|
|
|
title.set(truncate(new_value, max_len));
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
2023-03-27 14:22:35 +02:00
|
|
|
void SystemStatusView::on_converter() {
|
2023-06-23 16:13:39 -07:00
|
|
|
pmem::set_config_converter(!pmem::config_converter());
|
2023-06-11 11:47:13 -07:00
|
|
|
|
2023-06-17 16:15:21 -05:00
|
|
|
// Poke to update tuning
|
|
|
|
// NOTE: Code assumes here that a TX app isn't active, since RX & TX have diff tuning offsets
|
|
|
|
// (and there's only one tuner in the radio so can't update tuner for both).
|
2023-06-23 16:13:39 -07:00
|
|
|
// TODO: Maybe expose the 'enabled_' flag on models.
|
2023-06-11 11:47:13 -07:00
|
|
|
receiver_model.set_target_frequency(receiver_model.target_frequency());
|
2023-06-23 16:13:39 -07:00
|
|
|
refresh();
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2023-03-22 08:46:58 +01:00
|
|
|
|
2018-01-08 03:47:37 +00:00
|
|
|
void SystemStatusView::on_bias_tee() {
|
2023-05-23 21:22:30 -07:00
|
|
|
if (!portapack::get_antenna_bias()) {
|
2023-09-18 14:22:46 -07:00
|
|
|
nav_.display_modal(
|
|
|
|
"Bias voltage",
|
|
|
|
"Enable DC voltage on\nantenna connector?",
|
|
|
|
YESNO,
|
|
|
|
[this](bool v) {
|
|
|
|
if (v) {
|
|
|
|
portapack::set_antenna_bias(true);
|
|
|
|
receiver_model.set_antenna_bias();
|
|
|
|
transmitter_model.set_antenna_bias();
|
|
|
|
refresh();
|
|
|
|
}
|
|
|
|
});
|
2023-05-19 08:16:05 +12:00
|
|
|
} else {
|
|
|
|
portapack::set_antenna_bias(false);
|
|
|
|
receiver_model.set_antenna_bias();
|
|
|
|
transmitter_model.set_antenna_bias();
|
2023-05-23 21:22:30 -07:00
|
|
|
|
|
|
|
// Ensure this is disabled. The models don't actually
|
|
|
|
// update the radio unless they are 'enabled_'.
|
|
|
|
radio::set_antenna_bias(false);
|
2023-05-19 08:16:05 +12:00
|
|
|
refresh();
|
|
|
|
}
|
2018-01-08 03:47:37 +00:00
|
|
|
}
|
|
|
|
|
2016-02-18 21:35:46 -08:00
|
|
|
void SystemStatusView::on_camera() {
|
2024-03-25 02:44:49 -05:00
|
|
|
ensure_directory(screenshots_dir);
|
|
|
|
auto path = next_filename_matching_pattern(screenshots_dir / u"SCR_????.PNG");
|
2023-05-12 11:08:07 -07:00
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
if (path.empty())
|
2023-05-19 08:16:05 +12:00
|
|
|
return;
|
2016-04-20 10:22:05 -07:00
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
PNGWriter png;
|
2023-06-23 16:13:39 -07:00
|
|
|
auto error = png.create(path);
|
|
|
|
if (error)
|
2023-05-19 08:16:05 +12:00
|
|
|
return;
|
2016-02-19 15:34:02 -08:00
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
for (int i = 0; i < screen_height; i++) {
|
|
|
|
std::array<ColorRGB888, screen_width> row;
|
|
|
|
portapack::display.read_pixels({0, i, screen_width, 1}, row);
|
2023-05-19 08:16:05 +12:00
|
|
|
png.write_scanline(row);
|
|
|
|
}
|
2016-02-18 21:35:46 -08:00
|
|
|
}
|
|
|
|
|
2020-09-16 19:27:56 +08:00
|
|
|
void SystemStatusView::on_clk() {
|
2023-06-23 16:13:39 -07:00
|
|
|
pmem::set_clkout_enabled(!pmem::clkout_enabled());
|
|
|
|
portapack::clock_manager.enable_clock_output(pmem::clkout_enabled());
|
2023-05-19 08:16:05 +12:00
|
|
|
refresh();
|
2020-09-16 19:27:56 +08:00
|
|
|
}
|
|
|
|
|
2020-08-01 21:55:09 -04:00
|
|
|
void SystemStatusView::on_title() {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (nav_.is_top())
|
|
|
|
nav_.push<AboutView>();
|
|
|
|
else
|
|
|
|
nav_.pop();
|
2020-08-01 21:55:09 -04:00
|
|
|
}
|
|
|
|
|
2023-10-09 05:04:46 -05:00
|
|
|
void SystemStatusView::rtc_battery_workaround() {
|
|
|
|
if (sd_card::status() != sd_card::Status::Mounted)
|
|
|
|
return;
|
|
|
|
|
|
|
|
uint16_t year;
|
|
|
|
uint8_t month;
|
|
|
|
uint8_t day;
|
|
|
|
FATTimestamp timestamp;
|
|
|
|
rtc::RTC datetime;
|
|
|
|
|
|
|
|
rtcGetTime(&RTCD1, &datetime);
|
|
|
|
|
|
|
|
// if year is 0000, assume RTC battery is dead
|
|
|
|
if (datetime.year() == 0) {
|
|
|
|
// if timestamp file is present, use it's date and add 1 day
|
|
|
|
if (std::filesystem::file_exists(DATE_FILEFLAG)) {
|
|
|
|
timestamp = file_created_date(DATE_FILEFLAG);
|
|
|
|
|
|
|
|
year = (timestamp.FAT_date >> 9) + 1980;
|
|
|
|
month = (timestamp.FAT_date >> 5) & 0xF;
|
|
|
|
day = timestamp.FAT_date & 0x1F;
|
|
|
|
|
2024-01-24 16:37:21 -06:00
|
|
|
// bump to next month
|
|
|
|
if (++day > rtc_time::days_per_month(year, month)) {
|
2023-10-09 05:04:46 -05:00
|
|
|
day = 1;
|
|
|
|
if (++month > 12) {
|
|
|
|
month = 1;
|
|
|
|
++year;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2024-03-25 02:44:49 -05:00
|
|
|
ensure_directory(settings_dir);
|
2023-10-09 05:04:46 -05:00
|
|
|
make_new_file(DATE_FILEFLAG);
|
|
|
|
|
|
|
|
year = 1980;
|
|
|
|
month = 1;
|
|
|
|
day = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// update RTC (keeps ticking while powered on regardless of RTC battery condition)
|
|
|
|
rtc::RTC new_datetime{year, month, day, datetime.hour(), datetime.minute(), datetime.second()};
|
|
|
|
rtcSetTime(&RTCD1, &new_datetime);
|
|
|
|
|
|
|
|
// update file date
|
|
|
|
timestamp.FAT_date = ((year - 1980) << 9) | ((uint16_t)month << 5) | day;
|
|
|
|
timestamp.FAT_time = 0;
|
|
|
|
file_update_date(DATE_FILEFLAG, timestamp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-05 01:03:51 -04:00
|
|
|
/* Information View *****************************************************/
|
|
|
|
|
|
|
|
InformationView::InformationView(
|
2023-05-19 08:16:05 +12:00
|
|
|
NavigationView& nav)
|
|
|
|
: nav_(nav) {
|
|
|
|
add_children({&backdrop,
|
|
|
|
&version,
|
|
|
|
<ime});
|
|
|
|
|
2023-12-31 22:46:26 +01:00
|
|
|
#if GCC_VERSION_MISMATCH
|
2024-05-27 21:02:52 +02:00
|
|
|
version.set_style(Theme::getInstance()->warning_dark);
|
2023-12-31 22:46:26 +01:00
|
|
|
#else
|
2024-05-27 21:02:52 +02:00
|
|
|
version.set_style(Theme::getInstance()->bg_darker);
|
2023-12-31 22:46:26 +01:00
|
|
|
#endif
|
|
|
|
|
2024-01-24 16:37:21 -06:00
|
|
|
if (firmware_checksum_error()) {
|
2024-01-30 13:58:40 +08:00
|
|
|
version.set("FLASH ERR");
|
2024-05-27 21:02:52 +02:00
|
|
|
version.set_style(Theme::getInstance()->error_dark);
|
2024-01-24 16:37:21 -06:00
|
|
|
}
|
|
|
|
|
2024-05-27 21:02:52 +02:00
|
|
|
ltime.set_style(Theme::getInstance()->bg_darker);
|
2023-10-09 05:04:46 -05:00
|
|
|
refresh();
|
2023-05-19 08:16:05 +12:00
|
|
|
set_dirty();
|
2020-08-01 21:55:09 -04:00
|
|
|
}
|
|
|
|
|
2021-05-16 11:41:09 +02:00
|
|
|
void InformationView::refresh() {
|
2023-06-23 16:13:39 -07:00
|
|
|
ltime.set_hide_clock(pmem::hide_clock());
|
2023-05-19 08:16:05 +12:00
|
|
|
ltime.set_seconds_enabled(true);
|
2023-06-23 16:13:39 -07:00
|
|
|
ltime.set_date_enabled(pmem::clock_with_date());
|
2024-01-24 16:37:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
bool InformationView::firmware_checksum_error() {
|
|
|
|
static bool fw_checksum_checked{false};
|
|
|
|
static bool fw_checksum_error{false};
|
|
|
|
|
|
|
|
// only checking firmware checksum once per boot
|
|
|
|
if (!fw_checksum_checked) {
|
|
|
|
fw_checksum_error = (simple_checksum(FLASH_STARTING_ADDRESS, FLASH_ROM_SIZE) != FLASH_EXPECTED_CHECKSUM);
|
|
|
|
}
|
|
|
|
return fw_checksum_error;
|
2021-05-16 11:41:09 +02:00
|
|
|
}
|
|
|
|
|
2015-07-08 08:39:24 -07:00
|
|
|
/* Navigation ************************************************************/
|
|
|
|
|
2016-01-31 09:34:24 +01:00
|
|
|
bool NavigationView::is_top() const {
|
2023-05-19 08:16:05 +12:00
|
|
|
return view_stack.size() == 1;
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
2024-01-30 13:58:40 +08:00
|
|
|
bool NavigationView::is_valid() const {
|
|
|
|
return view_stack.size() != 0; // work around to check if nav is valid, not elegant i know. so TODO
|
|
|
|
}
|
|
|
|
|
2016-01-31 09:34:24 +01:00
|
|
|
View* NavigationView::push_view(std::unique_ptr<View> new_view) {
|
2023-05-19 08:16:05 +12:00
|
|
|
free_view();
|
|
|
|
const auto p = new_view.get();
|
2023-06-01 15:45:55 -07:00
|
|
|
view_stack.emplace_back(ViewState{std::move(new_view), {}});
|
2016-01-31 09:34:24 +01:00
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
update_view();
|
|
|
|
return p;
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
2023-09-27 12:03:02 -07:00
|
|
|
void NavigationView::pop(bool trigger_update) {
|
|
|
|
// Don't pop off the NavView.
|
|
|
|
if (view_stack.size() <= 1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto on_pop = view_stack.back().on_pop;
|
2015-07-08 08:39:24 -07:00
|
|
|
|
2023-09-27 12:03:02 -07:00
|
|
|
free_view();
|
|
|
|
view_stack.pop_back();
|
|
|
|
|
|
|
|
// NB: These are executed _after_ the view has been
|
|
|
|
// destroyed. The old view MUST NOT be referenced in
|
|
|
|
// these callbacks or it will cause crashes.
|
|
|
|
if (trigger_update) update_view();
|
|
|
|
if (on_pop) on_pop();
|
2016-12-06 13:31:38 +01:00
|
|
|
}
|
|
|
|
|
2024-01-12 17:33:41 +01:00
|
|
|
void NavigationView::home(bool trigger_update) {
|
|
|
|
while (view_stack.size() > 1) {
|
|
|
|
pop(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trigger_update) update_view();
|
|
|
|
}
|
|
|
|
|
2016-05-17 10:15:34 -07:00
|
|
|
void NavigationView::display_modal(
|
2023-05-19 08:16:05 +12:00
|
|
|
const std::string& title,
|
|
|
|
const std::string& message) {
|
|
|
|
display_modal(title, message, INFO, nullptr);
|
2016-12-06 13:31:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void NavigationView::display_modal(
|
2023-05-19 08:16:05 +12:00
|
|
|
const std::string& title,
|
|
|
|
const std::string& message,
|
2023-09-27 12:03:02 -07:00
|
|
|
modal_t type,
|
2024-04-29 17:38:27 +02:00
|
|
|
std::function<void(bool)> on_choice,
|
|
|
|
bool compact) {
|
|
|
|
push<ModalMessageView>(title, message, type, on_choice, compact);
|
2023-06-01 15:45:55 -07:00
|
|
|
}
|
|
|
|
|
2016-01-31 09:34:24 +01:00
|
|
|
void NavigationView::free_view() {
|
2023-09-27 12:03:02 -07:00
|
|
|
// The focus_manager holds a raw pointer to the currently focused Widget.
|
|
|
|
// It then tries to call blur() on that instance when the focus is changed.
|
|
|
|
// This causes crashes if focused_widget has been deleted (as is the case
|
|
|
|
// when a view is popped). Calling blur() here resets the focus_manager's
|
|
|
|
// focus_widget pointer so focus can be called safely.
|
|
|
|
this->blur();
|
2023-05-19 08:16:05 +12:00
|
|
|
remove_child(view());
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
2016-01-31 09:34:24 +01:00
|
|
|
void NavigationView::update_view() {
|
2023-09-27 12:03:02 -07:00
|
|
|
const auto& top = view_stack.back();
|
|
|
|
auto top_view = top.view.get();
|
2023-05-19 08:16:05 +12:00
|
|
|
|
2023-09-27 12:03:02 -07:00
|
|
|
add_child(top_view);
|
2024-09-26 14:29:35 +02:00
|
|
|
auto newSize = (is_top()) ? Size{size().width(), size().height() - 16} : size(); // if top(), then there is the info bar at the bottom, so leave space for it
|
|
|
|
top_view->set_parent_rect({{0, 0}, newSize});
|
2023-05-19 08:16:05 +12:00
|
|
|
focus();
|
|
|
|
set_dirty();
|
|
|
|
|
2023-09-27 12:03:02 -07:00
|
|
|
if (on_view_changed)
|
|
|
|
on_view_changed(*top_view);
|
2016-01-31 09:34:24 +01:00
|
|
|
}
|
2015-07-08 08:39:24 -07:00
|
|
|
|
2016-01-31 09:34:24 +01:00
|
|
|
Widget* NavigationView::view() const {
|
2023-05-19 08:16:05 +12:00
|
|
|
return children_.empty() ? nullptr : children_[0];
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void NavigationView::focus() {
|
2023-09-27 12:03:02 -07:00
|
|
|
if (view())
|
2023-05-19 08:16:05 +12:00
|
|
|
view()->focus();
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
2023-06-01 15:45:55 -07:00
|
|
|
bool NavigationView::set_on_pop(std::function<void()> on_pop) {
|
|
|
|
if (view_stack.size() <= 1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
auto& top = view_stack.back();
|
|
|
|
if (top.on_pop)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
top.on_pop = on_pop;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-04-03 16:27:13 +02:00
|
|
|
void NavigationView::handle_autostart() {
|
|
|
|
std::string autostart_app{""};
|
|
|
|
SettingsStore nav_setting{
|
|
|
|
"nav"sv,
|
|
|
|
{{"autostart_app"sv, &autostart_app}}};
|
|
|
|
if (!autostart_app.empty()) {
|
2024-11-02 15:15:33 +01:00
|
|
|
bool started = false;
|
2024-11-02 17:43:42 +08:00
|
|
|
// inner app
|
2024-10-07 03:58:03 +08:00
|
|
|
if (StartAppByName(autostart_app.c_str())) {
|
2024-11-02 15:15:33 +01:00
|
|
|
started = true;
|
2024-11-02 17:43:42 +08:00
|
|
|
}
|
|
|
|
|
2024-11-02 15:15:33 +01:00
|
|
|
if (!started) {
|
|
|
|
// ppma
|
|
|
|
|
|
|
|
std::string appwithpath = "/" + apps_dir.string() + "/" + autostart_app + ".ppma";
|
2024-11-02 17:43:42 +08:00
|
|
|
std::wstring_convert<std::codecvt_utf8_utf16<char16_t>, char16_t> conv;
|
|
|
|
std::filesystem::path pth = conv.from_bytes(appwithpath.c_str());
|
|
|
|
if (ui::ExternalItemsMenuLoader::run_external_app(*this, pth)) {
|
2024-11-02 15:15:33 +01:00
|
|
|
started = true;
|
2024-10-07 03:58:03 +08:00
|
|
|
}
|
2024-11-02 17:43:42 +08:00
|
|
|
|
2024-11-02 15:15:33 +01:00
|
|
|
if (!started) {
|
|
|
|
// ppmp / standalone
|
|
|
|
appwithpath = "/" + apps_dir.string() + "/" + autostart_app + ".ppmp";
|
|
|
|
pth = conv.from_bytes(appwithpath.c_str());
|
|
|
|
if (ui::ExternalItemsMenuLoader::run_standalone_app(*this, pth)) {
|
|
|
|
started = true;
|
|
|
|
}
|
|
|
|
}
|
2024-10-07 03:58:03 +08:00
|
|
|
}
|
2024-11-02 15:15:33 +01:00
|
|
|
if (!started) {
|
2024-10-07 03:58:03 +08:00
|
|
|
display_modal(
|
|
|
|
"Notice", "Autostart failed:\n" +
|
|
|
|
autostart_app +
|
|
|
|
"\nupdate sdcard content\n" +
|
|
|
|
"and check if .ppma exists");
|
|
|
|
}
|
2024-11-02 15:15:33 +01:00
|
|
|
} // autostart end
|
|
|
|
return;
|
2024-04-03 16:27:13 +02:00
|
|
|
}
|
|
|
|
|
2024-01-20 23:35:39 +01:00
|
|
|
/* Helpers **************************************************************/
|
|
|
|
|
|
|
|
static void add_apps(NavigationView& nav, BtnGridView& grid, app_location_t loc) {
|
|
|
|
for (auto& app : NavigationView::appList) {
|
|
|
|
if (app.menuLocation == loc) {
|
|
|
|
grid.add_item({app.displayName, app.iconColor, app.icon,
|
2024-10-06 22:14:27 +02:00
|
|
|
[&nav, &app]() {
|
|
|
|
i2cdev::I2CDevManager::set_autoscan_interval(0); //if i navigate away from any menu, turn off autoscan
|
|
|
|
nav.push_view(std::unique_ptr<View>(app.viewFactory->produce(nav))); }});
|
2024-01-20 23:35:39 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-03-26 11:32:24 -05:00
|
|
|
// clang-format off
|
2024-01-22 03:21:06 +08:00
|
|
|
void addExternalItems(NavigationView& nav, app_location_t location, BtnGridView& grid) {
|
|
|
|
auto externalItems = ExternalItemsMenuLoader::load_external_items(location, nav);
|
|
|
|
if (externalItems.empty()) {
|
2024-03-29 10:05:09 +08:00
|
|
|
grid.insert_item({"Notice!",
|
2024-05-27 21:02:52 +02:00
|
|
|
Theme::getInstance()->error_dark->foreground,
|
2024-03-29 10:05:09 +08:00
|
|
|
nullptr,
|
|
|
|
[&nav]() {
|
|
|
|
nav.display_modal(
|
|
|
|
"Notice",
|
|
|
|
"External app directory empty;\n"
|
|
|
|
"see Mayhem wiki and copy apps\n"
|
|
|
|
"to " + apps_dir.string() + " folder of SD card.");
|
|
|
|
}},
|
|
|
|
pmem::show_gui_return_icon() ? 1 : 0);
|
2024-01-22 03:21:06 +08:00
|
|
|
} else {
|
|
|
|
for (auto const& gridItem : externalItems) {
|
|
|
|
grid.add_item(gridItem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-03-26 11:32:24 -05:00
|
|
|
// clang-format on
|
2024-01-22 03:21:06 +08:00
|
|
|
|
2017-05-25 21:36:30 +01:00
|
|
|
/* ReceiversMenuView *****************************************************/
|
2016-01-31 09:34:24 +01:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
ReceiversMenuView::ReceiversMenuView(NavigationView& nav)
|
|
|
|
: nav_(nav) {}
|
|
|
|
|
|
|
|
void ReceiversMenuView::on_populate() {
|
2023-06-23 16:13:39 -07:00
|
|
|
if (pmem::show_gui_return_icon()) {
|
2024-05-27 21:02:52 +02:00
|
|
|
add_item({"..", Theme::getInstance()->fg_light->foreground, &bitmap_icon_previous, [this]() { nav_.pop(); }});
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2024-01-20 23:35:39 +01:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
add_apps(nav_, *this, RX);
|
2023-10-02 20:19:22 +02:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
addExternalItems(nav_, app_location_t::RX, *this);
|
2016-11-30 07:41:55 +01:00
|
|
|
}
|
|
|
|
|
2017-05-25 21:36:30 +01:00
|
|
|
/* TransmittersMenuView **************************************************/
|
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
TransmittersMenuView::TransmittersMenuView(NavigationView& nav)
|
|
|
|
: nav_(nav) {}
|
|
|
|
|
|
|
|
void TransmittersMenuView::on_populate() {
|
2023-06-23 16:13:39 -07:00
|
|
|
if (pmem::show_gui_return_icon()) {
|
2024-05-27 21:02:52 +02:00
|
|
|
add_items({{"..", Theme::getInstance()->fg_light->foreground, &bitmap_icon_previous, [this]() { nav_.pop(); }}});
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2024-01-20 23:35:39 +01:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
add_apps(nav_, *this, TX);
|
2023-10-02 20:19:22 +02:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
addExternalItems(nav_, app_location_t::TX, *this);
|
2016-01-31 09:34:24 +01:00
|
|
|
}
|
|
|
|
|
2017-05-25 21:36:30 +01:00
|
|
|
/* UtilitiesMenuView *****************************************************/
|
2016-12-06 13:31:38 +01:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
UtilitiesMenuView::UtilitiesMenuView(NavigationView& nav)
|
|
|
|
: nav_(nav) {
|
|
|
|
set_max_rows(2); // allow wider buttons
|
|
|
|
}
|
|
|
|
|
|
|
|
void UtilitiesMenuView::on_populate() {
|
2023-06-23 16:13:39 -07:00
|
|
|
if (pmem::show_gui_return_icon()) {
|
2024-05-27 21:02:52 +02:00
|
|
|
add_items({{"..", Theme::getInstance()->fg_light->foreground, &bitmap_icon_previous, [this]() { nav_.pop(); }}});
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2024-01-20 23:35:39 +01:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
add_apps(nav_, *this, UTILITIES);
|
2023-10-02 20:19:22 +02:00
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
addExternalItems(nav_, app_location_t::UTILITIES, *this);
|
2016-12-06 13:31:38 +01:00
|
|
|
}
|
2017-05-25 21:36:30 +01:00
|
|
|
|
2015-07-08 08:39:24 -07:00
|
|
|
/* SystemMenuView ********************************************************/
|
|
|
|
|
2016-12-26 01:31:38 +01:00
|
|
|
void SystemMenuView::hackrf_mode(NavigationView& nav) {
|
2023-09-18 14:22:46 -07:00
|
|
|
nav.push<ModalMessageView>(
|
|
|
|
"HackRF mode",
|
|
|
|
" This mode enables HackRF\n functionality. To return,\n press the reset button.\n\n Switch to HackRF mode?",
|
|
|
|
YESNO,
|
|
|
|
[this](bool choice) {
|
|
|
|
if (choice) {
|
|
|
|
EventDispatcher::request_stop();
|
|
|
|
}
|
|
|
|
});
|
2016-12-26 01:31:38 +01:00
|
|
|
}
|
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
SystemMenuView::SystemMenuView(NavigationView& nav)
|
|
|
|
: nav_(nav) {
|
2023-05-19 08:16:05 +12:00
|
|
|
set_max_rows(2); // allow wider buttons
|
|
|
|
set_arrow_enabled(false);
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
2024-03-10 14:49:36 -05:00
|
|
|
void SystemMenuView::on_populate() {
|
|
|
|
add_apps(nav_, *this, HOME);
|
2024-11-19 21:02:29 +01:00
|
|
|
addExternalItems(nav_, app_location_t::HOME, *this);
|
2024-05-27 21:02:52 +02:00
|
|
|
add_item({"HackRF", Theme::getInstance()->fg_cyan->foreground, &bitmap_icon_hackrf, [this]() { hackrf_mode(nav_); }});
|
2024-03-10 14:49:36 -05:00
|
|
|
}
|
|
|
|
|
2015-07-08 08:39:24 -07:00
|
|
|
/* SystemView ************************************************************/
|
|
|
|
|
|
|
|
SystemView::SystemView(
|
2023-05-19 08:16:05 +12:00
|
|
|
Context& context,
|
|
|
|
const Rect parent_rect)
|
|
|
|
: View{parent_rect},
|
|
|
|
context_(context) {
|
2024-05-27 21:02:52 +02:00
|
|
|
set_style(Theme::getInstance()->bg_darkest);
|
2023-05-19 08:16:05 +12:00
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
constexpr Dim status_view_height = 16;
|
|
|
|
constexpr Dim info_view_height = 16;
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
add_child(&status_view);
|
2023-09-18 14:22:46 -07:00
|
|
|
status_view.set_parent_rect(
|
|
|
|
{{0, 0},
|
|
|
|
{parent_rect.width(), status_view_height}});
|
2023-05-19 08:16:05 +12:00
|
|
|
status_view.on_back = [this]() {
|
|
|
|
this->navigation_view.pop();
|
|
|
|
};
|
|
|
|
|
|
|
|
add_child(&navigation_view);
|
2023-09-18 14:22:46 -07:00
|
|
|
navigation_view.set_parent_rect(
|
|
|
|
{{0, status_view_height},
|
|
|
|
{parent_rect.width(), static_cast<Dim>(parent_rect.height() - status_view_height)}});
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
add_child(&info_view);
|
2023-09-18 14:22:46 -07:00
|
|
|
info_view.set_parent_rect(
|
|
|
|
{{0, 19 * 16},
|
|
|
|
{parent_rect.width(), info_view_height}});
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
navigation_view.on_view_changed = [this](const View& new_view) {
|
|
|
|
if (!this->navigation_view.is_top()) {
|
|
|
|
remove_child(&info_view);
|
|
|
|
} else {
|
|
|
|
add_child(&info_view);
|
|
|
|
info_view.refresh();
|
2024-10-06 22:14:27 +02:00
|
|
|
i2cdev::I2CDevManager::set_autoscan_interval(3); // turn on autoscan in sysmainv
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
this->status_view.set_back_enabled(!this->navigation_view.is_top());
|
|
|
|
this->status_view.set_title_image_enabled(this->navigation_view.is_top());
|
|
|
|
this->status_view.set_title(new_view.title());
|
|
|
|
this->status_view.set_dirty();
|
|
|
|
};
|
|
|
|
|
|
|
|
navigation_view.push<SystemMenuView>();
|
|
|
|
|
2023-06-23 16:13:39 -07:00
|
|
|
if (pmem::config_splash()) {
|
2023-05-19 08:16:05 +12:00
|
|
|
navigation_view.push<BMPView>();
|
|
|
|
}
|
|
|
|
status_view.set_back_enabled(false);
|
|
|
|
status_view.set_title_image_enabled(true);
|
|
|
|
status_view.set_dirty();
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
Context& SystemView::context() const {
|
2023-05-19 08:16:05 +12:00
|
|
|
return context_;
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
2024-01-07 18:56:09 +01:00
|
|
|
NavigationView* SystemView::get_navigation_view() {
|
|
|
|
return &navigation_view;
|
|
|
|
}
|
2024-04-21 01:34:55 -05:00
|
|
|
SystemStatusView* SystemView::get_status_view() {
|
|
|
|
return &status_view;
|
|
|
|
}
|
2015-07-08 08:39:24 -07:00
|
|
|
|
2023-04-23 16:21:33 +02:00
|
|
|
void SystemView::toggle_overlay() {
|
2024-03-09 15:26:56 +01:00
|
|
|
static uint8_t last_perf_counter_status = shared_memory.request_m4_performance_counter;
|
2023-06-28 12:04:16 -05:00
|
|
|
switch (++overlay_active) {
|
|
|
|
case 1:
|
|
|
|
this->add_child(&this->overlay);
|
|
|
|
this->set_dirty();
|
|
|
|
shared_memory.request_m4_performance_counter = 1;
|
2024-01-07 15:31:08 +01:00
|
|
|
shared_memory.m4_performance_counter = 0;
|
2023-06-28 12:04:16 -05:00
|
|
|
shared_memory.m4_heap_usage = 0;
|
|
|
|
shared_memory.m4_stack_usage = 0;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
this->remove_child(&this->overlay);
|
|
|
|
this->add_child(&this->overlay2);
|
|
|
|
this->set_dirty();
|
2024-01-07 15:31:08 +01:00
|
|
|
shared_memory.request_m4_performance_counter = 2;
|
2023-06-28 12:04:16 -05:00
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
this->remove_child(&this->overlay2);
|
|
|
|
this->set_dirty();
|
2024-03-09 15:26:56 +01:00
|
|
|
shared_memory.request_m4_performance_counter = last_perf_counter_status;
|
2023-06-28 12:04:16 -05:00
|
|
|
overlay_active = 0;
|
|
|
|
break;
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2023-04-23 16:21:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SystemView::paint_overlay() {
|
2023-05-19 08:16:05 +12:00
|
|
|
static bool last_paint_state = false;
|
|
|
|
if (overlay_active) {
|
|
|
|
// paint background only every other second
|
|
|
|
if ((((chTimeNow() >> 10) & 0x01) == 0x01) == last_paint_state)
|
|
|
|
return;
|
2023-04-23 19:52:38 +02:00
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
last_paint_state = !last_paint_state;
|
2023-06-28 12:04:16 -05:00
|
|
|
if (overlay_active == 1)
|
|
|
|
this->overlay.set_dirty();
|
|
|
|
else
|
|
|
|
this->overlay2.set_dirty();
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2023-04-23 16:21:33 +02:00
|
|
|
}
|
|
|
|
|
2024-04-24 05:18:12 +02:00
|
|
|
void SystemView::set_app_fullscreen(bool fullscreen) {
|
|
|
|
auto parent_rect = screen_rect();
|
|
|
|
Dim status_view_height = (fullscreen) ? 0 : 16;
|
|
|
|
status_view.hidden(fullscreen);
|
|
|
|
navigation_view.set_parent_rect(
|
|
|
|
{{0, status_view_height},
|
|
|
|
{parent_rect.width(), static_cast<Dim>(parent_rect.height() - status_view_height)}});
|
|
|
|
}
|
|
|
|
|
2016-02-05 17:40:14 +01:00
|
|
|
/* ***********************************************************************/
|
|
|
|
|
|
|
|
void BMPView::focus() {
|
2023-05-19 08:16:05 +12:00
|
|
|
button_done.focus();
|
2016-02-05 17:40:14 +01:00
|
|
|
}
|
|
|
|
|
2024-09-24 02:57:15 +08:00
|
|
|
BMPView::BMPView(NavigationView& nav)
|
|
|
|
: nav_(nav) {
|
2023-05-19 08:16:05 +12:00
|
|
|
add_children({&button_done});
|
|
|
|
|
2024-09-24 02:57:15 +08:00
|
|
|
button_done.on_select = [this](Button&) {
|
|
|
|
handle_pop();
|
2023-05-19 08:16:05 +12:00
|
|
|
};
|
2016-02-05 17:40:14 +01:00
|
|
|
}
|
|
|
|
|
2016-12-09 18:21:47 +01:00
|
|
|
void BMPView::paint(Painter&) {
|
2023-07-05 15:45:43 -05:00
|
|
|
if (!portapack::display.drawBMP2({0, 0}, splash_dot_bmp))
|
2024-11-09 15:12:39 +01:00
|
|
|
portapack::display.drawBMP({0, 16}, splash_bmp, (const uint8_t[]){0x29, 0x18, 0x16});
|
2016-02-05 17:40:14 +01:00
|
|
|
}
|
|
|
|
|
2024-09-24 02:57:15 +08:00
|
|
|
bool BMPView::on_touch(const TouchEvent event) {
|
|
|
|
/* the event thing were resolved by HTotoo, talked here https://discord.com/channels/719669764804444213/956561375155589192/1287756910950486027
|
|
|
|
* the touch screen policy can be better, talked here https://discord.com/channels/719669764804444213/956561375155589192/1198926225897443328
|
2024-10-16 17:35:49 +08:00
|
|
|
* this workaround discussed here: https://discord.com/channels/719669764804444213/1170738202924044338/1295630640158478418
|
2024-09-24 02:57:15 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
if (!nav_.is_valid()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (event.type) {
|
|
|
|
case TouchEvent::Type::Start:
|
|
|
|
handle_pop();
|
2024-10-16 17:35:49 +08:00
|
|
|
return false;
|
2024-09-24 02:57:15 +08:00
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BMPView::handle_pop() {
|
|
|
|
if (nav_.is_valid()) {
|
|
|
|
nav_.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-08 08:39:24 -07:00
|
|
|
/* NotImplementedView ****************************************************/
|
|
|
|
|
2020-05-11 11:07:45 +02:00
|
|
|
/*NotImplementedView::NotImplementedView(NavigationView& nav) {
|
2023-05-19 08:16:05 +12:00
|
|
|
button_done.on_select = [&nav](Button&){
|
|
|
|
nav.pop();
|
|
|
|
};
|
2015-07-08 08:39:24 -07:00
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
add_children({
|
|
|
|
&text_title,
|
|
|
|
&button_done,
|
|
|
|
});
|
2015-07-08 08:39:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void NotImplementedView::focus() {
|
2023-05-19 08:16:05 +12:00
|
|
|
button_done.focus();
|
2020-05-11 11:07:45 +02:00
|
|
|
}*/
|
2015-07-08 08:39:24 -07:00
|
|
|
|
2016-05-12 21:58:54 -07:00
|
|
|
/* ModalMessageView ******************************************************/
|
2016-12-09 18:21:47 +01:00
|
|
|
|
2016-05-17 10:15:34 -07:00
|
|
|
ModalMessageView::ModalMessageView(
|
2023-05-19 08:16:05 +12:00
|
|
|
NavigationView& nav,
|
|
|
|
const std::string& title,
|
|
|
|
const std::string& message,
|
2023-09-27 12:03:02 -07:00
|
|
|
modal_t type,
|
2024-04-29 17:38:27 +02:00
|
|
|
std::function<void(bool)> on_choice,
|
|
|
|
bool compact)
|
2023-05-19 08:16:05 +12:00
|
|
|
: title_{title},
|
|
|
|
message_{message},
|
|
|
|
type_{type},
|
2024-04-29 17:38:27 +02:00
|
|
|
on_choice_{on_choice},
|
|
|
|
compact{compact} {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (type == INFO) {
|
|
|
|
add_child(&button_ok);
|
2023-05-19 13:39:35 -07:00
|
|
|
button_ok.on_select = [this, &nav](Button&) {
|
2023-09-27 12:03:02 -07:00
|
|
|
if (on_choice_) on_choice_(true);
|
|
|
|
nav.pop();
|
2023-05-19 08:16:05 +12:00
|
|
|
};
|
2023-09-27 12:03:02 -07:00
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
} else if (type == YESNO) {
|
|
|
|
add_children({&button_yes,
|
|
|
|
&button_no});
|
|
|
|
|
|
|
|
button_yes.on_select = [this, &nav](Button&) {
|
|
|
|
if (on_choice_) on_choice_(true);
|
|
|
|
nav.pop();
|
|
|
|
};
|
|
|
|
button_no.on_select = [this, &nav](Button&) {
|
|
|
|
if (on_choice_) on_choice_(false);
|
|
|
|
nav.pop();
|
|
|
|
};
|
|
|
|
|
|
|
|
} else { // ABORT
|
|
|
|
add_child(&button_ok);
|
|
|
|
|
|
|
|
button_ok.on_select = [this, &nav](Button&) {
|
|
|
|
if (on_choice_) on_choice_(true);
|
2023-09-27 12:03:02 -07:00
|
|
|
nav.pop(false); // Pop the modal.
|
|
|
|
nav.pop(); // Pop the underlying view.
|
2023-05-19 08:16:05 +12:00
|
|
|
};
|
|
|
|
}
|
2016-05-12 21:58:54 -07:00
|
|
|
}
|
|
|
|
|
2017-02-07 19:54:18 +00:00
|
|
|
void ModalMessageView::paint(Painter& painter) {
|
2024-09-28 20:50:37 +08:00
|
|
|
if (!compact) portapack::display.drawBMP({100, 48}, modal_warning_bmp, (const uint8_t[]){0, 0, 0});
|
2023-05-19 08:16:05 +12:00
|
|
|
|
2023-09-27 12:03:02 -07:00
|
|
|
// Break lines.
|
|
|
|
auto lines = split_string(message_, '\n');
|
|
|
|
for (size_t i = 0; i < lines.size(); ++i) {
|
2023-05-19 08:16:05 +12:00
|
|
|
painter.draw_string(
|
2024-04-29 17:38:27 +02:00
|
|
|
{1 * 8, (Coord)(((compact) ? 8 * 3 : 120) + (i * 16))},
|
2023-05-19 08:16:05 +12:00
|
|
|
style(),
|
2023-09-27 12:03:02 -07:00
|
|
|
lines[i]);
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
2016-08-23 08:45:33 +02:00
|
|
|
}
|
|
|
|
|
2016-05-12 21:58:54 -07:00
|
|
|
void ModalMessageView::focus() {
|
2023-09-27 12:03:02 -07:00
|
|
|
if ((type_ == YESNO)) {
|
2023-05-19 08:16:05 +12:00
|
|
|
button_yes.focus();
|
|
|
|
} else {
|
|
|
|
button_ok.focus();
|
|
|
|
}
|
2016-05-12 21:58:54 -07:00
|
|
|
}
|
|
|
|
|
2015-07-08 08:39:24 -07:00
|
|
|
} /* namespace ui */
|