mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-18 16:27:34 +01:00
[apps/shared] Templatize FloatParameterController to handle float/double
parameters. Fix bug: when entering "e^234" as a parameter of a model keeping floats, the FloatParameterController would accept the number (because e^234 is defined in double) and store an undefined value in the model (because e^234 is undefined in float).
This commit is contained in:
@@ -8,7 +8,7 @@ using namespace Shared;
|
||||
namespace Graph {
|
||||
|
||||
DomainParameterController::DomainParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate) :
|
||||
FloatParameterController(parentResponder),
|
||||
FloatParameterController<double>(parentResponder),
|
||||
m_domainCells{},
|
||||
m_record()
|
||||
{
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
|
||||
namespace Graph {
|
||||
|
||||
class DomainParameterController : public Shared::FloatParameterController {
|
||||
class DomainParameterController : public Shared::FloatParameterController<double> {
|
||||
public:
|
||||
DomainParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate);
|
||||
|
||||
|
||||
@@ -137,11 +137,11 @@ int ParametersController::reusableParameterCellCount(int type) {
|
||||
return m_distribution->numberOfParameter();
|
||||
}
|
||||
|
||||
double ParametersController::parameterAtIndex(int index) {
|
||||
float ParametersController::parameterAtIndex(int index) {
|
||||
return m_distribution->parameterValueAtIndex(index);
|
||||
}
|
||||
|
||||
bool ParametersController::setParameterAtIndex(int parameterIndex, double f) {
|
||||
bool ParametersController::setParameterAtIndex(int parameterIndex, float f) {
|
||||
if (!m_distribution->authorizedValueAtIndex(f, parameterIndex)) {
|
||||
Container::activeApp()->displayWarning(I18n::Message::ForbiddenValue);
|
||||
return false;
|
||||
|
||||
@@ -8,7 +8,7 @@
|
||||
|
||||
namespace Probability {
|
||||
|
||||
class ParametersController : public Shared::FloatParameterController {
|
||||
class ParametersController : public Shared::FloatParameterController<float> {
|
||||
public:
|
||||
ParametersController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, Distribution * m_distribution, CalculationController * calculationController);
|
||||
const char * title() override;
|
||||
@@ -23,8 +23,8 @@ private:
|
||||
HighlightCell * reusableParameterCell(int index, int type) override;
|
||||
int reusableParameterCellCount(int type) override;
|
||||
void buttonAction() override;
|
||||
double parameterAtIndex(int index) override;
|
||||
bool setParameterAtIndex(int parameterIndex, double f) override;
|
||||
float parameterAtIndex(int index) override;
|
||||
bool setParameterAtIndex(int parameterIndex, float f) override;
|
||||
bool textFieldDidFinishEditing(TextField * textField, const char * text, Ion::Events::Event event) override;
|
||||
class ContentView : public View {
|
||||
public:
|
||||
|
||||
@@ -8,7 +8,8 @@ using namespace Poincare;
|
||||
|
||||
namespace Shared {
|
||||
|
||||
FloatParameterController::FloatParameterController(Responder * parentResponder) :
|
||||
template<typename T>
|
||||
FloatParameterController<T>::FloatParameterController(Responder * parentResponder) :
|
||||
ViewController(parentResponder),
|
||||
m_selectableTableView(this, this, this),
|
||||
m_okButton(&m_selectableTableView, I18n::Message::Ok, Invocation([](void * context, void * sender) {
|
||||
@@ -19,7 +20,8 @@ FloatParameterController::FloatParameterController(Responder * parentResponder)
|
||||
{
|
||||
}
|
||||
|
||||
void FloatParameterController::didBecomeFirstResponder() {
|
||||
template<typename T>
|
||||
void FloatParameterController<T>::didBecomeFirstResponder() {
|
||||
if (selectedRow() >= 0) {
|
||||
int selRow = selectedRow();
|
||||
selRow = selRow >= numberOfRows() ? numberOfRows()-1 : selRow;
|
||||
@@ -30,7 +32,8 @@ void FloatParameterController::didBecomeFirstResponder() {
|
||||
Container::activeApp()->setFirstResponder(&m_selectableTableView);
|
||||
}
|
||||
|
||||
void FloatParameterController::viewWillAppear() {
|
||||
template<typename T>
|
||||
void FloatParameterController<T>::viewWillAppear() {
|
||||
if (selectedRow() == -1 || selectedRow() == numberOfRows()-1) {
|
||||
selectCellAtLocation(0, 0);
|
||||
} else {
|
||||
@@ -43,14 +46,16 @@ void FloatParameterController::viewWillAppear() {
|
||||
m_selectableTableView.reloadData();
|
||||
}
|
||||
|
||||
void FloatParameterController::willExitResponderChain(Responder * nextFirstResponder) {
|
||||
template<typename T>
|
||||
void FloatParameterController<T>::willExitResponderChain(Responder * nextFirstResponder) {
|
||||
if (parentResponder() == nullptr) {
|
||||
m_selectableTableView.deselectTable();
|
||||
m_selectableTableView.scrollToCell(0,0);
|
||||
}
|
||||
}
|
||||
|
||||
bool FloatParameterController::handleEvent(Ion::Events::Event event) {
|
||||
template<typename T>
|
||||
bool FloatParameterController<T>::handleEvent(Ion::Events::Event event) {
|
||||
if (event == Ion::Events::Back) {
|
||||
stackController()->pop();
|
||||
return true;
|
||||
@@ -58,49 +63,56 @@ bool FloatParameterController::handleEvent(Ion::Events::Event event) {
|
||||
return false;
|
||||
}
|
||||
|
||||
int FloatParameterController::typeAtLocation(int i, int j) {
|
||||
template<typename T>
|
||||
int FloatParameterController<T>::typeAtLocation(int i, int j) {
|
||||
if (j == numberOfRows()-1) {
|
||||
return 0;
|
||||
}
|
||||
return 1;
|
||||
}
|
||||
|
||||
int FloatParameterController::reusableCellCount(int type) {
|
||||
template<typename T>
|
||||
int FloatParameterController<T>::reusableCellCount(int type) {
|
||||
if (type == 0) {
|
||||
return 1;
|
||||
}
|
||||
return reusableParameterCellCount(type);
|
||||
}
|
||||
|
||||
HighlightCell * FloatParameterController::reusableCell(int index, int type) {
|
||||
template<typename T>
|
||||
HighlightCell * FloatParameterController<T>::reusableCell(int index, int type) {
|
||||
if (type == 0) {
|
||||
return &m_okButton;
|
||||
}
|
||||
return reusableParameterCell(index, type);
|
||||
}
|
||||
|
||||
KDCoordinate FloatParameterController::rowHeight(int j) {
|
||||
template<typename T>
|
||||
KDCoordinate FloatParameterController<T>::rowHeight(int j) {
|
||||
if (j == numberOfRows()-1) {
|
||||
return Metric::ParameterCellHeight+k_buttonMargin;
|
||||
}
|
||||
return Metric::ParameterCellHeight;
|
||||
}
|
||||
|
||||
KDCoordinate FloatParameterController::cumulatedHeightFromIndex(int j) {
|
||||
template<typename T>
|
||||
KDCoordinate FloatParameterController<T>::cumulatedHeightFromIndex(int j) {
|
||||
if (j == numberOfRows()) {
|
||||
return j*Metric::ParameterCellHeight+k_buttonMargin;
|
||||
}
|
||||
return Metric::ParameterCellHeight*j;
|
||||
}
|
||||
|
||||
int FloatParameterController::indexFromCumulatedHeight(KDCoordinate offsetY) {
|
||||
template<typename T>
|
||||
int FloatParameterController<T>::indexFromCumulatedHeight(KDCoordinate offsetY) {
|
||||
if (offsetY > numberOfRows()*Metric::ParameterCellHeight + k_buttonMargin) {
|
||||
return numberOfRows();
|
||||
}
|
||||
return (offsetY - 1) / Metric::ParameterCellHeight;
|
||||
}
|
||||
|
||||
void FloatParameterController::willDisplayCellForIndex(HighlightCell * cell, int index) {
|
||||
template<typename T>
|
||||
void FloatParameterController<T>::willDisplayCellForIndex(HighlightCell * cell, int index) {
|
||||
if (index == numberOfRows()-1) {
|
||||
return;
|
||||
}
|
||||
@@ -111,18 +123,20 @@ void FloatParameterController::willDisplayCellForIndex(HighlightCell * cell, int
|
||||
constexpr int precision = Preferences::LargeNumberOfSignificantDigits;
|
||||
constexpr int bufferSize = PrintFloat::bufferSizeForFloatsWithPrecision(precision);
|
||||
char buffer[bufferSize];
|
||||
PrintFloat::ConvertFloatToText<double>(parameterAtIndex(index), buffer, bufferSize, precision, Preferences::PrintFloatMode::Decimal);
|
||||
PrintFloat::ConvertFloatToText<T>(parameterAtIndex(index), buffer, bufferSize, precision, Preferences::PrintFloatMode::Decimal);
|
||||
myCell->setAccessoryText(buffer);
|
||||
}
|
||||
|
||||
bool FloatParameterController::textFieldShouldFinishEditing(TextField * textField, Ion::Events::Event event) {
|
||||
template<typename T>
|
||||
bool FloatParameterController<T>::textFieldShouldFinishEditing(TextField * textField, Ion::Events::Event event) {
|
||||
return (event == Ion::Events::Down && selectedRow() < numberOfRows()-1)
|
||||
|| (event == Ion::Events::Up && selectedRow() > 0)
|
||||
|| TextFieldDelegate::textFieldShouldFinishEditing(textField, event);
|
||||
}
|
||||
|
||||
bool FloatParameterController::textFieldDidFinishEditing(TextField * textField, const char * text, Ion::Events::Event event) {
|
||||
double floatBody;
|
||||
template<typename T>
|
||||
bool FloatParameterController<T>::textFieldDidFinishEditing(TextField * textField, const char * text, Ion::Events::Event event) {
|
||||
T floatBody;
|
||||
if (textFieldDelegateApp()->hasUndefinedValue(text, floatBody)) {
|
||||
return false;
|
||||
}
|
||||
@@ -139,17 +153,23 @@ bool FloatParameterController::textFieldDidFinishEditing(TextField * textField,
|
||||
return true;
|
||||
}
|
||||
|
||||
int FloatParameterController::activeCell() {
|
||||
template<typename T>
|
||||
int FloatParameterController<T>::activeCell() {
|
||||
return selectedRow();
|
||||
}
|
||||
|
||||
StackViewController * FloatParameterController::stackController() {
|
||||
template<typename T>
|
||||
StackViewController * FloatParameterController<T>::stackController() {
|
||||
return (StackViewController *)parentResponder();
|
||||
}
|
||||
|
||||
void FloatParameterController::buttonAction() {
|
||||
template<typename T>
|
||||
void FloatParameterController<T>::buttonAction() {
|
||||
StackViewController * stack = stackController();
|
||||
stack->pop();
|
||||
}
|
||||
|
||||
template class FloatParameterController<float>;
|
||||
template class FloatParameterController<double>;
|
||||
|
||||
}
|
||||
|
||||
@@ -10,6 +10,7 @@ namespace Shared {
|
||||
/* This controller edits float parameter of any model (given through
|
||||
* parameterAtIndex and setParameterAtIndex). */
|
||||
|
||||
template<typename T>
|
||||
class FloatParameterController : public ViewController, public ListViewDataSource, public SelectableTableViewDataSource, public ParameterTextFieldDelegate {
|
||||
public:
|
||||
FloatParameterController(Responder * parentResponder);
|
||||
@@ -31,7 +32,7 @@ public:
|
||||
protected:
|
||||
int activeCell();
|
||||
StackViewController * stackController();
|
||||
virtual double parameterAtIndex(int index) = 0;
|
||||
virtual T parameterAtIndex(int index) = 0;
|
||||
SelectableTableView m_selectableTableView;
|
||||
ButtonWithSeparator m_okButton;
|
||||
private:
|
||||
@@ -39,7 +40,7 @@ private:
|
||||
virtual void buttonAction();
|
||||
virtual int reusableParameterCellCount(int type) = 0;
|
||||
virtual HighlightCell * reusableParameterCell(int index, int type) = 0;
|
||||
virtual bool setParameterAtIndex(int parameterIndex, double f) = 0;
|
||||
virtual bool setParameterAtIndex(int parameterIndex, T f) = 0;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
@@ -4,7 +4,7 @@
|
||||
namespace Shared {
|
||||
|
||||
GoToParameterController::GoToParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, InteractiveCurveViewRange * graphRange, CurveViewCursor * cursor, I18n::Message symbol) :
|
||||
FloatParameterController(parentResponder),
|
||||
FloatParameterController<double>(parentResponder),
|
||||
m_cursor(cursor),
|
||||
m_graphRange(graphRange),
|
||||
m_abscisseCell(&m_selectableTableView, inputEventHandlerDelegate, this, symbol)
|
||||
|
||||
@@ -8,7 +8,7 @@
|
||||
|
||||
namespace Shared {
|
||||
|
||||
class GoToParameterController : public FloatParameterController {
|
||||
class GoToParameterController : public FloatParameterController<double> {
|
||||
public:
|
||||
GoToParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, InteractiveCurveViewRange * graphRange, CurveViewCursor * cursor, I18n::Message symbol);
|
||||
int numberOfRows() override;
|
||||
|
||||
@@ -4,7 +4,7 @@
|
||||
namespace Shared {
|
||||
|
||||
IntervalParameterController::IntervalParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, Interval * interval) :
|
||||
FloatParameterController(parentResponder),
|
||||
FloatParameterController<double>(parentResponder),
|
||||
m_interval(interval),
|
||||
m_intervalCells{}
|
||||
{
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
|
||||
namespace Shared {
|
||||
|
||||
class IntervalParameterController : public Shared::FloatParameterController {
|
||||
class IntervalParameterController : public Shared::FloatParameterController<double> {
|
||||
public:
|
||||
IntervalParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, Interval * interval);
|
||||
Interval * interval();
|
||||
|
||||
@@ -6,7 +6,7 @@ using namespace Poincare;
|
||||
namespace Shared {
|
||||
|
||||
RangeParameterController::RangeParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, InteractiveCurveViewRange * interactiveRange) :
|
||||
FloatParameterController(parentResponder),
|
||||
FloatParameterController<float>(parentResponder),
|
||||
m_interactiveRange(interactiveRange),
|
||||
m_xRangeCells{},
|
||||
m_yRangeCells{},
|
||||
@@ -79,14 +79,14 @@ bool RangeParameterController::handleEvent(Ion::Events::Event event) {
|
||||
return FloatParameterController::handleEvent(event);
|
||||
}
|
||||
|
||||
double RangeParameterController::parameterAtIndex(int parameterIndex) {
|
||||
float RangeParameterController::parameterAtIndex(int parameterIndex) {
|
||||
ParameterGetterPointer getters[k_numberOfTextCell] = {&InteractiveCurveViewRange::xMin,
|
||||
&InteractiveCurveViewRange::xMax, &InteractiveCurveViewRange::yMin, &InteractiveCurveViewRange::yMax};
|
||||
int index = parameterIndex > 2 ? parameterIndex - 1 : parameterIndex;
|
||||
return (m_interactiveRange->*getters[index])();
|
||||
}
|
||||
|
||||
bool RangeParameterController::setParameterAtIndex(int parameterIndex, double f) {
|
||||
bool RangeParameterController::setParameterAtIndex(int parameterIndex, float f) {
|
||||
ParameterSetterPointer setters[k_numberOfTextCell] = {&InteractiveCurveViewRange::setXMin,
|
||||
&InteractiveCurveViewRange::setXMax, &InteractiveCurveViewRange::setYMin, &InteractiveCurveViewRange::setYMax};
|
||||
int index = parameterIndex > 2 ? parameterIndex - 1 : parameterIndex;
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
|
||||
namespace Shared {
|
||||
|
||||
class RangeParameterController : public FloatParameterController {
|
||||
class RangeParameterController : public FloatParameterController<float> {
|
||||
public:
|
||||
RangeParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, InteractiveCurveViewRange * interactiveCurveViewRange);
|
||||
const char * title() override;
|
||||
@@ -32,8 +32,8 @@ private:
|
||||
};
|
||||
HighlightCell * reusableParameterCell(int index, int type) override;
|
||||
int reusableParameterCellCount(int type) override;
|
||||
double parameterAtIndex(int index) override;
|
||||
bool setParameterAtIndex(int parameterIndex, double f) override;
|
||||
float parameterAtIndex(int index) override;
|
||||
bool setParameterAtIndex(int parameterIndex, float f) override;
|
||||
constexpr static int k_numberOfEditableTextCell = 2;
|
||||
constexpr static int k_numberOfConvertibleTextCell = 2;
|
||||
constexpr static int k_numberOfTextCell = k_numberOfEditableTextCell+k_numberOfConvertibleTextCell;
|
||||
|
||||
@@ -40,8 +40,9 @@ bool TextFieldDelegateApp::isAcceptableText(const char * text) {
|
||||
return isAcceptable;
|
||||
}
|
||||
|
||||
bool TextFieldDelegateApp::hasUndefinedValue(const char * text, double & value) {
|
||||
value = PoincareHelpers::ApproximateToScalar<double>(text, localContext());
|
||||
template<typename T>
|
||||
bool TextFieldDelegateApp::hasUndefinedValue(const char * text, T & value) {
|
||||
value = PoincareHelpers::ApproximateToScalar<T>(text, localContext());
|
||||
bool isUndefined = std::isnan(value) || std::isinf(value);
|
||||
if (isUndefined) {
|
||||
displayWarning(I18n::Message::UndefinedValue);
|
||||
@@ -115,4 +116,7 @@ bool TextFieldDelegateApp::ExpressionCanBeSerialized(const Expression expression
|
||||
return true;
|
||||
}
|
||||
|
||||
template bool TextFieldDelegateApp::hasUndefinedValue(const char * text, float & value);
|
||||
template bool TextFieldDelegateApp::hasUndefinedValue(const char * text, double & value);
|
||||
|
||||
}
|
||||
|
||||
@@ -18,7 +18,8 @@ public:
|
||||
bool textFieldShouldFinishEditing(TextField * textField, Ion::Events::Event event) override;
|
||||
virtual bool textFieldDidReceiveEvent(TextField * textField, Ion::Events::Event event) override;
|
||||
bool isAcceptableText(const char * text);
|
||||
bool hasUndefinedValue(const char * text, double & value);
|
||||
template<typename T>
|
||||
bool hasUndefinedValue(const char * text, T & value);
|
||||
protected:
|
||||
TextFieldDelegateApp(Snapshot * snapshot, ViewController * rootViewController);
|
||||
bool fieldDidReceiveEvent(EditableField * field, Responder * responder, Ion::Events::Event event);
|
||||
|
||||
@@ -42,7 +42,7 @@ void IntervalController::ContentView::layoutSubviews() {
|
||||
/* IntervalController Controller */
|
||||
|
||||
IntervalController::IntervalController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, EquationStore * equationStore) :
|
||||
FloatParameterController(parentResponder),
|
||||
FloatParameterController<double>(parentResponder),
|
||||
m_contentView(&m_selectableTableView),
|
||||
m_intervalCell{},
|
||||
m_equationStore(equationStore)
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
|
||||
namespace Solver {
|
||||
|
||||
class IntervalController : public Shared::FloatParameterController {
|
||||
class IntervalController : public Shared::FloatParameterController<double> {
|
||||
public:
|
||||
IntervalController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, EquationStore * equationStore);
|
||||
const char * title() override;
|
||||
|
||||
@@ -8,7 +8,7 @@ using namespace Shared;
|
||||
namespace Statistics {
|
||||
|
||||
HistogramParameterController::HistogramParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegate, Store * store) :
|
||||
FloatParameterController(parentResponder),
|
||||
FloatParameterController<double>(parentResponder),
|
||||
m_cells{},
|
||||
m_store(store)
|
||||
{
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
|
||||
namespace Statistics {
|
||||
|
||||
class HistogramParameterController : public Shared::FloatParameterController {
|
||||
class HistogramParameterController : public Shared::FloatParameterController<double> {
|
||||
public:
|
||||
HistogramParameterController(Responder * parentResponder, InputEventHandlerDelegate * inputEventHandlerDelegateApp, Store * store);
|
||||
const char * title() override;
|
||||
|
||||
Reference in New Issue
Block a user