/* * Copyright (C) 2015 Jared Boone, ShareBrained Technology, Inc. * Copyright (C) 2016 Furrtek * Copyright (C) 2024 u-foka * Copyleft (ɔ) 2024 zxkmm under GPL license * * 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" #include "bmp_modal_warning.hpp" #include "bmp_splash.hpp" #include "event_m0.hpp" #include "portapack_persistent_memory.hpp" #include "portapack_shared_memory.hpp" #include "portapack.hpp" #include "ui_about_simple.hpp" #include "ui_adsb_rx.hpp" #include "ui_aprs_rx.hpp" #include "ui_aprs_tx.hpp" #include "ui_bht_tx.hpp" #include "ui_btle_rx.hpp" #include "ui_debug.hpp" #include "ui_encoders.hpp" #include "ui_fileman.hpp" #include "ui_flash_utility.hpp" #include "ui_font_fixed_8x16.hpp" #include "ui_freqman.hpp" #include "ui_iq_trim.hpp" #include "ui_level.hpp" #include "ui_looking_glass_app.hpp" #include "ui_mictx.hpp" #include "ui_playlist.hpp" #include "ui_pocsag_tx.hpp" #include "ui_rds.hpp" #include "ui_recon.hpp" #include "ui_scanner.hpp" #include "ui_sd_over_usb.hpp" #include "ui_sd_wipe.hpp" #include "ui_search.hpp" #include "ui_settings.hpp" #include "ui_siggen.hpp" #include "ui_sonde.hpp" #include "ui_ss_viewer.hpp" // #include "ui_test.hpp" #include "ui_text_editor.hpp" #include "ui_touchtunes.hpp" #include "ui_view_wav.hpp" #include "ui_weatherstation.hpp" #include "ui_subghzd.hpp" #include "ui_whipcalc.hpp" #include "ui_battinfo.hpp" #include "ui_external_items_menu_loader.hpp" #include "ais_app.hpp" #include "analog_audio_app.hpp" // #include "ble_comm_app.hpp" #include "ble_rx_app.hpp" #include "ble_tx_app.hpp" #include "capture_app.hpp" #include "ert_app.hpp" #include "pocsag_app.hpp" #include "soundboard_app.hpp" #include "core_control.hpp" #include "file.hpp" #include "file_reader.hpp" #include "png_writer.hpp" #include "file_path.hpp" #include "ff.h" #include #include using portapack::receiver_model; using portapack::transmitter_model; namespace pmem = portapack::persistent_memory; namespace ui { 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()}, {nullptr, "Transmit", HOME, Color::cyan(), &bitmap_icon_transmit, new ViewFactory()}, {"capture", "Capture", HOME, Color::red(), &bitmap_icon_capture, new ViewFactory()}, {"replay", "Replay", HOME, Color::green(), &bitmap_icon_replay, new ViewFactory()}, {"scanner", "Scanner", HOME, Color::green(), &bitmap_icon_scanner, new ViewFactory()}, {"microphone", "Microphone", HOME, Color::green(), &bitmap_icon_microphone, new ViewFactory()}, {"lookingglass", "Looking Glass", HOME, Color::green(), &bitmap_icon_looking, new ViewFactory()}, {nullptr, "Utilities", HOME, Color::cyan(), &bitmap_icon_utilities, new ViewFactory()}, {nullptr, "Settings", HOME, Color::cyan(), &bitmap_icon_setup, new ViewFactory()}, {nullptr, "Debug", HOME, Color::light_grey(), &bitmap_icon_debug, new ViewFactory()}, /* RX ********************************************************************/ {"adsbrx", "ADS-B", RX, Color::green(), &bitmap_icon_adsb, new ViewFactory()}, {"ais", "AIS Boats", RX, Color::green(), &bitmap_icon_ais, new ViewFactory()}, {"aprsrx", "APRS", RX, Color::green(), &bitmap_icon_aprs, new ViewFactory()}, {"audio", "Audio", RX, Color::green(), &bitmap_icon_speaker, new ViewFactory()}, //{"blecomm", "BLE Comm", RX, ui::Color::orange(), &bitmap_icon_btle, new ViewFactory()}, {"blerx", "BLE Rx", RX, Color::green(), &bitmap_icon_btle, new ViewFactory()}, {"ert", "ERT Meter", RX, Color::green(), &bitmap_icon_ert, new ViewFactory()}, {"level", "Level", RX, Color::green(), &bitmap_icon_options_radio, new ViewFactory()}, {"pocsag", "POCSAG", RX, Color::green(), &bitmap_icon_pocsag, new ViewFactory()}, {"radiosonde", "Radiosnde", RX, Color::green(), &bitmap_icon_sonde, new ViewFactory()}, {"recon", "Recon", RX, Color::green(), &bitmap_icon_scanner, new ViewFactory()}, {"search", "Search", RX, Color::yellow(), &bitmap_icon_search, new ViewFactory()}, {"subghzd", "SubGhzD", RX, Color::yellow(), &bitmap_icon_remote, new ViewFactory()}, {"weather", "Weather", RX, Color::green(), &bitmap_icon_thermometer, new ViewFactory()}, //{"fskrx", "FSK RX", RX, Color::yellow(), &bitmap_icon_remote, new ViewFactory()}, //for JT //{"dmr", "DMR", RX, Color::dark_grey(), &bitmap_icon_dmr, new ViewFactory()}, //{"sigfox", "SIGFOX", RX, Color::dark_grey(), &bitmap_icon_fox, new ViewFactory()}, //{"lora", "LoRa", RX, Color::dark_grey(), &bitmap_icon_lora, new ViewFactory()}, //{"sstv", "SSTV", RX, Color::dark_grey(), &bitmap_icon_sstv, new ViewFactory()}, //{"tetra", "TETRA", RX, Color::dark_grey(), &bitmap_icon_tetra, new ViewFactory()}, /* TX ********************************************************************/ //{"adsbtx", "ADS-B TX", TX, ui::Color::green(), &bitmap_icon_adsb, new ViewFactory()}, {"aprstx", "APRS TX", TX, ui::Color::green(), &bitmap_icon_aprs, new ViewFactory()}, {"bht", "BHT Xy/EP", TX, ui::Color::green(), &bitmap_icon_bht, new ViewFactory()}, {"bletx", "BLE Tx", TX, ui::Color::green(), &bitmap_icon_btle, new ViewFactory()}, {"ooktx", "OOK", TX, ui::Color::yellow(), &bitmap_icon_remote, new ViewFactory()}, {"pocsagtx", "POCSAG TX", TX, ui::Color::green(), &bitmap_icon_pocsag, new ViewFactory()}, {"rdstx", "RDS", TX, ui::Color::green(), &bitmap_icon_rds, new ViewFactory()}, {"soundbrd", "Soundbrd", TX, ui::Color::green(), &bitmap_icon_soundboard, new ViewFactory()}, {"touchtune", "TouchTune", TX, ui::Color::green(), &bitmap_icon_touchtunes, new ViewFactory()}, /* UTILITIES *************************************************************/ {"antennalength", "Antenna Length", UTILITIES, Color::green(), &bitmap_icon_tools_antenna, new ViewFactory()}, {"filemanager", "File Manager", UTILITIES, Color::green(), &bitmap_icon_dir, new ViewFactory()}, {"freqman", "Freq. Manager", UTILITIES, Color::green(), &bitmap_icon_freqman, new ViewFactory()}, {"notepad", "Notepad", UTILITIES, Color::dark_cyan(), &bitmap_icon_notepad, new ViewFactory()}, {"iqtrim", "IQ Trim", UTILITIES, Color::orange(), &bitmap_icon_trim, new ViewFactory()}, {nullptr, "SD Over USB", UTILITIES, Color::yellow(), &bitmap_icon_hackrf, new ViewFactory()}, {"signalgen", "Signal Gen", UTILITIES, Color::green(), &bitmap_icon_cwgen, new ViewFactory()}, //{"testapp", "Test App", UTILITIES, Color::dark_grey(), nullptr, new ViewFactory()}, {"wavview", "Wav View", UTILITIES, Color::yellow(), &bitmap_icon_soundboard, new ViewFactory()}, // Dangerous apps. {nullptr, "Flash Utility", UTILITIES, Color::red(), &bitmap_icon_peripherals_details, new ViewFactory()}, {nullptr, "Wipe SD card", UTILITIES, Color::red(), &bitmap_icon_tools_wipesd, new ViewFactory()}, }; const NavigationView::AppMap NavigationView::appMap = generate_app_map(NavigationView::appList); bool NavigationView::StartAppByName(const char* name) { home(false); auto it = appMap.find(name); if (it != appMap.end()) { push_view(std::unique_ptr(it->second.viewFactory->produce(*this))); return true; } return false; } /* 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&) { } /* SystemStatusView ******************************************************/ SystemStatusView::SystemStatusView( NavigationView& nav) : nav_(nav) { add_children({ &backdrop, &button_back, &title, &button_title, &status_icons, }); rtc_battery_workaround(); ui::load_blacklist(); if (pmem::should_use_sdcard_for_pmem()) { pmem::load_persistent_settings_from_file(); } // configure CLKOUT per pmem setting portapack::clock_manager.enable_clock_output(pmem::clkout_enabled()); // force apply of selected sdcard speed override at UI startup pmem::set_config_sdcard_high_speed_io(pmem::config_sdcard_high_speed_io(), false); button_back.id = -1; // Special ID used by FocusManager title.set_style(Theme::getInstance()->bg_dark); button_back.on_select = [this](ImageButton&) { if (pmem::should_use_sdcard_for_pmem()) { pmem::save_persistent_settings_to_file(); } 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(); }; toggle_speaker.on_change = [this](bool v) { pmem::set_config_speaker_disable(v); audio::output::update_audio_mute(); refresh(); }; toggle_mute.on_change = [this](bool v) { pmem::set_config_audio_mute(v); audio::output::update_audio_mute(); refresh(); }; toggle_stealth.on_change = [this, &nav](bool v) { pmem::set_stealth_mode(v); if (nav.is_valid() && v) { nav.display_modal( "Stealth", "You just enabled stealth mode.\n" "When you transmit,\n" "screen will turn off;\n"); } refresh(); }; battery_icon.on_select = [this]() { on_battery_details(); }; battery_text.on_select = [this]() { on_battery_details(); }; button_bias_tee.on_select = [this](ImageButton&) { this->on_bias_tee(); }; button_fake_brightness.on_select = [this](ImageButton&) { set_dirty(); pmem::toggle_fake_brightness_level(); refresh(); if (nullptr != parent()) { parent()->set_dirty(); // The parent of NavigationView shal be the SystemView } }; 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(); }; // 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()); audio::output::stop(); audio::output::update_audio_mute(); refresh(); } // 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(); nav_.set_on_pop([this]() { batt_info_up = false; }); } void SystemStatusView::on_battery_data(const BatteryStateMessage* msg) { if (!batt_was_inited) { batt_was_inited = true; refresh(); } if (!pmem::ui_hide_numeric_battery()) { battery_text.set_battery(msg->valid_mask, msg->percent, msg->on_charger); } if (!pmem::ui_hide_battery_icon()) { battery_icon.set_battery(msg->valid_mask, msg->percent, msg->on_charger); }; } void SystemStatusView::refresh() { // NB: Order of insertion is the display order Left->Right. // TODO: Might be better to support hide and only add once. status_icons.clear(); if (!pmem::ui_hide_camera()) status_icons.add(&button_camera); if (!pmem::ui_hide_sleep()) status_icons.add(&button_sleep); if (!pmem::ui_hide_stealth()) status_icons.add(&toggle_stealth); 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); if (!pmem::ui_hide_mute()) status_icons.add(&toggle_mute); // Display "Disable speaker" icon only if AK4951 Codec which has separate speaker/headphone control if (audio::speaker_disable_supported() && !pmem::ui_hide_speaker()) status_icons.add(&toggle_speaker); if (!pmem::ui_hide_fake_brightness() && !pmem::config_lcd_inverted_mode()) status_icons.add(&button_fake_brightness); if (battery::BatteryManagement::isDetected()) { batt_was_inited = true; if (!pmem::ui_hide_battery_icon()) { status_icons.add(&battery_icon); }; if (!pmem::ui_hide_numeric_battery()) { status_icons.add(&battery_text); } } if (!pmem::ui_hide_sd_card()) status_icons.add(&sd_card_status_view); 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); button_clock_status.set_foreground(pmem::clkout_enabled() ? *Theme::getInstance()->status_active : Theme::getInstance()->fg_light->foreground); // Antenna DC Bias if (portapack::get_antenna_bias()) { button_bias_tee.set_bitmap(&bitmap_icon_biast_on); button_bias_tee.set_foreground(Theme::getInstance()->warning_dark->foreground); } else { button_bias_tee.set_bitmap(&bitmap_icon_biast_off); button_bias_tee.set_foreground(Theme::getInstance()->fg_light->foreground); } // Converter button_converter.set_bitmap(pmem::config_updown_converter() ? &bitmap_icon_downconvert : &bitmap_icon_upconvert); button_converter.set_foreground(pmem::config_converter() ? Theme::getInstance()->fg_red->foreground : Theme::getInstance()->fg_light->foreground); // Fake Brightness button_fake_brightness.set_foreground((pmem::apply_fake_brightness() & (!pmem::config_lcd_inverted_mode())) ? *Theme::getInstance()->status_active : Theme::getInstance()->fg_light->foreground); set_dirty(); } void SystemStatusView::set_back_enabled(bool new_value) { if (new_value) { add_child(&button_back); } else { remove_child(&button_back); } } void SystemStatusView::set_back_hidden(bool new_value) { button_back.hidden(new_value); } void SystemStatusView::set_title_image_enabled(bool new_value) { if (new_value) { add_child(&button_title); } else { remove_child(&button_title); } } void SystemStatusView::set_title(const std::string new_value) { if (new_value.empty()) { title.set(default_title); } else { // Limit length of title string to prevent partial characters if too many StatusView icons size_t max_len = (status_icons.parent_rect().left() - title.parent_rect().left()) / 8; title.set(truncate(new_value, max_len)); } } void SystemStatusView::on_converter() { pmem::set_config_converter(!pmem::config_converter()); // 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). // TODO: Maybe expose the 'enabled_' flag on models. receiver_model.set_target_frequency(receiver_model.target_frequency()); refresh(); } void SystemStatusView::on_bias_tee() { if (!portapack::get_antenna_bias()) { 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(); } }); } else { portapack::set_antenna_bias(false); receiver_model.set_antenna_bias(); transmitter_model.set_antenna_bias(); // Ensure this is disabled. The models don't actually // update the radio unless they are 'enabled_'. radio::set_antenna_bias(false); refresh(); } } void SystemStatusView::on_camera() { ensure_directory(screenshots_dir); auto path = next_filename_matching_pattern(screenshots_dir / u"SCR_????.PNG"); if (path.empty()) return; PNGWriter png; auto error = png.create(path); if (error) return; for (int i = 0; i < screen_height; i++) { std::array row; portapack::display.read_pixels({0, i, screen_width, 1}, row); png.write_scanline(row); } } void SystemStatusView::on_clk() { pmem::set_clkout_enabled(!pmem::clkout_enabled()); portapack::clock_manager.enable_clock_output(pmem::clkout_enabled()); refresh(); } void SystemStatusView::on_title() { if (nav_.is_top()) nav_.push(); else nav_.pop(); } 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; // bump to next month if (++day > rtc_time::days_per_month(year, month)) { day = 1; if (++month > 12) { month = 1; ++year; } } } else { ensure_directory(settings_dir); 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); } } /* Information View *****************************************************/ InformationView::InformationView( NavigationView& nav) : nav_(nav) { add_children({&backdrop, &version, <ime}); #if GCC_VERSION_MISMATCH version.set_style(Theme::getInstance()->warning_dark); #else version.set_style(Theme::getInstance()->bg_darker); #endif if (firmware_checksum_error()) { version.set("FLASH ERR"); version.set_style(Theme::getInstance()->error_dark); } ltime.set_style(Theme::getInstance()->bg_darker); refresh(); set_dirty(); } void InformationView::refresh() { ltime.set_hide_clock(pmem::hide_clock()); ltime.set_seconds_enabled(true); ltime.set_date_enabled(pmem::clock_with_date()); } 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; } /* Navigation ************************************************************/ bool NavigationView::is_top() const { return view_stack.size() == 1; } bool NavigationView::is_valid() const { return view_stack.size() != 0; // work around to check if nav is valid, not elegant i know. so TODO } View* NavigationView::push_view(std::unique_ptr new_view) { free_view(); const auto p = new_view.get(); view_stack.emplace_back(ViewState{std::move(new_view), {}}); update_view(); return p; } 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; 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(); } void NavigationView::home(bool trigger_update) { while (view_stack.size() > 1) { pop(false); } if (trigger_update) update_view(); } void NavigationView::display_modal( const std::string& title, const std::string& message) { display_modal(title, message, INFO, nullptr); } void NavigationView::display_modal( const std::string& title, const std::string& message, modal_t type, std::function on_choice, bool compact) { push(title, message, type, on_choice, compact); } void NavigationView::free_view() { // 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(); remove_child(view()); } void NavigationView::update_view() { const auto& top = view_stack.back(); auto top_view = top.view.get(); add_child(top_view); 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}); focus(); set_dirty(); if (on_view_changed) on_view_changed(*top_view); } Widget* NavigationView::view() const { return children_.empty() ? nullptr : children_[0]; } void NavigationView::focus() { if (view()) view()->focus(); } bool NavigationView::set_on_pop(std::function 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; } void NavigationView::handle_autostart() { std::string autostart_app{""}; SettingsStore nav_setting{ "nav"sv, {{"autostart_app"sv, &autostart_app}}}; if (!autostart_app.empty()) { bool started = false; // inner app if (StartAppByName(autostart_app.c_str())) { started = true; } if (!started) { // ppma std::string appwithpath = "/" + apps_dir.string() + "/" + autostart_app + ".ppma"; std::wstring_convert, char16_t> conv; std::filesystem::path pth = conv.from_bytes(appwithpath.c_str()); if (ui::ExternalItemsMenuLoader::run_external_app(*this, pth)) { started = true; } 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; } } } if (!started) { display_modal( "Notice", "Autostart failed:\n" + autostart_app + "\nupdate sdcard content\n" + "and check if .ppma exists"); } } // autostart end return; } /* Helpers **************************************************************/ 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, [&nav, &app]() { i2cdev::I2CDevManager::set_autoscan_interval(0); //if i navigate away from any menu, turn off autoscan nav.push_view(std::unique_ptr(app.viewFactory->produce(nav))); }}, true); } }; grid.update_items(); } // clang-format off void add_external_items(NavigationView& nav, app_location_t location, BtnGridView& grid, uint8_t error_tile_pos) { auto externalItems = ExternalItemsMenuLoader::load_external_items(location, nav); if (externalItems.empty()) { grid.insert_item({"ExtAppErr", Theme::getInstance()->error_dark->foreground, 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."); }}, error_tile_pos); } else { std::sort(externalItems.begin(), externalItems.end(), [](const auto &a, const auto &b) { return a.desired_position < b.desired_position; }); for (auto const& gridItem : externalItems) { if (gridItem.desired_position < 0) { grid.add_item(gridItem, true); } else { grid.insert_item(gridItem, gridItem.desired_position, true); } } grid.update_items(); } } // clang-format on bool verify_sdcard_format() { FATFS* fs = &sd_card::fs; return (fs->fs_type == FS_FAT32 || fs->fs_type == FS_EXFAT) || !(sd_card::status() == sd_card::Status::Mounted); /* ^ to satisfy those users that not use an sd*/ } /* ReceiversMenuView *****************************************************/ ReceiversMenuView::ReceiversMenuView(NavigationView& nav) : nav_(nav) {} void ReceiversMenuView::on_populate() { bool return_icon = pmem::show_gui_return_icon(); if (return_icon) { add_item({"..", Theme::getInstance()->fg_light->foreground, &bitmap_icon_previous, [this]() { nav_.pop(); }}); } add_apps(nav_, *this, RX); add_external_items(nav_, app_location_t::RX, *this, return_icon ? 1 : 0); } /* TransmittersMenuView **************************************************/ TransmittersMenuView::TransmittersMenuView(NavigationView& nav) : nav_(nav) {} void TransmittersMenuView::on_populate() { bool return_icon = pmem::show_gui_return_icon(); if (return_icon) { add_items({{"..", Theme::getInstance()->fg_light->foreground, &bitmap_icon_previous, [this]() { nav_.pop(); }}}); } add_apps(nav_, *this, TX); add_external_items(nav_, app_location_t::TX, *this, return_icon ? 1 : 0); } /* UtilitiesMenuView *****************************************************/ UtilitiesMenuView::UtilitiesMenuView(NavigationView& nav) : nav_(nav) { set_max_rows(2); // allow wider buttons } void UtilitiesMenuView::on_populate() { bool return_icon = pmem::show_gui_return_icon(); if (return_icon) { add_items({{"..", Theme::getInstance()->fg_light->foreground, &bitmap_icon_previous, [this]() { nav_.pop(); }}}); } add_apps(nav_, *this, UTILITIES); add_external_items(nav_, app_location_t::UTILITIES, *this, return_icon ? 1 : 0); } /* SystemMenuView ********************************************************/ void SystemMenuView::hackrf_mode(NavigationView& nav) { nav.push( "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(); } }); } SystemMenuView::SystemMenuView(NavigationView& nav) : nav_(nav) { set_max_rows(2); // allow wider buttons set_arrow_enabled(false); } void SystemMenuView::on_populate() { if (!verify_sdcard_format()) { add_item({"SDCard Error", Theme::getInstance()->error_dark->foreground, nullptr, [this]() { nav_.display_modal("Error", "SD Card is not exFAT/FAT32,\nformat to exFAT or FAT32 on PC"); }}); } add_apps(nav_, *this, HOME); add_external_items(nav_, app_location_t::HOME, *this, 0); add_item({"HackRF", Theme::getInstance()->fg_cyan->foreground, &bitmap_icon_hackrf, [this]() { hackrf_mode(nav_); }}); } /* SystemView ************************************************************/ SystemView::SystemView( Context& context, const Rect parent_rect) : View{parent_rect}, context_(context) { set_style(Theme::getInstance()->bg_darkest); constexpr Dim status_view_height = 16; constexpr Dim info_view_height = 16; add_child(&status_view); status_view.set_parent_rect( {{0, 0}, {parent_rect.width(), status_view_height}}); status_view.on_back = [this]() { this->navigation_view.pop(); }; add_child(&navigation_view); navigation_view.set_parent_rect( {{0, status_view_height}, {parent_rect.width(), static_cast(parent_rect.height() - status_view_height)}}); add_child(&info_view); info_view.set_parent_rect( {{0, 19 * 16}, {parent_rect.width(), info_view_height}}); 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(); i2cdev::I2CDevManager::set_autoscan_interval(3); // turn on autoscan in sysmainv } 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(); if (pmem::config_splash()) { navigation_view.push(); } status_view.set_back_enabled(false); status_view.set_title_image_enabled(true); status_view.set_dirty(); } Context& SystemView::context() const { return context_; } NavigationView* SystemView::get_navigation_view() { return &navigation_view; } SystemStatusView* SystemView::get_status_view() { return &status_view; } void SystemView::toggle_overlay() { static uint8_t last_perf_counter_status = shared_memory.request_m4_performance_counter; switch (++overlay_active) { case 1: this->add_child(&this->overlay); this->set_dirty(); shared_memory.request_m4_performance_counter = 1; shared_memory.m4_performance_counter = 0; 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(); shared_memory.request_m4_performance_counter = 2; break; case 3: this->remove_child(&this->overlay2); this->set_dirty(); shared_memory.request_m4_performance_counter = last_perf_counter_status; overlay_active = 0; break; } } void SystemView::paint_overlay() { static bool last_paint_state = false; if (overlay_active) { // paint background only every other second if ((((chTimeNow() >> 10) & 0x01) == 0x01) == last_paint_state) return; last_paint_state = !last_paint_state; if (overlay_active == 1) this->overlay.set_dirty(); else this->overlay2.set_dirty(); } } 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(parent_rect.height() - status_view_height)}}); } /* ***********************************************************************/ void SplashScreenView::focus() { button_done.focus(); } SplashScreenView::SplashScreenView(NavigationView& nav) : nav_(nav) { add_children({&button_done}); button_done.on_select = [this](Button&) { handle_pop(); }; } void SplashScreenView::paint(Painter&) { if (!portapack::display.draw_bmp_from_sdcard_file({0, 0}, splash_dot_bmp)) // ^ try draw bmp file from sdcard at (0,0), and the (0,0) already bypassed the status bar, so actual pos is (0, STATUS_BAR_HEIGHT) portapack::display.draw_bmp_from_bmp_hex_arr({(240 - 230) / 2, (320 - 50) / 2 - 10}, splash_bmp, (const uint8_t[]){0x29, 0x18, 0x16}); // ^ draw BMP HEX arr in firmware, note that the BMP HEX arr only cover the image part (instead of fill the screen with background, this position is located it in the center) } bool SplashScreenView::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 * this workaround discussed here: https://discord.com/channels/719669764804444213/1170738202924044338/1295630640158478418 */ if (!nav_.is_valid()) { return false; } switch (event.type) { case TouchEvent::Type::Start: handle_pop(); return false; default: break; } return false; } void SplashScreenView::handle_pop() { if (nav_.is_valid()) { nav_.pop(); } } /* NotImplementedView ****************************************************/ /*NotImplementedView::NotImplementedView(NavigationView& nav) { button_done.on_select = [&nav](Button&){ nav.pop(); }; add_children({ &text_title, &button_done, }); } void NotImplementedView::focus() { button_done.focus(); }*/ /* ModalMessageView ******************************************************/ ModalMessageView::ModalMessageView( NavigationView& nav, const std::string& title, const std::string& message, modal_t type, std::function on_choice, bool compact) : title_{title}, message_{message}, type_{type}, on_choice_{on_choice}, compact{compact} { if (type == INFO) { add_child(&button_ok); button_ok.on_select = [this, &nav](Button&) { if (on_choice_) on_choice_(true); nav.pop(); }; } 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); nav.pop(false); // Pop the modal. nav.pop(); // Pop the underlying view. }; } } void ModalMessageView::paint(Painter& painter) { if (!compact) portapack::display.draw_bmp_from_bmp_hex_arr({100, 48}, modal_warning_bmp, (const uint8_t[]){0, 0, 0}); // Break lines. auto lines = split_string(message_, '\n'); for (size_t i = 0; i < lines.size(); ++i) { painter.draw_string( {1 * 8, (Coord)(((compact) ? 8 * 3 : 120) + (i * 16))}, style(), lines[i]); } } void ModalMessageView::focus() { if ((type_ == YESNO)) { button_yes.focus(); } else { button_ok.focus(); } } } /* namespace ui */