Branch data Line data Source code
1 : : /*************************************************************************** 2 : : qgspiediagram.cpp 3 : : --------------------- 4 : : begin : March 2011 5 : : copyright : (C) 2011 by Marco Hugentobler 6 : : email : marco dot hugentobler at sourcepole dot ch 7 : : *************************************************************************** 8 : : * * 9 : : * This program is free software; you can redistribute it and/or modify * 10 : : * it under the terms of the GNU General Public License as published by * 11 : : * the Free Software Foundation; either version 2 of the License, or * 12 : : * (at your option) any later version. * 13 : : * * 14 : : ***************************************************************************/ 15 : : #include "qgspiediagram.h" 16 : : #include "qgsdiagramrenderer.h" 17 : : #include "qgsrendercontext.h" 18 : : #include "qgsexpression.h" 19 : : 20 : : #include <QPainter> 21 : : 22 : : 23 : 0 : QgsPieDiagram::QgsPieDiagram() 24 : 0 : { 25 : 0 : mCategoryBrush.setStyle( Qt::SolidPattern ); 26 : 0 : mPen.setStyle( Qt::SolidLine ); 27 : 0 : } 28 : : 29 : 0 : QgsPieDiagram *QgsPieDiagram::clone() const 30 : : { 31 : 0 : return new QgsPieDiagram( *this ); 32 : 0 : } 33 : : 34 : 0 : QSizeF QgsPieDiagram::diagramSize( const QgsFeature &feature, const QgsRenderContext &c, const QgsDiagramSettings &s, const QgsDiagramInterpolationSettings &is ) 35 : : { 36 : 0 : Q_UNUSED( c ) 37 : : 38 : 0 : QVariant attrVal; 39 : 0 : if ( is.classificationAttributeIsExpression ) 40 : : { 41 : 0 : QgsExpressionContext expressionContext = c.expressionContext(); 42 : 0 : if ( !feature.fields().isEmpty() ) 43 : 0 : expressionContext.setFields( feature.fields() ); 44 : 0 : expressionContext.setFeature( feature ); 45 : : 46 : 0 : QgsExpression *expression = getExpression( is.classificationAttributeExpression, expressionContext ); 47 : 0 : attrVal = expression->evaluate( &expressionContext ); 48 : 0 : } 49 : : else 50 : : { 51 : 0 : attrVal = feature.attribute( is.classificationField ); 52 : : } 53 : : 54 : 0 : bool ok = false; 55 : 0 : double value = attrVal.toDouble( &ok ); 56 : 0 : if ( !ok ) 57 : : { 58 : 0 : return QSizeF(); //zero size if attribute is missing 59 : : } 60 : : 61 : 0 : return sizeForValue( value, s, is ); 62 : 0 : } 63 : : 64 : 0 : double QgsPieDiagram::legendSize( double value, const QgsDiagramSettings &s, const QgsDiagramInterpolationSettings &is ) const 65 : : { 66 : 0 : QSizeF size = sizeForValue( value, s, is ); 67 : 0 : return std::max( size.width(), size.height() ); 68 : : } 69 : : 70 : 0 : QString QgsPieDiagram::diagramName() const 71 : : { 72 : 0 : return DIAGRAM_NAME_PIE; 73 : : } 74 : : 75 : 0 : QSizeF QgsPieDiagram::diagramSize( const QgsAttributes &attributes, const QgsRenderContext &c, const QgsDiagramSettings &s ) 76 : : { 77 : 0 : Q_UNUSED( c ) 78 : 0 : Q_UNUSED( attributes ) 79 : 0 : return s.size; 80 : : } 81 : : 82 : 0 : void QgsPieDiagram::renderDiagram( const QgsFeature &feature, QgsRenderContext &c, const QgsDiagramSettings &s, QPointF position ) 83 : : { 84 : 0 : QPainter *p = c.painter(); 85 : 0 : if ( !p ) 86 : : { 87 : 0 : return; 88 : : } 89 : : 90 : : //get sum of values 91 : 0 : QList<double> values; 92 : 0 : double currentVal = 0; 93 : 0 : double valSum = 0; 94 : 0 : int valCount = 0; 95 : : 96 : 0 : QgsExpressionContext expressionContext = c.expressionContext(); 97 : 0 : expressionContext.setFeature( feature ); 98 : 0 : if ( !feature.fields().isEmpty() ) 99 : 0 : expressionContext.setFields( feature.fields() ); 100 : : 101 : 0 : QList<QString>::const_iterator catIt = s.categoryAttributes.constBegin(); 102 : 0 : for ( ; catIt != s.categoryAttributes.constEnd(); ++catIt ) 103 : : { 104 : 0 : QgsExpression *expression = getExpression( *catIt, expressionContext ); 105 : 0 : currentVal = expression->evaluate( &expressionContext ).toDouble(); 106 : 0 : values.push_back( currentVal ); 107 : 0 : valSum += currentVal; 108 : 0 : if ( currentVal ) valCount++; 109 : 0 : } 110 : : 111 : : //draw the slices 112 : 0 : double totalAngle = 0; 113 : : double currentAngle; 114 : : 115 : : //convert from mm / map units to painter units 116 : 0 : QSizeF spu = sizePainterUnits( s.size, s, c ); 117 : 0 : double w = spu.width(); 118 : 0 : double h = spu.height(); 119 : : 120 : 0 : double baseX = position.x(); 121 : 0 : double baseY = position.y() - h; 122 : : 123 : 0 : mPen.setColor( s.penColor ); 124 : 0 : setPenWidth( mPen, s, c ); 125 : 0 : p->setPen( mPen ); 126 : : 127 : : // there are some values > 0 available 128 : 0 : if ( valSum > 0 ) 129 : : { 130 : 0 : QList<double>::const_iterator valIt = values.constBegin(); 131 : 0 : QList< QColor >::const_iterator colIt = s.categoryColors.constBegin(); 132 : 0 : for ( ; valIt != values.constEnd(); ++valIt, ++colIt ) 133 : : { 134 : 0 : if ( *valIt ) 135 : : { 136 : 0 : currentAngle = ( *valIt / valSum * 360 * 16 ) * ( s.direction() == QgsDiagramSettings::Clockwise ? -1 : 1 ); 137 : 0 : mCategoryBrush.setColor( *colIt ); 138 : 0 : p->setBrush( mCategoryBrush ); 139 : : // if only 1 value is > 0, draw a circle 140 : 0 : if ( valCount == 1 ) 141 : : { 142 : 0 : p->drawEllipse( QRectF( baseX, baseY, w, h ) ); 143 : 0 : } 144 : : else 145 : : { 146 : 0 : p->drawPie( QRectF( baseX, baseY, w, h ), totalAngle - s.rotationOffset * 16.0, currentAngle ); 147 : : } 148 : 0 : totalAngle += currentAngle; 149 : 0 : } 150 : 0 : } 151 : 0 : } 152 : : else // valSum > 0 153 : : { 154 : : // draw empty circle if no values are defined at all 155 : 0 : mCategoryBrush.setColor( Qt::transparent ); 156 : 0 : p->setBrush( mCategoryBrush ); 157 : 0 : p->drawEllipse( QRectF( baseX, baseY, w, h ) ); 158 : : } 159 : 0 : }