mirror of
https://github.com/portapack-mayhem/mayhem-firmware.git
synced 2025-01-10 05:03:39 +00:00
a223bd1f21
* fix missing NULL initializers * fix mispelled OnKeyboad * set header version to 1
206 lines
6.0 KiB
C++
206 lines
6.0 KiB
C++
/*
|
|
* Copyright (C) 2024 Bernd Herzog
|
|
*
|
|
* 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_standalone_view.hpp"
|
|
#include "irq_controls.hpp"
|
|
|
|
#include "i2cdevmanager.hpp"
|
|
#include "i2cdev_ppmod.hpp"
|
|
|
|
#include "ui_font_fixed_5x8.hpp"
|
|
#include "ui_font_fixed_8x16.hpp"
|
|
|
|
namespace ui {
|
|
|
|
void create_thread(int32_t (*fn)(void*), void* arg, size_t stack_size, int priority) {
|
|
// TODO: collect memory on terminate, once this is used
|
|
chThdCreateFromHeap(NULL, stack_size, priority, fn, arg);
|
|
}
|
|
|
|
void fill_rectangle(int x, int y, int width, int height, uint16_t color) {
|
|
ui::Painter painter;
|
|
painter.fill_rectangle({x, y, width, height}, ui::Color(color));
|
|
}
|
|
|
|
void fill_rectangle_unrolled8(int x, int y, int width, int height, uint16_t color) {
|
|
ui::Painter painter;
|
|
painter.fill_rectangle_unrolled8({x, y, width, height}, ui::Color(color));
|
|
}
|
|
|
|
void draw_bitmap(int x, int y, int width, int height, const uint8_t* pixels, uint16_t foreground, uint16_t background) {
|
|
ui::Painter painter;
|
|
painter.draw_bitmap({x, y}, {{width, height}, pixels}, ui::Color(foreground), ui::Color(background));
|
|
}
|
|
|
|
void* alloc(size_t size) {
|
|
void* p = chHeapAlloc(0x0, size);
|
|
if (p == nullptr)
|
|
chDbgPanic("Out of Memory");
|
|
return p;
|
|
}
|
|
|
|
uint64_t get_switches_state_ulong() {
|
|
return get_switches_state().to_ulong();
|
|
}
|
|
|
|
ui::Coord scroll_area_y(const ui::Coord y) {
|
|
return portapack::display.scroll_area_y(y);
|
|
}
|
|
|
|
void scroll_set_area(const ui::Coord top_y, const ui::Coord bottom_y) {
|
|
portapack::display.scroll_set_area(top_y, bottom_y);
|
|
}
|
|
|
|
void scroll_disable() {
|
|
portapack::display.scroll_disable();
|
|
}
|
|
|
|
ui::Coord scroll_set_position(const ui::Coord position) {
|
|
return portapack::display.scroll_set_position(position);
|
|
}
|
|
|
|
ui::Coord scroll(const int32_t delta) {
|
|
return portapack::display.scroll(delta);
|
|
}
|
|
|
|
bool i2c_read(uint8_t* cmd, size_t cmd_len, uint8_t* data, size_t data_len) {
|
|
auto dev = (i2cdev::I2cDev_PPmod*)i2cdev::I2CDevManager::get_dev_by_model(I2C_DEVMDL::I2CDECMDL_PPMOD);
|
|
if (!dev) {
|
|
return false;
|
|
}
|
|
|
|
if (data_len == 0 || data == nullptr)
|
|
return dev->i2c_write(nullptr, 0, cmd, cmd_len);
|
|
|
|
return dev->i2c_read(cmd, cmd_len, data, data_len);
|
|
}
|
|
|
|
StandaloneView* standaloneView = nullptr;
|
|
|
|
void set_dirty() {
|
|
if (standaloneView != nullptr)
|
|
standaloneView->set_dirty();
|
|
}
|
|
|
|
standalone_application_api_t api = {
|
|
/* .malloc = */ &alloc,
|
|
/* .calloc = */ &calloc,
|
|
/* .realloc = */ &realloc,
|
|
/* .free = */ &chHeapFree,
|
|
/* .create_thread = */ &create_thread,
|
|
/* .fill_rectangle = */ &fill_rectangle,
|
|
/* .swizzled_switches = */ &swizzled_switches,
|
|
/* .get_switches_state = */ &get_switches_state_ulong,
|
|
/* .fixed_5x8_glyph_data = */ ui::font::fixed_5x8.get_data(),
|
|
/* .fixed_8x16_glyph_data = */ ui::font::fixed_8x16.get_data(),
|
|
/* .fill_rectangle_unrolled8 = */ &fill_rectangle_unrolled8,
|
|
/* .draw_bitmap = */ &draw_bitmap,
|
|
/* .scroll_area_y = */ &scroll_area_y,
|
|
/* .scroll_set_area = */ &scroll_set_area,
|
|
/* .scroll_disable = */ &scroll_disable,
|
|
/* .scroll_set_position = */ &scroll_set_position,
|
|
/* .scroll = */ &scroll,
|
|
/* .i2c_read = */ &i2c_read,
|
|
/* .panic = */ &chDbgPanic,
|
|
/* .set_dirty = */ &set_dirty,
|
|
};
|
|
|
|
StandaloneView::StandaloneView(NavigationView& nav, uint8_t* app_image)
|
|
: nav_(nav),
|
|
_app_image(*app_image) {
|
|
if (app_image == nullptr) {
|
|
chDbgPanic("Invalid application image");
|
|
}
|
|
|
|
set_focusable(true);
|
|
|
|
standaloneView = this;
|
|
}
|
|
|
|
void StandaloneView::focus() {
|
|
View::focus();
|
|
}
|
|
|
|
void StandaloneView::paint(Painter& painter) {
|
|
(void)painter;
|
|
|
|
if (initialized &&
|
|
get_application_information()->header_version > 1) {
|
|
get_application_information()->PaintViewMirror();
|
|
}
|
|
}
|
|
|
|
void StandaloneView::on_focus() {
|
|
if (get_application_information()->header_version > 1) {
|
|
get_application_information()->OnFocus();
|
|
}
|
|
}
|
|
|
|
bool StandaloneView::on_key(const KeyEvent key) {
|
|
if (get_application_information()->header_version > 1) {
|
|
return get_application_information()->OnKeyEvent((uint8_t)key);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool StandaloneView::on_encoder(const EncoderEvent event) {
|
|
if (get_application_information()->header_version > 1) {
|
|
return get_application_information()->OnEncoder((int32_t)event);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool StandaloneView::on_touch(const TouchEvent event) {
|
|
if (get_application_information()->header_version > 1) {
|
|
get_application_information()->OnTouchEvent(event.point.x(), event.point.y(), (uint32_t)event.type);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool StandaloneView::on_keyboard(const KeyboardEvent event) {
|
|
if (get_application_information()->header_version > 1) {
|
|
return get_application_information()->OnKeyboard((uint8_t)event);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void StandaloneView::frame_sync() {
|
|
// skip first refresh
|
|
if (!initialized) {
|
|
initialized = true;
|
|
} else {
|
|
get_application_information()->on_event(1);
|
|
}
|
|
}
|
|
|
|
void StandaloneView::on_after_attach() {
|
|
context().focus_manager().setMirror(this);
|
|
get_application_information()->initialize(api);
|
|
}
|
|
|
|
void StandaloneView::on_before_detach() {
|
|
get_application_information()->shutdown();
|
|
context().focus_manager().clearMirror();
|
|
}
|
|
|
|
} // namespace ui
|