mirror of
https://github.com/eried/portapack-mayhem.git
synced 2025-06-20 12:54:33 -04:00
Freqman UI (#1255)
* FreqmanDB direct file * Clear UI for short lists * Final touches on freqlist UI. * Support vertical alignment in NewButton * New buttons in FreqMan * Wiring up UI to filewrapper actions * Work around empty file
This commit is contained in:
parent
0c599f7d3a
commit
29e495a17f
23 changed files with 979 additions and 660 deletions
|
@ -1,5 +1,7 @@
|
|||
/*
|
||||
* Copyright (C) 2015 Jared Boone, ShareBrained Technology, Inc.
|
||||
* Copyright (C) 2023 gullradriel, Nilorea Studio Inc.
|
||||
* Copyright (C) 2023 Kyle Reed
|
||||
*
|
||||
* This file is part of PortaPack.
|
||||
*
|
||||
|
@ -24,154 +26,143 @@
|
|||
#include "baseband_api.hpp"
|
||||
#include "utility.hpp"
|
||||
|
||||
#include <algorithm>
|
||||
|
||||
namespace ui {
|
||||
FreqManUIList::FreqManUIList(
|
||||
Rect parent_rect,
|
||||
bool instant_exec)
|
||||
|
||||
FreqManUIList::FreqManUIList(Rect parent_rect)
|
||||
: Widget{parent_rect},
|
||||
instant_exec_{instant_exec} {
|
||||
visible_lines_{(unsigned)parent_rect.height() / char_height} {
|
||||
this->set_focusable(true);
|
||||
}
|
||||
|
||||
void FreqManUIList::set_highlighted_index(int index) {
|
||||
if (freqlist_db == nullptr || (unsigned)(current_index + index) >= freqlist_db->size())
|
||||
return;
|
||||
if (index < 0) {
|
||||
index = 0;
|
||||
if (current_index > 0)
|
||||
current_index--;
|
||||
}
|
||||
if (index >= freqlist_nb_lines) {
|
||||
index = freqlist_nb_lines - 1;
|
||||
if ((unsigned)(current_index + index) < freqlist_db->size())
|
||||
current_index++;
|
||||
else
|
||||
current_index = freqlist_db->size() - freqlist_nb_lines - 1;
|
||||
}
|
||||
highlighted_index = index;
|
||||
}
|
||||
|
||||
uint8_t FreqManUIList::get_index() {
|
||||
return current_index + highlighted_index;
|
||||
}
|
||||
|
||||
void FreqManUIList::paint(Painter& painter) {
|
||||
freqlist_nb_lines = 0;
|
||||
const auto r = screen_rect();
|
||||
uint8_t focused = has_focus();
|
||||
const Rect r_widget_screen{r.left() + focused, r.top() + focused, r.width() - 2 * focused, r.height() - 2 * +focused};
|
||||
painter.fill_rectangle(
|
||||
r_widget_screen,
|
||||
Color::black());
|
||||
// only return after clearing the screen so previous entries are not shown anymore
|
||||
if (freqlist_db == nullptr || freqlist_db->size() == 0)
|
||||
auto rect = screen_rect();
|
||||
|
||||
if (!db_ || db_->empty()) {
|
||||
auto line_position = rect.location() + Point{7 * 8, 6 * 16};
|
||||
painter.fill_rectangle(rect, Color::black());
|
||||
painter.draw_string(line_position, Styles::white, "Empty Category");
|
||||
return;
|
||||
// coloration if file is too big
|
||||
auto text_color = &Styles::white;
|
||||
if (freqlist_db->size() > FREQMAN_MAX_PER_FILE)
|
||||
text_color = &Styles::yellow;
|
||||
uint8_t nb_lines = 0;
|
||||
for (uint8_t it = current_index; it < freqlist_db->size(); it++) {
|
||||
uint8_t line_height = (int)nb_lines * char_height;
|
||||
if (line_height < (r.height() - char_height)) { // line is within the widget
|
||||
std::string description = freqman_item_string(*freqlist_db->at(it), 30);
|
||||
if (nb_lines == highlighted_index) {
|
||||
const Rect r_highlighted_freq{0, r.location().y() + (int)nb_lines * char_height, 240, char_height};
|
||||
painter.fill_rectangle(
|
||||
r_highlighted_freq,
|
||||
Color::white());
|
||||
painter.draw_string(
|
||||
{0, r.location().y() + (int)nb_lines * char_height},
|
||||
Styles::bg_white, description);
|
||||
} else {
|
||||
painter.draw_string(
|
||||
{0, r.location().y() + (int)nb_lines * char_height},
|
||||
*text_color, description);
|
||||
}
|
||||
}
|
||||
|
||||
nb_lines++;
|
||||
} else {
|
||||
// rect is filled, we can break
|
||||
break;
|
||||
}
|
||||
}
|
||||
freqlist_nb_lines = nb_lines;
|
||||
if (has_focus() || highlighted()) {
|
||||
const Rect r_focus{r.left(), r.top(), r.width(), r.height()};
|
||||
painter.draw_rectangle(
|
||||
r_focus,
|
||||
Color::white());
|
||||
}
|
||||
}
|
||||
// Indicate when a file is too large by drawing in yellow.
|
||||
auto over_max = db_->entry_count() > freqman_default_max_entries;
|
||||
auto base_style = over_max ? &Styles::yellow : &Styles::white;
|
||||
|
||||
void FreqManUIList::set_db(freqman_db& db) {
|
||||
freqlist_db = &db;
|
||||
if (db.size() == 0) {
|
||||
current_index = 0;
|
||||
highlighted_index = 0;
|
||||
} else {
|
||||
if ((unsigned)(current_index + highlighted_index) >= db.size()) {
|
||||
current_index = db.size() - 1 - highlighted_index;
|
||||
}
|
||||
if (current_index < 0) {
|
||||
current_index = 0;
|
||||
if (highlighted_index > 0)
|
||||
highlighted_index--;
|
||||
// TODO: could minimize redraw/re-read if necessary
|
||||
// with better change tracking.
|
||||
for (auto offset = 0u; offset < visible_lines_; ++offset) {
|
||||
// The whole frame needs to be cleared so every line 'slot'
|
||||
// is redrawn even when `text` just left empty.
|
||||
auto text = std::string{};
|
||||
auto index = start_index_ + offset;
|
||||
auto line_position = rect.location() + Point{4, 1 + (int)offset * char_height};
|
||||
|
||||
// Highlight the selected item.
|
||||
auto style = (offset == selected_index_) ? &Styles::bg_white : base_style;
|
||||
|
||||
if (index < db_->entry_count()) {
|
||||
auto entry = (*db_)[index];
|
||||
// db_ is directly backed by a file, so invalid lines cannot be
|
||||
// pre-filtered. Just show an empty 'slot' in this case.
|
||||
if (entry.type != freqman_type::Unknown)
|
||||
text = pretty_string(entry, line_max_length);
|
||||
}
|
||||
|
||||
// Pad right with ' ' so trailing chars are cleaned up.
|
||||
// draw_glyph has less flicker than fill_rect when drawing.
|
||||
if (text.length() < line_max_length)
|
||||
text.resize(line_max_length, ' ');
|
||||
|
||||
painter.draw_string(line_position, *style, text);
|
||||
}
|
||||
|
||||
// Draw a bounding rectangle when focused.
|
||||
painter.draw_rectangle(rect, (has_focus() ? Color::white() : Color::black()));
|
||||
}
|
||||
|
||||
void FreqManUIList::on_focus() {
|
||||
if (on_highlight)
|
||||
on_highlight(*this);
|
||||
set_dirty();
|
||||
}
|
||||
|
||||
void FreqManUIList::on_blur() {
|
||||
set_dirty();
|
||||
}
|
||||
|
||||
bool FreqManUIList::on_key(const KeyEvent key) {
|
||||
if (key == KeyEvent::Select) {
|
||||
if (on_select) {
|
||||
on_select(*this);
|
||||
return true;
|
||||
}
|
||||
} else {
|
||||
if (on_dir) {
|
||||
return on_dir(*this, key);
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
if (!db_ || db_->empty())
|
||||
return false;
|
||||
|
||||
bool FreqManUIList::on_touch(const TouchEvent event) {
|
||||
switch (event.type) {
|
||||
case TouchEvent::Type::Start:
|
||||
set_highlighted(true);
|
||||
set_dirty();
|
||||
if (on_touch_press) {
|
||||
on_touch_press(*this);
|
||||
}
|
||||
if (on_select && instant_exec_) {
|
||||
on_select(*this);
|
||||
}
|
||||
return true;
|
||||
case TouchEvent::Type::End:
|
||||
set_highlighted(false);
|
||||
set_dirty();
|
||||
if (on_touch_release) {
|
||||
on_touch_release(*this);
|
||||
}
|
||||
if (on_select && !instant_exec_) {
|
||||
on_select(*this);
|
||||
}
|
||||
return true;
|
||||
default:
|
||||
return false;
|
||||
if (key == KeyEvent::Select && on_select) {
|
||||
on_select(get_index());
|
||||
return true;
|
||||
} else if (key == KeyEvent::Right && on_leave) {
|
||||
on_leave();
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
bool FreqManUIList::on_encoder(EncoderEvent delta) {
|
||||
set_highlighted_index((int)highlighted_index + delta);
|
||||
auto delta = 0;
|
||||
if (key == KeyEvent::Up && get_index() > 0)
|
||||
delta = -1;
|
||||
else if (key == KeyEvent::Down && get_index() < db_->entry_count() - 1)
|
||||
delta = 1;
|
||||
else
|
||||
return false;
|
||||
|
||||
adjust_selected_index(delta);
|
||||
set_dirty();
|
||||
return true;
|
||||
}
|
||||
|
||||
bool FreqManUIList::on_encoder(EncoderEvent delta) {
|
||||
if (!db_ || db_->empty())
|
||||
return false;
|
||||
|
||||
adjust_selected_index(delta);
|
||||
set_dirty();
|
||||
return true;
|
||||
}
|
||||
|
||||
void FreqManUIList::set_parent_rect(Rect new_parent_rect) {
|
||||
visible_lines_ = new_parent_rect.height() / char_height;
|
||||
Widget::set_parent_rect(new_parent_rect);
|
||||
}
|
||||
|
||||
void FreqManUIList::set_index(size_t index) {
|
||||
start_index_ = 0;
|
||||
selected_index_ = 0;
|
||||
adjust_selected_index(index);
|
||||
}
|
||||
|
||||
size_t FreqManUIList::get_index() const {
|
||||
return start_index_ + selected_index_;
|
||||
}
|
||||
|
||||
void FreqManUIList::set_db(FreqmanDB& db) {
|
||||
db_ = &db;
|
||||
start_index_ = 0;
|
||||
selected_index_ = 0;
|
||||
set_dirty();
|
||||
}
|
||||
|
||||
void FreqManUIList::adjust_selected_index(int delta) {
|
||||
int32_t new_index = selected_index_ + delta;
|
||||
|
||||
// The selection went off the top of the screen, move up.
|
||||
if (new_index < 0) {
|
||||
start_index_ = std::max<int32_t>(start_index_ + new_index, 0);
|
||||
selected_index_ = 0;
|
||||
}
|
||||
|
||||
// Selection is off the bottom of the screen, move down.
|
||||
else if (new_index >= (int32_t)visible_lines_) {
|
||||
start_index_ = std::min<int32_t>(start_index_ + delta, db_->entry_count() - visible_lines_);
|
||||
selected_index_ = visible_lines_ - 1;
|
||||
}
|
||||
|
||||
// Otherwise, scroll within the screen, but not past the end.
|
||||
else {
|
||||
selected_index_ = std::min<int32_t>(new_index, db_->entry_count() - 1);
|
||||
}
|
||||
}
|
||||
|
||||
} /* namespace ui */
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue