mirror of
https://github.com/jhbruhn/eurorack.git
synced 2025-03-14 18:55:48 +00:00
140 lines
4.2 KiB
C++
140 lines
4.2 KiB
C++
#include "menu.h"
|
|
#include <algorithm>
|
|
#include <u8g2.h>
|
|
const int kMenuItemHeight = 12;
|
|
|
|
void Menu::render(U8G2* u8g2_, uint8_t xStart, uint8_t yStart, uint8_t width, uint8_t height)
|
|
{
|
|
if (this->visibility_dirty) {
|
|
this->update_visible_items();
|
|
this->visibility_dirty = false;
|
|
}
|
|
|
|
this->width = width;
|
|
this->height = height;
|
|
|
|
uint8_t maxVisibleItems = height / kMenuItemHeight;
|
|
|
|
uint8_t itemsToRender = std::min(maxVisibleItems, uint8_t(this->visibleItemCount - currentVisibleScrollStart));
|
|
u8g2_->setFont(u8g2_font_6x10_tf);
|
|
|
|
for (uint8_t i = 0; i < itemsToRender; i++) {
|
|
uint8_t yPosition = yStart + i * kMenuItemHeight;
|
|
|
|
uint8_t itemIndex = this->currentVisibleScrollStart + i;
|
|
|
|
AbstractMenuItem* item = this->visibleItems[itemIndex];
|
|
|
|
bool selected = this->selectedVisibleItem == itemIndex;
|
|
bool editing = this->currentEditingVisibleItem == itemIndex;
|
|
|
|
u8g2_->setDrawColor(selected ? 1 : 0);
|
|
|
|
if (editing) {
|
|
u8g2_->drawFrame(xStart, yPosition, width, kMenuItemHeight);
|
|
} else if (selected) {
|
|
u8g2_->drawBox(xStart, yPosition, width, kMenuItemHeight);
|
|
}
|
|
|
|
u8g2_->setDrawColor(editing || !selected ? 1 : 0);
|
|
u8g2_->drawStr(xStart + 2, yPosition + kMenuItemHeight - 3, item->get_label());
|
|
|
|
uint8_t valueStringWidth = u8g2_->getStrWidth(item->get_string_representation());
|
|
u8g2_->drawStr(xStart + width - valueStringWidth - 2, yPosition + kMenuItemHeight - 3, item->get_string_representation());
|
|
|
|
itemIndex++;
|
|
}
|
|
}
|
|
|
|
void Menu::update_visible_items()
|
|
{
|
|
AbstractMenuItem* currentlySelectedItem = this->visibleItems[this->selectedVisibleItem];
|
|
|
|
// set all visibleItems to null
|
|
this->visibleItemCount = 0;
|
|
for (size_t i = 0; i < MAXIMUM_MENU_ITEM_COUNT; i++) {
|
|
this->visibleItems[i] = NULL;
|
|
}
|
|
|
|
uint8_t newIndexSelectedItem = 0;
|
|
|
|
// find out which items are visible and only add these to the list.
|
|
for (size_t i = 0; i < itemCount; i++) {
|
|
if (this->items[i]->visible()) {
|
|
if (currentlySelectedItem && this->items[i] == currentlySelectedItem) {
|
|
newIndexSelectedItem = this->visibleItemCount;
|
|
}
|
|
this->visibleItems[this->visibleItemCount++] = this->items[i];
|
|
}
|
|
}
|
|
|
|
// if our visibleitem changed, chances are that the index of the selected item has changed:
|
|
if (currentlySelectedItem && this->visibleItems[this->selectedVisibleItem] != currentlySelectedItem) {
|
|
int8_t delta = this->selectedVisibleItem - newIndexSelectedItem;
|
|
this->selectedVisibleItem = this->currentEditingVisibleItem = newIndexSelectedItem;
|
|
|
|
this->currentVisibleScrollStart -= delta;
|
|
CONSTRAIN(this->currentVisibleScrollStart, 0, this->visibleItemCount);
|
|
CONSTRAIN(this->currentEditingVisibleItem, 0, this->visibleItemCount);
|
|
}
|
|
}
|
|
|
|
void Menu::up()
|
|
{
|
|
if (this->currentEditingVisibleItem >= 0) {
|
|
AbstractMenuItem* item = this->visibleItems[this->selectedVisibleItem];
|
|
if (item)
|
|
item->decrease();
|
|
} else if (this->selectedVisibleItem > 0) {
|
|
uint8_t newItem = this->selectedVisibleItem - 1;
|
|
|
|
if (newItem - this->currentVisibleScrollStart == 0) { // keep scroll start one up
|
|
this->currentVisibleScrollStart--;
|
|
}
|
|
|
|
this->selectedVisibleItem = newItem;
|
|
|
|
if (this->selectedVisibleItem == 0) {
|
|
this->currentVisibleScrollStart = 0;
|
|
}
|
|
}
|
|
this->visibility_dirty = true;
|
|
}
|
|
|
|
void Menu::down()
|
|
{
|
|
if (this->currentEditingVisibleItem >= 0) {
|
|
AbstractMenuItem* item = this->visibleItems[this->selectedVisibleItem];
|
|
if (item)
|
|
item->increase();
|
|
} else {
|
|
uint8_t maxVisibleItems = height / kMenuItemHeight;
|
|
if (this->selectedVisibleItem < this->visibleItemCount - 1) {
|
|
if (this->selectedVisibleItem - this->currentVisibleScrollStart == maxVisibleItems - 2 && this->visibleItemCount - this->currentVisibleScrollStart > maxVisibleItems) {
|
|
this->currentVisibleScrollStart++;
|
|
}
|
|
|
|
this->selectedVisibleItem++;
|
|
}
|
|
}
|
|
this->visibility_dirty = true;
|
|
}
|
|
|
|
bool Menu::enter()
|
|
{
|
|
if (this->currentEditingVisibleItem >= 0) {
|
|
this->currentEditingVisibleItem = -1;
|
|
return true;
|
|
}
|
|
this->currentEditingVisibleItem = this->selectedVisibleItem;
|
|
return false;
|
|
}
|
|
|
|
bool Menu::back()
|
|
{
|
|
if (this->currentEditingVisibleItem >= 0) {
|
|
this->currentEditingVisibleItem = -1;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|