summaryrefslogtreecommitdiffstats
path: root/karbon/karbon_part.cc
diff options
context:
space:
mode:
Diffstat (limited to 'karbon/karbon_part.cc')
-rw-r--r--karbon/karbon_part.cc679
1 files changed, 679 insertions, 0 deletions
diff --git a/karbon/karbon_part.cc b/karbon/karbon_part.cc
new file mode 100644
index 00000000..deab6739
--- /dev/null
+++ b/karbon/karbon_part.cc
@@ -0,0 +1,679 @@
+/* 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 <qdom.h>
+#include <qfileinfo.h>
+#include <qpainter.h>
+#include <qpaintdevicemetrics.h>
+
+#include <kconfig.h>
+#include <kdebug.h>
+#include <klocale.h>
+#include <ktempfile.h>
+#include <KoTemplateChooseDia.h>
+#include <KoStoreDevice.h>
+#include <KoOasisStyles.h>
+#include <KoOasisLoadingContext.h>
+#include <KoXmlWriter.h>
+#include <KoXmlNS.h>
+#include <KoDom.h>
+#include <KoOasisSettings.h>
+#include <KoMainWindow.h>
+
+#include "karbon_factory.h"
+#include "karbon_part.h"
+#include "karbon_part_iface.h"
+#include "karbon_view.h"
+#include "vcommand.h"
+#include "vglobal.h"
+#include "vpainter.h"
+#include "vpainterfactory.h"
+#include "vselection.h"
+#include "vcanvas.h"
+#include "vlayer.h"
+#include "vdocumentdocker.h"
+#include "vtoolcontroller.h"
+#include "KoApplication.h"
+#include "vtool.h"
+#include "commands/vtransformcmd.h"
+
+// Make sure an appropriate DTD is available in www/koffice/DTD if changing this value
+// static const char * CURRENT_DTD_VERSION = "1.2";
+
+KarbonPart::KarbonPart( QWidget* parentWidget, const char* widgetName,
+ QObject* parent, const char* name, bool singleViewMode )
+ : KoDocument( parentWidget, widgetName, parent, name, singleViewMode )
+{
+ setInstance( KarbonFactory::instance(), false );
+ setTemplateType( "karbon_template" );
+ m_bShowStatusBar = true;
+ dcop = 0L;
+
+ m_commandHistory = new VCommandHistory( this );
+ connect( m_commandHistory, SIGNAL( documentRestored() ), this, SLOT( slotDocumentRestored() ) );
+ connect( m_commandHistory, SIGNAL( commandExecuted( VCommand * ) ), this, SLOT( slotCommandExecuted( VCommand * ) ) );
+
+ initConfig();
+
+ m_merge = false;
+
+ m_maxRecentFiles = 10;
+
+ //if( name )
+ dcopObject();
+
+ // set as default paper
+ m_pageLayout.format = KoPageFormat::defaultFormat();
+ m_pageLayout.orientation = PG_PORTRAIT;
+ m_pageLayout.ptWidth = MM_TO_POINT( KoPageFormat::width( m_pageLayout.format, m_pageLayout.orientation ) );
+ m_pageLayout.ptHeight = MM_TO_POINT( KoPageFormat::height( m_pageLayout.format, m_pageLayout.orientation ) );
+ m_doc.setWidth( m_pageLayout.ptWidth );
+ m_doc.setHeight( m_pageLayout.ptHeight );
+ // enable selection drawing
+ m_doc.selection()->showHandle();
+ m_doc.selection()->setSelectObjects();
+ m_doc.selection()->setState( VObject::selected );
+ m_doc.selection()->selectNodes();
+}
+
+KarbonPart::~KarbonPart()
+{
+ // delete the command-history:
+ delete m_commandHistory;
+ delete dcop;
+}
+
+DCOPObject* KarbonPart::dcopObject()
+{
+ if( !dcop )
+ dcop = new KarbonPartIface( this );
+
+ return dcop;
+}
+
+void
+KarbonPart::setPageLayout( KoPageLayout& layout, KoUnit::Unit _unit )
+{
+ m_pageLayout = layout;
+ m_doc.setUnit( _unit );
+ m_doc.setWidth( m_pageLayout.ptWidth );
+ m_doc.setHeight( m_pageLayout.ptHeight );
+}
+
+bool
+KarbonPart::initDoc(InitDocFlags flags, QWidget* parentWidget)
+{
+ if (flags==KoDocument::InitDocEmpty)
+ {
+ return true;
+ }
+ QString file;
+ KoTemplateChooseDia::ReturnType result;
+
+ KoTemplateChooseDia::DialogType dlgtype;
+ if (flags != KoDocument::InitDocFileNew)
+ dlgtype = KoTemplateChooseDia::Everything;
+ else
+ dlgtype = KoTemplateChooseDia::OnlyTemplates;
+
+ result = KoTemplateChooseDia::choose( KarbonFactory::instance(), file, dlgtype, "karbon_template", parentWidget );
+
+ if( result == KoTemplateChooseDia::Template )
+ {
+ resetURL();
+ bool ok = loadNativeFormat( file );
+ if ( !ok )
+ showLoadingErrorDialog();
+ setEmpty();
+ return ok;
+ }
+ else if( result == KoTemplateChooseDia::Empty )
+ {
+ return true;
+ }
+ else if( result == KoTemplateChooseDia::File )
+ {
+ KURL url( file );
+ return openURL( url );
+ }
+
+ return false;
+}
+
+KoView*
+KarbonPart::createViewInstance( QWidget* parent, const char* name )
+{
+ KarbonView *result = new KarbonView( this, parent, name );
+ return result;
+}
+
+void
+KarbonPart::removeView( KoView *view )
+{
+ kdDebug(38000) << "KarbonPart::removeView" << endl;
+ KoDocument::removeView( view );
+}
+
+double getAttribute(QDomElement &element, const char *attributeName, double defaultValue)
+{
+ QString value;
+ if ( ( value = element.attribute( attributeName ) ) != QString::null )
+ return value.toDouble();
+ else
+ return defaultValue;
+}
+
+int getAttribute(QDomElement &element, const char *attributeName, int defaultValue)
+{
+ QString value;
+ if ( ( value = element.attribute( attributeName ) ) != QString::null )
+ return value.toInt();
+ else
+ return defaultValue;
+}
+
+bool
+KarbonPart::loadXML( QIODevice*, const QDomDocument& document )
+{
+ bool success = false;
+
+ QDomElement doc = document.documentElement();
+
+ if( m_merge )
+ {
+ m_doc.loadDocumentContent( doc );
+ return true;
+ }
+
+ success = m_doc.loadXML( doc );
+
+ //m_pageLayout = KoPageLayout::standardLayout();
+
+ // <PAPER>
+ QDomElement paper = doc.namedItem( "PAPER" ).toElement();
+ if ( !paper.isNull() )
+ {
+ m_pageLayout.format = static_cast<KoFormat>( getAttribute( paper, "format", 0 ) );
+ m_pageLayout.orientation = static_cast<KoOrientation>( getAttribute( paper, "orientation", 0 ) );
+
+ if( m_pageLayout.format == PG_CUSTOM )
+ {
+ m_pageLayout.ptWidth = m_doc.width();
+ m_pageLayout.ptHeight = m_doc.height();
+ }
+ else
+ {
+ m_pageLayout.ptWidth = getAttribute( paper, "width", 0.0 );
+ m_pageLayout.ptHeight = getAttribute( paper, "height", 0.0 );
+ }
+ }
+ else
+ {
+ m_pageLayout.ptWidth = getAttribute( doc, "width", 595.277);
+ m_pageLayout.ptHeight = getAttribute( doc, "height", 841.891 );
+ }
+
+ kdDebug() << " ptWidth=" << m_pageLayout.ptWidth << endl;
+ kdDebug() << " ptHeight=" << m_pageLayout.ptHeight << endl;
+ QDomElement borders = paper.namedItem( "PAPERBORDERS" ).toElement();
+ if( !borders.isNull() )
+ {
+ if( borders.hasAttribute( "ptLeft" ) )
+ m_pageLayout.ptLeft = borders.attribute( "ptLeft" ).toDouble();
+ if( borders.hasAttribute( "ptTop" ) )
+ m_pageLayout.ptTop = borders.attribute( "ptTop" ).toDouble();
+ if( borders.hasAttribute( "ptRight" ) )
+ m_pageLayout.ptRight = borders.attribute( "ptRight" ).toDouble();
+ if( borders.hasAttribute( "ptBottom" ) )
+ m_pageLayout.ptBottom = borders.attribute( "ptBottom" ).toDouble();
+ }
+
+ setUnit( m_doc.unit() );
+
+ return success;
+}
+
+QDomDocument
+KarbonPart::saveXML()
+{
+ QDomDocument doc = m_doc.saveXML();
+ QDomElement me = doc.documentElement();
+ QDomElement paper = doc.createElement( "PAPER" );
+ me.appendChild( paper );
+ paper.setAttribute( "format", static_cast<int>( m_pageLayout.format ) );
+ paper.setAttribute( "pages", pageCount() );
+ paper.setAttribute( "width", m_pageLayout.ptWidth );
+ paper.setAttribute( "height", m_pageLayout.ptHeight );
+ paper.setAttribute( "orientation", static_cast<int>( m_pageLayout.orientation ) );
+
+ QDomElement paperBorders = doc.createElement( "PAPERBORDERS" );
+ paperBorders.setAttribute( "ptLeft", m_pageLayout.ptLeft );
+ paperBorders.setAttribute( "ptTop", m_pageLayout.ptTop );
+ paperBorders.setAttribute( "ptRight", m_pageLayout.ptRight );
+ paperBorders.setAttribute( "ptBottom", m_pageLayout.ptBottom );
+ paper.appendChild(paperBorders);
+
+ return doc;
+}
+
+bool
+KarbonPart::loadOasis( const QDomDocument &doc, KoOasisStyles &styles, const QDomDocument &settings, KoStore *store )
+{
+ kdDebug(38000) << "Start loading OASIS document..." << doc.toString() << endl;
+
+ QDomElement contents = doc.documentElement();
+ kdDebug(38000) << "Start loading OASIS document..." << contents.text() << endl;
+ kdDebug(38000) << "Start loading OASIS contents..." << contents.lastChild().localName() << endl;
+ kdDebug(38000) << "Start loading OASIS contents..." << contents.lastChild().namespaceURI() << endl;
+ kdDebug(38000) << "Start loading OASIS contents..." << contents.lastChild().isElement() << endl;
+ QDomElement body( KoDom::namedItemNS( contents, KoXmlNS::office, "body" ) );
+ kdDebug(38000) << "Start loading OASIS document..." << body.text() << endl;
+ if( body.isNull() )
+ {
+ kdDebug(38000) << "No office:body found!" << endl;
+ setErrorMessage( i18n( "Invalid OASIS document. No office:body tag found." ) );
+ return false;
+ }
+
+ body = KoDom::namedItemNS( body, KoXmlNS::office, "drawing");
+ if(body.isNull())
+ {
+ kdDebug(38000) << "No office:drawing found!" << endl;
+ setErrorMessage( i18n( "Invalid OASIS document. No office:drawing tag found." ) );
+ return false;
+ }
+
+ QDomElement page( KoDom::namedItemNS( body, KoXmlNS::draw, "page" ) );
+ if(page.isNull())
+ {
+ kdDebug(38000) << "No office:drawing found!" << endl;
+ setErrorMessage( i18n( "Invalid OASIS document. No draw:page tag found." ) );
+ return false;
+ }
+
+ QString masterPageName = "Standard"; // use default layout as fallback
+ QDomElement *master = styles.masterPages()[ masterPageName ];
+ if ( !master ) //last test...
+ master = styles.masterPages()[ "Default" ];
+ // last resort, use the first found master page style
+ if ( ! master )
+ {
+ QDictIterator<QDomElement> it( styles.masterPages() );
+ master = it.current();
+ }
+ Q_ASSERT( master );
+ const QDomElement *style = master ? styles.findStyle( master->attributeNS( KoXmlNS::style, "page-layout-name", QString::null ) ) : 0;
+ if( style )
+ {
+ m_pageLayout.loadOasis( *style );
+ m_doc.setWidth( m_pageLayout.ptWidth );
+ m_doc.setHeight( m_pageLayout.ptHeight );
+ }
+ else
+ return false;
+
+ KoOasisLoadingContext context( this, styles, store );
+ m_doc.loadOasis( page, context );
+ // do y-mirroring here
+ QWMatrix mat;
+ mat.scale( 1, -1 );
+ mat.translate( 0, -m_doc.height() );
+ VTransformCmd trafo( 0L, mat );
+ trafo.visit( m_doc );
+
+ loadOasisSettings( settings );
+
+ return true;
+}
+
+void
+KarbonPart::loadOasisSettings( const QDomDocument&settingsDoc )
+{
+ if ( settingsDoc.isNull() )
+ return ; // not an error if some file doesn't have settings.xml
+ KoOasisSettings settings( settingsDoc );
+ KoOasisSettings::Items viewSettings = settings.itemSet( "view-settings" );
+ if ( !viewSettings.isNull() )
+ {
+ setUnit(KoUnit::unit(viewSettings.parseConfigItemString("unit")));
+ // FIXME: add other config here.
+ }
+}
+
+
+bool
+KarbonPart::saveOasis( KoStore *store, KoXmlWriter *manifestWriter )
+{
+ if( !store->open( "content.xml" ) )
+ return false;
+
+ KoStoreDevice storeDev( store );
+ KoXmlWriter* docWriter = createOasisXmlWriter( &storeDev, "office:document-content" );
+ KoGenStyles mainStyles;
+
+ KoGenStyle pageLayout = m_pageLayout.saveOasis();
+ QString layoutName = mainStyles.lookup( pageLayout, "PL" );
+ KoGenStyle masterPage( KoGenStyle::STYLE_MASTER );
+ masterPage.addAttribute( "style:page-layout-name", layoutName );
+ mainStyles.lookup( masterPage, "Default", KoGenStyles::DontForceNumbering );
+
+ KTempFile contentTmpFile;
+ contentTmpFile.setAutoDelete( true );
+ QFile* tmpFile = contentTmpFile.file();
+ KoXmlWriter contentTmpWriter( tmpFile, 1 );
+
+ contentTmpWriter.startElement( "office:body" );
+ contentTmpWriter.startElement( "office:drawing" );
+
+ m_doc.saveOasis( store, &contentTmpWriter, mainStyles ); // Save contents
+
+ contentTmpWriter.endElement(); // office:drawing
+ contentTmpWriter.endElement(); // office:body
+
+ docWriter->startElement( "office:automatic-styles" );
+
+ QValueList<KoGenStyles::NamedStyle> styles = mainStyles.styles( VDocument::STYLE_GRAPHICAUTO );
+ QValueList<KoGenStyles::NamedStyle>::const_iterator it = styles.begin();
+ for( ; it != styles.end() ; ++it )
+ (*it).style->writeStyle( docWriter, mainStyles, "style:style", (*it).name , "style:graphic-properties" );
+
+ docWriter->endElement(); // office:automatic-styles
+
+ // And now we can copy over the contents from the tempfile to the real one
+ tmpFile->close();
+ docWriter->addCompleteElement( tmpFile );
+ contentTmpFile.close();
+
+ docWriter->endElement(); // Root element
+ docWriter->endDocument();
+ delete docWriter;
+
+ if( !store->close() )
+ return false;
+
+ manifestWriter->addManifestEntry( "content.xml", "text/xml" );
+
+ if( !store->open( "styles.xml" ) )
+ return false;
+
+ KoXmlWriter* styleWriter = createOasisXmlWriter( &storeDev, "office:document-styles" );
+
+ styleWriter->startElement( "office:styles" );
+
+ styles = mainStyles.styles( VDocument::STYLE_LINEAR_GRADIENT );
+ it = styles.begin();
+ for( ; it != styles.end() ; ++it )
+ (*it).style->writeStyle( styleWriter, mainStyles, "svg:linearGradient", (*it).name, 0, true, true /*add draw:name*/);
+
+ styles = mainStyles.styles( VDocument::STYLE_RADIAL_GRADIENT );
+ it = styles.begin();
+ for( ; it != styles.end() ; ++it )
+ (*it).style->writeStyle( styleWriter, mainStyles, "svg:radialGradient", (*it).name, 0, true, true /*add draw:name*/);
+
+ styleWriter->endElement(); // office:styles
+
+ styleWriter->startElement( "office:automatic-styles" );
+
+ QValueList<KoGenStyles::NamedStyle> styleList = mainStyles.styles( KoGenStyle::STYLE_PAGELAYOUT );
+ it = styleList.begin();
+
+ for( ; it != styleList.end(); ++it )
+ (*it).style->writeStyle( styleWriter, mainStyles, "style:page-layout", (*it).name, "style:page-layout-properties" );
+
+ styleWriter->endElement(); // office:automatic-styles
+
+ styles = mainStyles.styles( KoGenStyle::STYLE_MASTER );
+ it = styles.begin();
+ styleWriter->startElement("office:master-styles");
+
+ for( ; it != styles.end(); ++it)
+ (*it).style->writeStyle( styleWriter, mainStyles, "style:master-page", (*it).name, "");
+
+ styleWriter->endElement(); // office:master-styles
+
+ styleWriter->endElement(); // Root element
+ styleWriter->endDocument();
+ delete styleWriter;
+
+ if( !store->close() )
+ return false;
+
+ manifestWriter->addManifestEntry( "styles.xml", "text/xml" );
+
+
+ if(!store->open("settings.xml"))
+ return false;
+
+
+ KoXmlWriter& settingsWriter = *createOasisXmlWriter(&storeDev, "office:document-settings");
+ settingsWriter.startElement("office:settings");
+ settingsWriter.startElement("config:config-item-set");
+ settingsWriter.addAttribute("config:name", "view-settings");
+
+ KoUnit::saveOasis(&settingsWriter, unit());
+ saveOasisSettings( settingsWriter );
+
+ settingsWriter.endElement(); // config:config-item-set
+ settingsWriter.endElement(); // office:settings
+ settingsWriter.endElement(); // Root element
+ settingsWriter.endDocument();
+ delete &settingsWriter;
+
+
+ if(!store->close())
+ return false;
+
+ manifestWriter->addManifestEntry("settings.xml", "text/xml");
+
+ setModified( false );
+ return true;
+}
+
+void
+KarbonPart::saveOasisSettings( KoXmlWriter &/*settingsWriter*/ )
+{
+ //todo
+}
+
+void
+KarbonPart::insertObject( VObject* object )
+{
+ // don't repaint here explicitly. some commands might want to insert many
+ // objects.
+ m_doc.append( object );
+ setModified( true );
+}
+
+void
+KarbonPart::addCommand( VCommand* cmd, bool repaint )
+{
+ m_commandHistory->addCommand( cmd );
+ setModified( true );
+
+ if( repaint )
+ repaintAllViews();
+}
+
+void
+KarbonPart::slotDocumentRestored()
+{
+ setModified( false );
+}
+
+void
+KarbonPart::slotCommandExecuted( VCommand *command )
+{
+ setModified( true );
+}
+
+void
+KarbonPart::clearHistory()
+{
+ m_commandHistory->clear();
+}
+
+void
+KarbonPart::repaintAllViews( bool repaint )
+{
+ QPtrListIterator<KoView> itr( views() );
+
+ for( ; itr.current() ; ++itr )
+ static_cast<KarbonView*>( itr.current() )->canvasWidget()->repaintAll( repaint );
+}
+
+void
+KarbonPart::repaintAllViews( const KoRect &rect )
+{
+ QPtrListIterator<KoView> itr( views() );
+
+ for( ; itr.current() ; ++itr )
+ static_cast<KarbonView*>( itr.current() )->canvasWidget()->repaintAll( rect );
+}
+
+void
+KarbonPart::paintContent( QPainter& painter, const QRect& rect,
+ bool /*transparent*/, double /*zoomX*/, double /*zoomY*/ )
+{
+ kdDebug(38000) << "**** part->paintContent()" << endl;
+
+ KoRect r = KoRect::fromQRect( rect );
+ double zoomFactorX = double( r.width() ) / double( document().width() );
+ double zoomFactorY = double( r.height() ) / double( document().height() );
+ double zoomFactor = kMin( zoomFactorX, zoomFactorY );
+
+ painter.eraseRect( rect );
+ VPainterFactory *painterFactory = new VPainterFactory;
+ //QPaintDeviceMetrics metrics( painter.device() );
+ painterFactory->setPainter( painter.device(), rect.width(), rect.height() );
+ VPainter *p = painterFactory->painter();
+ //VPainter *p = new VKoPainter( painter.device() );
+ p->begin();
+ p->setZoomFactor( zoomFactor );
+ kdDebug(38000) << "painter.worldMatrix().dx() : " << painter.worldMatrix().dx() << endl;
+ kdDebug(38000) << "painter.worldMatrix().dy() : " << painter.worldMatrix().dy() << endl;
+ kdDebug(38000) << "rect.x() : "<< rect.x() << endl;
+ kdDebug(38000) << "rect.y() : "<< rect.y() << endl;
+ kdDebug(38000) << "rect.width() : "<< rect.width() << endl;
+ kdDebug(38000) << "rect.height() : "<< rect.height() << endl;
+ r = document().boundingBox();
+ QWMatrix mat = painter.worldMatrix();
+ mat.scale( 1, -1 );
+ mat.translate( 0, -r.height() * zoomFactor );
+ p->setWorldMatrix( mat );
+
+ m_doc.selection()->clear();
+ QPtrListIterator<VLayer> itr( m_doc.layers() );
+
+ for( ; itr.current(); ++itr )
+ {
+ itr.current()->draw( p, &r );
+ }
+
+ p->end();
+ delete painterFactory;
+}
+
+void
+KarbonPart::setShowStatusBar( bool b )
+{
+ m_bShowStatusBar = b;
+}
+
+void
+KarbonPart::reorganizeGUI()
+{
+ QPtrListIterator<KoView> itr( views() );
+
+ for( ; itr.current(); ++itr )
+ {
+ static_cast<KarbonView*>( itr.current() )->reorganizeGUI();
+ }
+}
+
+void
+KarbonPart::setUndoRedoLimit( int undos )
+{
+ m_commandHistory->setUndoLimit( undos );
+ m_commandHistory->setRedoLimit( undos );
+}
+
+void
+KarbonPart::initConfig()
+{
+ KConfig* config = KarbonPart::instance()->config();
+
+ if( config->hasGroup( "Interface" ) )
+ {
+ config->setGroup( "Interface" );
+ setAutoSave( config->readNumEntry( "AutoSave", defaultAutoSave() / 60 ) * 60 );
+ m_maxRecentFiles = config->readNumEntry( "NbRecentFile", 10 );
+ setShowStatusBar( config->readBoolEntry( "ShowStatusBar" , true ) );
+ setBackupFile( config->readNumEntry( "BackupFile", true ) );
+ m_doc.saveAsPath( config->readBoolEntry( "SaveAsPath", true ) );
+ }
+ int undos = 30;
+ if( config->hasGroup( "Misc" ) )
+ {
+ config->setGroup( "Misc" );
+ undos = config->readNumEntry( "UndoRedo", -1 );
+ QString defaultUnit = "cm";
+
+ if( KGlobal::locale()->measureSystem() == KLocale::Imperial )
+ defaultUnit = "in";
+
+ setUnit( KoUnit::unit( config->readEntry( "Units", defaultUnit ) ) );
+ m_doc.setUnit( unit() );
+ }
+ if( undos != -1 )
+ setUndoRedoLimit( undos );
+}
+
+bool
+KarbonPart::mergeNativeFormat( const QString &file )
+{
+ m_merge = true;
+ bool result = loadNativeFormat( file );
+ if ( !result )
+ showLoadingErrorDialog();
+ m_merge = false;
+ return result;
+}
+
+void
+KarbonPart::addShell( KoMainWindow *shell )
+{
+ connect( shell, SIGNAL( documentSaved() ), m_commandHistory, SLOT( documentSaved() ) );
+ KoDocument::addShell( shell );
+}
+
+
+void
+KarbonPart::slotUnitChanged( KoUnit::Unit /*unit*/ )
+{
+#if 0
+ // VDocument has its own storage of the unit...
+ m_doc.setUnit( unit );
+ if( m_toolController->activeTool() )
+ m_toolController->activeTool()->refreshUnit();
+#endif
+}
+
+#include "karbon_part.moc"
+