eurorack/deprecated/midi2cv/menu/menu.cc

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;
}