mirror of
https://github.com/jkriege2/JKQtPlotter.git
synced 2025-01-15 02:02:10 +08:00
f40bb2010d
NEW: autoscaling for barcharts works now, also when stacked and unstacked charts are combined in one plot NEW: proper styling for financial graphs in style.ini-files REWORKED: separation and gruping factor for barcharts on autoscaling doc update
474 lines
14 KiB
C++
474 lines
14 KiB
C++
/*
|
|
Copyright (c) 2008-2024 Jan W. Krieger (<jan@jkrieger.de>)
|
|
|
|
|
|
|
|
This software is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License (LGPL) as published by
|
|
the Free Software Foundation, either version 2.1 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License (LGPL) for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License (LGPL)
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
|
|
#include "jkqtplotter/graphs/jkqtpbarchartbase.h"
|
|
#include "jkqtplotter/jkqtpbaseplotter.h"
|
|
#include <stdlib.h>
|
|
#include <QDebug>
|
|
#include <iostream>
|
|
#include "jkqtplotter/jkqtptools.h"
|
|
#include "jkqtplotter/jkqtplotter.h"
|
|
|
|
#define SmallestGreaterZeroCompare_xvsgz() if ((xvsgz>10.0*DBL_MIN)&&((smallestGreaterZero<10.0*DBL_MIN) || (xvsgz<smallestGreaterZero))) smallestGreaterZero=xvsgz;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
JKQTPBarGraphBase::JKQTPBarGraphBase(JKQTBasePlotter* parent):
|
|
JKQTPXYBaselineGraph(parent),
|
|
width(0.9),
|
|
shift(0),
|
|
rectRadiusAtValue(0),
|
|
rectRadiusAtBaseline(0),
|
|
m_drawBaseline(false),
|
|
m_stackSeparation(1),
|
|
m_fillMode(FillMode::SingleFilling),
|
|
m_lineColorDerivationModeForSpecialFill(parent->getCurrentPlotterStyle().graphsStyle.barchartStyle.graphColorDerivationMode),
|
|
m_useCustomDrawFunctor(false)
|
|
{
|
|
initFillStyle(parent, parentPlotStyle, JKQTPPlotStyleType::Barchart);
|
|
initLineStyle(parent, parentPlotStyle, JKQTPPlotStyleType::Barchart);
|
|
m_fillStyleBelow.initFillStyleInvertedColor(this);
|
|
m_baselineStyle.setLineStyle(Qt::SolidLine);
|
|
if (parent) {
|
|
m_baselineStyle.setLineColor(parent->getCurrentPlotterStyle().xAxisStyle.colorZeroAxis);
|
|
m_baselineStyle.setLineWidth(parent->getCurrentPlotterStyle().xAxisStyle.lineWidthZeroAxis);
|
|
rectRadiusAtBaseline= parent->getCurrentPlotterStyle().graphsStyle.barchartStyle.defaultRectRadiusAtBaseline;
|
|
rectRadiusAtValue= parent->getCurrentPlotterStyle().graphsStyle.barchartStyle.defaultRectRadiusAtValue;
|
|
m_drawBaseline=parent->getCurrentPlotterStyle().graphsStyle.barchartStyle.drawBaseline;
|
|
m_stackSeparation=parent->getCurrentPlotterStyle().graphsStyle.barchartStyle.stackSeparation;
|
|
}
|
|
}
|
|
|
|
|
|
JKQTPBarGraphBase::JKQTPBarGraphBase(JKQTPlotter* parent):
|
|
JKQTPBarGraphBase(parent->getPlotter())
|
|
{
|
|
}
|
|
|
|
void JKQTPBarGraphBase::drawKeyMarker(JKQTPEnhancedPainter& painter, const QRectF& r) {
|
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
QPen p=getLinePenForRects(painter, parent);
|
|
QBrush b=getFillBrush(painter, parent);
|
|
QRectF rect=r;
|
|
rect.setWidth(rect.width()-p.widthF());
|
|
rect.setHeight(rect.height()-p.widthF());
|
|
rect.setX(rect.x()+p.widthF()/2.0);
|
|
rect.setY(rect.y()+p.widthF()/2.0);
|
|
|
|
//int y=rect.top()+rect.height()/2.0;
|
|
painter.setPen(p);
|
|
painter.setBrush(b);
|
|
painter.drawRect(rect);
|
|
|
|
}
|
|
|
|
QColor JKQTPBarGraphBase::getKeyLabelColor() const {
|
|
return getFillColor();
|
|
}
|
|
|
|
|
|
void JKQTPBarGraphBase::autoscaleBarWidthAndShift(double maxWidth, double shrinkFactor)
|
|
{
|
|
if (parent) {
|
|
auto isStackBase=[](JKQTPBarGraphBase* other) {
|
|
const auto vg=dynamic_cast<JKQTPBarGraphStackInternalInterface*>(other);
|
|
return (vg && !other->hasStackParent());
|
|
};
|
|
auto isStackable=[](JKQTPBarGraphBase* other) {
|
|
return dynamic_cast<JKQTPBarGraphStackInternalInterface*>(other);
|
|
};
|
|
double cntH=0;
|
|
for (size_t i=0; i<parent->getGraphCount(); i++) {
|
|
JKQTPPlotElement* g=parent->getGraph(i);
|
|
JKQTPBarGraphBase* gb=qobject_cast<JKQTPBarGraphBase*>(g);
|
|
if (gb && considerForAutoscaling(gb) && (isStackBase(gb) || !isStackable(gb))) {
|
|
cntH++;
|
|
}
|
|
|
|
}
|
|
|
|
double widthH=1.0/cntH*maxWidth*shrinkFactor;
|
|
double dH=maxWidth/(cntH);
|
|
double h=0.1+dH/2.0;
|
|
for (size_t i=0; i<parent->getGraphCount(); i++) {
|
|
JKQTPPlotElement* g=parent->getGraph(i);
|
|
JKQTPBarGraphBase* gb=qobject_cast<JKQTPBarGraphBase*>(g);
|
|
if (gb && considerForAutoscaling(gb) && (isStackBase(gb) || !isStackable(gb))) {
|
|
if (cntH>1) {
|
|
gb->width=widthH;
|
|
gb->shift=h-0.5;
|
|
h=h+dH;
|
|
} else {
|
|
|
|
gb->width=maxWidth;
|
|
gb->shift=0.0;
|
|
}
|
|
}
|
|
|
|
}
|
|
for (size_t i=0; i<parent->getGraphCount(); i++) {
|
|
JKQTPPlotElement* g=parent->getGraph(i);
|
|
JKQTPBarGraphStackInternalInterface* gbo=dynamic_cast<JKQTPBarGraphStackInternalInterface*>(g);
|
|
JKQTPBarGraphBase* gb=qobject_cast<JKQTPBarGraphBase*>(g);
|
|
if (gbo && gb) {
|
|
auto stackBottom=gbo->getBottomOfStack();
|
|
if (g!=stackBottom && stackBottom) {
|
|
gb->width=stackBottom->width;
|
|
gb->shift=stackBottom->shift;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
void JKQTPBarGraphBase::autoscaleBarWidthAndShiftSeparatedGroups(double groupWidth) {
|
|
autoscaleBarWidthAndShift(groupWidth);
|
|
}
|
|
|
|
|
|
|
|
void JKQTPBarGraphBase::setColor(QColor c)
|
|
{
|
|
setFillColor(c);
|
|
setLineColor(JKQTPGetDerivedColor(parent->getCurrentPlotterStyle().graphsStyle.barchartStyle.graphColorDerivationMode, c));
|
|
c.setAlphaF(0.5);
|
|
setHighlightingLineColor(c);
|
|
m_fillStyleBelow.initFillStyleInvertedColor(this);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setBarPositionColumn(int column)
|
|
{
|
|
setKeyColumn(column);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setBarPositionColumn(size_t column)
|
|
{
|
|
setKeyColumn(static_cast<int>(column));
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setBarHeightColumn(int column)
|
|
{
|
|
setValueColumn(column);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setBarHeightColumn(size_t column)
|
|
{
|
|
setValueColumn(static_cast<int>(column));
|
|
}
|
|
|
|
JKQTPBarGraphBase::FillBrushFunctor JKQTPBarGraphBase::constructFillBrushFunctor() const
|
|
{
|
|
if (m_fillMode==FillMode::FunctorFilling) return m_fillBrushFunctor;
|
|
if (m_fillMode==FillMode::TwoColorFilling) return [](double , double value, JKQTPEnhancedPainter &painter, JKQTPBarGraphBase* graph) {
|
|
if (value<graph->getBaseline()) return graph->fillStyleBelow().getFillBrush(painter, graph->getParent());
|
|
return graph->getFillBrush(painter, graph->getParent());
|
|
};
|
|
else return [](double, double, JKQTPEnhancedPainter &painter, JKQTPBarGraphBase* graph) {
|
|
return graph->getFillBrush(painter, graph->getParent());
|
|
};
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setShift(double __value)
|
|
{
|
|
this->shift = __value;
|
|
}
|
|
|
|
double JKQTPBarGraphBase::getShift() const
|
|
{
|
|
return this->shift;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setWidth(double __value)
|
|
{
|
|
this->width = __value;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setDrawBaseline(bool __value)
|
|
{
|
|
m_drawBaseline=__value;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setStackSeparation(double __value)
|
|
{
|
|
m_stackSeparation=__value;
|
|
}
|
|
|
|
bool JKQTPBarGraphBase::getDrawBaseline() const
|
|
{
|
|
return this->m_drawBaseline;
|
|
}
|
|
|
|
double JKQTPBarGraphBase::getStackSeparation() const
|
|
{
|
|
return m_stackSeparation;
|
|
}
|
|
|
|
JKQTPGraphLineStyleMixin &JKQTPBarGraphBase::baselineStyle()
|
|
{
|
|
return m_baselineStyle;
|
|
}
|
|
|
|
const JKQTPGraphLineStyleMixin &JKQTPBarGraphBase::baselineStyle() const
|
|
{
|
|
return m_baselineStyle;
|
|
}
|
|
|
|
double JKQTPBarGraphBase::getWidth() const
|
|
{
|
|
return this->width;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setFillColor_and_darkenedColor(QColor fill, int colorDarker)
|
|
{
|
|
setFillColor(fill);
|
|
setLineColor(fill.darker(colorDarker));
|
|
}
|
|
|
|
int JKQTPBarGraphBase::getBarPositionColumn() const
|
|
{
|
|
return getKeyColumn();
|
|
}
|
|
|
|
int JKQTPBarGraphBase::getBarHeightColumn() const
|
|
{
|
|
return getValueColumn();
|
|
}
|
|
|
|
JKQTPGraphFillStyleMixin &JKQTPBarGraphBase::fillStyleBelow()
|
|
{
|
|
return m_fillStyleBelow;
|
|
}
|
|
|
|
const JKQTPGraphFillStyleMixin &JKQTPBarGraphBase::fillStyleBelow() const
|
|
{
|
|
return m_fillStyleBelow;
|
|
}
|
|
|
|
JKQTPBarGraphBase::FillMode JKQTPBarGraphBase::getFillMode() const
|
|
{
|
|
return m_fillMode;
|
|
}
|
|
|
|
double JKQTPBarGraphBase::getRectRadiusAtValue() const
|
|
{
|
|
return rectRadiusAtValue;
|
|
}
|
|
|
|
double JKQTPBarGraphBase::getRectRadiusAtBaseline() const
|
|
{
|
|
return rectRadiusAtBaseline;
|
|
}
|
|
|
|
JKQTPBarGraphBase::FillBrushFunctor &JKQTPBarGraphBase::getFillBrushFunctor() {
|
|
return m_fillBrushFunctor;
|
|
}
|
|
|
|
const JKQTPBarGraphBase::FillBrushFunctor &JKQTPBarGraphBase::getFillBrushFunctor() const {
|
|
return m_fillBrushFunctor;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setFillMode(FillMode mode)
|
|
{
|
|
m_fillMode=mode;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setFillBrushFunctor(const FillBrushFunctor &f) {
|
|
m_fillBrushFunctor=f;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setFillBrushFunctor(FillBrushFunctor &&f) {
|
|
m_fillBrushFunctor=std::forward<FillBrushFunctor>(f);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setFillBrushFunctor(const SimpleFillBrushFunctor &f) {
|
|
m_fillBrushFunctor=SimpleFillBrushFunctorAdaptor(f);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setFillBrushFunctor(SimpleFillBrushFunctor &&f) {
|
|
m_fillBrushFunctor=SimpleFillBrushFunctorAdaptor(std::forward<SimpleFillBrushFunctor>(f));
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setCustomDrawingFunctor(CustomDrawingFunctor &&f)
|
|
{
|
|
m_customDrawFunctor=std::forward<CustomDrawingFunctor>(f);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setCustomDrawingFunctor(const CustomDrawingFunctor &f)
|
|
{
|
|
m_customDrawFunctor=f;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setUseCustomDrawFunctor(bool enabled)
|
|
{
|
|
m_useCustomDrawFunctor=enabled;
|
|
}
|
|
|
|
double JKQTPBarGraphBase::getParentStackedMax(int /*index*/) const
|
|
{
|
|
return getBaseline();
|
|
}
|
|
|
|
bool JKQTPBarGraphBase::hasStackParent() const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool JKQTPBarGraphBase::getValuesMinMax(double &mmin, double &mmax, double &smallestGreaterZero)
|
|
{
|
|
mmin=0;
|
|
mmax=0;
|
|
smallestGreaterZero=0;
|
|
if (getBaseline()>0) {
|
|
smallestGreaterZero=getBaseline();
|
|
mmin=getBaseline();
|
|
mmax=getBaseline();
|
|
}
|
|
|
|
if (getBarPositionColumn()<0 || getBarHeightColumn()<0) return false;
|
|
|
|
const size_t datacol=static_cast<size_t>(getBarHeightColumn());
|
|
|
|
if (parent==nullptr) return false;
|
|
|
|
const JKQTPDatastore* datastore=parent->getDatastore();
|
|
int imin=0, imax=0;
|
|
if (getIndexRange(imin, imax)) {
|
|
|
|
|
|
for (int i=imin; i<imax; i++) {
|
|
double stack=0;
|
|
double yv=getBaseline();
|
|
const double boxstart=getParentStackedMax(i);
|
|
if (hasStackParent()) {
|
|
stack=boxstart;
|
|
yv=stack;
|
|
}
|
|
if (JKQTPIsOKFloat(yv)) {
|
|
if (yv>mmax) mmax=yv;
|
|
if (yv<mmin) mmin=yv;
|
|
double xvsgz;
|
|
xvsgz=yv; SmallestGreaterZeroCompare_xvsgz();
|
|
}
|
|
yv=stack+datastore->get(datacol,static_cast<size_t>(i));
|
|
if (JKQTPIsOKFloat(yv)) {
|
|
if (yv>mmax) mmax=yv;
|
|
if (yv<mmin) mmin=yv;
|
|
double xvsgz;
|
|
xvsgz=yv; SmallestGreaterZeroCompare_xvsgz();
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool JKQTPBarGraphBase::getPositionsMinMax(double &mmin, double &mmax, double &smallestGreaterZero)
|
|
{
|
|
bool start=true;
|
|
mmin=0;
|
|
mmax=0;
|
|
smallestGreaterZero=0;
|
|
|
|
if (getBarPositionColumn()<0 || getBarHeightColumn()<0) return false;
|
|
|
|
const size_t poscol=static_cast<size_t>(getBarPositionColumn());
|
|
|
|
if (parent==nullptr) return false;
|
|
|
|
const JKQTPDatastore* datastore=parent->getDatastore();
|
|
int imin=0, imax=0;
|
|
if (getIndexRange(imin, imax)) {
|
|
for (int i=imin; i<imax; i++) {
|
|
double xv=datastore->get(poscol,static_cast<size_t>(i));
|
|
int sr=datastore->getNextLowerIndex(poscol, i);
|
|
int lr=datastore->getNextHigherIndex(poscol, i);
|
|
double delta, deltap, deltam;
|
|
|
|
if (sr<0 && lr<0) { // only one x-value
|
|
deltam=0.5;
|
|
deltap=0.5;
|
|
} else if (lr<0) { // the right-most x-value
|
|
deltap=deltam=fabs(xv-datastore->get(poscol,sr))/2.0;
|
|
} else if (sr<0) { // the left-most x-value
|
|
deltam=deltap=fabs(datastore->get(poscol,lr)-xv)/2.0;
|
|
} else {
|
|
deltam=fabs(xv-datastore->get(poscol,sr))/2.0;
|
|
deltap=fabs(datastore->get(poscol,lr)-xv)/2.0;
|
|
}
|
|
delta=deltap+deltam;
|
|
|
|
if (JKQTPIsOKFloat(xv) && JKQTPIsOKFloat(delta) ) {
|
|
|
|
if (start || xv+shift*delta+width*delta/2.0>mmax) mmax=xv+shift*delta+width*delta/2.0;
|
|
if (start || xv+shift*delta-width*delta/2.0<mmin) mmin=xv+shift*delta-width*delta/2.0;
|
|
double xvsgz;
|
|
xvsgz=xv+shift*delta+width*delta/2.0; SmallestGreaterZeroCompare_xvsgz();
|
|
xvsgz=xv+shift*delta-width*delta/2.0; SmallestGreaterZeroCompare_xvsgz();
|
|
start=false;
|
|
}
|
|
}
|
|
return !start;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
void JKQTPBarGraphBase::setRectRadiusAtValue(double __value)
|
|
{
|
|
rectRadiusAtValue=__value;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setRectRadiusAtBaseline(double __value)
|
|
{
|
|
rectRadiusAtBaseline=__value;
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setRectRadius(double all)
|
|
{
|
|
setRectRadiusAtValue(all);
|
|
setRectRadiusAtBaseline(all);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setRectRadius(double atValue, double atBaseline)
|
|
{
|
|
setRectRadiusAtValue(atValue);
|
|
setRectRadiusAtBaseline(atBaseline);
|
|
}
|
|
|
|
void JKQTPBarGraphBase::setLineColorDerivationModeForSpecialFill(const JKQTPColorDerivationMode &m)
|
|
{
|
|
m_lineColorDerivationModeForSpecialFill=m;
|
|
}
|
|
|
|
JKQTPColorDerivationMode JKQTPBarGraphBase::getLineColorDerivationModeForSpecialFill() const
|
|
{
|
|
return m_lineColorDerivationModeForSpecialFill;
|
|
}
|
|
|
|
bool JKQTPBarGraphBase::usesCustomDrawFunctor() const
|
|
{
|
|
return m_useCustomDrawFunctor;
|
|
}
|