mirror of
https://github.com/jkriege2/JKQtPlotter.git
synced 2025-01-13 01:10:33 +08:00
20352285ca
new: notched boxplots new: example for styling boxplots
441 lines
22 KiB
C++
441 lines
22 KiB
C++
/*
|
|
Copyright (c) 2008-2019 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 <QString>
|
|
#include <QPainter>
|
|
#include "jkqtplottertools/jkqtptools.h"
|
|
#include "jkqtplottertools/jkqtp_imexport.h"
|
|
#include "jkqtplotter/jkqtpgraphsbase.h"
|
|
#include "jkqtplotter/jkqtpgraphsbasestylingmixins.h"
|
|
|
|
#ifndef jkqtpgraphsboxplotstylingmixins_H
|
|
#define jkqtpgraphsboxplotstylingmixins_H
|
|
|
|
|
|
|
|
|
|
|
|
/*! \brief Styling Mix-In for Boxplots
|
|
\ingroup jkqtplotter_basegraphs_stylemixins
|
|
|
|
|
|
\image html plot_boxplothorizontalelement.png
|
|
|
|
The example \ref JKQTPlotterBoxplotStyling discusses different options to style boxplots:
|
|
|
|
\image html test_styledboxplot.png
|
|
*/
|
|
class JKQTP_LIB_EXPORT JKQTPGraphBoxplotStyleMixin: public JKQTPGraphLineStyleMixin, public JKQTPGraphFillStyleMixin {
|
|
Q_GADGET
|
|
public:
|
|
/** \brief class constructor */
|
|
JKQTPGraphBoxplotStyleMixin();
|
|
|
|
void initBoxplotStyle(JKQTBasePlotter* parent, int &parentPlotStyle);
|
|
/*! \copydoc boxWidthAbsolute
|
|
\see see boxWidthAbsolute for details */
|
|
void setBoxWidthAbsolute(double __value);
|
|
/*! \copydoc boxWidthAbsolute
|
|
\see see boxWidthAbsolute for details */
|
|
double getBoxWidthAbsolute() const;
|
|
|
|
/*! \copydoc drawBox
|
|
\see see drawBox for details */
|
|
void setDrawBox(bool __value);
|
|
/*! \copydoc drawBox
|
|
\see see drawBox for details */
|
|
bool getDrawBox() const;
|
|
|
|
/*! \copydoc relativeWhiskerWidth
|
|
\see see relativeWhiskerWidth for details */
|
|
void setRelativeWhiskerWidth(double __value);
|
|
/*! \copydoc relativeWhiskerWidth
|
|
\see see relativeWhiskerWidth for details */
|
|
double getRelativeWhiskerWidth() const;
|
|
|
|
/*! \copydoc relativeNotchIndent
|
|
\see see relativeNotchIndent for details */
|
|
void setRelativeNotchIndent(double __value);
|
|
/*! \copydoc relativeNotchIndent
|
|
\see see relativeNotchIndent for details */
|
|
double getRelativeNotchIndent() const;
|
|
|
|
|
|
/*! \brief set the line style of whisker lines */
|
|
void setWhiskerLineStyle(Qt::PenStyle __value);
|
|
/*! \brief get the line style of whisker lines */
|
|
Qt::PenStyle getWhiskerLineStyle() const;
|
|
|
|
/*! \brief set the width [pt] of whisker lines */
|
|
void setWhiskerLineWidth(double __value);
|
|
/*! \brief get the width [pt] of whisker lines */
|
|
double getWhiskerLineWidth() const;
|
|
|
|
/*! \brief set the color of whisker lines */
|
|
void setWhiskerLineColor(QColor __value);
|
|
/*! \brief get the color of whisker lines */
|
|
QColor getWhiskerLineColor() const;
|
|
|
|
|
|
/** \brief sets the dash offset for a custom dash style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
void setWhiskerLineDashOffset(qreal offset);
|
|
/** \brief returns the dash offset for a custom dash style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
qreal getWhiskerLineDashOffset() const;
|
|
/** \brief sets the dash pattern for a custom dash style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
void setWhiskerLineDashPattern(const QVector<qreal> &pattern);
|
|
/** \brief gets the dash pattern for a custom dash style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
QVector<qreal> getWhiskerLineDashPattern() const;
|
|
/** \brief sets the join style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setJoinStyle
|
|
*/
|
|
void setWhiskerLineJoinStyle(Qt::PenJoinStyle style);
|
|
/** \brief returns the join style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setJoinStyle
|
|
*/
|
|
Qt::PenJoinStyle getWhiskerLineJoinStyle() const;
|
|
/** \brief sets the cap style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setCapStyle
|
|
*/
|
|
void setWhiskerLineCapStyle(Qt::PenCapStyle style);
|
|
/** \brief gets the cap style of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setCapStyle
|
|
*/
|
|
Qt::PenCapStyle getWhiskerLineCapStyle() const;
|
|
/** \brief sets the brush used to fill the line area of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setBrush
|
|
*/
|
|
void setWhiskerLineBrush(const QBrush& style);
|
|
/** \brief gets the brush used to fill the line area of whisker lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setBrush
|
|
*/
|
|
QBrush getWhiskerLineBrush() const;
|
|
|
|
/** \brief build a pen to be used for drawing whiskers */
|
|
QPen getWhiskerPen(JKQTPEnhancedPainter &painter, JKQTBasePlotter *parent) const;
|
|
|
|
|
|
|
|
/*! \brief set the line style of whisker cap lines */
|
|
void setWhiskerCapLineStyle(Qt::PenStyle __value);
|
|
/*! \brief get the line style of whisker cap lines */
|
|
Qt::PenStyle getWhiskerCapLineStyle() const;
|
|
|
|
/*! \brief set the width [pt] of whisker cap lines */
|
|
void setWhiskerCapLineWidth(double __value);
|
|
/*! \brief get the width [pt] of whisker cap lines */
|
|
double getWhiskerCapLineWidth() const;
|
|
|
|
/*! \brief set the color of whisker cap lines */
|
|
void setWhiskerCapLineColor(QColor __value);
|
|
/*! \brief get the color of whisker cap lines */
|
|
QColor getWhiskerCapLineColor() const;
|
|
|
|
|
|
/** \brief sets the dash offset for a custom dash style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
void setWhiskerCapLineDashOffset(qreal offset);
|
|
/** \brief returns the dash offset for a custom dash style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
qreal getWhiskerCapLineDashOffset() const;
|
|
/** \brief sets the dash pattern for a custom dash style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
void setWhiskerCapLineDashPattern(const QVector<qreal> &pattern);
|
|
/** \brief gets the dash pattern for a custom dash style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
QVector<qreal> getWhiskerCapLineDashPattern() const;
|
|
/** \brief sets the join style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setJoinStyle
|
|
*/
|
|
void setWhiskerCapLineJoinStyle(Qt::PenJoinStyle style);
|
|
/** \brief returns the join style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setJoinStyle
|
|
*/
|
|
Qt::PenJoinStyle getWhiskerCapLineJoinStyle() const;
|
|
/** \brief sets the cap style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setCapStyle
|
|
*/
|
|
void setWhiskerCapLineCapStyle(Qt::PenCapStyle style);
|
|
/** \brief gets the cap style of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setCapStyle
|
|
*/
|
|
Qt::PenCapStyle getWhiskerCapLineCapStyle() const;
|
|
/** \brief sets the brush used to fill the line area of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setBrush
|
|
*/
|
|
void setWhiskerCapLineBrush(const QBrush& style);
|
|
/** \brief gets the brush used to fill the line area of whisker cap lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setBrush
|
|
*/
|
|
QBrush getWhiskerCapLineBrush() const;
|
|
|
|
/** \brief build a pen to be used for drawing whisker caps */
|
|
QPen getWhiskerCapPen(JKQTPEnhancedPainter &painter, JKQTBasePlotter *parent) const;
|
|
|
|
|
|
|
|
|
|
/** \brief modes of how to draw the mean in a boxplot */
|
|
enum MeanMode {
|
|
MeanAsSymbol, /*!< \brief draw mean as a symbol (specified by the settings in JKQTPGraphSymbolStyleMixin) */
|
|
MeanAsLine /*!< \brief draw mean as a lie (specified by the pen settings in JKQTPGraphSymbolStyleMixin) */
|
|
};
|
|
|
|
/*! \copydoc meanMode
|
|
\see see meanMode for details */
|
|
void setMeanMode(MeanMode __value);
|
|
/*! \copydoc meanMode
|
|
\see see meanMode for details */
|
|
MeanMode getMeanMode() const;
|
|
|
|
|
|
/*! \brief set the line style of median lines */
|
|
void setMedianLineStyle(Qt::PenStyle __value);
|
|
/*! \brief get the line style of median lines */
|
|
Qt::PenStyle getMedianLineStyle() const;
|
|
|
|
/*! \brief set the width [pt] of median lines */
|
|
void setMedianLineWidth(double __value);
|
|
/*! \brief get the width [pt] of median lines */
|
|
double getMedianLineWidth() const;
|
|
|
|
/*! \brief set the color of median lines */
|
|
void setMedianLineColor(QColor __value);
|
|
/*! \brief get the color of median lines */
|
|
QColor getMedianLineColor() const;
|
|
|
|
|
|
/** \brief sets the dash offset for a custom dash style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
void setMedianLineDashOffset(qreal offset);
|
|
/** \brief returns the dash offset for a custom dash style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
qreal getMedianLineDashOffset() const;
|
|
/** \brief sets the dash pattern for a custom dash style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
void setMedianLineDashPattern(const QVector<qreal> &pattern);
|
|
/** \brief gets the dash pattern for a custom dash style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
QVector<qreal> getMedianLineDashPattern() const;
|
|
/** \brief sets the join style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setJoinStyle
|
|
*/
|
|
void setMedianLineJoinStyle(Qt::PenJoinStyle style);
|
|
/** \brief returns the join style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setJoinStyle
|
|
*/
|
|
Qt::PenJoinStyle getMedianLineJoinStyle() const;
|
|
/** \brief sets the cap style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setCapStyle
|
|
*/
|
|
void setMedianLineCapStyle(Qt::PenCapStyle style);
|
|
/** \brief gets the cap style of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setCapStyle
|
|
*/
|
|
Qt::PenCapStyle getMedianLineCapStyle() const;
|
|
/** \brief sets the brush used to fill the line area of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setBrush
|
|
*/
|
|
void setMedianLineBrush(const QBrush& style);
|
|
/** \brief gets the brush used to fill the line area of median lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setBrush
|
|
*/
|
|
QBrush getMedianLineBrush() const;
|
|
|
|
/** \brief build a pen to be used for drawing medians */
|
|
QPen getMedianPen(JKQTPEnhancedPainter &painter, JKQTBasePlotter *parent) const;
|
|
|
|
|
|
|
|
|
|
|
|
/*! \brief set the line style of Mean lines */
|
|
void setMeanLineStyle(Qt::PenStyle __value);
|
|
/*! \brief get the line style of Mean lines */
|
|
Qt::PenStyle getMeanLineStyle() const;
|
|
/** \brief sets the dash offset for a custom dash style of Mean lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
void setMeanLineDashOffset(qreal offset);
|
|
/** \brief returns the dash offset for a custom dash style of Mean lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashOffset
|
|
*/
|
|
qreal getMeanLineDashOffset() const;
|
|
/** \brief sets the dash pattern for a custom dash style of Mean lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
void setMeanLineDashPattern(const QVector<qreal> &pattern);
|
|
/** \brief gets the dash pattern for a custom dash style of Mean lines
|
|
* \see https://doc.qt.io/qt-5/qpen.html#setDashPattern
|
|
*/
|
|
QVector<qreal> getMeanLineDashPattern() const;
|
|
/** \brief set the type of the symbol for the mean */
|
|
void setMeanSymbolType(JKQTPGraphSymbols __value);
|
|
/** \brief get the type of the symbol for the mean */
|
|
JKQTPGraphSymbols getMeanSymbolType() const;
|
|
|
|
/** \brief set the size (=diameter in pt) of the symbol for the mean (in pt) */
|
|
void setMeanSize(double __value);
|
|
/** \brief get the size (=diameter in pt) of the symbol for the mean (in pt) */
|
|
double getMeanSize() const;
|
|
|
|
/** \brief set the color of the symbol for the mean, or mean line */
|
|
void setMeanColor(const QColor & __value);
|
|
/** \brief set the color of the symbol for the mean, or mean line */
|
|
QColor getMeanColor() const;
|
|
|
|
/** \brief set the color of filling of the symbol for the mean */
|
|
void setMeanFillColor(const QColor & __value);
|
|
/** \brief set the color of filling of the symbol for the mean */
|
|
QColor getMeanFillColor() const;
|
|
|
|
/** \brief set the line width of the symbol for the mean outline, or mean line (in pt) */
|
|
void setMeanLineWidth(double __value);
|
|
/** \brief get the line width of the symbol for the mean outline, or mean line (in pt) */
|
|
double getMeanLineWidth() const;
|
|
|
|
/** \brief constructs a QPen from the line styling properties to draw the mean line */
|
|
QPen getMeanLinePen(JKQTPEnhancedPainter &painter, JKQTBasePlotter* parent) const;
|
|
/** \brief constructs a QPen from the line styling properties to draw the mean symbol */
|
|
QPen getMeanSymbolPen(JKQTPEnhancedPainter &painter, JKQTBasePlotter* parent) const;
|
|
/** \brief constructs a QPen from the line styling properties */
|
|
QBrush getMeanSymbolBrush(JKQTPEnhancedPainter &painter, JKQTBasePlotter* parent) const;
|
|
|
|
/*! \brief set the color of the graph (colors all elements, based on the given color \a c )*/
|
|
void setBoxplotColor(QColor c, JKQTBasePlotter *parent);
|
|
|
|
/*! \brief set the color of the graph (colors all elements, based on the given color \a c , sets background colors from \a bc )*/
|
|
void setBoxplotColor(QColor c, QColor bc, JKQTBasePlotter *parent);
|
|
|
|
|
|
protected:
|
|
/*! \brief plot a symbol at location x,y (in painter coordinates), using the current style
|
|
|
|
\param parent parent JKQTBasePlotter of the graph that uses this mix-in (used e.g. for line-width transformation)
|
|
\param painter the <a href="http://doc.qt.io/qt-5/qpainter.html">QPainter</a> to draw to
|
|
\param x x-coordinate of the symbol center
|
|
\param y y-coordinate of the symbol center
|
|
*/
|
|
void plotStyledMeanSymbol(JKQTBasePlotter* parent, JKQTPEnhancedPainter &painter, double x, double y) const;
|
|
/** \brief draws a vertical boxplot, with all coordinates/sizes given in coordinates of the given painter,
|
|
* using the style properties declared in this class. Provide a parameter with \c JKQTP_NAN of you
|
|
* don't want it to be drawn, or don't know its value
|
|
*
|
|
* \param parent parent JKQTBasePlotter of the graph that uses this mix-in (used e.g. for line-width transformation)
|
|
* \param painter the <a href="http://doc.qt.io/qt-5/qpainter.html">QPainter</a> to draw to
|
|
* \param xp x-coordinate of the boxplot center
|
|
* \param xpleft x-coordinate of the boxplot box left edge
|
|
* \param xpright x-coordinate of the boxplot box right edge
|
|
* \param minp y-coordinate of the minimum (lower whisker)
|
|
* \param q25p y-coordinate of the 25% quartile (lower box border)
|
|
* \param medianp y-coordinate the median
|
|
* \param q75p y-coordinate of the 75% quartile (upper box border)
|
|
* \param maxp y-coordinate of the maximum (upper whisker)
|
|
* \param meanp y-coordinate of the mean (symbol or line)
|
|
* \param notchLowerp y-coordinate of the lower (near \a qe5p ) end of the notch interval
|
|
* \param notchUpperp y-coordinate of the upper (near \a q75p ) end of the notch interval
|
|
*/
|
|
void plotVerticalBoxplot(JKQTBasePlotter* parent, JKQTPEnhancedPainter &painter, double xp, double xpleft, double xpright, double minp, double q25p, double medianp, double q75p, double maxp, double meanp=JKQTP_NAN, double notchLowerp=JKQTP_NAN, double notchUpperp=JKQTP_NAN) const;
|
|
|
|
/** \brief draws a horizontal boxplot, with all coordinates/sizes given in coordinates of the given painter,
|
|
* using the style properties declared in this class. Provide a parameter with \c JKQTP_NAN of you
|
|
* don't want it to be drawn, or don't know its value
|
|
*
|
|
* \param parent parent JKQTBasePlotter of the graph that uses this mix-in (used e.g. for line-width transformation)
|
|
* \param painter the <a href="http://doc.qt.io/qt-5/qpainter.html">QPainter</a> to draw to
|
|
* \param yp y-coordinate of the boxplot center
|
|
* \param ypbottom y-coordinate of the boxplot box bottom edge
|
|
* \param yptop y-coordinate of the boxplot box top edge
|
|
* \param minp x-coordinate of the minimum (lower whisker)
|
|
* \param q25p x-coordinate of the 25% quartile (lower box border)
|
|
* \param medianp x-coordinate the median
|
|
* \param q75p x-coordinate of the 75% quartile (upper box border)
|
|
* \param maxp x-coordinate of the maximum (upper whisker)
|
|
* \param meanp x-coordinate of the mean (symbol or line)
|
|
* \param notchLowerp x-coordinate of the lower (near \a qe5p ) end of the notch interval
|
|
* \param notchUpperp x-coordinate of the upper (near \a q75p ) end of the notch interval
|
|
*/
|
|
void plotHorizontalBoxplot(JKQTBasePlotter* parent, JKQTPEnhancedPainter &painter, double yp, double ypbottom, double yptop, double minp, double q25p, double medianp, double q75p, double maxp, double meanp=JKQTP_NAN, double notchLowerp=JKQTP_NAN, double notchUpperp=JKQTP_NAN) const;
|
|
|
|
/** \brief draw a small, stylized, vertical symbol into \a rect that symbolizes a boxplot, e.g. in a plot legend */
|
|
void plotVerticalKeyMarker(JKQTBasePlotter* parent, JKQTPEnhancedPainter &painter, const QRectF& rect);
|
|
|
|
/** \brief draw a small, stylized, horizontal symbol into \a rect that symbolizes a boxplot, e.g. in a plot legend */
|
|
void plotHorizontalKeyMarker(JKQTBasePlotter* parent, JKQTPEnhancedPainter &painter, const QRectF& rect);
|
|
private:
|
|
/** \brief which symbol to use for the datapoints */
|
|
JKQTPGraphSymbols m_meanSymbolType;
|
|
/** \brief size (diameter in pt) of the symbol for the data points, given in pt */
|
|
double m_meanSymbolSize;
|
|
/** \brief outline color of the symbol or line pen of the mean-line */
|
|
QPen m_meanSymbolLinePen;
|
|
/** \brief color of the symbol filling */
|
|
QColor m_meanSymbolFillColor;
|
|
/** \brief width (in pt) of the lines used to plot the symbol for the data points, given in pt */
|
|
double m_meanSymbolLineWidth;
|
|
/** \brief line style of the whisker lines */
|
|
QPen m_whiskerLinePen;
|
|
/** \brief line width (in pt) of the whisker lines */
|
|
double whiskerLineWidth;
|
|
/** \brief line style of the whisker cap lines */
|
|
QPen m_whiskerCapLinePen;
|
|
/** \brief line width (in pt) of the whisker cap lines */
|
|
double whiskerCapLineWidth;
|
|
/** \brief line style of the median lines */
|
|
QPen m_medianLinePen;
|
|
/** \brief line width (in pt) of the median lines */
|
|
double medianLineWidth;
|
|
/** \brief width of box in pt.
|
|
*
|
|
* \note If several boxplots are drawn, the width is typically calculated,
|
|
* based on a relative width and the position distances, see JKQTPBoxplotVerticalGraph::boxWidthRelative */
|
|
double boxWidthAbsolute;
|
|
/** \brief width of the whiskers, relative to the box width (default: 0.5) */
|
|
double relativeWhiskerWidth;
|
|
/** \brief single-sided indention of the notch (if any), relative to the box width (default: 0.25) */
|
|
double relativeNotchIndent;
|
|
|
|
/** \brief specifies how to draw the mean (as a line of as a symbol, specified by the function in JKQTPGraphSymbolStyleMixin) */
|
|
MeanMode meanMode;
|
|
/** \brief enables/disables drawing of the actual box of the boxplot (\c false leads to Tufte Style boxplots ) */
|
|
bool drawBox;
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#endif // jkqtpgraphsboxplotstylingmixins_H
|