diff options
Diffstat (limited to 'karbon/shapes/vellipse.cc')
-rw-r--r-- | karbon/shapes/vellipse.cc | 316 |
1 files changed, 316 insertions, 0 deletions
diff --git a/karbon/shapes/vellipse.cc b/karbon/shapes/vellipse.cc new file mode 100644 index 00000000..c2afc804 --- /dev/null +++ b/karbon/shapes/vellipse.cc @@ -0,0 +1,316 @@ +/* This file is part of the KDE project + Copyright (C) 2001, 2002, 2003 The Karbon Developers + + 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 "vellipse.h" +#include "vtransformcmd.h" +#include <klocale.h> +#include <KoUnit.h> +#include <KoStore.h> +#include <KoXmlWriter.h> +#include <KoXmlNS.h> +#include <vglobal.h> +#include <vdocument.h> +#include <qdom.h> +#include <core/vfill.h> + +VEllipse::VEllipse( VObject* parent, VState state ) : VPath( parent, state ) +{ +} + +VEllipse::VEllipse( VObject* parent, + const KoPoint& topLeft, double width, double height, + VEllipseType type, double startAngle, double endAngle ) + : VPath( parent ), m_type( type ), m_startAngle( startAngle ), m_endAngle( endAngle ) +{ + setDrawCenterNode(); + + m_rx = width / 2.0; + m_ry = height / 2.0; + m_center.setX( topLeft.x() + m_rx ); + m_center.setY( topLeft.y() + m_ry ); + + init(); +} + +void +VEllipse::init() +{ + // to radials + int nsegs; + if( m_startAngle < m_endAngle ) + nsegs = int( floor( ( m_endAngle - m_startAngle ) / 90.0 ) ); + else + nsegs = 4 - int( ceil( ( m_startAngle - m_endAngle ) / 90.0 ) ); + double startAngle = m_startAngle - 90.0; + if( startAngle < 0 ) startAngle += 360.0; + startAngle = VGlobal::pi_2 * ( startAngle / 90.0 ); + double endAngle = m_endAngle - 90.0; + if( endAngle < 0 ) endAngle += 360.0; + endAngle = VGlobal::pi_2 * ( endAngle / 90.0 ); + // Create (half-)unity circle with topLeft at (0|0): + double currentAngle = -startAngle - VGlobal::pi_2; + KoPoint start( 0.5 * sin( -startAngle ), 0.5 * cos( -startAngle ) ); + moveTo( KoPoint( start.x(), start.y() ) ); + double midAngle = currentAngle + VGlobal::pi_2 / 2.0; + double midAmount = 0.5 / sin( VGlobal::pi_2 / 2.0 ); + for( int i = 0;i < nsegs;i++ ) + { + midAngle -= VGlobal::pi_2; + arcTo( KoPoint( cos( midAngle ) * midAmount, -sin( midAngle ) * midAmount ), + KoPoint( 0.5 * sin( currentAngle ), 0.5 * cos( currentAngle ) ), 0.5 ); + currentAngle -= VGlobal::pi_2; + } + double rest = ( -endAngle - VGlobal::pi_2 - currentAngle ) * 90.0 / VGlobal::pi_2; + if( rest > 0 ) + rest -= 360.0; + if( rest != 0 ) + { + midAngle = currentAngle - ( -rest / 360.0 ) * VGlobal::pi; + midAmount = 0.5 / cos( currentAngle - midAngle ); + KoPoint end( 0.5 * sin( -endAngle ), 0.5 * cos( -endAngle ) ); + arcTo( KoPoint( cos( midAngle ) * midAmount, -sin( midAngle ) * midAmount ), + KoPoint( 0.5 * sin( -endAngle ), 0.5 * cos( -endAngle ) ), 0.5 ); + } + if( m_type == cut ) + lineTo( KoPoint( 0.0, 0.0 ) ); + if( m_type != arc ) + close(); + + // Translate and scale: + QWMatrix m; + m.translate( m_center.x(), m_center.y() ); + m.scale( 2.0 * m_rx, 2.0 * m_ry ); + + // only tranform the path data + VTransformCmd cmd( 0L, m ); + cmd.VVisitor::visitVPath( *this ); + + m_matrix.reset(); +} + +QString +VEllipse::name() const +{ + QString result = VObject::name(); + return !result.isEmpty() ? result : i18n( "Ellipse" ); +} + +void +VEllipse::save( QDomElement& element ) const +{ + VDocument *doc = document(); + if( doc && doc->saveAsPath() ) + { + VPath::save( element ); + return; + } + + if( state() != deleted ) + { + QDomElement me = element.ownerDocument().createElement( "ELLIPSE" ); + element.appendChild( me ); + + // save fill/stroke untransformed + VPath path( *this ); + VTransformCmd cmd( 0L, m_matrix.invert() ); + cmd.visit( path ); + path.VObject::save( me ); + //VObject::save( me ); + + me.setAttribute( "cx", m_center.x() ); + me.setAttribute( "cy", m_center.y() ); + + me.setAttribute( "rx", m_rx ); + me.setAttribute( "ry", m_ry ); + + me.setAttribute( "start-angle", m_startAngle ); + me.setAttribute( "end-angle", m_endAngle ); + + if( m_type == cut ) + me.setAttribute( "kind", "cut" ); + else if( m_type == section ) + me.setAttribute( "kind", "section" ); + else if( m_type == arc ) + me.setAttribute( "kind", "arc" ); + else + me.setAttribute( "kind", "full" ); + + QString transform = buildSvgTransform(); + if( !transform.isEmpty() ) + me.setAttribute( "transform", transform ); + } +} + +void +VEllipse::saveOasis( KoStore *store, KoXmlWriter *docWriter, KoGenStyles &mainStyles, int &index ) const +{ + // do not save deleted objects + if( state() == deleted ) + return; + + docWriter->startElement( "draw:ellipse" ); + + //save all into pt + docWriter->addAttributePt( "svg:cx", m_center.x() ); + docWriter->addAttributePt( "svg:cy", m_center.y() ); + docWriter->addAttributePt( "svg:rx", m_rx ); + docWriter->addAttributePt( "svg:ry", m_ry ); + + if( m_type == full ) + docWriter->addAttribute( "draw:kind", "full" ); + else + { + // the meaning of cut and section is mixed up in karbon, so just set them so for now + // be sure to do the right thing tm for karbon 2.0 + if( m_type == section ) + docWriter->addAttribute( "draw:kind", "cut" ); + else if( m_type == cut ) + docWriter->addAttribute( "draw:kind", "section" ); + else + docWriter->addAttribute( "draw:kind", "arc" ); + + docWriter->addAttribute( "draw:start-angle", m_startAngle ); + docWriter->addAttribute( "draw:end-angle", m_endAngle ); + } + + VObject::saveOasis( store, docWriter, mainStyles, index ); + + QWMatrix tmpMat; + tmpMat.scale( 1, -1 ); + tmpMat.translate( 0, -document()->height() ); + + QString transform = buildOasisTransform( m_matrix*tmpMat ); + if( !transform.isEmpty() ) + docWriter->addAttribute( "draw:transform", transform ); + + docWriter->endElement(); +} + +bool +VEllipse::loadOasis( const QDomElement &element, KoOasisLoadingContext &context ) +{ + setState( normal ); + + if( element.tagName() == "ellipse" ) + { + if( element.hasAttributeNS( KoXmlNS::svg, "rx" ) ) + m_rx = KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "rx", QString::null ) ); + else + m_rx = 0.5 * KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "width", QString::null ) ); + + if( element.hasAttributeNS( KoXmlNS::svg, "ry" ) ) + m_ry = KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "ry", QString::null ) ); + else + m_ry = 0.5 * KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "height", QString::null ) ); + + } + else if( element.tagName() == "circle" ) + { + if( element.hasAttributeNS( KoXmlNS::svg, "r" ) ) + m_rx = m_ry = KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "r", QString::null ) ); + else + m_rx = m_ry = 0.5 * KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "width", QString::null ) ); + } + + if( element.hasAttributeNS( KoXmlNS::svg, "cx" ) ) + m_center.setX( KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "cx", QString::null ) ) ); + else + m_center.setX( m_rx + KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "x", QString::null ) ) ); + + if( element.hasAttributeNS( KoXmlNS::svg, "cy" ) ) + m_center.setY( KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "cy", QString::null ) ) ); + else + m_center.setY( m_ry + KoUnit::parseValue( element.attributeNS( KoXmlNS::svg, "y", QString::null ) ) ); + + // the meaning of cut and section is mixed up in karbon, so just set them so for now + // be sure to do the right thing tm for karbon 2.0 + QString kind = element.attributeNS( KoXmlNS::draw, "kind", QString::null ); + if( kind == "cut" ) + m_type = section; + else if( kind == "section" ) + m_type = cut; + else if( kind == "arc" ) + m_type = arc; + else + m_type = full; + + double startAngle = element.attributeNS( KoXmlNS::draw, "start-angle", QString::null ).toDouble(); + double endAngle = element.attributeNS( KoXmlNS::draw, "end-angle", QString::null ).toDouble(); + + // the shape gets mirrored in y-direction, so make the angles temporary clockwise + // just for creating the path + m_startAngle = 360.0 - endAngle; + m_endAngle = 360.0 - startAngle; + + init(); + + // now set the right angles + m_startAngle = startAngle; + m_endAngle = endAngle; + + transformByViewbox( element, element.attributeNS( KoXmlNS::svg, "viewBox", QString::null ) ); + + QString trafo = element.attributeNS( KoXmlNS::draw, "transform", QString::null ); + if( !trafo.isEmpty() ) + transformOasis( trafo ); + + return VObject::loadOasis( element, context ); +} + +void +VEllipse::load( const QDomElement& element ) +{ + setState( normal ); + + QDomNodeList list = element.childNodes(); + for( uint i = 0; i < list.count(); ++i ) + if( list.item( i ).isElement() ) + VObject::load( list.item( i ).toElement() ); + + m_rx = KoUnit::parseValue( element.attribute( "rx" ) ); + m_ry = KoUnit::parseValue( element.attribute( "ry" ) ); + + m_center.setX( KoUnit::parseValue( element.attribute( "cx" ) ) ); + m_center.setY( KoUnit::parseValue( element.attribute( "cy" ) ) ); + + m_startAngle = element.attribute( "start-angle" ).toDouble(); + m_endAngle = element.attribute( "end-angle" ).toDouble(); + + if( element.attribute( "kind" ) == "cut" ) + m_type = cut; + else if( element.attribute( "kind" ) == "section" ) + m_type = section; + else if( element.attribute( "kind" ) == "arc" ) + m_type = arc; + else + m_type = full; + + init(); + + QString trafo = element.attribute( "transform" ); + if( !trafo.isEmpty() ) + transform( trafo ); +} + +VPath* +VEllipse::clone() const +{ + return new VEllipse( *this ); +} |