portapack-mayhem/firmware/application/ui_navigation.cpp

498 lines
13 KiB
C++
Raw Normal View History

2015-07-08 11:39:24 -04:00
/*
* Copyright (C) 2015 Jared Boone, ShareBrained Technology, Inc.
2016-07-29 23:27:28 -04:00
* Copyright (C) 2016 Furrtek
2015-07-08 11:39:24 -04: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"
//#include "ui_loadmodule.hpp"
#include "modules.h"
2015-07-08 11:39:24 -04:00
#include "portapack.hpp"
2016-01-31 03:34:24 -05:00
#include "event_m0.hpp"
2016-04-21 16:12:51 -04:00
#include "portapack_persistent_memory.hpp"
#include "bmp_splash.hpp"
#include "bmp_modal_warning.hpp"
2015-07-08 11:39:24 -04:00
#include "ui_about.hpp"
2015-07-08 11:39:24 -04:00
#include "ui_setup.hpp"
#include "ui_debug.hpp"
//#include "ui_soundboard.hpp" // DEBUG
//#include "ui_closecall.hpp" // DEBUG
#include "ui_freqman.hpp" // DEBUG
2016-05-09 14:42:20 -04:00
#include "ui_encoders.hpp"
#include "ui_debug.hpp"
#include "ui_rds.hpp"
#include "ui_xylos.hpp"
#include "ui_epar.hpp"
#include "ui_lcr.hpp"
#include "analog_audio_app.hpp"
#include "ui_audiotx.hpp" // DEBUG
//#include "ui_jammer.hpp" // DEBUG
#include "analog_audio_app.hpp"
2016-01-31 03:34:24 -05:00
#include "ais_app.hpp"
#include "ert_app.hpp"
#include "tpms_app.hpp"
#include "pocsag_app.hpp"
2016-04-11 13:59:55 -04:00
#include "capture_app.hpp"
2016-01-31 03:34:24 -05:00
#include "core_control.hpp"
#include "file.hpp"
#include "png_writer.hpp"
2015-07-08 11:39:24 -04:00
namespace ui {
/* SystemStatusView ******************************************************/
SystemStatusView::SystemStatusView() {
uint8_t cfg;
2015-07-08 11:39:24 -04:00
add_children({ {
2016-01-31 03:34:24 -05:00
&button_back,
&title,
&button_textentry,
2016-02-19 00:35:46 -05:00
&button_camera,
2016-01-31 03:34:24 -05:00
&button_sleep,
&sd_card_status_view,
2015-07-08 11:39:24 -04:00
} });
cfg = portapack::persistent_memory::ui_config_textentry();
if (!cfg)
button_textentry.set_bitmap(&bitmap_keyboard);
else
button_textentry.set_bitmap(&bitmap_unistroke);
2016-01-31 03:34:24 -05:00
button_back.on_select = [this](Button&){
if( this->on_back ) {
this->on_back();
}
};
button_textentry.on_select = [this](ImageButton&) {
this->on_textentry();
};
2016-01-31 03:34:24 -05:00
2016-02-19 00:35:46 -05:00
button_camera.on_select = [this](ImageButton&) {
this->on_camera();
};
button_sleep.on_select = [this](ImageButton&) {
2016-01-31 03:34:24 -05:00
DisplaySleepMessage message;
EventDispatcher::send_message(message);
2016-01-31 03:34:24 -05:00
};
}
2016-02-07 13:33:15 -05:00
void SystemStatusView::set_back_enabled(bool new_value) {
button_back.set_text(new_value ? back_text_enabled : back_text_disabled);
button_back.set_focusable(new_value);
2016-01-31 03:34:24 -05:00
}
void SystemStatusView::set_title(const std::string new_value) {
if( new_value.empty() ) {
title.set(default_title);
} else {
title.set(new_value);
}
2015-07-08 11:39:24 -04:00
}
void SystemStatusView::on_textentry() {
uint8_t cfg;
cfg = portapack::persistent_memory::ui_config_textentry();
portapack::persistent_memory::set_config_textentry(cfg ^ 1);
if (!cfg)
button_textentry.set_bitmap(&bitmap_unistroke);
else
button_textentry.set_bitmap(&bitmap_keyboard);
}
2016-02-19 00:35:46 -05:00
void SystemStatusView::on_camera() {
2016-05-09 15:05:11 -04:00
const auto filename_stem = next_filename_stem_matching_pattern("SCR_????");
if( filename_stem.empty() ) {
return;
}
PNGWriter png;
auto create_error = png.create(filename_stem + ".PNG");
if( create_error.is_valid() ) {
return;
}
for(int i=0; i<320; i++) {
std::array<ColorRGB888, 240> row;
portapack::display.read_pixels({ 0, i, 240, 1 }, row);
png.write_scanline(row);
}
2016-02-19 00:35:46 -05:00
}
2015-07-08 11:39:24 -04:00
/* Navigation ************************************************************/
2016-01-31 03:34:24 -05:00
bool NavigationView::is_top() const {
return view_stack.size() == 1;
2015-07-08 11:39:24 -04:00
}
2016-01-31 03:34:24 -05:00
View* NavigationView::push_view(std::unique_ptr<View> new_view) {
free_view();
const auto p = new_view.get();
view_stack.emplace_back(std::move(new_view));
update_view();
return p;
2015-07-08 11:39:24 -04:00
}
void NavigationView::pop() {
if( view() == modal_view ) {
modal_view = nullptr;
}
2015-07-08 11:39:24 -04:00
// Can't pop last item from stack.
if( view_stack.size() > 1 ) {
2016-01-31 03:34:24 -05:00
free_view();
2015-07-08 11:39:24 -04:00
view_stack.pop_back();
2016-01-31 03:34:24 -05:00
update_view();
2015-07-08 11:39:24 -04:00
}
}
void NavigationView::display_modal(
const std::string& title,
const std::string& message
) {
/* If a modal view is already visible, don't display another */
if( !modal_view ) {
modal_view = push<ModalMessageView>(title, message, false);
}
}
2016-01-31 03:34:24 -05:00
void NavigationView::free_view() {
remove_child(view());
2015-07-08 11:39:24 -04:00
}
2016-01-31 03:34:24 -05:00
void NavigationView::update_view() {
const auto new_view = view_stack.back().get();
add_child(new_view);
new_view->set_parent_rect({ {0, 0}, size() });
focus();
set_dirty();
2015-07-08 11:39:24 -04:00
2016-01-31 03:34:24 -05:00
if( on_view_changed ) {
on_view_changed(*new_view);
2015-07-08 11:39:24 -04:00
}
2016-01-31 03:34:24 -05:00
}
2015-07-08 11:39:24 -04:00
2016-01-31 03:34:24 -05:00
Widget* NavigationView::view() const {
return children_.empty() ? nullptr : children_[0];
2015-07-08 11:39:24 -04:00
}
void NavigationView::focus() {
if( view() ) {
view()->focus();
}
}
2016-05-09 14:42:20 -04:00
/* TranspondersMenuView **************************************************/
2016-01-31 03:34:24 -05:00
TranspondersMenuView::TranspondersMenuView(NavigationView& nav) {
add_items<3>({ {
{ "AIS: Boats", ui::Color::white(), [&nav](){ nav.push<AISAppView>(); } },
{ "ERT: Utility Meters", ui::Color::white(), [&nav](){ nav.push<ERTAppView>(); } },
{ "TPMS: Cars", ui::Color::white(), [&nav](){ nav.push<TPMSAppView>(); } },
2016-01-31 03:34:24 -05:00
} });
on_left = [&nav](){ nav.pop(); };
2016-01-31 03:34:24 -05:00
}
/* ReceiverMenuView ******************************************************/
ReceiverMenuView::ReceiverMenuView(NavigationView& nav) {
add_items<3>({ {
2016-05-11 06:45:03 -04:00
{ "Audio", ui::Color::white(), [&nav](){ nav.push<AnalogAudioView>(); } },
{ "Transponders", ui::Color::white(), [&nav](){ nav.push<TranspondersMenuView>(); } },
{ "POCSAG", ui::Color::cyan(), [&nav](){ nav.push<POCSAGAppView>(); } },
2016-01-31 03:34:24 -05:00
} });
on_left = [&nav](){ nav.pop(); };
2016-01-31 03:34:24 -05:00
}
2015-07-08 11:39:24 -04:00
/* SystemMenuView ********************************************************/
SystemMenuView::SystemMenuView(NavigationView& nav) {
add_items<12>({ {
2016-07-29 23:27:28 -04:00
{ "Play dead", ui::Color::red(), [&nav](){ nav.push<PlayDeadView>(false); } },
{ "Receiver RX", ui::Color::cyan(), [&nav](){ nav.push<ReceiverMenuView>(); } },
2016-07-29 23:27:28 -04:00
{ "Capture RX", ui::Color::cyan(), [&nav](){ nav.push<CaptureAppView>(); } },
//{ "Close Call RX", ui::Color::cyan(), [&nav](){ nav.push<CloseCallView>(); } },
2016-07-29 23:27:28 -04:00
{ "Numbers station TX", ui::Color::purple(), [&nav](){ nav.push<NotImplementedView>(); } }, //nav.push<NumbersStationView>();
//{ "Pokemon GO Away TX", ui::Color::blue(), [&nav](){ nav.push<JammerView>(); } },
//{ "Soundboard TX", ui::Color::yellow(), [&nav](){ nav.push<LoadModuleView>(md5_baseband_tx, SoundBoard); } },
//{ "Audio TX", ui::Color::yellow(), [&nav](){ nav.push<LoadModuleView>(md5_baseband_tx, AudioTX); } },
2016-07-27 23:25:33 -04:00
//{ "Frequency manager", ui::Color::white(), [&nav](){ nav.push<FreqManView>(); } },
//{ "EPAR TX", ui::Color::green(), [&nav](){ nav.push<LoadModuleView>(md5_baseband_tx, EPAR); } },
{ "Xylos TX", ui::Color::green(), [&nav](){ nav.push<XylosView>(); } },
2016-07-29 23:27:28 -04:00
{ "TEDI/LCR TX", ui::Color::yellow(), [&nav](){ nav.push<LCRView>(); } },
{ "OOK encoders TX", ui::Color::orange(), [&nav](){ nav.push<EncodersView>(); } },
2016-07-29 23:27:28 -04:00
{ "RDS TX", ui::Color::red(), [&nav](){ nav.push<RDSView>(); } },
2016-05-09 14:42:20 -04:00
//{ "Analyze", ui::Color::white(), [&nav](){ nav.push<NotImplementedView>(); } },
2016-07-29 23:27:28 -04:00
{ "Setup", ui::Color::white(), [&nav](){ nav.push<SetupMenuView>(); } },
{ "Debug", ui::Color::white(), [&nav](){ nav.push<DebugMenuView>(); } },
{ "HackRF", ui::Color::white(), [&nav](){ nav.push<HackRFFirmwareView>(); } },
{ "About", ui::Color::white(), [&nav](){ nav.push<AboutView>(); } }
2015-07-08 11:39:24 -04:00
} });
//{ "Nordic/BTLE RX", ui::Color::cyan(), [&nav](){ nav.push(new NotImplementedView { nav }); } },
2016-05-10 19:03:42 -04:00
//{ "Jammer", ui::Color::white(), [&nav](){ nav.push<LoadModuleView>(md5_baseband, new JammerView(nav)); } },
//{ "Audio file TX", ui::Color::white(), [&nav](){ nav.push(new NotImplementedView { nav }); } },
//{ "Whistle", ui::Color::purple(), [&nav](){ nav.push(new LoadModuleView { nav, md5_baseband, new WhistleView { nav, transmitter_model }}); } },
2016-01-31 03:34:24 -05:00
//{ "SIGFOX RX", ui::Color::orange(), [&nav](){ nav.push(new LoadModuleView { nav, md5_baseband, new SIGFRXView { nav, receiver_model }}); } },
//{ "Xylos RX", ui::Color::green(), [&nav](){ nav.push(new LoadModuleView { nav, md5_baseband_tx, new XylosRXView { nav, receiver_model }}); } },
2016-01-31 03:34:24 -05:00
//{ "AFSK RX", ui::Color::cyan(), [&nav](){ nav.push(new LoadModuleView { nav, md5_baseband, new AFSKRXView { nav, receiver_model }}); } },
2016-05-10 19:03:42 -04:00
2015-07-08 11:39:24 -04:00
}
/* SystemView ************************************************************/
static constexpr ui::Style style_default {
.font = ui::font::fixed_8x16,
.background = ui::Color::black(),
.foreground = ui::Color::white()
2015-07-08 11:39:24 -04:00
};
SystemView::SystemView(
Context& context,
const Rect parent_rect
) : View { parent_rect },
context_(context)
2015-07-08 11:39:24 -04:00
{
2016-02-07 13:40:06 -05:00
set_style(&style_default);
2016-05-10 19:03:42 -04:00
2016-05-11 06:45:03 -04:00
//char debugtxt[21]; // DEBUG
2015-07-08 11:39:24 -04:00
constexpr ui::Dim status_view_height = 16;
2016-05-09 14:42:20 -04:00
2015-07-08 11:39:24 -04:00
add_child(&status_view);
status_view.set_parent_rect({
{ 0, 0 },
{ parent_rect.width(), status_view_height }
});
2016-01-31 03:34:24 -05:00
status_view.on_back = [this]() {
this->navigation_view.pop();
};
2015-07-08 11:39:24 -04:00
add_child(&navigation_view);
navigation_view.set_parent_rect({
{ 0, status_view_height },
{ parent_rect.width(), static_cast<ui::Dim>(parent_rect.height() - status_view_height) }
});
2016-01-31 03:34:24 -05:00
navigation_view.on_view_changed = [this](const View& new_view) {
2016-02-07 13:33:15 -05:00
this->status_view.set_back_enabled(!this->navigation_view.is_top());
2016-01-31 03:34:24 -05:00
this->status_view.set_title(new_view.title());
};
2015-07-08 11:39:24 -04:00
// Initial view.
// TODO: Restore from non-volatile memory?
2015-09-15 16:34:36 -04:00
2015-09-16 09:43:43 -04:00
//if (persistent_memory::playing_dead() == 0x59)
// navigation_view.push(new PlayDeadView { navigation_view, true });
//else
// navigation_view.push(new BMPView { navigation_view });
if (portapack::persistent_memory::ui_config() & 1)
2016-01-31 03:34:24 -05:00
navigation_view.push<BMPView>();
else
2016-05-09 14:42:20 -04:00
//navigation_view.push<SoundBoardView>();
//navigation_view.push<CloseCallView>();
2016-05-10 01:20:24 -04:00
//navigation_view.push<HandWriteView>(debugtxt, 20);
2016-05-10 01:20:24 -04:00
navigation_view.push<SystemMenuView>();
2015-07-08 11:39:24 -04:00
}
Context& SystemView::context() const {
return context_;
}
2015-08-01 16:46:15 -04:00
/* HackRFFirmwareView ****************************************************/
HackRFFirmwareView::HackRFFirmwareView(NavigationView& nav) {
button_yes.on_select = [](Button&){
EventDispatcher::request_stop();
2015-08-01 16:46:15 -04:00
};
button_no.on_select = [&nav](Button&){
nav.pop();
};
add_children({ {
&text_title,
2015-09-01 23:23:11 -04:00
&text_description_1,
&text_description_2,
&text_description_3,
&text_description_4,
2015-08-01 16:46:15 -04:00
&button_yes,
&button_no,
} });
}
2016-02-05 11:40:14 -05:00
/* ***********************************************************************/
void BMPView::focus() {
button_done.focus();
}
BMPView::BMPView(NavigationView& nav) {
add_children({ {
&text_info,
&button_done
} });
button_done.on_select = [this,&nav](Button&){
nav.pop();
2016-04-21 16:12:51 -04:00
nav.push<SystemMenuView>();
2016-02-05 11:40:14 -05:00
};
}
void BMPView::paint(Painter& painter) {
(void)painter;
portapack::display.drawBMP({(240-185)/2, 0}, splash_bmp, false);
2016-02-05 11:40:14 -05:00
}
/* PlayDeadView **********************************************************/
void PlayDeadView::focus() {
button_done.focus();
}
PlayDeadView::PlayDeadView(NavigationView& nav, bool booting) {
_booting = booting;
2016-04-21 16:12:51 -04:00
portapack::persistent_memory::set_playing_dead(0x59);
add_children({ {
&text_playdead1,
&text_playdead2,
&button_done,
} });
button_done.on_dir = [this,&nav](Button&, KeyEvent key){
sequence = (sequence<<3) | static_cast<std::underlying_type<KeyEvent>::type>(key);
};
button_done.on_select = [this,&nav](Button&){
2016-04-21 16:12:51 -04:00
if (sequence == portapack::persistent_memory::playdead_sequence()) {
portapack::persistent_memory::set_playing_dead(0);
if (_booting) {
nav.pop();
nav.push<SystemMenuView>();
} else {
nav.pop();
}
} else {
sequence = 0;
}
};
}
2015-08-01 16:46:15 -04:00
void HackRFFirmwareView::focus() {
button_no.focus();
}
2015-07-08 11:39:24 -04:00
/* 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,
bool yesno
) : title_ { title },
yesno_ { yesno }
{
if (!yesno) {
button_done.on_select = [&nav](Button&){
nav.pop();
};
add_children({ {
&text_message,
&button_done
} });
} else {
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();
};
add_children({ {
&text_message,
&button_yes,
&button_no
} });
}
text_message.set(message);
2016-05-17 13:29:49 -04:00
const int text_message_width = message.size() * 8;
text_message.set_parent_rect({
(240 - text_message_width) / 2, 7 * 16,
text_message_width, 16
});
}
void ModalMessageView::paint(Painter& painter) {
(void)painter;
portapack::display.drawBMP({64, 64}, modal_warning_bmp, false);
}
void ModalMessageView::focus() {
if (!yesno_) {
button_done.focus();
} else {
button_yes.focus();
}
}
2015-07-08 11:39:24 -04:00
} /* namespace ui */