mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-18 16:27:34 +01:00
309 lines
12 KiB
C++
309 lines
12 KiB
C++
#include "math_variable_box_controller.h"
|
|
#include "shared/global_context.h"
|
|
#include "shared/continuous_function.h"
|
|
#include <escher/metric.h>
|
|
#include <ion/unicode/utf8_decoder.h>
|
|
#include <poincare/exception_checkpoint.h>
|
|
#include <poincare/expression.h>
|
|
#include <poincare/layout_helper.h>
|
|
#include <poincare/matrix_layout.h>
|
|
#include <poincare/preferences.h>
|
|
#include <assert.h>
|
|
#include <algorithm>
|
|
#include <apps/shared/sequence.h>
|
|
#include <apps/apps_container.h>
|
|
#include "global_preferences.h"
|
|
|
|
using namespace Poincare;
|
|
using namespace Shared;
|
|
using namespace Ion;
|
|
|
|
MathVariableBoxController::MathVariableBoxController() :
|
|
AlternateEmptyNestedMenuController(I18n::Message::Variables),
|
|
m_currentPage(Page::RootMenu),
|
|
m_lockPageDelete(Page::RootMenu),
|
|
m_firstMemoizedLayoutIndex(0)
|
|
{
|
|
for (int i = 0; i < k_maxNumberOfDisplayedRows; i++) {
|
|
m_leafCells[i].setParentResponder(&m_selectableTableView);
|
|
}
|
|
}
|
|
|
|
void MathVariableBoxController::viewWillAppear() {
|
|
assert(m_currentPage == Page::RootMenu);
|
|
AlternateEmptyNestedMenuController::viewWillAppear();
|
|
}
|
|
|
|
void MathVariableBoxController::viewDidDisappear() {
|
|
AlternateEmptyNestedMenuController::viewDidDisappear();
|
|
|
|
/* NestedMenuController::viewDidDisappear might need cell heights, which would
|
|
* use the MathVariableBoxController cell heights memoization. We thus reset the
|
|
* MathVariableBoxController layouts only after calling the parent's
|
|
* viewDidDisappear. */
|
|
|
|
// Tidy the layouts displayed in the MathVariableBoxController to clean TreePool
|
|
for (int i = 0; i < k_maxNumberOfDisplayedRows; i++) {
|
|
m_leafCells[i].setLayout(Layout());
|
|
m_leafCells[i].setAccessoryLayout(Layout());
|
|
}
|
|
/* We reset the page when view disappears rather than when it appears because
|
|
* subview layout is done before viewWillAppear. If the page at that point is
|
|
* wrong, the memoized layouts are going be filled with wrong layouts. */
|
|
setPage(Page::RootMenu);
|
|
}
|
|
|
|
bool MathVariableBoxController::handleEvent(Ion::Events::Event event) {
|
|
/* We do not want to handle backspace event if:
|
|
* - On the root menu page
|
|
* The deletion on the current page is locked
|
|
* - The empty controller is displayed
|
|
*/
|
|
if (event == Ion::Events::Backspace && m_currentPage != Page::RootMenu && m_lockPageDelete != m_currentPage && !isDisplayingEmptyController()) {
|
|
int rowIndex = selectedRow();
|
|
m_selectableTableView.deselectTable();
|
|
destroyRecordAtRowIndex(rowIndex);
|
|
int newSelectedRow = rowIndex >= numberOfRows() ? numberOfRows()-1 : rowIndex;
|
|
selectCellAtLocation(selectedColumn(), newSelectedRow);
|
|
m_selectableTableView.reloadData();
|
|
displayEmptyControllerIfNeeded();
|
|
return true;
|
|
}
|
|
return AlternateEmptyNestedMenuController::handleEvent(event);
|
|
}
|
|
|
|
int MathVariableBoxController::numberOfRows() const {
|
|
switch (m_currentPage) {
|
|
case Page::RootMenu:
|
|
return k_numberOfMenuRows;
|
|
case Page::Expression:
|
|
return Storage::sharedStorage()->numberOfRecordsWithExtension(Ion::Storage::expExtension);
|
|
case Page::Function:
|
|
return Storage::sharedStorage()->numberOfRecordsWithExtension(Ion::Storage::funcExtension);
|
|
case Page::Sequence:
|
|
return Storage::sharedStorage()->numberOfRecordsWithExtension(Ion::Storage::seqExtension);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
int MathVariableBoxController::reusableCellCount(int type) {
|
|
assert(type < 2);
|
|
if (type == 0) {
|
|
return k_maxNumberOfDisplayedRows;
|
|
}
|
|
return k_numberOfMenuRows;
|
|
}
|
|
|
|
void MathVariableBoxController::willDisplayCellForIndex(HighlightCell * cell, int index) {
|
|
if (m_currentPage == Page::RootMenu) {
|
|
I18n::Message label = nodeLabelAtIndex(index);
|
|
MessageTableCell<> * myCell = (MessageTableCell<> *)cell;
|
|
myCell->setMessage(label);
|
|
myCell->reloadCell();
|
|
return;
|
|
}
|
|
ExpressionTableCellWithExpression * myCell = (ExpressionTableCellWithExpression *)cell;
|
|
Storage::Record record = recordAtIndex(index);
|
|
char symbolName[Shared::Function::k_maxNameWithArgumentSize];
|
|
size_t symbolLength = 0;
|
|
Layout symbolLayout;
|
|
if (m_currentPage == Page::Expression) {
|
|
static_assert(Shared::Function::k_maxNameWithArgumentSize > Poincare::SymbolAbstract::k_maxNameSize, "Forgot argument's size?");
|
|
symbolLength = SymbolAbstract::TruncateExtension(symbolName, record.fullName(), SymbolAbstract::k_maxNameSize);
|
|
} else if (m_currentPage == Page::Function) {
|
|
ContinuousFunction f(record);
|
|
symbolLength = f.nameWithArgument(
|
|
symbolName,
|
|
Shared::Function::k_maxNameWithArgumentSize
|
|
);
|
|
} else {
|
|
assert(m_currentPage == Page::Sequence);
|
|
Shared::Sequence u(record);
|
|
symbolLength = u.nameWithArgumentAndType(
|
|
symbolName,
|
|
Shared::Sequence::k_maxNameWithArgumentSize
|
|
);
|
|
Expression symbolExpression = Expression::Parse(symbolName, AppsContainer::sharedAppsContainer()->globalContext());
|
|
symbolLayout = symbolExpression.createLayout(Poincare::Preferences::sharedPreferences()->displayMode(), Poincare::Preferences::sharedPreferences()->numberOfSignificantDigits());
|
|
}
|
|
if (symbolLayout.isUninitialized()) {
|
|
symbolLayout = LayoutHelper::String(symbolName, symbolLength);
|
|
}
|
|
myCell->setLayout(symbolLayout);
|
|
myCell->setAccessoryLayout(expressionLayoutForRecord(record, index));
|
|
myCell->reloadScroll();
|
|
myCell->reloadCell();
|
|
}
|
|
|
|
KDCoordinate MathVariableBoxController::rowHeight(int index) {
|
|
if (m_currentPage != Page::RootMenu) {
|
|
Layout layoutR = expressionLayoutForRecord(recordAtIndex(index), index);
|
|
if (!layoutR.isUninitialized()) {
|
|
return std::max<KDCoordinate>(layoutR.layoutSize().height()+k_leafMargin, Metric::ToolboxRowHeight);
|
|
}
|
|
}
|
|
return AlternateEmptyNestedMenuController::rowHeight(index);
|
|
}
|
|
|
|
int MathVariableBoxController::typeAtLocation(int i, int j) {
|
|
if (m_currentPage == Page::RootMenu) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
ExpressionTableCellWithExpression * MathVariableBoxController::leafCellAtIndex(int index) {
|
|
assert(index >= 0 && index < k_maxNumberOfDisplayedRows);
|
|
return &m_leafCells[index];
|
|
}
|
|
|
|
MessageTableCellWithChevron<> * MathVariableBoxController::nodeCellAtIndex(int index) {
|
|
assert(index >= 0 && index < k_numberOfMenuRows);
|
|
return &m_nodeCells[index];
|
|
}
|
|
|
|
MathVariableBoxController::Page MathVariableBoxController::pageAtIndex(int index) {
|
|
Page pages[k_numberOfMenuRows] = {Page::Expression, Page::Function, Page::Sequence};
|
|
return pages[index];
|
|
}
|
|
|
|
void MathVariableBoxController::setPage(Page page) {
|
|
m_currentPage = page;
|
|
resetMemoization();
|
|
}
|
|
|
|
bool MathVariableBoxController::selectSubMenu(int selectedRow) {
|
|
m_selectableTableView.deselectTable();
|
|
setPage(pageAtIndex(selectedRow));
|
|
bool selectSubMenu = AlternateEmptyNestedMenuController::selectSubMenu(selectedRow);
|
|
if (displayEmptyControllerIfNeeded()) {
|
|
return true;
|
|
}
|
|
return selectSubMenu;
|
|
}
|
|
|
|
bool MathVariableBoxController::returnToPreviousMenu() {
|
|
if (isDisplayingEmptyController()) {
|
|
pop();
|
|
} else {
|
|
m_selectableTableView.deselectTable();
|
|
}
|
|
setPage(Page::RootMenu);
|
|
return AlternateEmptyNestedMenuController::returnToPreviousMenu();
|
|
}
|
|
|
|
bool MathVariableBoxController::selectLeaf(int selectedRow, bool quitToolbox) {
|
|
if (isDisplayingEmptyController()) {
|
|
/* We do not want to handle OK/EXE events in that case. */
|
|
return false;
|
|
}
|
|
|
|
// Deselect the table
|
|
assert(selectedRow >= 0 && selectedRow < numberOfRows());
|
|
m_selectableTableView.deselectTable();
|
|
|
|
// Get the name text to insert
|
|
Storage::Record record = recordAtIndex(selectedRow);
|
|
constexpr size_t nameToHandleMaxSize = Shared::Function::k_maxNameWithArgumentSize;
|
|
char nameToHandle[nameToHandleMaxSize];
|
|
size_t nameLength = SymbolAbstract::TruncateExtension(nameToHandle, record.fullName(), nameToHandleMaxSize);
|
|
|
|
if (m_currentPage == Page::Function || m_currentPage == Page::Sequence) {
|
|
// Add parentheses to a function name
|
|
assert(nameLength < nameToHandleMaxSize);
|
|
nameLength += UTF8Decoder::CodePointToChars('(', nameToHandle+nameLength, nameToHandleMaxSize - nameLength);
|
|
assert(nameLength < nameToHandleMaxSize);
|
|
nameLength+= UTF8Decoder::CodePointToChars(UCodePointEmpty, nameToHandle+nameLength, nameToHandleMaxSize - nameLength);
|
|
assert(nameLength < nameToHandleMaxSize);
|
|
nameLength += UTF8Decoder::CodePointToChars(')', nameToHandle+nameLength, nameToHandleMaxSize - nameLength);
|
|
assert(nameLength < nameToHandleMaxSize);
|
|
nameToHandle[nameLength] = 0;
|
|
}
|
|
|
|
// Handle the text
|
|
sender()->handleEventWithText(nameToHandle);
|
|
Container::activeApp()->dismissModalViewController();
|
|
return true;
|
|
}
|
|
|
|
I18n::Message MathVariableBoxController::nodeLabelAtIndex(int index) {
|
|
assert(m_currentPage == Page::RootMenu);
|
|
I18n::Message labels[k_numberOfMenuRows] = {I18n::Message::Expressions, I18n::Message::Functions, I18n::Message::Sequences};
|
|
return labels[index];
|
|
}
|
|
|
|
Layout MathVariableBoxController::expressionLayoutForRecord(Storage::Record record, int index) {
|
|
assert(m_currentPage != Page::RootMenu);
|
|
assert(index >= 0);
|
|
if (index >= m_firstMemoizedLayoutIndex+k_maxNumberOfDisplayedRows || index < m_firstMemoizedLayoutIndex) {
|
|
// Change range of layout memoization
|
|
int deltaIndex = index >= m_firstMemoizedLayoutIndex + k_maxNumberOfDisplayedRows ? index - k_maxNumberOfDisplayedRows + 1 - m_firstMemoizedLayoutIndex : index - m_firstMemoizedLayoutIndex;
|
|
for (int i = 0; i < k_maxNumberOfDisplayedRows; i++) {
|
|
int j = deltaIndex + i;
|
|
m_layouts[i] = (j >= m_firstMemoizedLayoutIndex && j < k_maxNumberOfDisplayedRows) ? m_layouts[j] : Layout();
|
|
}
|
|
m_firstMemoizedLayoutIndex += deltaIndex;
|
|
assert(m_firstMemoizedLayoutIndex >= 0);
|
|
}
|
|
assert(index >= m_firstMemoizedLayoutIndex && index < m_firstMemoizedLayoutIndex + k_maxNumberOfDisplayedRows);
|
|
if (m_layouts[index-m_firstMemoizedLayoutIndex].isUninitialized()) {
|
|
/* Creating the layout of a very long variable might throw a pool exception.
|
|
* We want to catch it and return a dummy layout instead, otherwise the user
|
|
* won't be able to open the variable box again, until she deletes the
|
|
* problematic variable -> and she has no help to remember its name, as she
|
|
* can't open the variable box. */
|
|
Layout result;
|
|
Poincare::ExceptionCheckpoint ecp;
|
|
if (ExceptionRun(ecp)) {
|
|
result = GlobalContext::LayoutForRecord(record);
|
|
}
|
|
m_layouts[index-m_firstMemoizedLayoutIndex] = result;
|
|
}
|
|
return m_layouts[index-m_firstMemoizedLayoutIndex];
|
|
}
|
|
|
|
const char * MathVariableBoxController::extension() const {
|
|
assert(m_currentPage != Page::RootMenu);
|
|
if (m_currentPage == Page::Function) {
|
|
return Ion::Storage::funcExtension;
|
|
} else if (m_currentPage == Page::Expression) {
|
|
return Ion::Storage::expExtension;
|
|
} else {
|
|
assert(m_currentPage == Page::Sequence);
|
|
return Ion::Storage::seqExtension;
|
|
}
|
|
}
|
|
|
|
Storage::Record MathVariableBoxController::recordAtIndex(int rowIndex) {
|
|
assert(m_currentPage != Page::RootMenu);
|
|
assert(!Storage::sharedStorage()->recordWithExtensionAtIndex(extension(), rowIndex).isNull());
|
|
return Storage::sharedStorage()->recordWithExtensionAtIndex(extension(), rowIndex);
|
|
}
|
|
|
|
ViewController * MathVariableBoxController::emptyViewController() {
|
|
m_emptyViewController.setType((MathVariableBoxEmptyController::Type)m_currentPage);
|
|
return &m_emptyViewController;
|
|
}
|
|
|
|
void MathVariableBoxController::resetMemoization() {
|
|
for (int i = 0; i < k_maxNumberOfDisplayedRows; i++) {
|
|
m_layouts[i] = Layout();
|
|
}
|
|
m_firstMemoizedLayoutIndex = 0;
|
|
}
|
|
|
|
void MathVariableBoxController::destroyRecordAtRowIndex(int rowIndex) {
|
|
// Destroy the record
|
|
recordAtIndex(rowIndex).destroy();
|
|
// Shift the memoization if needed
|
|
if (rowIndex >= m_firstMemoizedLayoutIndex + k_maxNumberOfDisplayedRows) {
|
|
// The deleted row is after the memoization
|
|
return;
|
|
}
|
|
for (int i = std::max(0, rowIndex - m_firstMemoizedLayoutIndex); i < k_maxNumberOfDisplayedRows - 1; i++) {
|
|
m_layouts[i] = m_layouts[i+1];
|
|
}
|
|
m_layouts[k_maxNumberOfDisplayedRows - 1] = Layout();
|
|
}
|