#include "list_controller.h" #include "../app.h" #include 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; } View * 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(View * 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); } }