Files
Upsilon/apps/graph/list/list_controller.cpp
Émilie Feral ce88f34179 [escher] In text view, fix bug: color all background in background color
Change-Id: I45bab2759cde8d6af51b927750ba16610afc36cd
2017-01-27 10:39:26 +01:00

249 lines
7.6 KiB
C++

#include "list_controller.h"
#include "../app.h"
#include <assert.h>
namespace Graph {
ListController::ListController(Responder * parentResponder, FunctionStore * functionStore, HeaderViewController * header) :
ViewController(parentResponder),
HeaderViewDelegate(header),
m_functionTitleCells{FunctionTitleCell(FunctionTitleCell::Orientation::VerticalIndicator), FunctionTitleCell(FunctionTitleCell::Orientation::VerticalIndicator), FunctionTitleCell(FunctionTitleCell::Orientation::VerticalIndicator),
FunctionTitleCell(FunctionTitleCell::Orientation::VerticalIndicator), FunctionTitleCell(FunctionTitleCell::Orientation::VerticalIndicator), FunctionTitleCell(FunctionTitleCell::Orientation::VerticalIndicator)},
m_selectableTableView(SelectableTableView(this, this)),
m_functionStore(functionStore),
m_parameterController(ParameterController(this, functionStore))
{
}
const char * ListController::title() const {
return "Fonctions";
}
View * ListController::view() {
return &m_selectableTableView;
}
Responder * ListController::tabController() const{
return (parentResponder()->parentResponder()->parentResponder());
}
StackViewController * ListController::stackController() const{
return (StackViewController *)(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_selectableTableView.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_selectableTableView.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_selectableTableView.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::didBecomeFirstResponder() {
if (m_selectableTableView.selectedRow() == -1) {
m_selectableTableView.selectCellAtLocation(1, 0);
} else {
m_selectableTableView.selectCellAtLocation(m_selectableTableView.selectedColumn(), m_selectableTableView.selectedRow());
}
app()->setFirstResponder(&m_selectableTableView);
}
void ListController::configureFunction(Function * function) {
StackViewController * stack = stackController();
m_parameterController.setFunction(function);
stack->push(&m_parameterController);
}
void ListController::editExpression(FunctionExpressionView * functionCell, Ion::Events::Event event) {
char * initialText = nullptr;
char initialTextContent[255];
if (event == Ion::Events::OK) {
strlcpy(initialTextContent, functionCell->function()->text(), sizeof(initialTextContent));
initialText = initialTextContent;
}
App * myApp = (App *)app();
InputViewController * inputController = myApp->inputViewController();
inputController->edit(this, event, functionCell, initialText,
[](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) {
if (event == Ion::Events::Up) {
m_selectableTableView.deselectTable();
assert(m_selectableTableView.selectedRow() == -1);
app()->setFirstResponder(tabController());
return true;
}
if (event == Ion::Events::OK) {
return handleEnter();
}
if ((!event.hasText() && event != Ion::Events::XNT) || m_selectableTableView.selectedColumn() == 0
|| m_selectableTableView.selectedRow() == numberOfRows() - 1) {
return false;
}
FunctionExpressionView * functionCell = (FunctionExpressionView *)(m_selectableTableView.cellAtLocation(m_selectableTableView.selectedColumn(), m_selectableTableView.selectedRow()));
editExpression(functionCell, event);
return true;
}
bool ListController::handleEnter() {
switch (m_selectableTableView.selectedColumn()) {
case 0:
{
if (m_selectableTableView.selectedRow() == numberOfRows() - 1) {
return true;
}
configureFunction(m_functionStore->functionAtIndex(m_selectableTableView.selectedRow()));
return true;
}
case 1:
{
if (m_selectableTableView.selectedRow() == numberOfRows() - 1) {
if (m_functionStore->numberOfFunctions() < FunctionStore::k_maxNumberOfFunctions) {
m_functionStore->addEmptyFunction();
m_selectableTableView.reloadData();
return true;
}
// Add a warning to tell the user there is no more space for new functions
return false;
}
FunctionExpressionView * functionCell = (FunctionExpressionView *)(m_selectableTableView.cellAtLocation(m_selectableTableView.selectedColumn(), m_selectableTableView.selectedRow()));
editExpression(functionCell, Ion::Events::OK);
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] = {*function->name(),'(','x',')', 0};
myFunctionCell->setText(bufferName);
KDColor functionNameColor = function->isActive() ? function->color() : Palette::DesactiveTextColor;
myFunctionCell->setColor(functionNameColor);
} else {
FunctionExpressionView * myCell = (FunctionExpressionView *)cell;
myCell->setFunction(m_functionStore->functionAtIndex(j));
}
}
EvenOddCell * myCell = (EvenOddCell *)cell;
myCell->setEven(j%2 == 0);
myCell->setHighlighted(i == m_selectableTableView.selectedColumn() && j == m_selectableTableView.selectedRow());
}
}