mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-19 00:37:25 +01:00
284 lines
7.8 KiB
C++
284 lines
7.8 KiB
C++
#include "list_controller.h"
|
|
#include "../app.h"
|
|
#include <assert.h>
|
|
|
|
namespace Graph {
|
|
|
|
ListController::ListController(Responder * parentResponder, FunctionStore * functionStore) :
|
|
HeaderViewController(parentResponder, &m_tableView),
|
|
m_tableView(TableView(this)),
|
|
m_activeCellx(0),
|
|
m_activeCelly(-1),
|
|
m_manualScrolling(0),
|
|
m_functionStore(functionStore),
|
|
m_parameterController(ParameterController(this, functionStore))
|
|
{
|
|
setVisibleHeader(false);
|
|
}
|
|
|
|
const char * ListController::title() const {
|
|
return "Fonctions";
|
|
}
|
|
|
|
Responder * ListController::tabController() const{
|
|
return (parentResponder()->parentResponder());
|
|
}
|
|
|
|
int ListController::numberOfRows() {
|
|
return 1 + m_functionStore->numberOfFunctions();
|
|
};
|
|
|
|
int ListController::numberOfColumns() {
|
|
return 2;
|
|
};
|
|
|
|
KDCoordinate ListController::rowHeight(int j) {
|
|
if (j == numberOfRows() - 1) {
|
|
return k_emptyRowHeight;
|
|
}
|
|
Function * function = m_functionStore->functionAtIndex(j);
|
|
if (function->layout() == nullptr) {
|
|
return k_emptyRowHeight;
|
|
}
|
|
KDCoordinate functionSize = function->layout()->size().height();
|
|
return functionSize + k_verticalFunctionMargin;
|
|
}
|
|
|
|
KDCoordinate ListController::columnWidth(int i) {
|
|
switch (i) {
|
|
case 0:
|
|
return k_functionNameWidth;
|
|
case 1:
|
|
return m_tableView.bounds().width()-k_functionNameWidth;
|
|
default:
|
|
assert(false);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
KDCoordinate ListController::cumulatedWidthFromIndex(int i) {
|
|
switch (i) {
|
|
case 0:
|
|
return 0;
|
|
case 1:
|
|
return k_functionNameWidth;
|
|
case 2:
|
|
return m_tableView.bounds().width();
|
|
default:
|
|
assert(false);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
KDCoordinate ListController::cumulatedHeightFromIndex(int j) {
|
|
int result = 0;
|
|
for (int k = 0; k < j; k++) {
|
|
result += rowHeight(k);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
int ListController::indexFromCumulatedWidth(KDCoordinate offsetX) {
|
|
if (offsetX <= k_functionNameWidth) {
|
|
return 0;
|
|
} else {
|
|
if (offsetX <= m_tableView.bounds().width())
|
|
return 1;
|
|
else {
|
|
return 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
int ListController::indexFromCumulatedHeight(KDCoordinate offsetY) {
|
|
int result = 0;
|
|
int j = 0;
|
|
while (result < offsetY && j < numberOfRows()) {
|
|
result += rowHeight(j++);
|
|
}
|
|
return (result < offsetY || offsetY == 0) ? j : j - 1;
|
|
}
|
|
|
|
void ListController::setActiveCell(int i, int j) {
|
|
if (i < 0 || i >= numberOfColumns()) {
|
|
return;
|
|
}
|
|
if (j < -1 || j >= numberOfRows()) {
|
|
return;
|
|
}
|
|
|
|
if (m_activeCelly >= 0) {
|
|
EvenOddCell * previousCell = (EvenOddCell *)(m_tableView.cellAtLocation(m_activeCellx, m_activeCelly));
|
|
previousCell->setHighlighted(false);
|
|
}
|
|
|
|
m_activeCellx = i;
|
|
m_activeCelly = j;
|
|
if (m_activeCelly >= 0) {
|
|
m_tableView.scrollToCell(i, j);
|
|
EvenOddCell * cell = (EvenOddCell *)(m_tableView.cellAtLocation(i, j));
|
|
cell->setHighlighted(true);
|
|
}
|
|
}
|
|
|
|
void ListController::didBecomeFirstResponder() {
|
|
m_tableView.reloadData();
|
|
if (m_activeCelly == -1) {
|
|
setActiveCell(1,0);
|
|
} else {
|
|
if (m_activeCelly <= m_functionStore->numberOfFunctions()) {
|
|
setActiveCell(m_activeCellx, m_activeCelly);
|
|
} else {
|
|
setActiveCell(m_activeCellx, m_functionStore->numberOfFunctions());
|
|
}
|
|
}
|
|
}
|
|
|
|
void ListController::configureFunction(Function * function) {
|
|
StackViewController * stack = ((StackViewController *)parentResponder());
|
|
m_parameterController.setFunction(function);
|
|
stack->push(&m_parameterController);
|
|
}
|
|
|
|
void ListController::editExpression(FunctionExpressionView * functionCell, bool overwrite, char initialDigit) {
|
|
char initialTextContent[255];
|
|
if (overwrite) {
|
|
initialTextContent[0] = initialDigit;
|
|
initialTextContent[1] = 0;
|
|
} else {
|
|
strlcpy(initialTextContent, functionCell->function()->text(), sizeof(initialTextContent));
|
|
}
|
|
App * myApp = (App *)app();
|
|
InputViewController * inputController = myApp->inputViewController();
|
|
inputController->edit(this, initialTextContent, functionCell,
|
|
[](void * context, void * sender){
|
|
FunctionExpressionView * myCell = (FunctionExpressionView *) context;
|
|
Function * myFunction = myCell->function();
|
|
InputViewController * myInputViewController = (InputViewController *)sender;
|
|
const char * textBody = myInputViewController->textBody();
|
|
myFunction->setContent(textBody);
|
|
myCell->reloadCell();
|
|
},
|
|
[](void * context, void * sender){
|
|
});
|
|
}
|
|
|
|
bool ListController::handleEvent(Ion::Events::Event event) {
|
|
switch (event) {
|
|
case Ion::Events::Event::DOWN_ARROW:
|
|
setActiveCell(m_activeCellx, m_activeCelly+1);
|
|
return true;
|
|
case Ion::Events::Event::UP_ARROW:
|
|
setActiveCell(m_activeCellx, m_activeCelly-1);
|
|
if (m_activeCelly == -1) {
|
|
app()->setFirstResponder(tabController());
|
|
}
|
|
return true;
|
|
case Ion::Events::Event::LEFT_ARROW:
|
|
setActiveCell(m_activeCellx-1, m_activeCelly);
|
|
return true;
|
|
case Ion::Events::Event::RIGHT_ARROW:
|
|
setActiveCell(m_activeCellx+1, m_activeCelly);
|
|
return true;
|
|
case Ion::Events::Event::PLUS:
|
|
m_manualScrolling += 10;
|
|
m_tableView.setContentOffset({0, m_manualScrolling});
|
|
return true;
|
|
case Ion::Events::Event::ENTER:
|
|
return handleEnter();
|
|
default:
|
|
if ((int)event >= 0x100 || m_activeCellx == 0) {
|
|
return false;
|
|
}
|
|
FunctionExpressionView * functionCell = (FunctionExpressionView *)(m_tableView.cellAtLocation(m_activeCellx, m_activeCelly));
|
|
editExpression(functionCell, true, (char)event);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool ListController::handleEnter() {
|
|
switch (m_activeCellx) {
|
|
case 0:
|
|
{
|
|
if (m_activeCelly == numberOfRows() - 1) {
|
|
return true;
|
|
}
|
|
configureFunction(m_functionStore->functionAtIndex(m_activeCelly));
|
|
return true;
|
|
}
|
|
case 1:
|
|
{
|
|
if (m_activeCelly == numberOfRows() - 1) {
|
|
if (m_functionStore->numberOfFunctions() < FunctionStore::k_maxNumberOfFunctions) {
|
|
m_functionStore->addEmptyFunction();
|
|
m_tableView.reloadData();
|
|
return true;
|
|
}
|
|
// Add a warning to tell the user there is no more space for new functions
|
|
return false;
|
|
}
|
|
FunctionExpressionView * functionCell = (FunctionExpressionView *)(m_tableView.cellAtLocation(m_activeCellx, m_activeCelly));
|
|
editExpression(functionCell, false);
|
|
return true;
|
|
}
|
|
default:
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
int ListController::typeAtLocation(int i, int j) {
|
|
if (j == numberOfRows() - 1) {
|
|
return i + 2;
|
|
}
|
|
return i;
|
|
}
|
|
|
|
TableViewCell * ListController::reusableCell(int index, int type) {
|
|
assert(index >= 0);
|
|
assert(index < k_maxNumberOfRows);
|
|
switch (type) {
|
|
case 0:
|
|
return &m_functionTitleCells[index];
|
|
case 1:
|
|
return &m_expressionCells[index];
|
|
case 2:
|
|
return &m_emptyCell;
|
|
case 3:
|
|
return &m_addNewFunction;
|
|
default:
|
|
assert(false);
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
int ListController::reusableCellCount(int type) {
|
|
if (type > 1) {
|
|
return 1;
|
|
}
|
|
return k_maxNumberOfRows;
|
|
}
|
|
|
|
void ListController::willDisplayCellAtLocation(TableViewCell * cell, int i, int j) {
|
|
if (j < numberOfRows() - 1) {
|
|
if (i == 0) {
|
|
FunctionTitleCell * myFunctionCell = (FunctionTitleCell *)cell;
|
|
Function * function = m_functionStore->functionAtIndex(j);
|
|
char bufferName[5] = "*(x)";
|
|
bufferName[0] = *function->name();
|
|
myFunctionCell->setText(bufferName);
|
|
KDColor functionNameColor = function->isActive() ? function->color() : Palette::k_desactiveTextColor;
|
|
myFunctionCell->setColor(functionNameColor);
|
|
myFunctionCell->setOrientation(FunctionTitleCell::Orientation::VerticalIndicator);
|
|
} else {
|
|
FunctionExpressionView * myCell = (FunctionExpressionView *)cell;
|
|
myCell->setFunction(m_functionStore->functionAtIndex(j));
|
|
}
|
|
}
|
|
EvenOddCell * myCell = (EvenOddCell *)cell;
|
|
myCell->setEven(j%2 == 0);
|
|
myCell->setHighlighted(i == m_activeCellx && j == m_activeCelly);
|
|
}
|
|
|
|
} |