summaryrefslogtreecommitdiffstats
path: root/kpresenter/KPrPieObject.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kpresenter/KPrPieObject.cpp')
-rw-r--r--kpresenter/KPrPieObject.cpp607
1 files changed, 607 insertions, 0 deletions
diff --git a/kpresenter/KPrPieObject.cpp b/kpresenter/KPrPieObject.cpp
new file mode 100644
index 00000000..9ed85109
--- /dev/null
+++ b/kpresenter/KPrPieObject.cpp
@@ -0,0 +1,607 @@
+/* This file is part of the KDE project
+ Copyright (C) 1998, 1999 Reginald Stadlbauer <reggie@kde.org>
+ Copyright (C) 2005-2006 Thorsten Zachmann <zachmann@kde.org>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License as published by the Free Software Foundation; either
+ version 2 of the License, or (at your option) any later version.
+
+ This library 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
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ * Boston, MA 02110-1301, USA.
+*/
+
+#include "KPrPieObject.h"
+#include "KPrGradient.h"
+#include "KPrUtils.h"
+#include "KPrPieObjectIface.h"
+
+#include <KoTextZoomHandler.h>
+#include <KoOasisContext.h>
+#include <KoStyleStack.h>
+#include <KoXmlNS.h>
+
+#include <kdebug.h>
+
+#include <qregion.h>
+#include <qpicture.h>
+#include <qdom.h>
+#include <qpainter.h>
+#include <qbitmap.h>
+using namespace std;
+
+KPrPieObject::KPrPieObject()
+: KPr2DObject()
+, KPrStartEndLine( L_NORMAL, L_NORMAL )
+{
+ pieType = PT_PIE;
+ p_angle = 45 * 16;
+ p_len = 270 * 16;
+}
+
+KPrPieObject::KPrPieObject( const KoPen &_pen, const QBrush &_brush, FillType _fillType,
+ const QColor &_gColor1, const QColor &_gColor2, BCType _gType,
+ PieType _pieType, int _p_angle, int _p_len,
+ LineEnd _lineBegin, LineEnd _lineEnd,
+ bool _unbalanced, int _xfactor, int _yfactor )
+: KPr2DObject( _pen, _brush, _fillType, _gColor1, _gColor2, _gType, _unbalanced, _xfactor, _yfactor )
+, KPrStartEndLine( _lineBegin, _lineEnd )
+{
+ pieType = _pieType;
+ p_angle = _p_angle;
+ p_len = _p_len;
+}
+
+DCOPObject* KPrPieObject::dcopObject()
+{
+ if ( !dcop )
+ dcop = new KPrPieObjectIface( this );
+ return dcop;
+}
+
+KPrPieObject &KPrPieObject::operator=( const KPrPieObject & )
+{
+ return *this;
+}
+
+QDomDocumentFragment KPrPieObject::save( QDomDocument& doc, double offset )
+{
+ QDomDocumentFragment fragment=KPr2DObject::save(doc, offset);
+ KPrStartEndLine::save( fragment, doc );
+ if (p_angle!=720)
+ fragment.appendChild(KPrObject::createValueElement("PIEANGLE", p_angle, doc));
+ if (p_len!=1440)
+ fragment.appendChild(KPrObject::createValueElement("PIELENGTH", p_len, doc));
+ if (pieType!=PT_PIE)
+ fragment.appendChild(KPrObject::createValueElement("PIETYPE", static_cast<int>(pieType), doc));
+ return fragment;
+}
+
+bool KPrPieObject::saveOasisObjectAttributes( KPOasisSaveContext &sc ) const
+{
+ switch( pieType )
+ {
+ case PT_PIE:
+ sc.xmlWriter.addAttribute( "draw:kind", "section" );
+ break;
+ case PT_CHORD:
+ sc.xmlWriter.addAttribute( "draw:kind", "cut" );
+ break;
+ case PT_ARC:
+ sc.xmlWriter.addAttribute( "draw:kind", "arc" );
+ break;
+ default:
+ kdDebug() << " type of pie not supported" << endl;
+ }
+
+ int startangle = ( (int)p_angle / 16 );
+ sc.xmlWriter.addAttribute( "draw:start-angle", startangle );
+
+ int endangle = ( (int) p_len / 16 ) + startangle;
+ sc.xmlWriter.addAttribute( "draw:end-angle", endangle );
+
+ return true;
+}
+
+void KPrPieObject::fillStyle( KoGenStyle& styleObjectAuto, KoGenStyles& mainStyles ) const
+{
+ KPrShadowObject::fillStyle( styleObjectAuto, mainStyles );
+ if ( pieType == PT_ARC )
+ {
+ saveOasisMarkerElement( mainStyles, styleObjectAuto );
+ }
+ else
+ {
+ m_brush.saveOasisFillStyle( styleObjectAuto, mainStyles );
+ }
+}
+
+const char * KPrPieObject::getOasisElementName() const
+{
+ return ext.width() == ext.height() ? "draw:circle" : "draw:ellipse";
+}
+
+
+void KPrPieObject::loadOasis(const QDomElement &element, KoOasisContext & context, KPrLoadingInfo *info)
+{
+ kdDebug()<<"void KPrPieObject::loadOasis(const QDomElement &element) ***************\n";
+ KPr2DObject::loadOasis(element, context, info);
+ QString kind = element.attributeNS( KoXmlNS::draw, "kind", QString::null );
+ if ( kind == "section" )
+ pieType = PT_PIE;
+ else if ( kind == "cut" )
+ pieType = PT_CHORD;
+ else if ( kind == "arc" )
+ pieType =PT_ARC;
+ else
+ {
+ kdDebug()<<" KPrPieObject::loadOasis(const QDomElement &element) type indefined :"<<kind<<endl;
+ pieType = PT_PIE;
+ }
+ kdDebug()<<" type of pie object :"<<( ( pieType == PT_PIE ) ? "pie" : ( pieType == PT_CHORD )?"cut" : "arc" )<<endl;
+
+ int start = (int) ( element.attributeNS( KoXmlNS::draw, "start-angle", QString::null ).toDouble() );
+ p_angle=start*16;
+
+ int end = (int) ( element.attributeNS( KoXmlNS::draw, "end-angle", QString::null ).toDouble() );
+ if ( end < start )
+ p_len = ( ( 360 - start + end ) * 16 );
+ else
+ p_len = ( ( end - start ) * 16 );
+
+ kdDebug()<<"KPrPieObject::loadOasis(const QDomElement &element) : p_angle :"<<p_angle<<" p_len :"<<p_len<<endl;
+ if ( pieType == PT_ARC )
+ {
+ loadOasisMarkerElement( context, "marker-start", lineBegin );
+ loadOasisMarkerElement( context, "marker-end", lineEnd );
+ }
+}
+
+double KPrPieObject::load(const QDomElement &element)
+{
+ double offset=KPr2DObject::load(element);
+ KPrStartEndLine::load( element );
+ QDomElement e=element.namedItem("PIEANGLE").toElement();
+ if(!e.isNull()) {
+ int tmp=0;
+ if(e.hasAttribute("value"))
+ tmp=e.attribute("value").toInt();
+ p_angle=tmp;
+ }
+ e=element.namedItem("PIELENGTH").toElement();
+ if(!e.isNull()) {
+ int tmp=0;
+ if(e.hasAttribute("value"))
+ tmp=e.attribute("value").toInt();
+ p_len=tmp;
+ }
+ else
+ p_len=1440; //necessary to reinitialise p_len
+ //I don't know who change default value
+ e=element.namedItem("PIETYPE").toElement();
+ if(!e.isNull()) {
+ int tmp=0;
+ if(e.hasAttribute("value"))
+ tmp=e.attribute("value").toInt();
+ pieType=static_cast<PieType>(tmp);
+ }
+ return offset;
+}
+
+void KPrPieObject::paint( QPainter* _painter, KoTextZoomHandler*_zoomHandler,
+ int /* pageNum */, bool drawingShadow, bool drawContour )
+{
+ double ow = ext.width();
+ double oh = ext.height();
+ double pw = ( ( pen.style() == Qt::NoPen ) ? 1 : pen.pointWidth() ) / 2.0;
+
+ if ( drawContour ) {
+ QPen pen3( Qt::black, 1, Qt::DotLine );
+ _painter->setPen( pen3 );
+ _painter->setRasterOp( Qt::NotXorROP );
+ }
+ else {
+ QPen pen2 = pen.zoomedPen( _zoomHandler );
+ _painter->setPen( pen2 );
+ if ( drawingShadow || getFillType() == FT_BRUSH || !gradient )
+ {
+ _painter->setBrush( getBrush() );
+ }
+ else
+ {
+ if ( pieType != PT_ARC )
+ {
+ QSize size( _zoomHandler->zoomSize( ext ) );
+
+ if ( m_redrawGradientPix || gradient->size() != size )
+ {
+ m_redrawGradientPix = false;
+ gradient->setSize( size );
+
+ m_gradientPix.resize ( size );
+ m_gradientPix.fill( Qt::white );
+ QPainter p;
+ p.begin( &m_gradientPix );
+ p.drawPixmap( 0, 0, gradient->pixmap() );
+ p.end();
+
+ QBitmap mask( size, true );
+ p.begin( &mask );
+ p.setPen( QPen( Qt::color1 ) );
+ p.setBrush( QBrush( Qt::color1 ) );
+ if ( pieType == PT_CHORD )
+ {
+ p.drawChord( _zoomHandler->zoomItX(pw), _zoomHandler->zoomItY(pw),
+ _zoomHandler->zoomItX(ow - 2 * pw),
+ _zoomHandler->zoomItY(oh - 2 * pw), p_angle, p_len );
+ }
+ else
+ {
+ p.drawPie( _zoomHandler->zoomItX(pw), _zoomHandler->zoomItY(pw),
+ _zoomHandler->zoomItX( ow - 2 * pw),
+ _zoomHandler->zoomItY( oh - 2 * pw), p_angle, p_len );
+ }
+ p.end();
+ m_gradientPix.setMask( mask );
+ }
+ _painter->drawPixmap( 0, 0, m_gradientPix, 0, 0, size.width(), size.height() );
+ _painter->setBrush( Qt::NoBrush );
+ }
+ }
+ if ( pieType == PT_ARC )
+ {
+ KoPointArray points( 2 );
+ setEndPoints( points );
+ KoPoint start( points.point( 0 ) );
+ KoPoint end( points.point( 1 ) );
+
+ double ys = ( ( 1 - start.x() / ( ext.width() * ext.width() / 4 ) ) * ext.height() * ext.height() / 4 ) / start.y();
+ double s_angle = 90 + ( atan( ( start.x() - 1 ) / ( start.y() - ys ) ) * 180 / M_PI );
+ if ( p_angle / 16 >= 90 && p_angle / 16 <= 270 )
+ {
+ s_angle += 180.0;
+ }
+ double ye = ( ( 1 - end.x() / ( ext.width() * ext.width() / 4 ) ) * ext.height() * ext.height() / 4 ) / end.y();
+ double e_angle = 270 + ( atan( ( end.x() - 1 ) / ( end.y() - ye ) ) * 180 / M_PI );
+ if ( ( ( p_angle + p_len ) / 16 ) % 360 >= 90 && ( ( p_angle + p_len ) / 16 ) % 360 <= 270 )
+ {
+ e_angle -= 180.0;
+ }
+
+ start = KoPoint( ext.width() / 2.0 + start.x(), ext.height() / 2.0 - start.y() );
+ end = KoPoint( ext.width() / 2.0 + end.x(), ext.height() / 2.0 - end.y() );
+
+
+
+ if ( lineBegin != L_NORMAL )
+ drawFigureWithOffset( lineBegin, _painter, start,
+ pen2.color(), int( pen.pointWidth() ), s_angle, _zoomHandler, true );
+
+ if ( lineEnd != L_NORMAL )
+ drawFigureWithOffset( lineEnd, _painter, end,
+ pen2.color(), int( pen.pointWidth() ), e_angle, _zoomHandler, false );
+ }
+ }
+ switch ( pieType )
+ {
+ case PT_PIE:
+ _painter->drawPie( _zoomHandler->zoomItX(pw), _zoomHandler->zoomItY( pw),
+ _zoomHandler->zoomItX( ow - 2 * pw),
+ _zoomHandler->zoomItY( oh - 2 * pw), p_angle, p_len );
+ break;
+ case PT_ARC:
+ _painter->drawArc( _zoomHandler->zoomItX(pw), _zoomHandler->zoomItY(pw),
+ _zoomHandler->zoomItX(ow - 2 * pw),
+ _zoomHandler->zoomItY(oh - 2 * pw), p_angle, p_len );
+ break;
+ case PT_CHORD:
+ _painter->drawChord( _zoomHandler->zoomItX(pw), _zoomHandler->zoomItY(pw),
+ _zoomHandler->zoomItX(ow - 2 * pw),
+ _zoomHandler->zoomItY(oh - 2 * pw), p_angle, p_len );
+ break;
+ default: break;
+ }
+}
+
+void KPrPieObject::flip( bool horizontal )
+{
+ KPr2DObject::flip( horizontal );
+ if ( ! horizontal )
+ {
+ p_angle = 360*16 - p_angle -p_len;
+ }
+ else
+ {
+ p_angle = 180*16 - p_angle - p_len;
+ }
+ // angle smaller 0
+ while ( p_angle < 0 ) {
+ p_angle += 360*16;
+ }
+
+}
+
+
+void KPrPieObject::setMinMax( double &min_x, double &min_y,
+ double &max_x, double &max_y, KoPoint point ) const
+{
+ double tmp_x = point.x();
+ double tmp_y = point.y();
+
+ if ( tmp_x < min_x ) {
+ min_x = tmp_x;
+ }
+ else if ( tmp_x > max_x ) {
+ max_x = tmp_x;
+ }
+
+ if ( tmp_y < min_y ) {
+ min_y = tmp_y;
+ }
+ else if ( tmp_y > max_y ) {
+ max_y = tmp_y;
+ }
+}
+
+
+/*
+ * The calculation of the real size and origin for a pie object is a little more
+ * complicated. It took me quite a whlie to get it right.
+ * Here is how it works:
+ * 1. calculate the position of the end points
+ * 2. calculate the 4 maximal points, the points with max x or y position, of the
+ * hole ellipse
+ * 3. find minimal and maximal points
+ * 4. check if the maximal points lie on the arc
+ *
+ */
+void KPrPieObject::getRealSizeAndOrig( KoSize &size, KoPoint &realOrig ) const {
+ double radius1 = size.width() / 2.0;
+ double radius2 = size.height() / 2.0;
+
+ // the rotation angle
+ double angInRad = angle * M_PI / 180;
+
+ // 1. calulate position of end points
+ KoPointArray points(2);
+ setEndPoints( points );
+
+ // rotate point
+ for ( int i = 0; i < 2; i++ ) {
+ if ( angle != 0 ) {
+ double sinus = sin( angInRad );
+ double cosinus = cos( angInRad );
+
+ double tmp_x = points.point( i ).x();
+ double tmp_y = points.point( i ).y();
+
+ double x = tmp_x * cosinus + tmp_y * sinus;
+ double y = - tmp_x * sinus + tmp_y * cosinus;
+ points.setPoint( i, x, y );
+ }
+ }
+
+ KoPoint firstPoint( points.point(0) );
+ KoPoint secondPoint( points.point(1) );
+
+ // 2. calulate maximal points
+ KoPointArray maxPoints(4);
+ if ( angle == 0 ) {
+ maxPoints.setPoint( 0, 0, radius2 );
+ maxPoints.setPoint( 1, radius1, 0 );
+ maxPoints.setPoint( 2, 0, -radius2 );
+ maxPoints.setPoint( 3, -radius1, 0 );
+ }
+ else {
+ double sinus = sin( angInRad );
+ double cosinus = cos( angInRad );
+
+ double x = sqrt( pow( radius1 * cosinus , 2 ) + pow(radius2 * sinus, 2));
+ double y = ( pow( radius2, 2 ) - pow( radius1, 2) ) * sinus * cosinus / x;
+ maxPoints.setPoint( 0, x, y );
+ maxPoints.setPoint( 1, -x, -y );
+
+ y = sqrt( pow( radius1 * sinus , 2 ) + pow(radius2 * cosinus, 2));
+ x = ( pow( radius1, 2 ) - pow( radius2, 2) ) * sinus * cosinus / y;
+ maxPoints.setPoint( 2, x, y);
+ maxPoints.setPoint( 3, -x, -y );
+ }
+
+ // 3. find minimal and maximal points
+ double min_x = firstPoint.x();
+ double min_y = firstPoint.y();
+ double max_x = firstPoint.x();
+ double max_y = firstPoint.y();
+
+ if ( pieType == PT_PIE ) {
+ KoPoint zero(0,0);
+ setMinMax( min_x, min_y, max_x, max_y, zero );
+ }
+ setMinMax( min_x, min_y, max_x, max_y, secondPoint );
+
+ /* 4. check if maximal points lie on the arc.
+ * There are three posibilities how many sections have to
+ * been checked.
+ * 1. the arc is only once on one side of the x axis
+ * 2. the arc is on both sides of the x axis
+ * 3. the arc is twice on one one side of the x axis
+ *
+ * 1) 2) 3)
+ * y y y
+ * ex|xx xx|xs s |
+ * | x x | x | e
+ * | s x | x | x
+ * ----+---- x ----+---- x ----+---- x
+ * | x | x | x
+ * | x | x | x
+ * | e | xx|xx
+ *
+ */
+ if ( firstPoint.y() >= 0 ) {
+ if ( secondPoint.y() >= 0 ) {
+ if ( firstPoint.x() > secondPoint.x() || p_len == 0 ) {
+ // 1 section
+ // f.x() <= x <= s.x() && y >= 0
+ KoPointArray::ConstIterator it( maxPoints.begin() );
+ for ( ; it != maxPoints.end(); ++it ) {
+ if ( (*it).y() >= 0 &&
+ (*it).x() <= firstPoint.x() && (*it).x() >= secondPoint.x() )
+ {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ }
+ else {
+ // 3 sections
+ // x <= f.x() && y >= 0
+ // y < 0
+ // x >= s.x() && y >= 0
+ KoPointArray::ConstIterator it( maxPoints.begin() );
+ for ( ; it != maxPoints.end(); ++it ) {
+ if ( (*it).y() >= 0 ) {
+ if ( (*it).x() <= firstPoint.x() || (*it).x() >= secondPoint.x() ) {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ else {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ }
+ }
+ else {
+ // 2 sections
+ // x <= f.x() && y >= 0
+ // x <= s.x() && y < 0
+ KoPointArray::ConstIterator it( maxPoints.begin() );
+ for ( ; it != maxPoints.end(); ++it ) {
+ if ( (*it).y() >= 0 ) {
+ if ( (*it).x() <= firstPoint.x() ) {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ else {
+ if ( (*it).x() <= secondPoint.x() ) {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ }
+ }
+ }
+ else {
+ if ( secondPoint.y() >= 0 ) {
+ // 2 sections
+ // x >= f.x() && y < 0
+ // x >= s.x() && y >= 0
+ KoPointArray::ConstIterator it( maxPoints.begin() );
+ for ( ; it != maxPoints.end(); ++it ) {
+ if ( (*it).y() < 0 ) {
+ if ( (*it).x() >= firstPoint.x() ) {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ else {
+ if ( (*it).x() >= secondPoint.x() ) {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ }
+ }
+ else {
+ if ( firstPoint.x() < secondPoint.x() || p_len == 0 ) {
+ // 1 section
+ // f.x() <= x <= s.x() && y < 0
+ KoPointArray::ConstIterator it( maxPoints.begin() );
+ for ( ; it != maxPoints.end(); ++it ) {
+ if ( (*it).y() < 0 &&
+ (*it).x() >= firstPoint.x() && (*it).x() <= secondPoint.x() )
+ {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ }
+ else {
+ // 3 sections
+ // x >= f.x() && y < 0
+ // y >= 0
+ // x <= s.x() && y < 0
+ KoPointArray::ConstIterator it( maxPoints.begin() );
+ for ( ; it != maxPoints.end(); ++it ) {
+ if ( (*it).y() < 0 ) {
+ if ( (*it).x() >= firstPoint.x() || (*it).x() <= secondPoint.x() ) {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ else {
+ setMinMax( min_x, min_y, max_x, max_y, *it );
+ }
+ }
+ }
+ }
+ }
+
+ double mid_x = size.width() / 2;
+ double mid_y = size.height() / 2;
+
+ size.setWidth( max_x - min_x );
+ size.setHeight( max_y - min_y );
+
+ realOrig.setX( realOrig.x() + mid_x + min_x );
+ realOrig.setY( realOrig.y() + mid_y - max_y );
+}
+
+void KPrPieObject::setEndPoints( KoPointArray &points ) const
+{
+ int angles[] = { p_angle, ( p_angle + p_len ) % ( 16 * 360 ) };
+ double anglesInRad[] = { p_angle / 16.0 * M_PI / 180, ( angles[1] ) / 16.0 * M_PI / 180 };
+
+ double radius1 = ext.width() / 2.0;
+ double radius2 = ext.height() / 2.0;
+
+ double prop = radius2 / radius1;
+
+ for ( int i = 0; i < 2; i++ ) {
+ double x = 0;
+ double y = 0;
+
+ // be carefull
+ if ( angles[i] == 90 * 16 ) {
+ y = radius2;
+ }
+ else if ( angles[i] == 270 * 16 ) {
+ y = -radius2;
+ }
+ else {
+ // The real angle is not what was given. It is only ok if radius1 == radius2,
+ // otherwise it is arctan ( radius2 / radius1 tan ( angle ) )
+ double tanalpha = tan( anglesInRad[i] ) * prop;
+ x = sqrt( 1 / ( pow ( 1 / radius1, 2 ) + pow( tanalpha / radius2, 2 ) ) );
+ if ( angles[i] > 90 * 16 && angles[i] < 270 * 16 )
+ x = -x;
+ y = tanalpha * x;
+ }
+ points.setPoint( i, x, y );
+ }
+}
+
+KoSize KPrPieObject::getRealSize() const {
+ KoSize size( ext );
+ KoPoint realOrig( orig );
+ getRealSizeAndOrig( size, realOrig );
+ return size;
+}
+
+
+KoPoint KPrPieObject::getRealOrig() const {
+ KoSize size( ext );
+ KoPoint realOrig( orig );
+ getRealSizeAndOrig( size, realOrig );
+ return realOrig;
+}