[apps/statistics] Show several histograms

This commit is contained in:
Léa Saviot
2018-05-22 16:30:54 +02:00
parent b1e732e135
commit c84fd45b97
7 changed files with 159 additions and 47 deletions

View File

@@ -10,20 +10,77 @@ using namespace Shared;
namespace Statistics {
HistogramController::HistogramController(Responder * parentResponder, ButtonRowController * header, Store * store, uint32_t * storeVersion, uint32_t * barVersion, uint32_t * rangeVersion, int * selectedBarIndex) :
HistogramController::ContentView::ContentView(HistogramController * controller, Store * store) :
m_histogramView1(controller, store, 0, &m_bannerView),
m_histogramView2(controller, store, 1, &m_bannerView),
m_histogramView3(controller, store, 2, &m_bannerView),
m_bannerView(),
m_store(store)
{
}
HistogramView * HistogramController::ContentView::histogramViewAtIndex(int index) {
assert(index >= 0 && index < 3);
HistogramView * views[] = {&m_histogramView1, &m_histogramView2, &m_histogramView3};
return views[index];
}
int HistogramController::ContentView::seriesOfSubviewAtIndex(int index) {
assert(index >= 0 && index < numberOfSubviews());
return static_cast<HistogramView *>(subviewAtIndex(index))->series();
}
void HistogramController::ContentView::layoutSubviews() {
int numberSubviews = numberOfSubviews();
KDCoordinate subviewHeight = bounds().height()/numberSubviews;
int displayedSubviewIndex = 0;
for (int i = 0; i < 3; i++) {
if (!m_store->seriesIsEmpty(i)) {
KDRect frame = KDRect(0, displayedSubviewIndex*subviewHeight, bounds().width(), subviewHeight);
subviewAtIndex(displayedSubviewIndex)->setFrame(frame);
displayedSubviewIndex++;
}
}
}
int HistogramController::ContentView::numberOfSubviews() const {
int result = m_store->numberOfNonEmptySeries();
assert(result <= Store::k_numberOfSeries);
return result;
}
View * HistogramController::ContentView::subviewAtIndex(int index) {
int seriesIndex = 0;
int nonEmptySeriesIndex = 0;
while (nonEmptySeriesIndex < index && seriesIndex < Store::k_numberOfSeries) {
if (!m_store->seriesIsEmpty(seriesIndex)) {
nonEmptySeriesIndex++;
}
seriesIndex++;
}
if (nonEmptySeriesIndex == index) {
assert(seriesIndex >=0 && seriesIndex < 3);
View * views[] = {&m_histogramView1, &m_histogramView2, &m_histogramView3};
return views[seriesIndex];
}
assert(false);
return nullptr;
}
HistogramController::HistogramController(Responder * parentResponder, ButtonRowController * header, Store * store, int series, uint32_t * storeVersion, uint32_t * barVersion, uint32_t * rangeVersion, int * selectedBarIndex) :
ViewController(parentResponder),
ButtonRowDelegate(header, nullptr),
m_bannerView(),
m_view(store, &m_bannerView),
m_settingButton(this, I18n::Message::HistogramSet, Invocation([](void * context, void * sender) {
HistogramController * histogramController = (HistogramController *) context;
StackViewController * stack = ((StackViewController *)histogramController->stackController());
stack->push(histogramController->histogramParameterController());
}, this)),
m_store(store),
m_view(this, store),
m_storeVersion(storeVersion),
m_barVersion(barVersion),
m_rangeVersion(rangeVersion),
m_selectedSeries(-1),
m_selectedBarIndex(selectedBarIndex),
m_histogramParameterController(nullptr, store)
{
@@ -34,6 +91,10 @@ StackViewController * HistogramController::stackController() {
return stack;
}
void HistogramController::setCurrentDrawnSeries(int series) {
initYRangeParameters(series);
}
int HistogramController::numberOfButtons(ButtonRowController::Position) const {
return isEmpty() ? 0 : 1;
}
@@ -58,8 +119,9 @@ const char * HistogramController::title() {
}
void HistogramController::viewWillAppear() {
if (!m_view.isMainViewSelected()) {
m_view.selectMainView(true);
if (m_selectedSeries < 0) {
m_selectedSeries = m_view.seriesOfSubviewAtIndex(0);
m_view.histogramViewAtIndex(m_selectedSeries)->selectMainView(true);
header()->setSelectedButton(-1);
}
reloadBannerView();
@@ -68,9 +130,9 @@ void HistogramController::viewWillAppear() {
bool HistogramController::handleEvent(Ion::Events::Event event) {
if (event == Ion::Events::Down) {
if (!m_view.isMainViewSelected()) {
if (!m_view.histogramViewAtIndex(m_selectedSeries)->isMainViewSelected()) {
header()->setSelectedButton(-1);
m_view.selectMainView(true);
m_view.histogramViewAtIndex(m_selectedSeries)->selectMainView(true);
reloadBannerView();
m_view.reload();
app()->setFirstResponder(this);
@@ -79,16 +141,16 @@ bool HistogramController::handleEvent(Ion::Events::Event event) {
return false;
}
if (event == Ion::Events::Up) {
if (!m_view.isMainViewSelected()) {
if (!m_view.histogramViewAtIndex(m_selectedSeries)->isMainViewSelected()) {
header()->setSelectedButton(-1);
app()->setFirstResponder(tabController());
return true;
}
m_view.selectMainView(false);
m_view.histogramViewAtIndex(m_selectedSeries)->selectMainView(false);
header()->setSelectedButton(0);
return true;
}
if (m_view.isMainViewSelected() && (event == Ion::Events::Left || event == Ion::Events::Right)) {
if (m_view.histogramViewAtIndex(m_selectedSeries)->isMainViewSelected() && (event == Ion::Events::Left || event == Ion::Events::Right)) {
int direction = event == Ion::Events::Left ? -1 : 1;
if (moveSelection(direction)) {
reloadBannerView();
@@ -117,16 +179,16 @@ void HistogramController::didBecomeFirstResponder() {
initBarSelection();
reloadBannerView();
}
if (!m_view.isMainViewSelected()) {
if (!m_view.histogramViewAtIndex(m_selectedSeries)->isMainViewSelected()) {
header()->setSelectedButton(0);
} else {
m_view.setHighlight(m_store->startOfBarAtIndex(0, *m_selectedBarIndex), m_store->endOfBarAtIndex(0, *m_selectedBarIndex)); //TODO
m_view.histogramViewAtIndex(m_selectedSeries)->setHighlight(m_store->startOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex), m_store->endOfBarAtIndex(0, *m_selectedBarIndex));
}
}
void HistogramController::willExitResponderChain(Responder * nextFirstResponder) {
if (nextFirstResponder == nullptr || nextFirstResponder == tabController()) {
m_view.selectMainView(false);
m_view.histogramViewAtIndex(m_selectedSeries)->selectMainView(false);
header()->setSelectedButton(-1);
m_view.reload();
}
@@ -147,13 +209,13 @@ void HistogramController::reloadBannerView() {
numberOfChar += legendLength;
// Add lower bound
double lowerBound = m_store->startOfBarAtIndex(0, *m_selectedBarIndex); //TODO
double lowerBound = m_store->startOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex);
numberOfChar += PrintFloat::convertFloatToText<double>(lowerBound, buffer+numberOfChar, PrintFloat::bufferSizeForFloatsWithPrecision(Constant::LargeNumberOfSignificantDigits), Constant::LargeNumberOfSignificantDigits);
buffer[numberOfChar++] = ';';
// Add upper bound
double upperBound = m_store->endOfBarAtIndex(0, *m_selectedBarIndex); //TODO
double upperBound = m_store->endOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex);
numberOfChar += PrintFloat::convertFloatToText<double>(upperBound, buffer+numberOfChar, PrintFloat::bufferSizeForFloatsWithPrecision(Constant::LargeNumberOfSignificantDigits), Constant::LargeNumberOfSignificantDigits);
buffer[numberOfChar++] = '[';
@@ -163,7 +225,7 @@ void HistogramController::reloadBannerView() {
buffer[numberOfChar++] = ' ';
}
buffer[k_maxIntervalLegendLength] = 0;
m_bannerView.setLegendAtIndex(buffer, 1);
m_view.bannerView()->setLegendAtIndex(buffer, 1);
// Add Size Data
numberOfChar = 0;
@@ -171,14 +233,14 @@ void HistogramController::reloadBannerView() {
legendLength = strlen(legend);
strlcpy(buffer, legend, legendLength+1);
numberOfChar += legendLength;
double size = m_store->heightOfBarAtIndex(0, *m_selectedBarIndex); //TODO
double size = m_store->heightOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex);
numberOfChar += PrintFloat::convertFloatToText<double>(size, buffer+numberOfChar, PrintFloat::bufferSizeForFloatsWithPrecision(Constant::LargeNumberOfSignificantDigits), Constant::LargeNumberOfSignificantDigits);
// Padding
for (int i = numberOfChar; i < k_maxLegendLength; i++) {
buffer[numberOfChar++] = ' ';
}
buffer[k_maxLegendLength] = 0;
m_bannerView.setLegendAtIndex(buffer, 3);
m_view.bannerView()->setLegendAtIndex(buffer, 3);
// Add Frequency Data
numberOfChar = 0;
@@ -186,14 +248,14 @@ void HistogramController::reloadBannerView() {
legendLength = strlen(legend);
strlcpy(buffer, legend, legendLength+1);
numberOfChar += legendLength;
double frequency = size/m_store->sumOfOccurrences(0); //TODO
double frequency = size/m_store->sumOfOccurrences(m_selectedSeries);
numberOfChar += PrintFloat::convertFloatToText<double>(frequency, buffer+numberOfChar, PrintFloat::bufferSizeForFloatsWithPrecision(Constant::LargeNumberOfSignificantDigits), Constant::LargeNumberOfSignificantDigits);
// Padding
for (int i = numberOfChar; i < k_maxLegendLength; i++) {
buffer[numberOfChar++] = ' ';
}
buffer[k_maxLegendLength] = 0;
m_bannerView.setLegendAtIndex(buffer, 5);
m_view.bannerView()->setLegendAtIndex(buffer, 5);
}
bool HistogramController::moveSelection(int deltaIndex) {
@@ -201,16 +263,15 @@ bool HistogramController::moveSelection(int deltaIndex) {
if (deltaIndex > 0) {
do {
newSelectedBarIndex++;
} while (m_store->heightOfBarAtIndex(0, newSelectedBarIndex) == 0 && newSelectedBarIndex < m_store->numberOfBars(0)); //TODO
} else {
} while (m_store->heightOfBarAtIndex(m_selectedSeries, newSelectedBarIndex) == 0 && newSelectedBarIndex < m_store->numberOfBars(m_selectedSeries)); } else {
do {
newSelectedBarIndex--;
} while (m_store->heightOfBarAtIndex(0, newSelectedBarIndex) == 0 && newSelectedBarIndex >= 0); //TODO
} while (m_store->heightOfBarAtIndex(m_selectedSeries, newSelectedBarIndex) == 0 && newSelectedBarIndex >= 0);
}
if (newSelectedBarIndex >= 0 && newSelectedBarIndex < m_store->numberOfBars(0) && *m_selectedBarIndex != newSelectedBarIndex) {//TODO
if (newSelectedBarIndex >= 0 && newSelectedBarIndex < m_store->numberOfBars(m_selectedSeries) && *m_selectedBarIndex != newSelectedBarIndex) {
*m_selectedBarIndex = newSelectedBarIndex;
m_view.setHighlight(m_store->startOfBarAtIndex(0, *m_selectedBarIndex), m_store->endOfBarAtIndex(0, *m_selectedBarIndex)); //TODO
m_store->scrollToSelectedBarIndex(0, *m_selectedBarIndex);//TODO
m_view.histogramViewAtIndex(m_selectedSeries)->setHighlight(m_store->startOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex), m_store->endOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex));
m_store->scrollToSelectedBarIndex(m_selectedSeries, *m_selectedBarIndex);
return true;
}
return false;
@@ -218,7 +279,7 @@ bool HistogramController::moveSelection(int deltaIndex) {
void HistogramController::initRangeParameters() {
float min = m_store->firstDrawnBarAbscissa();
float max = m_store->maxValue(0); //TODO
float max = m_store->maxValue(m_selectedSeries);
float barWidth = m_store->barWidth();
float xMin = min;
float xMax = max + barWidth;
@@ -232,22 +293,27 @@ void HistogramController::initRangeParameters() {
}
m_store->setXMin(xMin - Store::k_displayLeftMarginRatio*(xMax-xMin));
m_store->setXMax(xMax + Store::k_displayRightMarginRatio*(xMax-xMin));
initYRangeParameters(m_selectedSeries);
}
void HistogramController::initYRangeParameters(int series) {
float yMax = -FLT_MAX;
for (int index = 0; index < m_store->numberOfBars(0); index++) { //TODO
float size = m_store->heightOfBarAtIndex(0, index); //TODO
for (int index = 0; index < m_store->numberOfBars(series); index++) {
float size = m_store->heightOfBarAtIndex(series, index);
if (size > yMax) {
yMax = size;
}
}
yMax = yMax/m_store->sumOfOccurrences(0); //TODO
yMax = yMax/m_store->sumOfOccurrences(series);
yMax = yMax < 0 ? 1 : yMax;
m_store->setYMin(-Store::k_displayBottomMarginRatio*yMax);
m_store->setYMax(yMax*(1.0f+Store::k_displayTopMarginRatio));
}
void HistogramController::initBarParameters() {
float min = m_store->minValue(0); //TODO
float max = m_store->maxValue(0); //TODO
float min = m_store->minValue(m_selectedSeries);
float max = m_store->maxValue(m_selectedSeries);
max = min >= max ? min + std::pow(10.0f, std::floor(std::log10(std::fabs(min)))-1.0f) : max;
m_store->setFirstDrawnBarAbscissa(min);
float barWidth = m_store->computeGridUnit(CurveViewRange::Axis::X, min, max);
@@ -259,18 +325,18 @@ void HistogramController::initBarParameters() {
void HistogramController::initBarSelection() {
*m_selectedBarIndex = 0;
while ((m_store->heightOfBarAtIndex(0, *m_selectedBarIndex) == 0 || //TODO
m_store->startOfBarAtIndex(0, *m_selectedBarIndex) < m_store->firstDrawnBarAbscissa()) && *m_selectedBarIndex < m_store->numberOfBars(0)) {//TODO
while ((m_store->heightOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex) == 0 ||
m_store->startOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex) < m_store->firstDrawnBarAbscissa()) && *m_selectedBarIndex < m_store->numberOfBars(m_selectedSeries)) {
*m_selectedBarIndex = *m_selectedBarIndex+1;
}
if (*m_selectedBarIndex >= m_store->numberOfBars(0)) { //TODO
if (*m_selectedBarIndex >= m_store->numberOfBars(m_selectedSeries)) {
/* No bar is after m_firstDrawnBarAbscissa, so we select the first bar */
*m_selectedBarIndex = 0;
while (m_store->heightOfBarAtIndex(0, *m_selectedBarIndex) == 0 && *m_selectedBarIndex < m_store->numberOfBars(0)) { //TODO
while (m_store->heightOfBarAtIndex(m_selectedSeries, *m_selectedBarIndex) == 0 && *m_selectedBarIndex < m_store->numberOfBars(m_selectedSeries)) {
*m_selectedBarIndex = *m_selectedBarIndex+1;
}
}
m_store->scrollToSelectedBarIndex(0, *m_selectedBarIndex);//TODO
m_store->scrollToSelectedBarIndex(m_selectedSeries, *m_selectedBarIndex);
}
}