mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-19 00:37:25 +01:00
[apps/statistics] Show several histograms
This commit is contained in:
@@ -55,7 +55,7 @@ App::App(Container * container, Snapshot * snapshot) :
|
||||
m_boxController(&m_boxAlternateEmptyViewController, &m_boxHeader, snapshot->store(), snapshot->selectedBoxQuantile()),
|
||||
m_boxAlternateEmptyViewController(&m_boxHeader, &m_boxController, &m_boxController),
|
||||
m_boxHeader(&m_tabViewController, &m_boxAlternateEmptyViewController, &m_boxController),
|
||||
m_histogramController(&m_histogramAlternateEmptyViewController, &m_histogramHeader, snapshot->store(), snapshot->storeVersion(), snapshot->barVersion(), snapshot->rangeVersion(), snapshot->selectedHistogramBarIndex()),
|
||||
m_histogramController(&m_histogramAlternateEmptyViewController, &m_histogramHeader, snapshot->store(), 0, snapshot->storeVersion(), snapshot->barVersion(), snapshot->rangeVersion(), snapshot->selectedHistogramBarIndex()),
|
||||
m_histogramAlternateEmptyViewController(&m_histogramHeader, &m_histogramController, &m_histogramController),
|
||||
m_histogramHeader(&m_histogramStackViewController, &m_histogramAlternateEmptyViewController, &m_histogramController),
|
||||
m_histogramStackViewController(&m_tabViewController, &m_histogramHeader),
|
||||
|
||||
@@ -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);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -13,9 +13,10 @@ namespace Statistics {
|
||||
class HistogramController : public ViewController, public ButtonRowDelegate, public AlternateEmptyViewDelegate {
|
||||
|
||||
public:
|
||||
HistogramController(Responder * parentResponder, ButtonRowController * header, Store * store, uint32_t * m_storeVersion, uint32_t * m_barVersion, uint32_t * m_rangeVersion, int * m_selectedBarIndex);
|
||||
HistogramController(Responder * parentResponder, ButtonRowController * header, Store * store, int series, uint32_t * m_storeVersion, uint32_t * m_barVersion, uint32_t * m_rangeVersion, int * m_selectedBarIndex);
|
||||
StackViewController * stackController();
|
||||
HistogramParameterController * histogramParameterController() { return &m_histogramParameterController; }
|
||||
void setCurrentDrawnSeries(int series);
|
||||
|
||||
// ButtonRowDelegate
|
||||
int numberOfButtons(ButtonRowController::Position) const override;
|
||||
@@ -40,20 +41,38 @@ private:
|
||||
constexpr static int k_maxIntervalLegendLength = 33;
|
||||
constexpr static int k_maxLegendLength = 13;
|
||||
constexpr static int k_maxNumberOfCharacters = 30;
|
||||
class ContentView : public View {
|
||||
public:
|
||||
ContentView(HistogramController * controller, Store * store);
|
||||
void reload() {}
|
||||
HistogramView * histogramViewAtIndex(int index);
|
||||
int seriesOfSubviewAtIndex(int index);
|
||||
HistogramBannerView * bannerView() { return &m_bannerView; }
|
||||
void layoutSubviews() override;
|
||||
private:
|
||||
int numberOfSubviews() const override;
|
||||
View * subviewAtIndex(int index) override;
|
||||
HistogramView m_histogramView1;
|
||||
HistogramView m_histogramView2;
|
||||
HistogramView m_histogramView3;
|
||||
HistogramBannerView m_bannerView;
|
||||
Store * m_store; // TODO Do not duplicate this pointer
|
||||
};
|
||||
Responder * tabController() const;
|
||||
void reloadBannerView();
|
||||
void initRangeParameters();
|
||||
void initYRangeParameters(int series);
|
||||
void initBarParameters();
|
||||
void initBarSelection();
|
||||
// return true if the window has scrolled
|
||||
bool moveSelection(int deltaIndex);
|
||||
HistogramBannerView m_bannerView;
|
||||
HistogramView m_view;
|
||||
Button m_settingButton;
|
||||
Store * m_store;
|
||||
ContentView m_view;
|
||||
uint32_t * m_storeVersion;
|
||||
uint32_t * m_barVersion;
|
||||
uint32_t * m_rangeVersion;
|
||||
int m_selectedSeries;
|
||||
int * m_selectedBarIndex;
|
||||
HistogramParameterController m_histogramParameterController;
|
||||
};
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
#include "histogram_view.h"
|
||||
#include "histogram_controller.h"
|
||||
#include <assert.h>
|
||||
#include <cmath>
|
||||
|
||||
@@ -6,12 +7,14 @@ using namespace Shared;
|
||||
|
||||
namespace Statistics {
|
||||
|
||||
HistogramView::HistogramView(Store * store, BannerView * bannerView) :
|
||||
HistogramView::HistogramView(HistogramController * controller, Store * store, int series, Shared::BannerView * bannerView) :
|
||||
CurveView(store, nullptr, bannerView, nullptr),
|
||||
m_controller(controller),
|
||||
m_store(store),
|
||||
m_labels{},
|
||||
m_highlightedBarStart(NAN),
|
||||
m_highlightedBarEnd(NAN)
|
||||
m_highlightedBarEnd(NAN),
|
||||
m_series(series)
|
||||
{
|
||||
}
|
||||
|
||||
@@ -27,13 +30,14 @@ void HistogramView::reload() {
|
||||
}
|
||||
|
||||
void HistogramView::drawRect(KDContext * ctx, KDRect rect) const {
|
||||
m_controller->setCurrentDrawnSeries(m_series);
|
||||
ctx->fillRect(rect, KDColorWhite);
|
||||
drawAxes(ctx, rect, Axis::Horizontal);
|
||||
drawLabels(ctx, rect, Axis::Horizontal, false);
|
||||
/* We memoize the total size to avoid recomputing it in double precision at
|
||||
* every call to EvaluateHistogramAtAbscissa() */
|
||||
float totalSize = m_store->sumOfColumn(0, 1); // TODO Draw all the histograms
|
||||
float context[] = {totalSize, 0};
|
||||
float totalSize = m_store->sumOfOccurrences(m_series);
|
||||
float context[] = {totalSize, static_cast<float>(m_series)};
|
||||
if (isMainViewSelected()) {
|
||||
drawHistogram(ctx, rect, EvaluateHistogramAtAbscissa, m_store, context, m_store->firstDrawnBarAbscissa(), m_store->barWidth(), true, Palette::Select, Palette::YellowDark, m_highlightedBarStart, m_highlightedBarEnd);
|
||||
} else {
|
||||
|
||||
@@ -8,19 +8,24 @@
|
||||
|
||||
namespace Statistics {
|
||||
|
||||
class HistogramController;
|
||||
|
||||
class HistogramView : public Shared::CurveView {
|
||||
public:
|
||||
HistogramView(Store * store, Shared::BannerView * bannerView);
|
||||
HistogramView(HistogramController * controller, Store * store, int series, Shared::BannerView * bannerView);
|
||||
int series() const { return m_series; }
|
||||
void reload() override;
|
||||
void drawRect(KDContext * ctx, KDRect rect) const override;
|
||||
void setHighlight(float start, float end);
|
||||
private:
|
||||
char * label(Axis axis, int index) const override;
|
||||
HistogramController * m_controller;
|
||||
Store * m_store;
|
||||
char m_labels[k_maxNumberOfXLabels][Poincare::PrintFloat::bufferSizeForFloatsWithPrecision(Constant::ShortNumberOfSignificantDigits)];
|
||||
static float EvaluateHistogramAtAbscissa(float abscissa, void * model, void * context);
|
||||
float m_highlightedBarStart;
|
||||
float m_highlightedBarEnd;
|
||||
int m_series;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
@@ -78,13 +78,28 @@ bool Store::scrollToSelectedBarIndex(int series, int index) {
|
||||
|
||||
bool Store::isEmpty() {
|
||||
for (int i = 0; i < k_numberOfSeries; i ++) {
|
||||
if (sumOfOccurrences(i) > 0) {
|
||||
if (!seriesIsEmpty(i)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
int Store::numberOfNonEmptySeries() {
|
||||
int result = 0;
|
||||
for (int i = 0; i < k_numberOfSeries; i ++) {
|
||||
if (!seriesIsEmpty(i)) {
|
||||
result++;
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
bool Store::seriesIsEmpty(int i) {
|
||||
return sumOfOccurrences(i) == 0;
|
||||
}
|
||||
|
||||
|
||||
/* Calculation */
|
||||
|
||||
double Store::sumOfOccurrences(int series) {
|
||||
|
||||
@@ -23,6 +23,8 @@ public:
|
||||
// return true if the window has scrolled
|
||||
bool scrollToSelectedBarIndex(int series, int index);
|
||||
bool isEmpty();
|
||||
int numberOfNonEmptySeries();
|
||||
bool seriesIsEmpty(int i);
|
||||
|
||||
// Calculation
|
||||
double sumOfOccurrences(int series);
|
||||
@@ -46,6 +48,7 @@ public:
|
||||
constexpr static float k_displayRightMarginRatio = 0.04f;
|
||||
constexpr static float k_displayBottomMarginRatio = 0.4f;
|
||||
constexpr static float k_displayLeftMarginRatio = 0.04f;
|
||||
|
||||
private:
|
||||
double defaultValue(int series, int i, int j) override;
|
||||
double sumOfValuesBetween(int series, double x1, double x2);
|
||||
|
||||
Reference in New Issue
Block a user