mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-19 16:57:31 +01:00
190 lines
7.5 KiB
C++
190 lines
7.5 KiB
C++
#include "values_controller.h"
|
|
#include <assert.h>
|
|
#include "../../constant.h"
|
|
#include "../app.h"
|
|
|
|
using namespace Shared;
|
|
using namespace Poincare;
|
|
|
|
namespace Graph {
|
|
|
|
ValuesController::ValuesController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, Interval * interval, ButtonRowController * header) :
|
|
Shared::ValuesController(parentResponder, header, interval),
|
|
m_functionTitleCells{},
|
|
m_floatCells{},
|
|
m_abscissaTitleCells{},
|
|
m_abscissaCells{},
|
|
m_functionParameterController(this),
|
|
m_intervalParameterController(this, inputEventHandlerDelegate, m_interval),
|
|
m_derivativeParameterController(this),
|
|
m_setIntervalButton(this, I18n::Message::IntervalSet, Invocation([](void * context, void * sender) {
|
|
ValuesController * valuesController = (ValuesController *) context;
|
|
StackViewController * stack = ((StackViewController *)valuesController->stackController());
|
|
stack->push(valuesController->intervalParameterSelectorController());
|
|
return true;
|
|
}, this), k_font)
|
|
{
|
|
for (int i = 0; i < k_maxNumberOfFunctions; i++) {
|
|
m_functionTitleCells[i].setOrientation(FunctionTitleCell::Orientation::HorizontalIndicator);
|
|
m_functionTitleCells[i].setFont(KDFont::SmallFont);
|
|
}
|
|
setupAbscissaCellsAndTitleCells(inputEventHandlerDelegate);
|
|
}
|
|
|
|
void ValuesController::willDisplayCellAtLocation(HighlightCell * cell, int i, int j) {
|
|
Shared::ValuesController::willDisplayCellAtLocation(cell, i, j);
|
|
int typeAtLoc = typeAtLocation(i,j);
|
|
if (typeAtLoc == k_abscissaTitleCellType) {
|
|
AbscissaTitleCell * myTitleCell = (AbscissaTitleCell *)cell;
|
|
Ion::Storage::Record record = recordAtColumn(i+1);
|
|
Shared::ExpiringPointer<CartesianFunction> function = functionStore()->modelForRecord(record);
|
|
myTitleCell->setMessage(function->parameterMessageName());
|
|
myTitleCell->setSeparatorLeft(i > 0);
|
|
return;
|
|
}
|
|
if (typeAtLoc == k_functionTitleCellType) {
|
|
Shared::BufferFunctionTitleCell * myFunctionCell = (Shared::BufferFunctionTitleCell *)cell;
|
|
const size_t bufferNameSize = Shared::Function::k_maxNameWithArgumentSize + 1;
|
|
char bufferName[bufferNameSize];
|
|
bool isDerivative = false;
|
|
Ion::Storage::Record record = recordAtColumn(i, &isDerivative);
|
|
Shared::ExpiringPointer<CartesianFunction> function = functionStore()->modelForRecord(record);
|
|
if (isDerivative) {
|
|
function->derivativeNameWithArgument(bufferName, bufferNameSize);
|
|
} else {
|
|
function->nameWithArgument(bufferName, bufferNameSize);
|
|
}
|
|
myFunctionCell->setText(bufferName);
|
|
myFunctionCell->setColor(function->color());
|
|
return;
|
|
}
|
|
if (typeAtLoc == k_editableValueCellType) {
|
|
StoreCell * storeCell = (StoreCell *)cell;
|
|
storeCell->setSeparatorLeft(i > 0);
|
|
}
|
|
}
|
|
|
|
int ValuesController::typeAtLocation(int i, int j) {
|
|
int plotTypeIndex = 0;
|
|
while (plotTypeIndex < CartesianFunction::k_numberOfPlotTypes && i >= m_numberOfColumnsForType[plotTypeIndex]) {
|
|
i -= m_numberOfColumnsForType[plotTypeIndex++];
|
|
}
|
|
return Shared::ValuesController::typeAtLocation(i, j);
|
|
}
|
|
|
|
I18n::Message ValuesController::emptyMessage() {
|
|
if (functionStore()->numberOfDefinedModels() == 0) {
|
|
return I18n::Message::NoFunction;
|
|
}
|
|
return I18n::Message::NoActivatedFunction;
|
|
}
|
|
|
|
Ion::Storage::Record ValuesController::recordAtColumn(int i) {
|
|
bool isDerivative = false;
|
|
return recordAtColumn(i, &isDerivative);
|
|
}
|
|
|
|
Ion::Storage::Record ValuesController::recordAtColumn(int i, bool * isDerivative) {
|
|
assert(typeAtLocation(i, 0) == k_functionTitleCellType);
|
|
int plotTypeIndex = 0;
|
|
while (plotTypeIndex < CartesianFunction::k_numberOfPlotTypes && i >= m_numberOfColumnsForType[plotTypeIndex]) {
|
|
i -= m_numberOfColumnsForType[plotTypeIndex++];
|
|
}
|
|
CartesianFunction::PlotType plotType = static_cast<CartesianFunction::PlotType>(plotTypeIndex);
|
|
int index = 1;
|
|
for (int k = 0; k < functionStore()->numberOfActiveFunctionsOfType(plotType); k++) {
|
|
Ion::Storage::Record record = functionStore()->activeRecordOfTypeAtIndex(plotType, k);
|
|
const int numberOfColumnsForCurrentRecord = numberOfColumnsForRecord(record);
|
|
if (index <= i && i < index + numberOfColumnsForCurrentRecord) {
|
|
ExpiringPointer<CartesianFunction> f = functionStore()->modelForRecord(record);
|
|
*isDerivative = i != index && f->plotType() == CartesianFunction::PlotType::Cartesian;
|
|
return record;
|
|
}
|
|
index += numberOfColumnsForCurrentRecord;
|
|
}
|
|
assert(false);
|
|
return nullptr;
|
|
}
|
|
|
|
int ValuesController::numberOfColumnsForRecord(Ion::Storage::Record record) const {
|
|
ExpiringPointer<CartesianFunction> f = functionStore()->modelForRecord(record);
|
|
CartesianFunction::PlotType plotType = f->plotType();
|
|
return 1 +
|
|
(plotType == CartesianFunction::PlotType::Cartesian && f->displayDerivative()) +
|
|
(plotType == CartesianFunction::PlotType::Parametric);
|
|
}
|
|
|
|
int ValuesController::maxNumberOfCells() {
|
|
return k_maxNumberOfCells;
|
|
}
|
|
|
|
int ValuesController::maxNumberOfFunctions() {
|
|
return k_maxNumberOfFunctions;
|
|
}
|
|
|
|
Shared::BufferFunctionTitleCell * ValuesController::functionTitleCells(int j) {
|
|
assert(j >= 0 && j < k_maxNumberOfFunctions);
|
|
return &m_functionTitleCells[j];
|
|
}
|
|
|
|
EvenOddBufferTextCell * ValuesController::floatCells(int j) {
|
|
assert(j >= 0 && j < k_maxNumberOfCells);
|
|
return &m_floatCells[j];
|
|
}
|
|
|
|
ViewController * ValuesController::functionParameterController() {
|
|
bool isDerivative = false;
|
|
Ion::Storage::Record record = recordAtColumn(selectedColumn(), &isDerivative);
|
|
if (functionStore()->modelForRecord(record)->plotType() != CartesianFunction::PlotType::Cartesian) {
|
|
return nullptr;
|
|
}
|
|
if (isDerivative) {
|
|
m_derivativeParameterController.setRecord(record);
|
|
return &m_derivativeParameterController;
|
|
}
|
|
m_functionParameterController.setRecord(record);
|
|
return &m_functionParameterController;
|
|
}
|
|
|
|
I18n::Message ValuesController::valuesParameterControllerPageTitle() const {
|
|
return I18n::Message::X;
|
|
}
|
|
|
|
double ValuesController::evaluationOfAbscissaAtColumn(double abscissa, int columnIndex) {
|
|
bool isDerivative = false;
|
|
Ion::Storage::Record record = recordAtColumn(columnIndex, &isDerivative);
|
|
Shared::ExpiringPointer<CartesianFunction> function = functionStore()->modelForRecord(record);
|
|
Poincare::Context * context = textFieldDelegateApp()->localContext();
|
|
if (isDerivative) {
|
|
return function->approximateDerivative(abscissa, context);
|
|
}
|
|
Poincare::Coordinate2D<double> eval = function->evaluate2DAtParameter(abscissa, context);
|
|
if (function->plotType() != CartesianFunction::PlotType::Parametric
|
|
|| (columnIndex == numberOfColumns() - 1
|
|
|| !((typeAtLocation(columnIndex+1, 0) == k_functionTitleCellType)
|
|
&& recordAtColumn(columnIndex+1) == record)))
|
|
{
|
|
return eval.x2();
|
|
}
|
|
return eval.x1();
|
|
}
|
|
|
|
void ValuesController::updateNumberOfColumns() {
|
|
for (int plotTypeIndex = 0; plotTypeIndex < CartesianFunction::k_numberOfPlotTypes; plotTypeIndex++) {
|
|
m_numberOfColumnsForType[plotTypeIndex] = 0;
|
|
}
|
|
for (int i = 0; i < functionStore()->numberOfActiveFunctions(); i++) {
|
|
Ion::Storage::Record record = functionStore()->activeRecordAtIndex(i);
|
|
ExpiringPointer<CartesianFunction> f = functionStore()->modelForRecord(record);
|
|
int plotTypeIndex = static_cast<int>(f->plotType());
|
|
m_numberOfColumnsForType[plotTypeIndex] += numberOfColumnsForRecord(record);
|
|
}
|
|
m_numberOfColumns = 0;
|
|
for (int plotTypeIndex = 0; plotTypeIndex < CartesianFunction::k_numberOfPlotTypes; plotTypeIndex++) {
|
|
m_numberOfColumnsForType[plotTypeIndex] += (m_numberOfColumnsForType[plotTypeIndex] > 0);
|
|
m_numberOfColumns += m_numberOfColumnsForType[plotTypeIndex];
|
|
}
|
|
}
|
|
|
|
}
|