diff options
Diffstat (limited to 'lib/kofficecore')
128 files changed, 37174 insertions, 0 deletions
diff --git a/lib/kofficecore/DESIGN b/lib/kofficecore/DESIGN new file mode 100644 index 00000000..c5409b30 --- /dev/null +++ b/lib/kofficecore/DESIGN @@ -0,0 +1,113 @@ +------------------------------------------------------------------------- +| This document answers some questions about how the startup of KOffice +| applications works. This file was put together by +| Eric R. Turner <eric.r.turner@bitbreather.com> +| from the answers given by David Faure <faure@kde.org> +------------------------------------------------------------------------- + +Q: Where is the pointer to the application's instance of KAboutData +data stored? + +A: In a static member of the KCmdLineArgs class. + + + +Q: One of the arguments to KAboutData's constructor is the program +name. Will this be used later to run the program? + +A: No (by the time the KAboutData's constructor is called, the program +is already running). The program name is in fact the "instance name", +which is used to locate the application's own data +(/share/apps/<application name>/...) and many other things. + + + +Q: What is the "+[file]" option passed to KCmdLineArgs? + +A: KCmdLineOptions describe the KCmdLineArgs. "+[file]" means that the +command line arguments can be one or more file names, and that these +arguments are optional. + + + +Q: What is I18N_NOOP()? + +A: I18N_NOOP() marks the options description for translation to other +languages. + + + +Q: What is a KoApplicationInterface used for? + +A: It allows a KOffice application to participate in Inter-Process +Communication and Remote Procedure Calls via DCOP. + + + +Q: What is the .desktop file? + +A: It's responsible for the menu entry, standard KDE stuff. In KOffice +we add a few useful tags to it, in particular the "native mime type" +of the component (e.g. application/x-kword for kword). + + + +Q: Is the .desktop file generated by hand? + +A: Yes. + + + +Q: What's a KoDocumentEntry? I couldn't find it in lib/kofficecore. + +A: KoDocumentEntry describes a KOffice component (i.e. it's a +representation in memory of the data contained in the .desktop +file). It's in kofficecore/koQueryTrader.h. + + + +Q: Is the show() method of KoMainWindow where the application actually +appears to the user? + +A: Yes. + + + +Q: How does KoMainWindow->show() bring up the correct application (how +does it know we want to show one app or another)? + +A: The KoMainWindow is the same for all. But KoDocumentEntry was +created from specific application data, and koQueryTrader klopens the +library containing this component. The very same mechanism is used to +1) embed other components inside the app, 2) create any kind of +koffice document, e.g. in koshell. All the KOffice applications are in +fact very small wrappers that simply dlopen their own compnent - with +the code in kofficecore, which can also dlopen ANY component. + + + +Q: The KoApplication object doesn't store pointers to the KoDocument +nd KoMainWindow objects. Do these objects communicate with each other +later on, and if so, is that done through DCOP? + +A: The KoMainWindow knows about the KoDocument[s] inside it. The +KoMainWindows are in a static linked list, if one really wants to +iterateover all windows in the application, but this is rarely needed. + + +Q: How does KoMainWindow relate to KoDocument and KoView? + +A: KOffice applications reimplement KoDocument and KoView. But the +KoMainWindow is the same class for all, since it doesn't have any +app-specific stuff. It's GUI, i.e. menus and toolbars, are provided by +1) itself (the standard stuff like the File and Help menus), 2) the +document's actions, and 3) the view's actions. + + + +Q: How do KoDocument and KoView relate to each other? + +A: KoDocument can have one or more KoViews, which allows the user to +have multiple views of the same document. In a single KoMainWindow +this might be accomplished using the "split view" feature. KoView[s] +of the same document can be in separate KoMainWindow[s]. diff --git a/lib/kofficecore/KoApplication.cpp b/lib/kofficecore/KoApplication.cpp new file mode 100644 index 00000000..3f2f6b23 --- /dev/null +++ b/lib/kofficecore/KoApplication.cpp @@ -0,0 +1,249 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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 "KoApplication.h" +#include <config.h> +#include <qfile.h> +#include <qregexp.h> +#include <dcopclient.h> +#include <KoApplicationIface.h> +#include <KoQueryTrader.h> +#include <KoDocument.h> +#include <KoMainWindow.h> +#include <klocale.h> +#include <kcmdlineargs.h> +#include <kdebug.h> +#include <kdesktopfile.h> +#include <kmessagebox.h> +#include <kstandarddirs.h> +#include <stdlib.h> + +void qt_generate_epsf( bool b ); + +static const KCmdLineOptions options[]= +{ + {"print", I18N_NOOP("Only print and exit"),0}, + {"template", I18N_NOOP("Open a new document with a template"), 0}, + {"dpi <dpiX,dpiY>", I18N_NOOP("Override display DPI"), 0}, + KCmdLineLastOption +}; + +bool KoApplication::m_starting = true; + +class KoApplicationPrivate +{ +public: + KoApplicationPrivate() { + m_appIface = 0L; + } + KoApplicationIface *m_appIface; // to avoid a leak +}; + +KoApplication::KoApplication() + : KApplication( initHack() ) +{ + d = new KoApplicationPrivate; + + // Initialize all KOffice directories etc. + KoGlobal::initialize(); + + // Prepare a DCOP interface + d->m_appIface = new KoApplicationIface; + dcopClient()->setDefaultObject( d->m_appIface->objId() ); + + m_starting = true; +} + +// This gets called before entering KApplication::KApplication +bool KoApplication::initHack() +{ + KCmdLineArgs::addCmdLineOptions( options, I18N_NOOP("KOffice"), "koffice", "kde" ); + return true; +} + +// Small helper for start() so that we don't forget to reset m_starting before a return +class KoApplication::ResetStarting +{ +public: + ~ResetStarting() { + KoApplication::m_starting = false; + } +}; + +bool KoApplication::start() +{ + ResetStarting resetStarting; // reset m_starting to false when we're done + Q_UNUSED( resetStarting ); + + // Find the *.desktop file corresponding to the kapp instance name + KoDocumentEntry entry = KoDocumentEntry( KoDocument::readNativeService() ); + if ( entry.isEmpty() ) + { + kdError( 30003 ) << instanceName() << "part.desktop not found." << endl; + kdError( 30003 ) << "Run 'kde-config --path services' to see which directories were searched, assuming kde startup had the same environment as your current shell." << endl; + kdError( 30003 ) << "Check your installation (did you install KOffice in a different prefix than KDE, without adding the prefix to /etc/kderc ?)" << endl; + return false; + } + + // Get the command line arguments which we have to parse + KCmdLineArgs *args= KCmdLineArgs::parsedArgs(); + int argsCount = args->count(); + + KCmdLineArgs *koargs = KCmdLineArgs::parsedArgs("koffice"); + QCString dpiValues = koargs->getOption( "dpi" ); + if ( !dpiValues.isEmpty() ) { + int sep = dpiValues.find( QRegExp( "[x, ]" ) ); + int dpiX; + int dpiY = 0; + bool ok = true; + if ( sep != -1 ) { + dpiY = dpiValues.mid( sep+1 ).toInt( &ok ); + dpiValues.truncate( sep ); + } + if ( ok ) { + dpiX = dpiValues.toInt( &ok ); + if ( ok ) { + if ( !dpiY ) dpiY = dpiX; + KoGlobal::setDPI( dpiX, dpiY ); + } + } + } + + // No argument -> create an empty document + if ( !argsCount ) { + KoDocument* doc = entry.createDoc( 0, "Document" ); + if ( !doc ) + return false; + KoMainWindow *shell = new KoMainWindow( doc->instance() ); + shell->show(); + QObject::connect(doc, SIGNAL(sigProgress(int)), shell, SLOT(slotProgress(int))); + // for initDoc to fill in the recent docs list + // and for KoDocument::slotStarted + doc->addShell( shell ); + + if ( doc->checkAutoSaveFile() ) { + shell->setRootDocument( doc ); + } else { + doc->showStartUpWidget( shell ); + } + + // FIXME This needs to be moved someplace else + QObject::disconnect(doc, SIGNAL(sigProgress(int)), shell, SLOT(slotProgress(int))); + } else { + bool print = koargs->isSet("print"); + bool doTemplate = koargs->isSet("template"); + koargs->clear(); + + // Loop through arguments + + short int n=0; // number of documents open + short int nPrinted = 0; + for(int i=0; i < argsCount; i++ ) + { + // For now create an empty document + KoDocument* doc = entry.createDoc( 0 ); + if ( doc ) + { + // show a shell asap + KoMainWindow *shell = new KoMainWindow( doc->instance() ); + if (!print) + shell->show(); + // are we just trying to open a template? + if ( doTemplate ) { + QStringList paths; + if ( args->url(i).isLocalFile() && QFile::exists(args->url(i).path()) ) + { + paths << QString(args->url(i).path()); + kdDebug(30003) << "using full path..." << endl; + } else { + QString desktopName(args->arg(i)); + QString appName = KGlobal::instance()->instanceName(); + + paths = KGlobal::dirs()->findAllResources("data", appName +"/templates/*/" + desktopName ); + if ( paths.isEmpty()) { + paths = KGlobal::dirs()->findAllResources("data", appName +"/templates/" + desktopName ); + } + if ( paths.isEmpty()) { + KMessageBox::error(0L, i18n("No template found for: %1 ").arg(desktopName) ); + delete shell; + } else if ( paths.count() > 1 ) { + KMessageBox::error(0L, i18n("Too many templates found for: %1").arg(desktopName) ); + delete shell; + } + } + + if ( !paths.isEmpty() ) { + KURL templateBase; + templateBase.setPath(paths[0]); + KDesktopFile templateInfo(paths[0]); + + QString templateName = templateInfo.readURL(); + KURL templateURL; + templateURL.setPath( templateBase.directory() + "/" + templateName ); + if ( shell->openDocument(doc, templateURL )) { + doc->resetURL(); + doc->setEmpty(); + doc->setTitleModified(); + kdDebug(30003) << "Template loaded..." << endl; + n++; + } else { + KMessageBox::error(0L, i18n("Template %1 failed to load.").arg(templateURL.prettyURL()) ); + delete shell; + } + } + // now try to load + } else if ( shell->openDocument( doc, args->url(i) ) ) { + if ( print ) { + shell->print(false /*we want to get the dialog*/); + // delete shell; done by ~KoDocument + nPrinted++; + } else { + // Normal case, success + n++; + } + } else { + // .... if failed + // delete doc; done by openDocument + // delete shell; done by ~KoDocument + } + } + } + if ( print ) + return nPrinted > 0; + if (n == 0) // no doc, e.g. all URLs were malformed + return false; + } + + args->clear(); + // not calling this before since the program will quit there. + return true; +} + +KoApplication::~KoApplication() +{ + delete d->m_appIface; + delete d; +} + +bool KoApplication::isStarting() +{ + return KoApplication::m_starting; +} + +#include <KoApplication.moc> diff --git a/lib/kofficecore/KoApplication.h b/lib/kofficecore/KoApplication.h new file mode 100644 index 00000000..ecd9b943 --- /dev/null +++ b/lib/kofficecore/KoApplication.h @@ -0,0 +1,85 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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. +*/ + +#ifndef __ko_app_h__ +#define __ko_app_h__ + +namespace std { } +using namespace std; +#include <kapplication.h> +#include <koffice_export.h> + +class KoApplicationPrivate; + +/** + * @brief Base class for all %KOffice apps + * + * This class handles arguments given on the command line and + * shows a generic about dialog for all KOffice apps. + * + * In addition it adds the standard directories where KOffice applications + * can find their images etc. + * + * If the last mainwindow becomes closed, KoApplication automatically + * calls QApplication::quit. + */ +class KOFFICECORE_EXPORT KoApplication : public KApplication +{ + Q_OBJECT + +public: + /** + * Creates an application object, adds some standard directories and + * initializes kimgio. + */ + KoApplication(); + + /** + * Destructor. + */ + virtual ~KoApplication(); + + // ######### Bad name + /** + * Call this to start the application. + * + * Parses command line arguments and creates the initial shells and docs + * from them (or an empty doc if no cmd-line argument is specified ). + * + * You must call this method directly before calling QApplication::exec. + * + * It is valid behaviour not to call this method at all. In this case you + * have to process your command line parameters by yourself. + */ + virtual bool start(); + + /** + * @return true if the application is starting + */ + static bool isStarting(); + +private: + bool initHack(); + KoApplicationPrivate* d; + static bool m_starting ; ///< is the application starting or not + class ResetStarting; + friend class ResetStarting; +}; + +#endif diff --git a/lib/kofficecore/KoApplicationIface.cc b/lib/kofficecore/KoApplicationIface.cc new file mode 100644 index 00000000..eb6f4d2d --- /dev/null +++ b/lib/kofficecore/KoApplicationIface.cc @@ -0,0 +1,97 @@ +/* This file is part of the KDE project + Copyright (C) 2000 David Faure <faure@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. + + 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 <dcopclient.h> +#include <kdebug.h> +#include <klocale.h> +#include <kmessagebox.h> + +#include "KoApplication.h" +#include "KoApplicationIface.h" +#include "KoDocument.h" +#include "KoDocumentIface.h" +#include "KoMainWindow.h" +#include "KoQueryTrader.h" +#include "KoView.h" + +KoApplicationIface::KoApplicationIface() + : DCOPObject( "KoApplicationIface" ) +{ +} + +KoApplicationIface::~KoApplicationIface() +{ +} + +DCOPRef KoApplicationIface::createDocument( const QString &nativeFormat ) +{ + KoDocumentEntry entry = KoDocumentEntry::queryByMimeType( nativeFormat ); + if ( entry.isEmpty() ) + { + KMessageBox::questionYesNo( 0, i18n( "Unknown KOffice MimeType %s. Check your installation." ).arg( nativeFormat ) ); + return DCOPRef(); + } + KoDocument* doc = entry.createDoc( 0 ); + return DCOPRef( kapp->dcopClient()->appId(), doc->dcopObject()->objId() ); +} + +QValueList<DCOPRef> KoApplicationIface::getDocuments() +{ + QValueList<DCOPRef> lst; + QPtrList<KoDocument> *documents = KoDocument::documentList(); + if ( documents ) + { + QPtrListIterator<KoDocument> it( *documents ); + for (; it.current(); ++it ) + lst.append( DCOPRef( kapp->dcopClient()->appId(), it.current()->dcopObject()->objId() ) ); + } + return lst; +} + +QValueList<DCOPRef> KoApplicationIface::getViews() +{ + QValueList<DCOPRef> lst; + QPtrList<KoDocument> *documents = KoDocument::documentList(); + if ( documents ) + { + QPtrListIterator<KoDocument> it( *documents ); + for (; it.current(); ++it ) + { + QPtrListIterator<KoView> itview( it.current()->views() ); + for ( ; itview.current(); ++itview ) + lst.append( DCOPRef( kapp->dcopClient()->appId(), itview.current()->dcopObject()->objId() ) ); + } + } + return lst; +} + +QValueList<DCOPRef> KoApplicationIface::getWindows() +{ + QValueList<DCOPRef> lst; + QPtrList<KMainWindow> *mainWindows = KMainWindow::memberList; + if ( mainWindows ) + { + QPtrListIterator<KMainWindow> it( *mainWindows ); + for (; it.current(); ++it ) + lst.append( DCOPRef( kapp->dcopClient()->appId(), + static_cast<KoMainWindow *>(it.current())->dcopObject()->objId() ) ); + } + return lst; +} diff --git a/lib/kofficecore/KoApplicationIface.h b/lib/kofficecore/KoApplicationIface.h new file mode 100644 index 00000000..ac72f09a --- /dev/null +++ b/lib/kofficecore/KoApplicationIface.h @@ -0,0 +1,66 @@ +/* This file is part of the KDE project + Copyright (C) 2000 David Faure <faure@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. + + 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. +*/ + +#ifndef __KoApplicationIface_h__ +#define __KoApplicationIface_h__ + +#include <dcopobject.h> +#include <qvaluelist.h> +#include <dcopref.h> + +/** + * DCOP interface for any KOffice application (entry point) + */ +class KoApplicationIface : public DCOPObject +{ + K_DCOP +public: + + KoApplicationIface(); + ~KoApplicationIface(); + +k_dcop: + /** + * Creates a new document for the given native mimetype + * Use it to create a shell and to load an existing file, if any + */ + DCOPRef createDocument( const QString &nativeFormat ); + + /** + * @return a list of references to all the documents + * (see KoDocumentIface) + */ + QValueList<DCOPRef> getDocuments(); + + /** + * @return a list of references to all the views + * (see KoViewIface) + * Convenience method to avoid iterating over all documents to get all the views. + */ + QValueList<DCOPRef> getViews(); + + /** + * @return a list of references to all the windows + * (see KoMainWindowIface) + */ + QValueList<DCOPRef> getWindows(); +}; + +#endif + diff --git a/lib/kofficecore/KoChild.cpp b/lib/kofficecore/KoChild.cpp new file mode 100644 index 00000000..8207d086 --- /dev/null +++ b/lib/kofficecore/KoChild.cpp @@ -0,0 +1,366 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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 <KoChild.h> + +#include <qpainter.h> + +#include <kdebug.h> + +class KoChild::KoChildPrivate +{ +public: + KoChildPrivate() + { + m_contentsX = m_contentsY = 0; + } + ~KoChildPrivate() + { + } + + QRect m_geometry; + + double m_rotation; + double m_shearX; + double m_shearY; + QPoint m_rotationPoint; + double m_scaleX; + double m_scaleY; + QWMatrix m_matrix; + bool m_lock; + QPointArray m_old; + bool m_transparent; + int m_contentsX; + int m_contentsY; +}; + +KoChild::KoChild( QObject *parent, const char *name ) +: QObject( parent, name ) +{ + d = new KoChildPrivate; + + d->m_scaleX = d->m_scaleY = 1.0; + d->m_shearX = d->m_shearY = 0.0; + d->m_rotation = 0.0; + d->m_lock = false; + d->m_transparent = false; + + updateMatrix(); +} + +KoChild::~KoChild() +{ + delete d; +} + +void KoChild::setGeometry( const QRect &rect, bool noEmit ) +{ + if ( !d->m_lock ) + d->m_old = framePointArray(); + + d->m_geometry = rect; + + // Embedded objects should have a minimum size of 3, so they can be selected + if( d->m_geometry.width() < 3 ) + d->m_geometry.setWidth( 3 ); + + if( d->m_geometry.height() < 3 ) + d->m_geometry.setHeight( 3 ); + + updateMatrix(); + + if ( !d->m_lock && !noEmit ) + emit changed( this ); +} + +QRect KoChild::geometry() const +{ + return d->m_geometry; +} + +QRegion KoChild::region( const QWMatrix &matrix ) const +{ + return QRegion( pointArray( matrix ) ); +} + +QPointArray KoChild::pointArray( const QWMatrix &matrix ) const +{ + return pointArray( QRect( 0, 0, d->m_geometry.width(), d->m_geometry.height() ), matrix ); +} + +//bool KoChild::contains( const QPoint &point ) const +//{ +// return region().contains( point ); +//} + +QRect KoChild::boundingRect() const +{ + return pointArray().boundingRect(); +} + +bool KoChild::isRectangle() const +{ + return !( d->m_shearX != 0.0 || d->m_shearY != 0.0 || d->m_rotation != 0.0 ); +} + +void KoChild::setClipRegion( QPainter &painter, bool combine ) +{ + painter.setClipping( true ); + if ( combine && !painter.clipRegion().isEmpty() ) + painter.setClipRegion( region( painter.worldMatrix() ).intersect( painter.clipRegion() ) ); + else + painter.setClipRegion( region( painter.worldMatrix() ) ); +} + +void KoChild::setScaling( double x, double y ) +{ + if ( !d->m_lock ) + d->m_old = framePointArray(); + + d->m_scaleX = x; + d->m_scaleY = y; + + // why is that commented out? (Simon) + // This is commented out, because KoChild::transform() scales + // the world matrix explicitly and updateMatrix() doesn't even + // handle scaling (Werner) + //updateMatrix() + + if ( !d->m_lock ) + emit changed( this ); +} + +double KoChild::xScaling() const +{ + return d->m_scaleX; +} + +double KoChild::yScaling() const +{ + return d->m_scaleY; +} + +void KoChild::setShearing( double x, double y ) +{ + if ( !d->m_lock ) + d->m_old = framePointArray(); + + d->m_shearX = x; + d->m_shearY = y; + + updateMatrix(); + + if ( !d->m_lock ) + emit changed( this ); +} + +double KoChild::xShearing() const +{ + return d->m_shearX; +} + +double KoChild::yShearing() const +{ + return d->m_shearY; +} + +void KoChild::setRotation( double rot ) +{ + if ( !d->m_lock ) + d->m_old = framePointArray(); + + d->m_rotation = rot; + updateMatrix(); + + if ( !d->m_lock ) + emit changed( this ); +} + +double KoChild::rotation() const +{ + return d->m_rotation; +} + +void KoChild::setRotationPoint( const QPoint &pos ) +{ + if ( !d->m_lock ) + d->m_old = framePointArray(); + + d->m_rotationPoint = pos; + updateMatrix(); + + if ( !d->m_lock ) + emit changed( this ); +} + +QPoint KoChild::rotationPoint() const +{ + return d->m_rotationPoint; +} + +void KoChild::transform( QPainter &painter ) +{ + setClipRegion( painter, true ); + + QWMatrix m = painter.worldMatrix(); + m = d->m_matrix * m; + m.scale( d->m_scaleX, d->m_scaleY ); + painter.setWorldMatrix( m ); +} + +void KoChild::setContentsPos( int x, int y ) +{ + d->m_contentsX = x; + d->m_contentsY = y; +} + +QRect KoChild::contentRect() const +{ + return QRect( d->m_contentsX, d->m_contentsY, int(d->m_geometry.width() / d->m_scaleX), + int(d->m_geometry.height() / d->m_scaleY) ); +} + +QPointArray KoChild::framePointArray( const QWMatrix &matrix ) const +{ + return pointArray( QRect( -6, -6, d->m_geometry.width() + 12, d->m_geometry.height() + 12 ), matrix ); +} + +QRegion KoChild::frameRegion( const QWMatrix &matrix, bool solid ) const +{ + const QPointArray arr = framePointArray( matrix ); + const QRegion frameReg( arr ); + + if ( solid ) + return frameReg; + + const QRegion reg = region( matrix ); + return frameReg.subtract( reg ); +} + +QPointArray KoChild::pointArray( const QRect &r, const QWMatrix &matrix ) const +{ + QPoint topleft = d->m_matrix.map( QPoint( r.left(), r.top() ) ); + QPoint topright = d->m_matrix.map( QPoint( r.right(), r.top() ) ); + QPoint bottomleft = d->m_matrix.map( QPoint( r.left(), r.bottom() ) ); + QPoint bottomright = d->m_matrix.map( QPoint( r.right(), r.bottom() ) ); + + QPointArray arr( 4 ); + arr.setPoint( 0, topleft ); + arr.setPoint( 1, topright ); + arr.setPoint( 2, bottomright ); + arr.setPoint( 3, bottomleft ); + + for( int i = 0; i < 4; ++i ) + arr.setPoint( i, matrix.map( arr.point( i ) ) ); + + return arr; +} + +void KoChild::updateMatrix() +{ + QWMatrix r; + r.rotate( - d->m_rotation ); + QPoint p = r.map( QPoint( d->m_rotationPoint.x(), + d->m_rotationPoint.y() ) ); + + QWMatrix m; + m.rotate( d->m_rotation ); + m.translate( -d->m_rotationPoint.x() + d->m_geometry.x(), -d->m_rotationPoint.y() + d->m_geometry.y() ); + m.translate( p.x(), p.y() ); + m.shear( d->m_shearX, d->m_shearY ); + + d->m_matrix = m; +} + +QWMatrix KoChild::matrix() const +{ + return d->m_matrix; +} + +void KoChild::lock() +{ + if ( d->m_lock ) + return; + + d->m_old = framePointArray(); + d->m_lock = true; +} + +void KoChild::unlock() +{ + if ( !d->m_lock ) + return; + + d->m_lock = false; + emit changed( this ); +} + +bool KoChild::locked() const +{ + return d->m_lock; +} + +QPointArray KoChild::oldPointArray( const QWMatrix &matrix ) +{ + QPointArray arr = d->m_old; + + for( int i = 0; i < 4; ++i ) + arr.setPoint( i, matrix.map( arr.point( i ) ) ); + + return arr; +} + +void KoChild::setTransparent( bool transparent ) +{ + d->m_transparent = transparent; +} + +bool KoChild::isTransparent() const +{ + return d->m_transparent; +} + +KoChild::Gadget KoChild::gadgetHitTest( const QPoint &p ) +{ + if ( !frameRegion().contains( p ) ) + return NoGadget; + + if ( QRegion( pointArray( QRect( -5, -5, 5, 5 ) ) ).contains( p ) ) + return TopLeft; + if ( QRegion( pointArray( QRect( d->m_geometry.width() / 2 - 3, -5, 5, 5 ) ) ).contains( p ) ) + return TopMid; + if ( QRegion( pointArray( QRect( d->m_geometry.width(), -5, 5, 5 ) ) ).contains( p ) ) + return TopRight; + if ( QRegion( pointArray( QRect( -5, d->m_geometry.height() / 2 - 3, 5, 5 ) ) ).contains( p ) ) + return MidLeft; + if ( QRegion( pointArray( QRect( -5, d->m_geometry.height(), 5, 5 ) ) ).contains( p ) ) + return BottomLeft; + if ( QRegion( pointArray( QRect( d->m_geometry.width() / 2 - 3, + d->m_geometry.height(), 5, 5 ) ) ).contains( p ) ) + return BottomMid; + if ( QRegion( pointArray( QRect( d->m_geometry.width(), d->m_geometry.height(), 5, 5 ) ) ).contains( p ) ) + return BottomRight; + if ( QRegion( pointArray( QRect( d->m_geometry.width(), + d->m_geometry.height() / 2 - 3, 5, 5 ) ) ).contains( p ) ) + return MidRight; + + return Move; +} + +#include <KoChild.moc> diff --git a/lib/kofficecore/KoChild.h b/lib/kofficecore/KoChild.h new file mode 100644 index 00000000..26ad4a08 --- /dev/null +++ b/lib/kofficecore/KoChild.h @@ -0,0 +1,312 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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. +*/ +#ifndef __koChild_h__ +#define __koChild_h__ + +#include <qobject.h> +#include <qwmatrix.h> +#include <koffice_export.h> + +/** + * KoChild is an abstract base class that represents the geometry + * associated with an embedded document. In general it handles its position + * relative to the embedded document's parent. + * + * In detail it handles size, matrix operations and can give you + * a clip region. It can deal with scaling, rotation etc. because it + * makes heavy usage of QWMatrix. + * + * After applying the matrix, viewGeometry() applies zooming, but can be + * reimplemented to also apply e.g. some translation by the application + * (e.g. for centering the page). + * + * @see KoDocumentChild KoViewChild + */ +class KOFFICECORE_EXPORT KoChild : public QObject +{ + Q_OBJECT +public: + + /** + * The gadget generally identifies where a child has been hit (generally + * by the mouse pointer). + * Based on this information different actions can be taken, for example + * moving the child or opening a context menu. NoGadget means that + * this child has not been hit. + * + * @see #gadgetHitTest + */ + enum Gadget { NoGadget, TopLeft, TopMid, TopRight, MidLeft, MidRight, + BottomLeft, BottomMid, BottomRight, Move }; + + KoChild( QObject *parent = 0, const char *name = 0 ); + virtual ~KoChild(); + + /** + * Sets a new geometry for this child document. + * Use noEmit = true if you do not want the 'changed'-signal to be emitted + */ + void setGeometry( const QRect &rect, bool noEmit = false ); + + /** + * @return the rectangle that would be used to display this + * child document if the child is not rotated or + * subject to some other geometric transformation. + * The rectangle is in the coordinate system of the parent, + * using unzoomed coordinates in points. + * + * @see #setGeometry + */ + QRect geometry() const; + + /** + * @return the region of this child part relative to the + * coordinate system of the parent. + * The region is transformed with the passed + * matrix. + */ + virtual QRegion region( const QWMatrix& = QWMatrix() ) const; + + /** + * @return the polygon which surrounds the child part. The points + * are in coordinates of the parent. + * The points are transformed with the + * passed matrix. + */ + virtual QPointArray pointArray( const QWMatrix &matrix = QWMatrix() ) const; + + /** + * Tests whether the part contains a certain point. The point is + * in the coordinate system of the parent. + */ + //virtual bool contains( const QPoint& ) const; + + /** + * @return the effective bounding rect after all transformations. + * The coordinates of the rectangle are in the coordinate system + * of the parent. + */ + QRect boundingRect() const; + + /** + * Scales the content of the child part. However, that does not + * affect the size of the child part. + */ + virtual void setScaling( double x, double y ); + + /** + * @return the x axis scaling of the child part + */ + virtual double xScaling() const; + + /** + * @return the y axis scaling of the child part + */ + virtual double yScaling() const; + + /** + * Shears the content of the child part. + */ + virtual void setShearing( double x, double y ); + + /** + * @return the x axis shearing of the child part + */ + virtual double xShearing() const; + + /** + * @return the y axis shearing of the child part + */ + virtual double yShearing() const; + + /** + * Sets the angle of rotation. + */ + virtual void setRotation( double ); + + /** + * @return the angle of rotation + */ + virtual double rotation() const; + + /** + * Sets the center of the rotation to the point @p pos. + */ + virtual void setRotationPoint( const QPoint& pos ); + + /** + * @return the center of the rotation + */ + virtual QPoint rotationPoint() const; + + /** + * @return true if the child part is an orthogonal rectangle relative + * to its parents coordinate system. + */ + bool isRectangle() const; + + /** + * Sets the clip region of the painter, so that only pixels of the + * child part can be drawn. + * + * @param painter the painter do modify. + * @param combine tells whether the new clip region is an intersection + * of the current region with the childs region or whether only + * the childs region is set. + */ + virtual void setClipRegion( QPainter& painter, bool combine = true ); + + /** + * Transforms the painter (its worldmatrix and the clipping) + * in such a way that the painter can be passed to the child part + * for drawing. + */ + virtual void transform( QPainter& painter ); + + /** + * Sets the position of the content relative to the child frame. + * This can be used to create a border between the frame border + * and the actual content. + */ + virtual void setContentsPos( int x, int y ); + + /** + * @return the contents rectangle that is visible. + * This value depends on the scaling and the geometry. + * This is the value that is passed to KoDocument::paintContent. + * + * @see #xScaling #geometry + */ + virtual QRect contentRect() const; + + /** + * @return the region of the child frame. + * If solid is set to true the complete area of the child region + * is returned, otherwise only the child border is returned. + */ + virtual QRegion frameRegion( const QWMatrix& matrix = QWMatrix(), bool solid = false ) const; + + /** + * @return the frame geometry including a border (6 pixels) as a point + * array with 4 points, one for each corner, transformed by given matrix. + */ + virtual QPointArray framePointArray( const QWMatrix &matrix = QWMatrix() ) const; + + /** + * @return the current transformation of this child as matrix. + * This includes translation, scale, rotation, shearing. + * + * @see #updateMatrix + */ + virtual QWMatrix matrix() const; + + /** + * Locks this child and stores the current transformation. + * A locked child does not emit changed signals. + * + * This is useful if a series of changes are done on this + * child and only the final result is of interest (GUI updating,...). + * + * @see #locked #unlock + */ + void lock(); + + /** + * Unlocks this child and emits a changed signal. + */ + void unlock(); + + /** + * If the child is locked, geometry changes + * (including scaling, rotation, ...) are not backed up. + * + * As long as this child is locked, the backed up + * geometry state can be recovered with oldPointArray. + * + * @return true when this child is locked. + * + * @see #locked #unlock #oldPointArray + */ + bool locked() const; + + /** + * @return the backed up geometry transformed by given matrix. + */ + virtual QPointArray oldPointArray( const QWMatrix &matrix ); + + /** + * Marks this child as either transparent or not. + * @param transparent set this child to transparent (true) + * or opaque (false). + * + * @see #isTransparent + */ + virtual void setTransparent( bool transparent ); + + /** + * It might be interesting for view updates and repainting in general + * whether a child is transparent or not. + * @return true when this child is marked as transparent. + */ + virtual bool isTransparent() const; + + /** + * Different actions are taken depending on where a child frame is + * hit. Two gadgets are known: one for the border (5 pixels) and one + * for the inner area. + * @return the gadget identification for the hit area. + * @param p the hit position. + * + * @see #Gadget + */ + virtual Gadget gadgetHitTest( const QPoint& p ); + +signals: + + /** + * Emitted every time this child changes, but only if this child is not + * locked. + * @see #locked + */ + void changed( KoChild *thisChild ); + +protected: + + /** + * @return point array with the 4 corners of given rectangle, which is + * transformed by given matrix. + * + * @param matrix the transformation of r. + * @param r the rectangle for which the point array should be created. + */ + virtual QPointArray pointArray( const QRect& r, const QWMatrix& matrix = QWMatrix() ) const; + + /** + * Stores the current transformation of this child into a matrix. + * + * @see #matrix + */ + virtual void updateMatrix(); +private: + + class KoChildPrivate; + KoChildPrivate *d; +}; + +#endif diff --git a/lib/kofficecore/KoContainerHandler.cpp b/lib/kofficecore/KoContainerHandler.cpp new file mode 100644 index 00000000..78060592 --- /dev/null +++ b/lib/kofficecore/KoContainerHandler.cpp @@ -0,0 +1,395 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@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 "KoContainerHandler.h" +#include <KoView.h> +#include <math.h> +#include <kcursor.h> +#include <kdebug.h> + +KoEventHandler::KoEventHandler( QObject* target ) +{ + m_target = target; + + m_target->installEventFilter( this ); +} + +KoEventHandler::~KoEventHandler() +{ +} + +QObject* KoEventHandler::target() +{ + return m_target; +} + +// ------------------------------------------------------ + +class KoPartResizeHandlerPrivate { +public: + KoPartResizeHandlerPrivate( const QWMatrix& matrix, KoView *view, KoChild* child, + KoChild::Gadget gadget, const QPoint& point ) : + m_gadget(gadget), m_view(view), m_child(child), m_parentMatrix(matrix) { + + m_geometryStart = child->geometry(); + m_matrix = child->matrix() * matrix; + m_invertParentMatrix = matrix.invert(); + + bool ok = true; + m_invert = m_matrix.invert( &ok ); + Q_ASSERT( ok ); + m_mouseStart = m_invert.map( m_invertParentMatrix.map( point ) ); + } + ~KoPartResizeHandlerPrivate() {} + + KoChild::Gadget m_gadget; + QPoint m_mouseStart; + QRect m_geometryStart; + KoView* m_view; + KoChild* m_child; + QWMatrix m_invert; + QWMatrix m_matrix; + QWMatrix m_parentMatrix; + QWMatrix m_invertParentMatrix; +}; + +KoPartResizeHandler::KoPartResizeHandler( QWidget* widget, const QWMatrix& matrix, KoView* view, KoChild* child, + KoChild::Gadget gadget, const QPoint& point ) + : KoEventHandler( widget ) +{ + child->lock(); + d=new KoPartResizeHandlerPrivate(matrix, view, child, gadget, point); +} + +KoPartResizeHandler::~KoPartResizeHandler() +{ + d->m_child->unlock(); + delete d; + d=0L; +} + +void KoPartResizeHandler::repaint(QRegion &rgn) +{ + rgn = rgn.unite( d->m_child->frameRegion( d->m_parentMatrix, true ) ); + // rgn.translate(- d->m_view->canvasXOffset(), - d->m_view->canvasYOffset()); + ((QWidget*)target())->repaint( rgn ); +} + +bool KoPartResizeHandler::eventFilter( QObject*, QEvent* ev ) +{ + if ( ev->type() == QEvent::MouseButtonRelease ) + { + delete this; + return true; + } + else if ( ev->type() == QEvent::MouseMove ) + { + QMouseEvent* e = (QMouseEvent*)ev; + QPoint p = d->m_invert.map( d->m_invertParentMatrix.map( e->pos() /*+ QPoint(d->m_view->canvasXOffset(), d->m_view->canvasYOffset()) */ ) ); + QRegion rgn( d->m_child->frameRegion( d->m_parentMatrix, true ) ); + + double x1_x, x1_y, x2_x, x2_y; + d->m_matrix.map( double( p.x() ), 0.0, &x1_x, &x1_y ); + d->m_matrix.map( double( d->m_mouseStart.x() ), 0.0, &x2_x, &x2_y ); + double y1_x, y1_y, y2_x, y2_y; + d->m_matrix.map( 0.0, double( p.y() ), &y1_x, &y1_y ); + d->m_matrix.map( 0.0, double( d->m_mouseStart.y() ), &y2_x, &y2_y ); + + double dx = x2_x - x1_x; + double dy = x2_y - x1_y; + int x = int( sqrt( dx * dx + dy * dy ) * ( d->m_mouseStart.x() < p.x() ? 1.0 : -1.0 ) ); + + dx = y2_x - y1_x; + dy = y2_y - y1_y; + int y = int( sqrt( dx * dx + dy * dy ) * ( d->m_mouseStart.y() < p.y() ? 1.0 : -1.0 ) ); + + switch( d->m_gadget ) + { + case KoChild::TopLeft: + { + x = QMIN( d->m_geometryStart.width() - 1, x ); + y = QMIN( d->m_geometryStart.height() - 1, y ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x() + x, d->m_geometryStart.y() + y, + d->m_geometryStart.width() - x, d->m_geometryStart.height() - y ) ); + repaint(rgn); + } + break; + case KoChild::TopMid: + { + y = QMIN( d->m_geometryStart.height() - 1, y ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x(), d->m_geometryStart.y() + y, + d->m_geometryStart.width(), d->m_geometryStart.height() - y ) ); + repaint(rgn); + } + break; + case KoChild::TopRight: + { + x = QMAX( -d->m_geometryStart.width() + 1, x ); + y = QMIN( d->m_geometryStart.height() - 1, y ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x(), d->m_geometryStart.y() + y, + d->m_geometryStart.width() + x, d->m_geometryStart.height() - y ) ); + repaint(rgn); + } + break; + case KoChild::MidLeft: + { + x = QMIN( d->m_geometryStart.width() - 1, x ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x() + x, d->m_geometryStart.y(), + d->m_geometryStart.width() - x, d->m_geometryStart.height() ) ); + repaint(rgn); + } + break; + case KoChild::MidRight: + { + x = QMAX( -d->m_geometryStart.width() + 1, x ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x(), d->m_geometryStart.y(), + d->m_geometryStart.width() + x, d->m_geometryStart.height() ) ); + repaint(rgn); + } + break; + case KoChild::BottomLeft: + { + x = QMIN( d->m_geometryStart.width() - 1, x ); + y = QMAX( -d->m_geometryStart.height() + 1, y ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x() + x, d->m_geometryStart.y(), + d->m_geometryStart.width() - x, d->m_geometryStart.height() + y ) ); + repaint(rgn); + } + break; + case KoChild::BottomMid: + { + y = QMAX( -d->m_geometryStart.height() + 1, y ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x(), d->m_geometryStart.y(), + d->m_geometryStart.width(), d->m_geometryStart.height() + y ) ); + repaint(rgn); + } + break; + case KoChild::BottomRight: + { + x = QMAX( -d->m_geometryStart.width() + 1, x ); + y = QMAX( -d->m_geometryStart.height() + 1, y ); + + d->m_child->setGeometry( QRect( d->m_geometryStart.x(), d->m_geometryStart.y(), + d->m_geometryStart.width() + x, d->m_geometryStart.height() + y ) ); + repaint(rgn); + } + break; + default: + Q_ASSERT( 0 ); + } + return true; + } + return false; +} + +// -------------------------------------------------------------- + +class KoPartMoveHandlerPrivate { +public: + KoPartMoveHandlerPrivate( const QWMatrix& matrix, KoView* view, KoChild* child, + const QPoint& point) : m_view(view), m_dragChild(child), + m_parentMatrix(matrix) { + m_invertParentMatrix = matrix.invert(); + m_mouseDragStart = m_invertParentMatrix.map( point ); + m_geometryDragStart = m_dragChild->geometry(); + m_rotationDragStart = m_dragChild->rotationPoint(); + } + ~KoPartMoveHandlerPrivate() {} + + KoView* m_view; + KoChild* m_dragChild; + QPoint m_mouseDragStart; + QRect m_geometryDragStart; + QPoint m_rotationDragStart; + QWMatrix m_invertParentMatrix; + QWMatrix m_parentMatrix; +}; + +KoPartMoveHandler::KoPartMoveHandler( QWidget* widget, const QWMatrix& matrix, KoView* view, KoChild* child, + const QPoint& point ) + : KoEventHandler( widget ) +{ + child->lock(); + d=new KoPartMoveHandlerPrivate(matrix, view, child, point); +} + +KoPartMoveHandler::~KoPartMoveHandler() +{ + d->m_dragChild->unlock(); + delete d; + d=0L; +} + +bool KoPartMoveHandler::eventFilter( QObject*, QEvent* ev ) +{ + if ( ev->type() == QEvent::MouseButtonRelease ) + { + delete this; + return true; + } + else if ( ev->type() == QEvent::MouseMove ) + { + QMouseEvent* e = (QMouseEvent*)ev; + + QRegion bound = d->m_dragChild->frameRegion( d->m_parentMatrix, true ); + QPoint pos = d->m_invertParentMatrix.map( e->pos() /* + QPoint(d->m_view->canvasXOffset(), d->m_view->canvasYOffset()) */ ); + d->m_dragChild->setGeometry( QRect( d->m_geometryDragStart.x() + pos.x() - d->m_mouseDragStart.x(), + d->m_geometryDragStart.y() + pos.y() - d->m_mouseDragStart.y(), + d->m_geometryDragStart.width(), d->m_geometryDragStart.height() ) ); + d->m_dragChild->setRotationPoint( QPoint( d->m_rotationDragStart.x() + pos.x() - d->m_mouseDragStart.x(), + d->m_rotationDragStart.y() + pos.y() - d->m_mouseDragStart.y() ) ); + bound = bound.unite( d->m_dragChild->frameRegion( d->m_parentMatrix, false ) ); + // bound.translate(- d->m_view->canvasXOffset(), - d->m_view->canvasYOffset()); + ((QWidget*)target())->repaint( bound ); + + return true; + } + + return false; +} + +// ------------------------------------------------------- + +KoContainerHandler::KoContainerHandler( KoView* view, QWidget* widget ) + : KoEventHandler( widget ) +{ + m_view = view; +} + +KoContainerHandler::~KoContainerHandler() +{ +} + +bool KoContainerHandler::eventFilter( QObject*, QEvent* ev ) +{ + if ( ev->type() == QEvent::KeyPress ) + { + QKeyEvent* keyEvent=(QKeyEvent*)ev; + + KoChild* child=m_view->selectedChild(); + + if (child != 0) + { + if (keyEvent->key()==Qt::Key_Delete) + emit deleteChild(child); + } + } + + if ( ev->type() == QEvent::MouseButtonPress ) + { + KoChild::Gadget gadget; + QPoint pos; + QMouseEvent *e=static_cast<QMouseEvent*>(ev); + KoChild *ch=child(gadget, pos, e); + + if ( e->button() == RightButton && gadget != KoChild::NoGadget ) + { + emit popupMenu( ch, e->globalPos() ); + return true; + } + else if ( e->button() == LeftButton && gadget == KoChild::Move ) + { + (void)new KoPartMoveHandler( static_cast<QWidget*>(target()), m_view->matrix(), m_view, ch, pos ); + return true; + } + else if ( e->button() == LeftButton && gadget != KoChild::NoGadget ) + { + (void)new KoPartResizeHandler( static_cast<QWidget*>(target()), m_view->matrix(), m_view, ch, gadget, pos ); + return true; + } + return false; + } + else if ( ev->type() == QEvent::MouseMove ) + { + QWidget *targetWidget = static_cast<QWidget *>( target() ); + KoChild::Gadget gadget; + QPoint pos; + QMouseEvent *e=static_cast<QMouseEvent*>(ev); + child(gadget, pos, e); + + bool retval = true; + if ( gadget == KoChild::NoGadget ) + retval = false; + + if ( gadget == KoChild::TopLeft || gadget == KoChild::BottomRight ) + targetWidget->setCursor( sizeFDiagCursor ); + else if ( gadget == KoChild::TopRight || gadget == KoChild::BottomLeft ) + targetWidget->setCursor( sizeBDiagCursor ); + else if ( gadget == KoChild::TopMid || gadget == KoChild::BottomMid ) + targetWidget->setCursor( sizeVerCursor ); + else if ( gadget == KoChild::MidLeft || gadget == KoChild::MidRight ) + targetWidget->setCursor( sizeHorCursor ); + else if ( gadget == KoChild::Move ) + targetWidget->setCursor( KCursor::handCursor() ); + else + { +// targetWidget->setCursor( arrowCursor ); + return false; + } + return retval; + } + return false; +} + +KoChild *KoContainerHandler::child(KoChild::Gadget &gadget, QPoint &pos, const QMouseEvent *ev) { + + pos = ev->pos(); //+ QPoint(m_view->canvasXOffset(), m_view->canvasYOffset()); + + pos = m_view->reverseViewTransformations( pos ); + + KoChild *child = 0; + KoDocumentChild* docChild = m_view->selectedChild(); + gadget = KoChild::NoGadget; + if ( docChild ) + { + KoViewChild *viewChild = m_view->child( docChild->document() ); + + if ( viewChild ) + child = viewChild; + else + child = docChild; + + gadget = child->gadgetHitTest( pos ); + } + if ( gadget == KoChild::NoGadget ) + { + docChild = m_view->activeChild(); + if ( docChild ) + { + KoViewChild *viewChild = m_view->child( docChild->document() ); + + if ( viewChild ) + child = viewChild; + else + child = docChild; + + gadget = child->gadgetHitTest( pos ); + } + } + return child; +} + +#include "KoContainerHandler.moc" diff --git a/lib/kofficecore/KoContainerHandler.h b/lib/kofficecore/KoContainerHandler.h new file mode 100644 index 00000000..d34d0500 --- /dev/null +++ b/lib/kofficecore/KoContainerHandler.h @@ -0,0 +1,137 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@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. +*/ + +#ifndef HANDLER_H +#define HANDLER_H + +#include <qobject.h> +#include <KoDocumentChild.h> + +class QWMatrix; + +class KoView; +class KoPartResizeHandlerPrivate; +class KoPartMoveHandlerPrivate; + +/** + * @brief An abstract base class for event handlers. + * + * The idea of an event handler is that it is created for a + * certain purpose, for example moving or resizing of a part. + * Once that action is finished, the handler will destroy + * itself. + * + * This design pattern helps you to keep your event filters + * and your mousePressEvent, mouseMoveEvent etc. methods clean. + */ +class KOFFICECORE_EXPORT KoEventHandler : public QObject +{ + Q_OBJECT +public: + KoEventHandler( QObject* target ); + ~KoEventHandler(); + + QObject* target(); + +private: + QObject* m_target; +}; + +/** + * Used by @ref KoContainerHandler internally to handle resizing of + * embedded documents. + */ +class KoPartResizeHandler : public KoEventHandler +{ + Q_OBJECT +public: + KoPartResizeHandler( QWidget* widget, const QWMatrix& matrix, KoView* view, KoChild* child, + KoChild::Gadget gadget, const QPoint& point ); + ~KoPartResizeHandler(); + +protected: + void repaint(QRegion &rgn); + bool eventFilter( QObject*, QEvent* ); + +private: + KoPartResizeHandlerPrivate *d; +}; + +/** + * Used by @ref KoContainerHandler internally to handle moving of + * embedded documents. + */ +class KoPartMoveHandler : public KoEventHandler +{ + Q_OBJECT +public: + KoPartMoveHandler( QWidget* widget, const QWMatrix& matrix, KoView* view, KoChild* child, + const QPoint& point ); + ~KoPartMoveHandler(); + +protected: + bool eventFilter( QObject*, QEvent* ); + +private: + KoPartMoveHandlerPrivate *d; +}; + +/** + * This class can handle moving and resizing of embedded + * documents in your class derived from @ref KoView. + * + * Just create one instance per view of this class and parts + * will magically be moved around on your document. + * + * This class acts like an event filter on your view, so the + * mouse events which are used for parts moving and resizing + * will never show up in your view. + * + * @see KoPartMoveHandlerPrivate + * @see KoPartResizeHandlerPrivate + */ +class KOFFICECORE_EXPORT KoContainerHandler : public KoEventHandler +{ + Q_OBJECT +public: + KoContainerHandler( KoView* view, QWidget* widget ); + ~KoContainerHandler(); + +signals: + /** + * Emitted if the user wants to open the popup menu for some + * child object. + */ + void popupMenu( KoChild*, const QPoint& global_pos ); + + /** + * Emitted if the user pressed the delete key whilst a child was selected + */ + void deleteChild( KoChild* ); + +protected: + bool eventFilter( QObject*, QEvent* ); + +private: + /// This is a little helper function to get rid of some duplicated code + KoChild *child(KoChild::Gadget &gadget, QPoint &pos, const QMouseEvent *ev); + KoView* m_view; +}; + +#endif diff --git a/lib/kofficecore/KoDetailsPane.cpp b/lib/kofficecore/KoDetailsPane.cpp new file mode 100644 index 00000000..d94a01f7 --- /dev/null +++ b/lib/kofficecore/KoDetailsPane.cpp @@ -0,0 +1,470 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Peter Simonsson <psn@linux.se> + + 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 "KoDetailsPane.h" + +#include <qcheckbox.h> +#include <qlabel.h> +#include <qfile.h> +#include <qimage.h> +#include <qheader.h> +#include <qrect.h> +#include <qbrush.h> +#include <qpainter.h> +#include <qsimplerichtext.h> +#include <qevent.h> +#include <qsplitter.h> + +#include <kinstance.h> +#include <klocale.h> +#include <klistview.h> +#include <kpushbutton.h> +#include <kconfig.h> +#include <kurl.h> +#include <kfileitem.h> +#include <kio/previewjob.h> +#include <kdebug.h> +#include <ktextbrowser.h> +#include <kapplication.h> + +#include "KoTemplates.h" + +class KoFileListItem : public KListViewItem +{ + public: + KoFileListItem(KListView* listView, QListViewItem* after, const QString& filename, + const QString& fullPath, KFileItem* fileItem) + : KListViewItem(listView, after, filename, fullPath), m_fileItem(fileItem) + { + } + + ~KoFileListItem() + { + delete m_fileItem; + } + + KFileItem* fileItem() const + { + return m_fileItem; + } + + private: + KFileItem* m_fileItem; +}; + +class KoTemplatesPanePrivate +{ + public: + KoTemplatesPanePrivate() + : m_instance(0), m_selected(false) + { + } + + KInstance* m_instance; + bool m_selected; + QString m_alwaysUseTemplate; +}; + + +KoTemplatesPane::KoTemplatesPane(QWidget* parent, KInstance* instance, + KoTemplateGroup *group, KoTemplate* /*defaultTemplate*/) + : KoDetailsPaneBase(parent, "TemplatesPane") +{ + d = new KoTemplatesPanePrivate; + d->m_instance = instance; + m_previewLabel->installEventFilter(this); + m_documentList->installEventFilter(this); + setFocusProxy(m_documentList); +#if KDE_IS_VERSION(3,4,0) + m_documentList->setShadeSortColumn(false); +#endif + + KGuiItem openGItem(i18n("Use This Template")); + m_openButton->setGuiItem(openGItem); + m_documentList->header()->hide(); + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + QString fullTemplateName = cfgGrp.readPathEntry("FullTemplateName"); + d->m_alwaysUseTemplate = cfgGrp.readPathEntry("AlwaysUseTemplate"); + connect(m_alwaysUseCheckBox, SIGNAL(clicked()), this, SLOT(alwaysUseClicked())); + changePalette(); + + if(kapp) { + connect(kapp, SIGNAL(kdisplayPaletteChanged()), this, SLOT(changePalette())); + } + + KListViewItem* selectItem = 0; + + for (KoTemplate* t = group->first(); t != 0L; t = group->next()) { + KListViewItem* item = new KListViewItem(m_documentList, t->name(), t->description(), t->file()); + QImage icon = t->loadPicture(instance).convertToImage(); + icon = icon.smoothScale(64, 64, QImage::ScaleMin); + icon.setAlphaBuffer(true); + icon = icon.copy((icon.width() - 64) / 2, (icon.height() - 64) / 2, 64, 64); + item->setPixmap(0, QPixmap(icon)); + item->setPixmap(2, t->loadPicture(instance)); + + if(d->m_alwaysUseTemplate == t->file()) { + selectItem = item; + } else if(!selectItem && (t->file() == fullTemplateName)) { + selectItem = item; + } + } + + connect(m_documentList, SIGNAL(selectionChanged(QListViewItem*)), + this, SLOT(selectionChanged(QListViewItem*))); + connect(m_documentList, SIGNAL(doubleClicked(QListViewItem*, const QPoint&, int)), + this, SLOT(openTemplate(QListViewItem*))); + connect(m_documentList, SIGNAL(returnPressed(QListViewItem*)), + this, SLOT(openTemplate(QListViewItem*))); + connect(m_openButton, SIGNAL(clicked()), this, SLOT(openTemplate())); + + if(selectItem) { + m_documentList->setSelected(selectItem, true); + d->m_selected = true; + } else { + m_documentList->setSelected(m_documentList->firstChild(), true); + } +} + +KoTemplatesPane::~KoTemplatesPane() +{ + delete d; +} + +void KoTemplatesPane::selectionChanged(QListViewItem* item) +{ + if(item) { + m_openButton->setEnabled(true); + m_alwaysUseCheckBox->setEnabled(true); + m_titleLabel->setText(item->text(0)); + m_previewLabel->setPixmap(*(item->pixmap(2))); + m_detailsLabel->setText(item->text(1)); + m_alwaysUseCheckBox->setChecked(item->text(2) == d->m_alwaysUseTemplate); + } else { + m_openButton->setEnabled(false); + m_alwaysUseCheckBox->setEnabled(false); + m_alwaysUseCheckBox->setChecked(false); + m_titleLabel->setText(QString::null); + m_previewLabel->setPixmap(QPixmap()); + } +} + +void KoTemplatesPane::openTemplate() +{ + QListViewItem* item = m_documentList->selectedItem(); + openTemplate(item); +} + +void KoTemplatesPane::openTemplate(QListViewItem* item) +{ + if(item) { + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + cfgGrp.writePathEntry("FullTemplateName", item->text(2)); + cfgGrp.writeEntry("LastReturnType", "Template"); + cfgGrp.writeEntry("AlwaysUseTemplate", d->m_alwaysUseTemplate); + emit openTemplate(item->text(2)); + } +} + +void KoTemplatesPane::changePalette() +{ + QPalette p = kapp ? kapp->palette() : palette(); + p.setBrush(QColorGroup::Base, p.brush(QPalette::Normal, QColorGroup::Background)); + p.setColor(QColorGroup::Text, p.color(QPalette::Normal, QColorGroup::Foreground)); + m_detailsLabel->setPalette(p); +} + +bool KoTemplatesPane::isSelected() +{ + return d->m_selected; +} + +void KoTemplatesPane::alwaysUseClicked() +{ + QListViewItem* item = m_documentList->selectedItem(); + + if(!m_alwaysUseCheckBox->isChecked()) { + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + cfgGrp.writeEntry("AlwaysUseTemplate", QString::null); + d->m_alwaysUseTemplate = QString::null; + } else { + d->m_alwaysUseTemplate = item->text(2); + } + + emit alwaysUseChanged(this, d->m_alwaysUseTemplate); +} + +void KoTemplatesPane::changeAlwaysUseTemplate(KoTemplatesPane* sender, const QString& alwaysUse) +{ + if(this == sender) + return; + + QListViewItem* item = m_documentList->selectedItem(); + + // If the old always use template is selected uncheck the checkbox + if(item && (item->text(2) == d->m_alwaysUseTemplate)) { + m_alwaysUseCheckBox->setChecked(false); + } + + d->m_alwaysUseTemplate = alwaysUse; +} + +bool KoTemplatesPane::eventFilter(QObject* watched, QEvent* e) +{ + if(watched == m_previewLabel) { + if(e->type() == QEvent::MouseButtonDblClick) { + openTemplate(); + } + } + + if(watched == m_documentList) { + if((e->type() == QEvent::Resize) && isShown()) { + emit splitterResized(this, m_splitter->sizes()); + } + } + + return false; +} + +void KoTemplatesPane::resizeSplitter(KoDetailsPaneBase* sender, const QValueList<int>& sizes) +{ + if(sender == this) + return; + + m_splitter->setSizes(sizes); +} + + +class KoRecentDocumentsPanePrivate +{ + public: + KoRecentDocumentsPanePrivate() + : m_previewJob(0), m_instance(0) + { + } + + ~KoRecentDocumentsPanePrivate() + { + if(m_previewJob) + m_previewJob->kill(); + } + + KIO::PreviewJob* m_previewJob; + KInstance* m_instance; +}; + +KoRecentDocumentsPane::KoRecentDocumentsPane(QWidget* parent, KInstance* instance) + : KoDetailsPaneBase(parent, "RecentDocsPane") +{ + d = new KoRecentDocumentsPanePrivate; + d->m_instance = instance; + m_previewLabel->installEventFilter(this); + m_documentList->installEventFilter(this); + setFocusProxy(m_documentList); + KGuiItem openGItem(i18n("Open This Document"), "fileopen"); + m_openButton->setGuiItem(openGItem); + m_alwaysUseCheckBox->hide(); + m_documentList->header()->hide(); + m_documentList->setSorting(-1); // Disable sorting + changePalette(); + + if(kapp) { + connect(kapp, SIGNAL(kdisplayPaletteChanged()), this, SLOT(changePalette())); + } + + QString oldGroup = instance->config()->group(); + instance->config()->setGroup("RecentFiles"); + + int i = 0; + QString value; + KFileItemList fileList; + fileList.setAutoDelete(false); + + do { + QString key = QString("File%1").arg(i); + value = instance->config()->readPathEntry(key); + + if(!value.isEmpty()) { + QString path = value; + QString name; + + // Support for kdelibs-3.5's new RecentFiles format: name[url] + if(path.endsWith("]")) { + int pos = path.find("["); + name = path.mid(0, pos - 1); + path = path.mid(pos + 1, path.length() - pos - 2); + } + + KURL url(path); + + if(name.isEmpty()) + name = url.filename(); + + if(!url.isLocalFile() || QFile::exists(url.path())) { + KFileItem* fileItem = new KFileItem(KFileItem::Unknown, KFileItem::Unknown, url); + fileList.append(fileItem); + KoFileListItem* item = new KoFileListItem(m_documentList, + m_documentList->lastItem(), name, url.url(), fileItem); + //center all icons in 64x64 area + QImage icon = fileItem->pixmap(64).convertToImage(); + icon.setAlphaBuffer(true); + icon = icon.copy((icon.width() - 64) / 2, (icon.height() - 64) / 2, 64, 64); + item->setPixmap(0, QPixmap(icon)); + item->setPixmap(2, fileItem->pixmap(128)); + } + } + + i++; + } while ( !value.isEmpty() || i<=10 ); + + instance->config()->setGroup( oldGroup ); + + connect(m_documentList, SIGNAL(selectionChanged(QListViewItem*)), + this, SLOT(selectionChanged(QListViewItem*))); + connect(m_documentList, SIGNAL(clicked(QListViewItem*)), + this, SLOT(selectionChanged(QListViewItem*))); + connect(m_documentList, SIGNAL(doubleClicked(QListViewItem*, const QPoint&, int)), + this, SLOT(openFile(QListViewItem*))); + connect(m_documentList, SIGNAL(returnPressed(QListViewItem*)), + this, SLOT(openFile(QListViewItem*))); + connect(m_openButton, SIGNAL(clicked()), this, SLOT(openFile())); + + m_documentList->setSelected(m_documentList->firstChild(), true); + + d->m_previewJob = KIO::filePreview(fileList, 200, 200); + + connect(d->m_previewJob, SIGNAL(result(KIO::Job*)), this, SLOT(previewResult(KIO::Job*))); + connect(d->m_previewJob, SIGNAL(gotPreview(const KFileItem*, const QPixmap&)), + this, SLOT(updatePreview(const KFileItem*, const QPixmap&))); +} + +KoRecentDocumentsPane::~KoRecentDocumentsPane() +{ + delete d; +} + +void KoRecentDocumentsPane::selectionChanged(QListViewItem* item) +{ + if(item) { + m_openButton->setEnabled(true); + m_titleLabel->setText(item->text(0)); + m_previewLabel->setPixmap(*(item->pixmap(2))); + + if(static_cast<KoFileListItem*>(item)->fileItem()) { + KFileItem* fileItem = static_cast<KoFileListItem*>(item)->fileItem(); + QString details = "<center><table border=\"0\">"; + details += i18n("File modification date and time. %1 is date time", "<tr><td><b>Modified:</b></td><td>%1</td></tr>") + .arg(fileItem->timeString(KIO::UDS_MODIFICATION_TIME)); + details += i18n("File access date and time. %1 is date time", "<tr><td><b>Accessed:</b></td><td>%1</td></tr>") + .arg(fileItem->timeString(KIO::UDS_ACCESS_TIME)); + details += "</table></center>"; + m_detailsLabel->setText(details); + } else { + m_detailsLabel->setText(QString::null); + } + } else { + m_openButton->setEnabled(false); + m_titleLabel->setText(QString::null); + m_previewLabel->setPixmap(QPixmap()); + m_detailsLabel->setText(QString::null); + } +} + +void KoRecentDocumentsPane::openFile() +{ + QListViewItem* item = m_documentList->selectedItem(); + openFile(item); +} + +void KoRecentDocumentsPane::openFile(QListViewItem* item) +{ + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + cfgGrp.writeEntry("LastReturnType", "File"); + + if(item) + emit openFile(item->text(1)); +} + +void KoRecentDocumentsPane::previewResult(KIO::Job* job) +{ + if(d->m_previewJob == job) + d->m_previewJob = 0; +} + +void KoRecentDocumentsPane::updatePreview(const KFileItem* fileItem, const QPixmap& preview) +{ + if(preview.isNull()) { + return; + } + + QListViewItemIterator it(m_documentList); + + while(it.current()) { + if(it.current()->text(1) == fileItem->url().url()) { + it.current()->setPixmap(2, preview); + QImage icon = preview.convertToImage(); + icon = icon.smoothScale(64, 64, QImage::ScaleMin); + icon.setAlphaBuffer(true); + icon = icon.copy((icon.width() - 64) / 2, (icon.height() - 64) / 2, 64, 64); + it.current()->setPixmap(0, QPixmap(icon)); + + if(it.current()->isSelected()) { + m_previewLabel->setPixmap(preview); + } + + break; + } + + it++; + } +} + +void KoRecentDocumentsPane::changePalette() +{ + QPalette p = kapp ? kapp->palette() : palette(); + p.setBrush(QColorGroup::Base, p.brush(QPalette::Normal, QColorGroup::Background)); + p.setColor(QColorGroup::Text, p.color(QPalette::Normal, QColorGroup::Foreground)); + m_detailsLabel->setPalette(p); +} + +bool KoRecentDocumentsPane::eventFilter(QObject* watched, QEvent* e) +{ + if(watched == m_previewLabel) { + if(e->type() == QEvent::MouseButtonDblClick) { + openFile(); + } + } + + if(watched == m_documentList) { + if((e->type() == QEvent::Resize) && isShown()) { + emit splitterResized(this, m_splitter->sizes()); + } + } + + return false; +} + +void KoRecentDocumentsPane::resizeSplitter(KoDetailsPaneBase* sender, const QValueList<int>& sizes) +{ + if(sender == this) + return; + + m_splitter->setSizes(sizes); +} + +#include "KoDetailsPane.moc" diff --git a/lib/kofficecore/KoDetailsPane.h b/lib/kofficecore/KoDetailsPane.h new file mode 100644 index 00000000..96a92b98 --- /dev/null +++ b/lib/kofficecore/KoDetailsPane.h @@ -0,0 +1,132 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Peter Simonsson <psn@linux.se> + + 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. +*/ +#ifndef KODETAILSPANE_H +#define KODETAILSPANE_H + +#include <klistview.h> + +#include "koDetailsPaneBase.h" + +class KoTemplateGroup; +class KoTemplate; +class KInstance; +class QListViewItem; +class KoRecentDocumentsPanePrivate; +class KoRichTextListItemPrivate; +class KFileItem; +class QPixmap; + +namespace KIO { + class Job; +} + +class KoTemplatesPanePrivate; + +/** + * This widget is the right-side part of the template opening widget. + * The parent widget is initial widget in the document space of each KOffice component. + * This widget shows a list of templates and can show their details or open it. + */ +class KoTemplatesPane : public KoDetailsPaneBase +{ + Q_OBJECT + public: + /** + * Constructor. + * @param parent the parent widget + * @param instance the instance object for the app + * @param group the group of templates this widget will show. + * @param defaultTemplate pointer to the default template. Used to select a + * template when none has been selected before. + */ + KoTemplatesPane(QWidget* parent, KInstance* instance, + KoTemplateGroup* group, KoTemplate* defaultTemplate); + ~KoTemplatesPane(); + + /// Returns true if a template in this group was the last one selected + bool isSelected(); + + virtual bool eventFilter(QObject* watched, QEvent* e); + + signals: + void openTemplate(const QString&); + /// Emited when the always use checkbox is selected + void alwaysUseChanged(KoTemplatesPane* sender, const QString& alwaysUse); + + void splitterResized(KoDetailsPaneBase* sender, const QValueList<int>& sizes); + + public slots: + void resizeSplitter(KoDetailsPaneBase* sender, const QValueList<int>& sizes); + + protected slots: + void selectionChanged(QListViewItem* item); + void openTemplate(); + void openTemplate(QListViewItem* item); + void alwaysUseClicked(); + void changeAlwaysUseTemplate(KoTemplatesPane* sender, const QString& alwaysUse); + + void changePalette(); + + private: + KoTemplatesPanePrivate* d; +}; + + +/** + * This widget is the recent doc part of the template opening widget. + * The parent widget is initial widget in the document space of each KOffice component. + * This widget shows a list of recent documents and can show their details or open it. + */ +class KoRecentDocumentsPane : public KoDetailsPaneBase +{ + Q_OBJECT + public: + /** + * Constructor. + * @param parent the parent widget + * @param instance the instance object for the app + */ + KoRecentDocumentsPane(QWidget* parent, KInstance* instance); + ~KoRecentDocumentsPane(); + + virtual bool eventFilter(QObject* watched, QEvent* e); + + signals: + void openFile(const QString&); + + void splitterResized(KoDetailsPaneBase* sender, const QValueList<int>& sizes); + + public slots: + void resizeSplitter(KoDetailsPaneBase* sender, const QValueList<int>& sizes); + + protected slots: + void selectionChanged(QListViewItem* item); + void openFile(); + void openFile(QListViewItem* item); + + void previewResult(KIO::Job* job); + void updatePreview(const KFileItem* fileItem, const QPixmap& preview); + + void changePalette(); + + private: + KoRecentDocumentsPanePrivate* d; +}; + +#endif //KODETAILSPANE_H diff --git a/lib/kofficecore/KoDocInfoPropsFactory.cpp b/lib/kofficecore/KoDocInfoPropsFactory.cpp new file mode 100644 index 00000000..b08deeb2 --- /dev/null +++ b/lib/kofficecore/KoDocInfoPropsFactory.cpp @@ -0,0 +1,27 @@ +/* This file is part of the KDE project + Copyright (c) 2000 Simon Hausmann <hausmann@kde.org> + + $Id: KoDocInfoPropsFactory.cpp 508787 2006-02-12 18:28:12Z ingwa $ + + 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 <KoDocumentInfoDlg.h> + +#include <kgenericfactory.h> + +typedef KGenericFactory<KoDocumentInfoPropsPage, KPropertiesDialog> PropsDlgFactory; +K_EXPORT_COMPONENT_FACTORY( kodocinfopropspage, PropsDlgFactory( "koffice" ) ) diff --git a/lib/kofficecore/KoDocument.cpp b/lib/kofficecore/KoDocument.cpp new file mode 100644 index 00000000..50003627 --- /dev/null +++ b/lib/kofficecore/KoDocument.cpp @@ -0,0 +1,2672 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright (C) 2000-2005 David Faure <faure@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 "KoDocument.h" + +#include "KoDocument_p.h" +#include "KoDocumentIface.h" +#include "KoDocumentChild.h" +#include "KoView.h" +#include "KoMainWindow.h" +#include "KoFilterManager.h" +#include "KoDocumentInfo.h" +#include "KoOasisStyles.h" +#include "KoOasisStore.h" +#include "KoXmlNS.h" +#include "KoOpenPane.h" + +#include <KoStoreDevice.h> +#include <KoXmlWriter.h> + +#include <kapplication.h> +#include <kdebug.h> +#include <kdeversion.h> +#include <kfileitem.h> +#include <kiconloader.h> +#include <kio/job.h> +#include <kio/netaccess.h> +#include <klocale.h> +#include <kmessagebox.h> +#include <kmimetype.h> +#include <kparts/partmanager.h> +#include <kprinter.h> +#include <ksavefile.h> + +#include <qbuffer.h> +#include <qcursor.h> +#include <qdir.h> +#include <qfile.h> +#include <qfileinfo.h> +#include <qimage.h> +#include <qmap.h> +#include <qpainter.h> +#include <qtimer.h> +#include <qxml.h> +#include <qlayout.h> + +#include <config.h> +#include <assert.h> +#include <locale.h> + + +// Define the protocol used here for embedded documents' URL +// This used to "store" but KURL didn't like it, +// so let's simply make it "tar" ! +#define STORE_PROTOCOL "tar" +// The internal path is a hack to make KURL happy and still pass +// some kind of relative path to KoDocumentChild +#define INTERNAL_PROTOCOL "intern" +#define INTERNAL_PREFIX "intern:/" +// Warning, keep it sync in koStore.cc and koDocumentChild.cc + +QPtrList<KoDocument> *KoDocument::s_documentList=0L; + +using namespace std; +class KoViewWrapperWidget; + +/********************************************************** + * + * KoDocument + * + **********************************************************/ + +const int KoDocument::s_defaultAutoSave = 300; // 5 minutes + +class KoDocument::Private +{ +public: + Private() : + m_dcopObject( 0L ), + filterManager( 0L ), + m_specialOutputFlag( 0 ), // default is native format + m_isImporting( false ), m_isExporting( false ), + m_numOperations( 0 ), + modifiedAfterAutosave( false ), + m_autosaving( false ), + m_shouldCheckAutoSaveFile( true ), + m_autoErrorHandlingEnabled( true ), + m_backupFile( true ), + m_backupPath( QString::null ), + m_doNotSaveExtDoc( false ), + m_current( false ), + m_storeInternal( false ), + m_bLoading( false ), + m_startUpWidget( 0 ) + { + m_confirmNonNativeSave[0] = true; + m_confirmNonNativeSave[1] = true; + if ( KGlobal::locale()->measureSystem() == KLocale::Imperial ) { + m_unit = KoUnit::U_INCH; + } else { + m_unit = KoUnit::U_CM; + } + } + + QPtrList<KoView> m_views; + QPtrList<KoDocumentChild> m_children; + QPtrList<KoMainWindow> m_shells; + QValueList<QDomDocument> m_viewBuildDocuments; + + KoViewWrapperWidget *m_wrapperWidget; + KoDocumentIface * m_dcopObject; + KoDocumentInfo *m_docInfo; + KoView* hitTestView; + + KoUnit::Unit m_unit; + + KoFilterManager * filterManager; // The filter-manager to use when loading/saving [for the options] + + QCString mimeType; // The actual mimetype of the document + QCString outputMimeType; // The mimetype to use when saving + bool m_confirmNonNativeSave [2]; // used to pop up a dialog when saving for the + // first time if the file is in a foreign format + // (Save/Save As, Export) + int m_specialOutputFlag; // See KoFileDialog in koMainWindow.cc + bool m_isImporting, m_isExporting; // File --> Import/Export vs File --> Open/Save + + QTimer m_autoSaveTimer; + QString lastErrorMessage; // see openFile() + int m_autoSaveDelay; // in seconds, 0 to disable. + int m_numOperations; + bool modifiedAfterAutosave; + bool m_bSingleViewMode; + bool m_autosaving; + bool m_shouldCheckAutoSaveFile; // usually true + bool m_autoErrorHandlingEnabled; // usually true + bool m_backupFile; + QString m_backupPath; + bool m_doNotSaveExtDoc; // makes it possible to save only internally stored child documents + bool m_current; + bool m_storeInternal; // Store this doc internally even if url is external + bool m_bLoading; // True while loading (openURL is async) + + KoOpenPane* m_startUpWidget; + QString m_templateType; +}; + +// Used in singleViewMode +class KoViewWrapperWidget : public QWidget +{ +public: + KoViewWrapperWidget( QWidget *parent, const char *name ) + : QWidget( parent, name ) + { + KGlobal::locale()->insertCatalogue("koffice"); + // Tell the iconloader about share/apps/koffice/icons + KGlobal::iconLoader()->addAppDir("koffice"); + m_view = 0L; + // Avoid warning from KParts - we'll have the KoView as focus proxy anyway + setFocusPolicy( ClickFocus ); + } + + virtual ~KoViewWrapperWidget() { + setFocusProxy( 0 ); // to prevent a crash due to clearFocus (#53466) + } + + virtual void resizeEvent( QResizeEvent * ) + { + QObject *wid = child( 0, "QWidget" ); + if ( wid ) + static_cast<QWidget *>(wid)->setGeometry( 0, 0, width(), height() ); + } + + virtual void childEvent( QChildEvent *ev ) + { + if ( ev->type() == QEvent::ChildInserted ) + resizeEvent( 0L ); + } + + // Called by openFile() + void setKoView( KoView * view ) { + m_view = view; + setFocusProxy( m_view ); + } + KoView * koView() const { return m_view; } +private: + KoView* m_view; +}; + +KoBrowserExtension::KoBrowserExtension( KoDocument * doc, const char * name ) + : KParts::BrowserExtension( doc, name ) +{ + emit enableAction( "print", true ); +} + +void KoBrowserExtension::print() +{ + KoDocument * doc = static_cast<KoDocument *>( parent() ); + KoViewWrapperWidget * wrapper = static_cast<KoViewWrapperWidget *>( doc->widget() ); + KoView * view = wrapper->koView(); + // TODO remove code duplication (KoMainWindow), by moving this to KoView + KPrinter printer; + // ### TODO: apply global koffice settings here + view->setupPrinter( printer ); + if ( printer.setup( view ) ) + view->print( printer ); +} + +KoDocument::KoDocument( QWidget * parentWidget, const char *widgetName, QObject* parent, const char* name, bool singleViewMode ) + : KParts::ReadWritePart( parent, name ) +{ + if(s_documentList==0L) + s_documentList=new QPtrList<KoDocument>; + s_documentList->append(this); + + d = new Private; + m_bEmpty = TRUE; + connect( &d->m_autoSaveTimer, SIGNAL( timeout() ), this, SLOT( slotAutoSave() ) ); + setAutoSave( s_defaultAutoSave ); + d->m_bSingleViewMode = singleViewMode; + + + // the parent setting *always* overrides! (Simon) + if ( parent ) + { + if ( parent->inherits( "KoDocument" ) ) + d->m_bSingleViewMode = ((KoDocument *)parent)->isSingleViewMode(); + else if ( parent->inherits( "KParts::Part" ) ) + d->m_bSingleViewMode = true; + } + + if ( singleViewMode ) + { + d->m_wrapperWidget = new KoViewWrapperWidget( parentWidget, widgetName ); + setWidget( d->m_wrapperWidget ); + kdDebug(30003) << "creating KoBrowserExtension" << endl; + (void) new KoBrowserExtension( this ); // ## only if embedded into a browser? + } + + d->m_docInfo = new KoDocumentInfo( this, "document info" ); + + m_pageLayout.ptWidth = 0; + m_pageLayout.ptHeight = 0; + m_pageLayout.ptTop = 0; + m_pageLayout.ptBottom = 0; + m_pageLayout.ptLeft = 0; + m_pageLayout.ptRight = 0; + + // A way to 'fix' the job's window, since we have no widget known to KParts + if ( !singleViewMode ) + connect( this, SIGNAL( started( KIO::Job* ) ), SLOT( slotStarted( KIO::Job* ) ) ); +} + +KoDocument::~KoDocument() +{ + d->m_autoSaveTimer.stop(); + + QPtrListIterator<KoDocumentChild> childIt( d->m_children ); + for (; childIt.current(); ++childIt ) + disconnect( childIt.current(), SIGNAL( destroyed() ), + this, SLOT( slotChildDestroyed() ) ); + + // Tell our views that the document is already destroyed and + // that they shouldn't try to access it. + QPtrListIterator<KoView> vIt( d->m_views ); + for (; vIt.current(); ++vIt ) + vIt.current()->setDocumentDeleted(); + + delete d->m_startUpWidget; + d->m_startUpWidget = 0; + + d->m_children.setAutoDelete( true ); + d->m_children.clear(); + + d->m_shells.setAutoDelete( true ); + d->m_shells.clear(); + + delete d->m_dcopObject; + delete d->filterManager; + delete d; + s_documentList->removeRef(this); + // last one? + if(s_documentList->isEmpty()) { + delete s_documentList; + s_documentList=0; + } +} + +bool KoDocument::isSingleViewMode() const +{ + return d->m_bSingleViewMode; +} + +bool KoDocument::isEmbedded() const +{ + return dynamic_cast<KoDocument *>( parent() ) != 0; +} + +KoView *KoDocument::createView( QWidget *parent, const char *name ) +{ + KoView *view=createViewInstance(parent, name); + addView(view); + return view; +} + +bool KoDocument::exp0rt( const KURL & _url ) +{ + bool ret; + + d->m_isExporting = true; + + // + // Preserve a lot of state here because we need to restore it in order to + // be able to fake a File --> Export. Can't do this in saveFile() because, + // for a start, KParts has already set m_url and m_file and because we need + // to restore the modified flag etc. and don't want to put a load on anyone + // reimplementing saveFile() (Note: import() and export() will remain + // non-virtual). + // + KURL oldURL = m_url; + QString oldFile = m_file; + + bool wasModified = isModified (); + QCString oldMimeType = mimeType (); + + + // save... + ret = saveAs( _url ); + + + // + // This is sooooo hacky :( + // Hopefully we will restore enough state. + // + kdDebug(30003) << "Restoring KoDocument state to before export" << endl; + + // always restore m_url & m_file because KParts has changed them + // (regardless of failure or success) + m_url = oldURL; + m_file = oldFile; + + // on successful export we need to restore modified etc. too + // on failed export, mimetype/modified hasn't changed anyway + if (ret) + { + setModified (wasModified); + d->mimeType = oldMimeType; + } + + + d->m_isExporting = false; + + return ret; +} + +bool KoDocument::saveFile() +{ + kdDebug(30003) << "KoDocument::saveFile() doc='" << url().url() <<"'"<< endl; + + // set local again as it can happen that it gets resetted + // so that all saved numbers have a . and not e.g a , as a + // decimal seperator + setlocale( LC_NUMERIC, "C" ); + + // Save it to be able to restore it after a failed save + const bool wasModified = isModified (); + + // The output format is set by koMainWindow, and by openFile + QCString outputMimeType = d->outputMimeType; + //Q_ASSERT( !outputMimeType.isEmpty() ); // happens when using the DCOP method saveAs + if ( outputMimeType.isEmpty() ) + outputMimeType = d->outputMimeType = nativeFormatMimeType(); + + QApplication::setOverrideCursor( waitCursor ); + + if ( backupFile() ) { + if ( url().isLocalFile() ) + KSaveFile::backupFile( url().path(), d->m_backupPath ); + else { + KIO::UDSEntry entry; + if ( KIO::NetAccess::stat( url(), entry, shells().current() ) ) { // this file exists => backup + emit sigStatusBarMessage( i18n("Making backup...") ); + KURL backup; + if ( d->m_backupPath.isEmpty()) + backup = url(); + else + backup = d->m_backupPath +"/"+url().fileName(); + backup.setPath( backup.path() + QString::fromLatin1("~") ); + KFileItem item( entry, url() ); + Q_ASSERT( item.name() == url().fileName() ); + KIO::NetAccess::file_copy( url(), backup, item.permissions(), true /*overwrite*/, false /*resume*/, shells().current() ); + } + } + } + + emit sigStatusBarMessage( i18n("Saving...") ); + bool ret = false; + bool suppressErrorDialog = false; + if ( !isNativeFormat( outputMimeType ) ) { + kdDebug(30003) << "Saving to format " << outputMimeType << " in " << m_file << endl; + // Not native format : save using export filter + if ( !d->filterManager ) + d->filterManager = new KoFilterManager( this ); + + KoFilter::ConversionStatus status = d->filterManager->exp0rt( m_file, outputMimeType ); + ret = status == KoFilter::OK; + suppressErrorDialog = (status == KoFilter::UserCancelled || status == KoFilter::BadConversionGraph ); + } else { + // Native format => normal save + Q_ASSERT( !m_file.isEmpty() ); + ret = saveNativeFormat( m_file ); + } + + if ( ret ) { + removeAutoSaveFiles(); + // Restart the autosave timer + // (we don't want to autosave again 2 seconds after a real save) + setAutoSave( d->m_autoSaveDelay ); + } + + QApplication::restoreOverrideCursor(); + if ( !ret ) + { + if ( !suppressErrorDialog ) + { + showSavingErrorDialog(); + } + + // couldn't save file so this new URL is invalid + // FIXME: we should restore the current document's true URL instead of + // setting it to nothing otherwise anything that depends on the URL + // being correct will not work (i.e. the document will be called + // "Untitled" which may not be true) + // + // Update: now the URL is restored in KoMainWindow but really, this + // should still be fixed in KoDocument/KParts (ditto for m_file). + // We still resetURL() here since we may or may not have been called + // by KoMainWindow - Clarence + resetURL(); + + // As we did not save, restore the "was modified" status + setModified( wasModified ); + } + + if ( ret ) + { + d->mimeType = outputMimeType; + setConfirmNonNativeSave ( isExporting (), false ); + } + emit sigClearStatusBarMessage(); + + return ret; +} + +QCString KoDocument::mimeType() const +{ + return d->mimeType; +} + +void KoDocument::setMimeType( const QCString & mimeType ) +{ + d->mimeType = mimeType; +} + +void KoDocument::setOutputMimeType( const QCString & mimeType, int specialOutputFlag ) +{ + d->outputMimeType = mimeType; + d->m_specialOutputFlag = specialOutputFlag; +} + +QCString KoDocument::outputMimeType() const +{ + return d->outputMimeType; +} + +int KoDocument::specialOutputFlag() const +{ + return d->m_specialOutputFlag; +} + +bool KoDocument::confirmNonNativeSave( const bool exporting ) const +{ + // "exporting ? 1 : 0" is different from "exporting" because a bool is + // usually implemented like an "int", not "unsigned : 1" + return d->m_confirmNonNativeSave [ exporting ? 1 : 0 ]; +} + +void KoDocument::setConfirmNonNativeSave( const bool exporting, const bool on ) +{ + d->m_confirmNonNativeSave [ exporting ? 1 : 0] = on; +} + +bool KoDocument::wantExportConfirmation() const +{ + return true; +} + +bool KoDocument::isImporting() const +{ + return d->m_isImporting; +} + +bool KoDocument::isExporting() const +{ + return d->m_isExporting; +} + +void KoDocument::setCheckAutoSaveFile( bool b ) +{ + d->m_shouldCheckAutoSaveFile = b; +} + +void KoDocument::setAutoErrorHandlingEnabled( bool b ) +{ + d->m_autoErrorHandlingEnabled = b; +} + +bool KoDocument::isAutoErrorHandlingEnabled() const +{ + return d->m_autoErrorHandlingEnabled; +} + +void KoDocument::slotAutoSave() +{ + if ( isModified() && d->modifiedAfterAutosave && !d->m_bLoading ) + { + connect( this, SIGNAL( sigProgress( int ) ), shells().current(), SLOT( slotProgress( int ) ) ); + emit sigStatusBarMessage( i18n("Autosaving...") ); + d->m_autosaving = true; + bool ret = saveNativeFormat( autoSaveFile( m_file ) ); + setModified( true ); + if ( ret ) { + d->modifiedAfterAutosave = false; + d->m_autoSaveTimer.stop(); // until the next change + } + d->m_autosaving = false; + emit sigClearStatusBarMessage(); + disconnect( this, SIGNAL( sigProgress( int ) ), shells().current(), SLOT( slotProgress( int ) ) ); + if ( !ret ) + emit sigStatusBarMessage( i18n("Error during autosave! Partition full?") ); + } +} + +KAction *KoDocument::action( const QDomElement &element ) const +{ + // First look in the document itself + KAction* act = KParts::ReadWritePart::action( element ); + if ( act ) + return act; + + Q_ASSERT( d->m_bSingleViewMode ); + // Then look in the first view (this is for the single view mode) + if ( !d->m_views.isEmpty() ) + return d->m_views.getFirst()->action( element ); + else + return 0L; +} + +QDomDocument KoDocument::domDocument() const +{ + // When embedded into e.g. konqueror, we want the view's GUI (hopefully a reduced one) + // to be used. + Q_ASSERT( d->m_bSingleViewMode ); + if ( d->m_views.isEmpty() ) + return QDomDocument(); + else + return d->m_views.getFirst()->domDocument(); +} + +void KoDocument::setManager( KParts::PartManager *manager ) +{ + KParts::ReadWritePart::setManager( manager ); + if ( d->m_bSingleViewMode && d->m_views.count() == 1 ) + d->m_views.getFirst()->setPartManager( manager ); + + if ( manager ) + { + QPtrListIterator<KoDocumentChild> it( d->m_children ); + for (; it.current(); ++it ) + if ( it.current()->document() ) + manager->addPart( it.current()->document(), false ); + } +} + +void KoDocument::setReadWrite( bool readwrite ) +{ + KParts::ReadWritePart::setReadWrite( readwrite ); + + QPtrListIterator<KoView> vIt( d->m_views ); + for (; vIt.current(); ++vIt ) + vIt.current()->updateReadWrite( readwrite ); + + QPtrListIterator<KoDocumentChild> dIt( d->m_children ); + for (; dIt.current(); ++dIt ) + if ( dIt.current()->document() ) + dIt.current()->document()->setReadWrite( readwrite ); + + setAutoSave( d->m_autoSaveDelay ); +} + +void KoDocument::setAutoSave( int delay ) +{ + d->m_autoSaveDelay = delay; + if ( isReadWrite() && !isEmbedded() && d->m_autoSaveDelay > 0 ) + d->m_autoSaveTimer.start( d->m_autoSaveDelay * 1000 ); + else + d->m_autoSaveTimer.stop(); +} + +void KoDocument::addView( KoView *view ) +{ + if ( !view ) + return; + + d->m_views.append( view ); + view->updateReadWrite( isReadWrite() ); +} + +void KoDocument::removeView( KoView *view ) +{ + d->m_views.removeRef( view ); +} + +const QPtrList<KoView>& KoDocument::views() const +{ + return d->m_views; +} + +int KoDocument::viewCount() const +{ + return d->m_views.count(); +} + +void KoDocument::insertChild( KoDocumentChild *child ) +{ + setModified( true ); + + d->m_children.append( child ); + + connect( child, SIGNAL( changed( KoChild * ) ), + this, SLOT( slotChildChanged( KoChild * ) ) ); + connect( child, SIGNAL( destroyed() ), + this, SLOT( slotChildDestroyed() ) ); + + // It may be that insertChild is called without the KoDocumentChild + // having a KoDocument attached, yet. This happens for example + // when KPresenter loads a document with embedded objects. For those + // KPresenterChild objects are allocated and insertChild is called. + // Later in loadChildren() KPresenter iterates over the child list + // and calls loadDocument for each child. That's exactly where we + // will try to do what we cannot do now: Register the child document + // at the partmanager (Simon) + if ( manager() && !isSingleViewMode() && child->document() ) + manager()->addPart( child->document(), false ); +} + +void KoDocument::slotChildChanged( KoChild *c ) +{ + assert( c->inherits( "KoDocumentChild" ) ); + emit childChanged( static_cast<KoDocumentChild *>( c ) ); +} + +void KoDocument::slotChildDestroyed() +{ + setModified( true ); + + const KoDocumentChild *child = static_cast<const KoDocumentChild *>( sender() ); + d->m_children.removeRef( child ); +} + +const QPtrList<KoDocumentChild>& KoDocument::children() const +{ + return d->m_children; +} + +KParts::Part *KoDocument::hitTest( QWidget *widget, const QPoint &globalPos ) +{ + QPtrListIterator<KoView> it( d->m_views ); + for (; it.current(); ++it ) + if ( static_cast<QWidget *>(it.current()) == widget ) + { + KoView* view = it.current(); + d->hitTestView = view; // koffice-1.x hack + QPoint canvasPos( view->canvas()->mapFromGlobal( globalPos ) ); + canvasPos.rx() += view->canvasXOffset(); + canvasPos.ry() += view->canvasYOffset(); + + KParts::Part *part = view->hitTest( canvasPos ); + d->hitTestView = 0; + if ( part ) + return part; + } + + return 0L; +} + +KoView* KoDocument::hitTestView() +{ + return d->hitTestView; +} + +KoDocument* KoDocument::hitTest( const QPoint &pos, const QWMatrix &matrix ) +{ + // Call KoDocumentChild::hitTest for any child document + QPtrListIterator<KoDocumentChild> it( d->m_children ); + for (; it.current(); ++it ) + { + KoDocument *doc = it.current()->hitTest( pos, matrix ); + if ( doc ) + return doc; + } + + // Unless we hit an embedded document, the hit is on this document itself. + return this; +} + +KoDocumentChild *KoDocument::child( KoDocument *doc ) +{ + QPtrListIterator<KoDocumentChild> it( d->m_children ); + for (; it.current(); ++it ) + if ( it.current()->document() == doc ) + return it.current(); + + return 0L; +} + +KoDocumentInfo *KoDocument::documentInfo() const +{ + return d->m_docInfo; +} + +void KoDocument::setViewBuildDocument( KoView *view, const QDomDocument &doc ) +{ + if ( d->m_views.find( view ) == -1 ) + return; + + uint viewIdx = d->m_views.at(); + + if ( d->m_viewBuildDocuments.count() == viewIdx ) + d->m_viewBuildDocuments.append( doc ); + else if ( d->m_viewBuildDocuments.count() > viewIdx ) + d->m_viewBuildDocuments[ viewIdx ] = doc; +} + +QDomDocument KoDocument::viewBuildDocument( KoView *view ) +{ + QDomDocument res; + + if ( d->m_views.find( view ) == -1 ) + return res; + + uint viewIdx = d->m_views.at(); + + if ( viewIdx >= d->m_viewBuildDocuments.count() ) + return res; + + res = d->m_viewBuildDocuments[ viewIdx ]; + + // make this entry empty. otherwise we get a segfault in QMap ;-( + d->m_viewBuildDocuments[ viewIdx ] = QDomDocument(); + + return res; +} + +void KoDocument::paintEverything( QPainter &painter, const QRect &rect, bool transparent, KoView *view, double zoomX, double zoomY ) +{ + paintContent( painter, rect, transparent, zoomX, zoomY ); + paintChildren( painter, rect, view, zoomX, zoomY ); +} + +void KoDocument::paintChildren( QPainter &painter, const QRect &/*rect*/, KoView *view, double zoomX, double zoomY ) +{ + QPtrListIterator<KoDocumentChild> it( d->m_children ); + for (; it.current(); ++it ) + { + // #### todo: paint only if child is visible inside rect + painter.save(); + paintChild( it.current(), painter, view, zoomX, zoomY ); + painter.restore(); + } +} + +void KoDocument::paintChild( KoDocumentChild *child, QPainter &painter, KoView *view, double zoomX, double zoomY ) +{ + if ( child->isDeleted() ) + return; + + // QRegion rgn = painter.clipRegion(); + + child->transform( painter ); + child->document()->paintEverything( painter, child->contentRect(), child->isTransparent(), view, zoomX, zoomY ); + + if ( view && view->partManager() ) + { + // ### do we need to apply zoomX and zoomY here ? + KParts::PartManager *manager = view->partManager(); + + painter.scale( 1.0 / child->xScaling(), 1.0 / child->yScaling() ); + + int w = int( (double)child->contentRect().width() * child->xScaling() ); + int h = int( (double)child->contentRect().height() * child->yScaling() ); + if ( ( manager->selectedPart() == (KParts::Part *)child->document() && + manager->selectedWidget() == (QWidget *)view ) || + ( manager->activePart() == (KParts::Part *)child->document() && + manager->activeWidget() == (QWidget *)view ) ) + { + // painter.setClipRegion( rgn ); + painter.setClipping( FALSE ); + + painter.setPen( black ); + painter.fillRect( -5, -5, w + 10, 5, white ); + painter.fillRect( -5, h, w + 10, 5, white ); + painter.fillRect( -5, -5, 5, h + 10, white ); + painter.fillRect( w, -5, 5, h + 10, white ); + painter.fillRect( -5, -5, w + 10, 5, BDiagPattern ); + painter.fillRect( -5, h, w + 10, 5, BDiagPattern ); + painter.fillRect( -5, -5, 5, h + 10, BDiagPattern ); + painter.fillRect( w, -5, 5, h + 10, BDiagPattern ); + + if ( manager->selectedPart() == (KParts::Part *)child->document() && + manager->selectedWidget() == (QWidget *)view ) + { + QColor color; + if ( view->koDocument() == this ) + color = black; + else + color = gray; + painter.fillRect( -5, -5, 5, 5, color ); + painter.fillRect( -5, h, 5, 5, color ); + painter.fillRect( w, h, 5, 5, color ); + painter.fillRect( w, -5, 5, 5, color ); + painter.fillRect( w / 2 - 3, -5, 5, 5, color ); + painter.fillRect( w / 2 - 3, h, 5, 5, color ); + painter.fillRect( -5, h / 2 - 3, 5, 5, color ); + painter.fillRect( w, h / 2 - 3, 5, 5, color ); + } + + painter.setClipping( TRUE ); + } + } +} + +bool KoDocument::isModified() const +{ + if ( KParts::ReadWritePart::isModified() ) + { + //kdDebug(30003)<<k_funcinfo<<" Modified doc='"<<url().url()<<"' extern="<<isStoredExtern()<<endl; + return true; + } + // Then go through internally stored children (considered to be part of this doc) + QPtrListIterator<KoDocumentChild> it = children(); + for (; it.current(); ++it ) + { + KoDocument *doc = it.current()->document(); + if ( doc && !it.current()->isStoredExtern() && !it.current()->isDeleted() && doc->isModified() ) + return true; + } + return false; +} + +bool KoDocument::saveChildren( KoStore* _store ) +{ + //kdDebug(30003)<<k_funcinfo<<" checking children of doc='"<<url().url()<<"'"<<endl; + int i = 0; + QPtrListIterator<KoDocumentChild> it( children() ); + for( ; it.current(); ++it ) { + KoDocument* childDoc = it.current()->document(); + if (childDoc && !it.current()->isDeleted()) + { + if ( !childDoc->isStoredExtern() ) + { + //kdDebug(30003) << "KoDocument::saveChildren internal url: /" << i << endl; + if ( !childDoc->saveToStore( _store, QString::number( i++ ) ) ) + return FALSE; + + if (!isExporting ()) + childDoc->setModified( false ); + } + //else kdDebug(30003)<<k_funcinfo<<" external (don't save) url:" << childDoc->url().url()<<endl; + } + } + return true; +} + +bool KoDocument::saveChildrenOasis( KoStore* store, KoXmlWriter* manifestWriter ) +{ + //kdDebug(30003)<<k_funcinfo<<" checking children of doc='"<<url().url()<<"'"<<endl; + QPtrListIterator<KoDocumentChild> it( children() ); + for( ; it.current(); ++it ) { + KoDocument* childDoc = it.current()->document(); + if ( childDoc && !it.current()->isDeleted() ) + { + if ( !it.current()->saveOasis( store, manifestWriter ) ) + return false; + if ( !childDoc->isStoredExtern() && !isExporting () ) + childDoc->setModified( false ); + } + } + return true; +} + +bool KoDocument::saveExternalChildren() +{ + if ( d->m_doNotSaveExtDoc ) + { + //kdDebug(30003)<<k_funcinfo<<" Don't save external docs in doc='"<<url().url()<<"'"<<endl; + d->m_doNotSaveExtDoc = false; + return true; + } + + //kdDebug(30003)<<k_funcinfo<<" checking children of doc='"<<url().url()<<"'"<<endl; + KoDocumentChild *ch; + QPtrListIterator<KoDocumentChild> it = children(); + for (; (ch = it.current()); ++it ) + { + if ( !ch->isDeleted() ) + { + KoDocument* doc = ch->document(); + if ( doc && doc->isStoredExtern() && doc->isModified() ) + { + kdDebug(30003)<<" save external doc='"<<url().url()<<"'"<<endl; + doc->setDoNotSaveExtDoc(); // Only save doc + it's internal children + if ( !doc->save() ) + return false; // error + } + //kdDebug(30003)<<k_funcinfo<<" not modified doc='"<<url().url()<<"'"<<endl; + // save possible external docs inside doc + if ( doc && !doc->saveExternalChildren() ) + return false; + } + } + return true; +} + +bool KoDocument::saveNativeFormat( const QString & file ) +{ + d->lastErrorMessage = QString::null; + //kdDebug(30003) << "Saving to store" << endl; + + KoStore::Backend backend = KoStore::Auto; +#if 0 + if ( d->m_specialOutputFlag == SaveAsKOffice1dot1 ) + { + kdDebug(30003) << "Saving as KOffice-1.1 format, using a tar.gz" << endl; + backend = KoStore::Tar; // KOffice-1.0/1.1 used tar.gz for the native mimetype + //// TODO more backwards compat stuff (embedded docs etc.) + } + else +#endif + if ( d->m_specialOutputFlag == SaveAsDirectoryStore ) + { + backend = KoStore::Directory; + kdDebug(30003) << "Saving as uncompressed XML, using directory store." << endl; + } + else if ( d->m_specialOutputFlag == SaveAsFlatXML ) + { + kdDebug(30003) << "Saving as a flat XML file." << endl; + QFile f( file ); + if ( f.open( IO_WriteOnly | IO_Translate ) ) + { + bool success = saveToStream( &f ); + f.close(); + return success; + } + else + return false; + } + + kdDebug(30003) << "KoDocument::saveNativeFormat nativeFormatMimeType=" << nativeFormatMimeType() << endl; + // OLD: bool oasis = d->m_specialOutputFlag == SaveAsOASIS; + // OLD: QCString mimeType = oasis ? nativeOasisMimeType() : nativeFormatMimeType(); + QCString mimeType = d->outputMimeType; + QCString nativeOasisMime = nativeOasisMimeType(); + bool oasis = !mimeType.isEmpty() && ( mimeType == nativeOasisMime || mimeType == nativeOasisMime + "-template" ); + // TODO: use std::auto_ptr or create store on stack [needs API fixing], + // to remove all the 'delete store' in all the branches + KoStore* store = KoStore::createStore( file, KoStore::Write, mimeType, backend ); + if ( store->bad() ) + { + d->lastErrorMessage = i18n( "Could not create the file for saving" ); // more details needed? + delete store; + return false; + } + + if ( oasis ) + { + kdDebug(30003) << "Saving to OASIS format" << endl; + // Tell KoStore not to touch the file names + store->disallowNameExpansion(); + KoOasisStore oasisStore( store ); + KoXmlWriter* manifestWriter = oasisStore.manifestWriter( mimeType ); + + if ( !saveOasis( store, manifestWriter ) ) + { + kdDebug(30003) << "saveOasis failed" << endl; + delete store; + return false; + } + + // Save embedded objects + if ( !saveChildrenOasis( store, manifestWriter ) ) + { + kdDebug(30003) << "saveChildrenOasis failed" << endl; + delete store; + return false; + } + + if ( store->open( "meta.xml" ) ) + { + if ( !d->m_docInfo->saveOasis( store ) || !store->close() ) { + delete store; + return false; + } + manifestWriter->addManifestEntry( "meta.xml", "text/xml" ); + } + else + { + d->lastErrorMessage = i18n( "Not able to write '%1'. Partition full?" ).arg( "meta.xml" ); + delete store; + return false; + } + + if ( store->open( "Thumbnails/thumbnail.png" ) ) + { + if ( !saveOasisPreview( store, manifestWriter ) || !store->close() ) { + d->lastErrorMessage = i18n( "Error while trying to write '%1'. Partition full?" ).arg( "Thumbnails/thumbnail.png" ); + delete store; + return false; + } + // No manifest entry! + } + else + { + d->lastErrorMessage = i18n( "Not able to write '%1'. Partition full?" ).arg( "Thumbnails/thumbnail.png" ); + delete store; + return false; + } + + // Write out manifest file + if ( !oasisStore.closeManifestWriter() ) + { + d->lastErrorMessage = i18n( "Error while trying to write '%1'. Partition full?" ).arg( "META-INF/manifest.xml" ); + delete store; + return false; + } + + delete store; + } + else + { + // Save internal children first since they might get a new url + if ( !saveChildren( store ) && !oasis ) + { + if ( d->lastErrorMessage.isEmpty() ) + d->lastErrorMessage = i18n( "Error while saving embedded documents" ); // more details needed + delete store; + return false; + } + + kdDebug(30003) << "Saving root" << endl; + if ( store->open( "root" ) ) + { + KoStoreDevice dev( store ); + if ( !saveToStream( &dev ) || !store->close() ) + { + kdDebug(30003) << "saveToStream failed" << endl; + delete store; + return false; + } + } + else + { + d->lastErrorMessage = i18n( "Not able to write '%1'. Partition full?" ).arg( "maindoc.xml" ); + delete store; + return false; + } + if ( store->open( "documentinfo.xml" ) ) + { + QDomDocument doc = d->m_docInfo->save(); + KoStoreDevice dev( store ); + + QCString s = doc.toCString(); // this is already Utf8! + (void)dev.writeBlock( s.data(), s.size()-1 ); + (void)store->close(); + } + + if ( store->open( "preview.png" ) ) + { + // ### TODO: missing error checking (The partition could be full!) + savePreview( store ); + (void)store->close(); + } + + if ( !completeSaving( store ) ) + { + delete store; + return false; + } + kdDebug(30003) << "Saving done of url: " << url().url() << endl; + delete store; + } + if ( !saveExternalChildren() ) + { + return false; + } + return true; +} + +bool KoDocument::saveToStream( QIODevice * dev ) +{ + QDomDocument doc = saveXML(); + // Save to buffer + QCString s = doc.toCString(); // utf8 already + // We use QCString::size()-1 here, not the official QCString::length + // It works here, as we are not modifying QCString as QByteArray + int nwritten = dev->writeBlock( s.data(), s.size()-1 ); + if ( nwritten != (int)s.size()-1 ) + kdWarning(30003) << "KoDocument::saveToStream wrote " << nwritten << " - expected " << s.size()-1 << endl; + return nwritten == (int)s.size()-1; +} + +// Called for embedded documents +bool KoDocument::saveToStore( KoStore* _store, const QString & _path ) +{ + kdDebug(30003) << "Saving document to store " << _path << endl; + + // Use the path as the internal url + if ( _path.startsWith( STORE_PROTOCOL ) ) + m_url = KURL( _path ); + else // ugly hack to pass a relative URI + m_url = KURL( INTERNAL_PREFIX + _path ); + + // To make the children happy cd to the correct directory + _store->pushDirectory(); + _store->enterDirectory( _path ); + + // Save childen first since they might get a new url + if ( !saveChildren( _store ) ) + return false; + + // In the current directory we're the king :-) + if ( _store->open( "root" ) ) + { + KoStoreDevice dev( _store ); + if ( !saveToStream( &dev ) ) + { + _store->close(); + return false; + } + if ( !_store->close() ) + return false; + } + + if ( !completeSaving( _store ) ) + return false; + + // Now that we're done leave the directory again + _store->popDirectory(); + + kdDebug(30003) << "Saved document to store" << endl; + + return true; +} + +bool KoDocument::saveOasisPreview( KoStore* store, KoXmlWriter* manifestWriter ) +{ + const QPixmap pix = generatePreview( QSize( 128, 128 ) ); + QImage preview ( pix.convertToImage().convertDepth( 32, Qt::ColorOnly ) ); + if ( !preview.hasAlphaBuffer() ) + { + preview.setAlphaBuffer( true ); + } + // ### TODO: freedesktop.org Thumbnail specification (date...) + KoStoreDevice io ( store ); + if ( !io.open( IO_WriteOnly ) ) + return false; + if ( ! preview.save( &io, "PNG", 0 ) ) + return false; + io.close(); + manifestWriter->addManifestEntry( "Thumbnails/", "" ); + manifestWriter->addManifestEntry( "Thumbnails/thumbnail.png", "" ); + return true; +} + +bool KoDocument::savePreview( KoStore* store ) +{ + QPixmap pix = generatePreview(QSize(256, 256)); + // Reducing to 8bpp reduces file sizes quite a lot. + const QImage preview ( pix.convertToImage().convertDepth( 8, Qt::AvoidDither | Qt::DiffuseDither) ); + KoStoreDevice io ( store ); + if ( !io.open( IO_WriteOnly ) ) + return false; + if ( ! preview.save( &io, "PNG" ) ) // ### TODO What is -9 in quality terms? + return false; + io.close(); + return true; +} + +QPixmap KoDocument::generatePreview( const QSize& size ) +{ + double docWidth, docHeight; + int pixmapSize = QMAX(size.width(), size.height()); + + if (m_pageLayout.ptWidth > 1.0) { + docWidth = m_pageLayout.ptWidth / 72 * KoGlobal::dpiX(); + docHeight = m_pageLayout.ptHeight / 72 * KoGlobal::dpiY(); + + } else { + // If we don't have a page layout, just draw the top left hand corner + docWidth = 500.0; + docHeight = 500.0; + } + + double ratio = docWidth / docHeight; + + QPixmap pix; + int previewWidth, previewHeight; + if (ratio > 1.0) + { + previewWidth = (int) pixmapSize; + previewHeight = (int) (pixmapSize / ratio); + } + else + { + previewWidth = (int) (pixmapSize * ratio); + previewHeight = (int) pixmapSize; + } + + pix.resize((int)docWidth, (int)docHeight); + + pix.fill( QColor( 245, 245, 245 ) ); + + QRect rc(0, 0, pix.width(), pix.height()); + + QPainter p; + p.begin(&pix); + paintEverything(p, rc, false); + p.end(); + + // ### TODO: why re-convert it to a QPixmap, when mostly it will be re-converted back to a QImage in the calling function + pix.convertFromImage(pix.convertToImage().smoothScale(previewWidth, previewHeight)); + + return pix; +} + +QString KoDocument::autoSaveFile( const QString & path ) const +{ + // set local again as it can happen that it gets resetted + // so that all saved numbers have a . and not e.g a , as a + // decimal seperator + setlocale( LC_NUMERIC, "C" ); + + // Using the extension allows to avoid relying on the mime magic when opening + KMimeType::Ptr mime = KMimeType::mimeType( nativeFormatMimeType() ); + QString extension = mime->property( "X-KDE-NativeExtension" ).toString(); + if ( path.isEmpty() ) + { + // Never saved? Use a temp file in $HOME then + // Yes, two open unnamed docs will overwrite each other's autosave file, + // but hmm, we can only do something if that's in the same process anyway... + QString ret = QDir::homeDirPath() + "/." + QString::fromLatin1(instance()->instanceName()) + ".autosave" + extension; + return ret; + } + else + { + KURL url( path ); + Q_ASSERT( url.isLocalFile() ); + QString dir = url.directory(false); + QString filename = url.fileName(); + return dir + "." + filename + ".autosave" + extension; + } +} + +bool KoDocument::checkAutoSaveFile() +{ + QString asf = autoSaveFile( QString::null ); // the one in $HOME + //kdDebug(30003) << "asf=" << asf << endl; + if ( QFile::exists( asf ) ) + { + QDateTime date = QFileInfo(asf).lastModified(); + QString dateStr = date.toString(Qt::LocalDate); + int res = KMessageBox::warningYesNoCancel( + 0, i18n( "An autosaved file for an unnamed document exists in %1.\nThis file is dated %2\nDo you want to open it?" ) + .arg(asf, dateStr) ); + switch(res) { + case KMessageBox::Yes : { + KURL url; + url.setPath( asf ); + bool ret = openURL( url ); + if ( ret ) + resetURL(); + return ret; + } + case KMessageBox::No : + QFile::remove( asf ); + return false; + default: // Cancel + return false; + } + } + return false; +} + +bool KoDocument::import( const KURL & _url ) +{ + bool ret; + + kdDebug (30003) << "KoDocument::import url=" << _url.url() << endl; + d->m_isImporting = true; + + // open... + ret = openURL (_url); + + // reset m_url & m_file (kindly? set by KParts::openURL()) to simulate a + // File --> Import + if (ret) + { + kdDebug (30003) << "KoDocument::import success, resetting url" << endl; + resetURL (); + setTitleModified (); + } + + d->m_isImporting = false; + + return ret; +} + +bool KoDocument::openURL( const KURL & _url ) +{ + kdDebug(30003) << "KoDocument::openURL url=" << _url.url() << endl; + d->lastErrorMessage = QString::null; + + // Reimplemented, to add a check for autosave files and to improve error reporting + if ( !_url.isValid() ) + { + d->lastErrorMessage = i18n( "Malformed URL\n%1" ).arg( _url.url() ); // ## used anywhere ? + return false; + } + if ( !closeURL() ) + return false; + + KURL url( _url ); + bool autosaveOpened = false; + d->m_bLoading = true; + if ( url.isLocalFile() && d->m_shouldCheckAutoSaveFile ) + { + QString file = url.path(); + QString asf = autoSaveFile( file ); + if ( QFile::exists( asf ) ) + { + //kdDebug(30003) << "KoDocument::openURL asf=" << asf << endl; + // ## TODO compare timestamps ? + int res = KMessageBox::warningYesNoCancel( 0, + i18n( "An autosaved file exists for this document.\nDo you want to open it instead?" )); + switch(res) { + case KMessageBox::Yes : + url.setPath( asf ); + autosaveOpened = true; + break; + case KMessageBox::No : + QFile::remove( asf ); + break; + default: // Cancel + d->m_bLoading = false; + return false; + } + } + } + + bool ret = KParts::ReadWritePart::openURL( url ); + + if ( autosaveOpened ) + resetURL(); // Force save to act like 'Save As' + else + { + // We have no koffice shell when we are being embedded as a readonly part. + //if ( d->m_shells.isEmpty() ) + // kdWarning(30003) << "KoDocument::openURL no shell yet !" << endl; + // Add to recent actions list in our shells + QPtrListIterator<KoMainWindow> it( d->m_shells ); + for (; it.current(); ++it ) + it.current()->addRecentURL( _url ); + } + return ret; +} + +bool KoDocument::openFile() +{ + //kdDebug(30003) << "KoDocument::openFile for " << m_file << endl; + if ( !QFile::exists(m_file) ) + { + QApplication::restoreOverrideCursor(); + if ( d->m_autoErrorHandlingEnabled ) + // Maybe offer to create a new document with that name ? + KMessageBox::error(0L, i18n("The file %1 does not exist.").arg(m_file) ); + d->m_bLoading = false; + return false; + } + + QApplication::setOverrideCursor( waitCursor ); + + d->m_specialOutputFlag = 0; + QCString _native_format = nativeFormatMimeType(); + + KURL u; + u.setPath( m_file ); + QString typeName = KMimeType::findByURL( u, 0, true )->name(); + + // Allow to open backup files, don't keep the mimetype application/x-trash. + if ( typeName == "application/x-trash" ) + { + QString path = u.path(); + QStringList patterns = KMimeType::mimeType( typeName )->patterns(); + // Find the extension that makes it a backup file, and remove it + for( QStringList::Iterator it = patterns.begin(); it != patterns.end(); ++it ) { + QString ext = *it; + if ( !ext.isEmpty() && ext[0] == '*' ) + { + ext.remove(0, 1); + if ( path.endsWith( ext ) ) { + path.truncate( path.length() - ext.length() ); + break; + } + } + } + typeName = KMimeType::findByPath( path, 0, true )->name(); + } + + // Special case for flat XML files (e.g. using directory store) + if ( u.fileName() == "maindoc.xml" || u.fileName() == "content.xml" || typeName == "inode/directory" ) + { + typeName = _native_format; // Hmm, what if it's from another app? ### Check mimetype + d->m_specialOutputFlag = SaveAsDirectoryStore; + kdDebug(30003) << "KoDocument::openFile loading " << u.fileName() << ", using directory store for " << m_file << "; typeName=" << typeName << endl; + } + kdDebug(30003) << "KoDocument::openFile " << m_file << " type:" << typeName << endl; + + QString importedFile = m_file; + + if ( !isNativeFormat( typeName.latin1() ) ) { + if ( !d->filterManager ) + d->filterManager = new KoFilterManager( this ); + KoFilter::ConversionStatus status; + importedFile = d->filterManager->import( m_file, status ); + if ( status != KoFilter::OK ) + { + QApplication::restoreOverrideCursor(); + + QString msg; + switch( status ) + { + case KoFilter::OK: break; + + case KoFilter::CreationError: + msg = i18n( "Creation error" ); break; + + case KoFilter::FileNotFound: + msg = i18n( "File not found" ); break; + + case KoFilter::StorageCreationError: + msg = i18n( "Cannot create storage" ); break; + + case KoFilter::BadMimeType: + msg = i18n( "Bad MIME type" ); break; + + case KoFilter::EmbeddedDocError: + msg = i18n( "Error in embedded document" ); break; + + case KoFilter::WrongFormat: + msg = i18n( "Format not recognized" ); break; + + case KoFilter::NotImplemented: + msg = i18n( "Not implemented" ); break; + + case KoFilter::ParsingError: + msg = i18n( "Parsing error" ); break; + + case KoFilter::PasswordProtected: + msg = i18n( "Document is password protected" ); break; + + case KoFilter::InternalError: + case KoFilter::UnexpectedEOF: + case KoFilter::UnexpectedOpcode: + case KoFilter::StupidError: // ?? what is this ?? + case KoFilter::UsageError: + msg = i18n( "Internal error" ); break; + + case KoFilter::OutOfMemory: + msg = i18n( "Out of memory" ); break; + + case KoFilter::UserCancelled: + case KoFilter::BadConversionGraph: + // intentionally we do not prompt the error message here + break; + + default: msg = i18n( "Unknown error" ); break; + } + + if( d->m_autoErrorHandlingEnabled && !msg.isEmpty()) + { + QString errorMsg( i18n( "Could not open\n%2.\nReason: %1" ) ); + QString docUrl = url().prettyURL( 0, KURL::StripFileProtocol ); + KMessageBox::error( 0L, errorMsg.arg(msg).arg(docUrl) ); + } + + d->m_bLoading = false; + return false; + } + kdDebug(30003) << "KoDocument::openFile - importedFile '" << importedFile + << "', status: " << static_cast<int>( status ) << endl; + } + + QApplication::restoreOverrideCursor(); + + bool ok = true; + + if (!importedFile.isEmpty()) // Something to load (tmp or native file) ? + { + // The filter, if any, has been applied. It's all native format now. + if ( !loadNativeFormat( importedFile ) ) + { + ok = false; + if ( d->m_autoErrorHandlingEnabled ) + { + showLoadingErrorDialog(); + } + } + } + + if ( importedFile != m_file ) + { + // We opened a temporary file (result of an import filter) + // Set document URL to empty - we don't want to save in /tmp ! + // But only if in readwrite mode (no saving problem otherwise) + // -- + // But this isn't true at all. If this is the result of an + // import, then importedFile=temporary_file.kwd and + // m_file/m_url=foreignformat.ext so m_url is correct! + // So don't resetURL() or else the caption won't be set when + // foreign files are opened (an annoying bug). + // - Clarence + // +#if 0 + if ( isReadWrite() ) + resetURL(); +#endif + + // remove temp file - uncomment this to debug import filters + if(!importedFile.isEmpty()) { + QFile::remove( importedFile ); + } + } + + if ( ok && d->m_bSingleViewMode ) + { + // See addClient below + KXMLGUIFactory* guiFactory = factory(); + if( guiFactory ) // 0L when splitting views in konq, for some reason + guiFactory->removeClient( this ); + + if ( !d->m_views.isEmpty() ) + { + // We already had a view (this happens when doing reload in konqueror) + KoView* v = d->m_views.first(); + if( guiFactory ) + guiFactory->removeClient( v ); + removeView( v ); + delete v; + Q_ASSERT( d->m_views.isEmpty() ); + } + + KoView *view = createView( d->m_wrapperWidget ); + d->m_wrapperWidget->setKoView( view ); + view->show(); + + // Ok, now we have a view, so action() and domDocument() will work as expected + // -> rebuild GUI + if ( guiFactory ) + guiFactory->addClient( this ); + } + + if ( ok ) + { + setMimeTypeAfterLoading( typeName ); + } + d->m_bLoading = false; + return ok; +} + +// shared between openFile and koMainWindow's "create new empty document" code +void KoDocument::setMimeTypeAfterLoading( const QString& mimeType ) +{ + d->mimeType = mimeType.latin1(); + + d->outputMimeType = d->mimeType; + + const bool needConfirm = !isNativeFormat( d->mimeType ); + setConfirmNonNativeSave( false, needConfirm ); + setConfirmNonNativeSave( true, needConfirm ); +} + +// The caller must call store->close() if loadAndParse returns true. +bool KoDocument::oldLoadAndParse(KoStore* store, const QString& filename, QDomDocument& doc) +{ + //kdDebug(30003) << "oldLoadAndParse: Trying to open " << filename << endl; + + if (!store->open(filename)) + { + kdWarning(30003) << "Entry " << filename << " not found!" << endl; + d->lastErrorMessage = i18n( "Could not find %1" ).arg( filename ); + return false; + } + // Error variables for QDomDocument::setContent + QString errorMsg; + int errorLine, errorColumn; + bool ok = doc.setContent( store->device(), &errorMsg, &errorLine, &errorColumn ); + if ( !ok ) + { + kdError(30003) << "Parsing error in " << filename << "! Aborting!" << endl + << " In line: " << errorLine << ", column: " << errorColumn << endl + << " Error message: " << errorMsg << endl; + d->lastErrorMessage = i18n( "Parsing error in %1 at line %2, column %3\nError message: %4" ) + .arg( filename ).arg( errorLine ).arg( errorColumn ) + .arg( i18n ( "QXml", errorMsg.utf8() ) ); + store->close(); + return false; + } + kdDebug(30003) << "File " << filename << " loaded and parsed" << endl; + return true; +} + +bool KoDocument::loadNativeFormat( const QString & file ) +{ + QFileInfo fileInfo( file ); + if ( !fileInfo.exists() ) // check duplicated from openURL, but this is useful for templates + { + d->lastErrorMessage = i18n("The file %1 does not exist.").arg(file); + return false; + } + if ( !fileInfo.isFile() ) + { + d->lastErrorMessage = i18n( "%1 is not a file." ).arg(file); + return false; + } + + QApplication::setOverrideCursor( waitCursor ); + + kdDebug(30003) << "KoDocument::loadNativeFormat( " << file << " )" << endl; + + QFile in; + bool isRawXML = false; + if ( d->m_specialOutputFlag != SaveAsDirectoryStore ) // Don't try to open a directory ;) + { + in.setName(file); + if ( !in.open( IO_ReadOnly ) ) + { + QApplication::restoreOverrideCursor(); + d->lastErrorMessage = i18n( "Could not open the file for reading (check read permissions)." ); + return false; + } + + // Try to find out whether it is a mime multi part file + char buf[5]; + if ( in.readBlock( buf, 4 ) < 4 ) + { + QApplication::restoreOverrideCursor(); + in.close(); + d->lastErrorMessage = i18n( "Could not read the beginning of the file." ); + return false; + } + // ### TODO: allow UTF-16 + isRawXML = (strncasecmp( buf, "<?xm", 4 ) == 0); + if ( !isRawXML ) { + // Still check for broken MathML files, which seem to be rather common + in.reset(); + // Remove spaces + do { + if ( in.readBlock( buf , 1 ) < 1 ) + { + QApplication::restoreOverrideCursor(); + in.close(); + d->lastErrorMessage = i18n( "Could not read the beginning of the file." ); + return false; + } + } while ( QChar( buf[0] ).isSpace() ); + if ( buf[0] == '<' ) { // First not-space character + if ( in.readBlock( buf , 4 ) < 4 ) + { + QApplication::restoreOverrideCursor(); + in.close(); + d->lastErrorMessage = i18n( "Could not read the beginning of the file." ); + return false; + } + isRawXML = (strncasecmp( buf, "math", 4 ) == 0); // file begins with <math ? + } + } + //kdDebug(30003) << "PATTERN=" << buf << endl; + } + // Is it plain XML? + if ( isRawXML ) + { + in.at(0); + QString errorMsg; + int errorLine; + int errorColumn; + QDomDocument doc; + bool res; + if ( doc.setContent( &in, true, &errorMsg, &errorLine, &errorColumn ) ) + { + res = loadXML( &in, doc ); + if ( res ) + res = completeLoading( 0L ); + } + else + { + kdError (30003) << "Parsing Error! Aborting! (in KoDocument::loadNativeFormat (QFile))" << endl + << " Line: " << errorLine << " Column: " << errorColumn << endl + << " Message: " << errorMsg << endl; + d->lastErrorMessage = i18n( "parsing error in the main document at line %1, column %2\nError message: %3" ) + .arg( errorLine ).arg( errorColumn ).arg( i18n ( errorMsg.utf8() ) ); + res=false; + } + + QApplication::restoreOverrideCursor(); + in.close(); + m_bEmpty = false; + return res; + } else + { // It's a koffice store (tar.gz, zip, directory, etc.) + in.close(); + + return loadNativeFormatFromStore( file ); + } +} + +bool KoDocument::loadNativeFormatFromStore( const QString& file ) +{ + KoStore::Backend backend = (d->m_specialOutputFlag == SaveAsDirectoryStore) ? KoStore::Directory : KoStore::Auto; + KoStore * store = KoStore::createStore( file, KoStore::Read, "", backend ); + + if ( store->bad() ) + { + d->lastErrorMessage = i18n( "Not a valid KOffice file: %1" ).arg( file ); + delete store; + QApplication::restoreOverrideCursor(); + return false; + } + + bool oasis = true; + // OASIS/OOo file format? + if ( store->hasFile( "content.xml" ) ) + { + store->disallowNameExpansion(); + + KoOasisStore oasisStore( store ); + // We could check the 'mimetype' file, but let's skip that and be tolerant. + + if ( !loadOasisFromStore( store ) ) { + delete store; + QApplication::restoreOverrideCursor(); + return false; + } + + } else if ( store->hasFile( "root" ) ) // Fallback to "old" file format (maindoc.xml) + { + oasis = false; + + QDomDocument doc; + bool ok = oldLoadAndParse( store, "root", doc ); + if ( ok ) + ok = loadXML( store->device(), doc ); + if ( !ok ) + { + delete store; + QApplication::restoreOverrideCursor(); + return false; + } + store->close(); + + if ( !loadChildren( store ) ) + { + kdError(30003) << "ERROR: Could not load children" << endl; + // Don't abort, proceed nonetheless + } + + } else + { + kdError(30003) << "ERROR: No maindoc.xml" << endl; + d->lastErrorMessage = i18n( "Invalid document: no file 'maindoc.xml'." ); + delete store; + QApplication::restoreOverrideCursor(); + return false; + } + + if ( oasis && store->hasFile( "meta.xml" ) ) { + QDomDocument metaDoc; + KoOasisStore oasisStore( store ); + if ( oasisStore.loadAndParse( "meta.xml", metaDoc, d->lastErrorMessage ) ) { + d->m_docInfo->loadOasis( metaDoc ); + } + } + else if ( !oasis && store->hasFile( "documentinfo.xml" ) ) + { + QDomDocument doc; + if ( oldLoadAndParse( store, "documentinfo.xml", doc ) ) { + store->close(); + d->m_docInfo->load( doc ); + } + } + else + { + //kdDebug( 30003 ) << "cannot open document info" << endl; + delete d->m_docInfo; + d->m_docInfo = new KoDocumentInfo( this, "document info" ); + } + + bool res = completeLoading( store ); + delete store; + QApplication::restoreOverrideCursor(); + m_bEmpty = false; + return res; +} + +// For embedded documents +bool KoDocument::loadFromStore( KoStore* _store, const QString& url ) +{ + if ( _store->open( url ) ) + { + QDomDocument doc; + doc.setContent( _store->device() ); + if ( !loadXML( _store->device(), doc ) ) + { + _store->close(); + return false; + } + _store->close(); + } else { + kdWarning() << "couldn't open " << url << endl; + } + + _store->pushDirectory(); + // Store as document URL + if ( url.startsWith( STORE_PROTOCOL ) ) { + m_url = KURL( url ); + } else { + m_url = KURL( INTERNAL_PREFIX + url ); + _store->enterDirectory( url ); + } + + if ( !loadChildren( _store ) ) + { + kdError(30003) << "ERROR: Could not load children" << endl; +#if 0 + return false; +#endif + } + + bool result = completeLoading( _store ); + + // Restore the "old" path + _store->popDirectory(); + + return result; +} + +bool KoDocument::loadOasisFromStore( KoStore* store ) +{ + KoOasisStyles oasisStyles; + QDomDocument contentDoc; + QDomDocument settingsDoc; + KoOasisStore oasisStore( store ); + bool ok = oasisStore.loadAndParse( "content.xml", contentDoc, d->lastErrorMessage ); + if ( !ok ) + return false; + + QDomDocument stylesDoc; + (void)oasisStore.loadAndParse( "styles.xml", stylesDoc, d->lastErrorMessage ); + // Load styles from style.xml + oasisStyles.createStyleMap( stylesDoc, true ); + // Also load styles from content.xml + oasisStyles.createStyleMap( contentDoc, false ); + + // TODO post 1.4, pass manifestDoc to the apps so that they don't have to do it themselves + // (when calling KoDocumentChild::loadOasisDocument) + //QDomDocument manifestDoc; + //KoOasisStore oasisStore( store ); + //if ( !oasisStore.loadAndParse( "tar:/META-INF/manifest.xml", manifestDoc, d->lastErrorMessage ) ) + // return false; + + if ( store->hasFile( "settings.xml" ) ) { + (void)oasisStore.loadAndParse( "settings.xml", settingsDoc, d->lastErrorMessage ); + } + if ( !loadOasis( contentDoc, oasisStyles, settingsDoc, store ) ) + return false; + + return true; +} + +bool KoDocument::isInOperation() const +{ + return d->m_numOperations > 0; +} + +void KoDocument::emitBeginOperation() +{ + + /* if we're already in an operation, don't send the signal again */ + if (!isInOperation()) + emit sigBeginOperation(); + d->m_numOperations++; +} + +void KoDocument::emitEndOperation() +{ + d->m_numOperations--; + + /* don't end the operation till we've cleared all the nested operations */ + if (d->m_numOperations == 0) + emit sigEndOperation(); + else if (d->m_numOperations < 0) + /* ignore 'end' calls with no matching 'begin' call */ + d->m_numOperations = 0; +} + + +bool KoDocument::isStoredExtern() const +{ + return !storeInternal() && hasExternURL(); +} + +void KoDocument::setModified( bool mod ) +{ + if ( isAutosaving() ) // ignore setModified calls due to autosaving + return; + + //kdDebug(30003)<<k_funcinfo<<" url:" << m_url.path() << endl; + //kdDebug(30003)<<k_funcinfo<<" mod="<<mod<<" MParts mod="<<KParts::ReadWritePart::isModified()<<" isModified="<<isModified()<<endl; + + if ( mod && !d->modifiedAfterAutosave ) { + // First change since last autosave -> start the autosave timer + setAutoSave( d->m_autoSaveDelay ); + } + d->modifiedAfterAutosave = mod; + + if ( mod == isModified() ) + return; + + KParts::ReadWritePart::setModified( mod ); + + if ( mod ) { + m_bEmpty = FALSE; + } else { + // When saving this document, all non-external child documents get saved too. + QPtrListIterator<KoDocumentChild> it = children(); + for (; it.current(); ++it ) + { + KoDocument *doc = it.current()->document(); + if ( doc && !it.current()->isStoredExtern() && !it.current()->isDeleted() && doc->isModified() ) + doc->setModified( false ); + } + } + + // This influences the title + setTitleModified(); + emit modified( mod ); +} + +void KoDocument::setDoNotSaveExtDoc( bool on ) +{ + d->m_doNotSaveExtDoc = on; +} + +int KoDocument::queryCloseDia() +{ + //kdDebug(30003)<<k_funcinfo<<endl; + + QString name; + if ( documentInfo() ) + { + name = documentInfo()->title(); + } + if ( name.isEmpty() ) + name = url().fileName(); + + if ( name.isEmpty() ) + name = i18n( "Untitled" ); + + int res = KMessageBox::warningYesNoCancel( 0L, + i18n( "<p>The document <b>'%1'</b> has been modified.</p><p>Do you want to save it?</p>" ).arg(name)); + + switch(res) + { + case KMessageBox::Yes : + setDoNotSaveExtDoc(); // Let save() only save myself and my internal docs + save(); // NOTE: External files always in native format. ###TODO: Handle non-native format + setModified( false ); // Now when queryClose() is called by closeEvent it won't do anything. + break; + case KMessageBox::No : + removeAutoSaveFiles(); + setModified( false ); // Now when queryClose() is called by closeEvent it won't do anything. + break; + default : // case KMessageBox::Cancel : + return res; // cancels the rest of the files + } + return res; +} + +int KoDocument::queryCloseExternalChildren() +{ + //kdDebug(30003)<<k_funcinfo<<" checking for children in: "<<url().url()<<endl; + setDoNotSaveExtDoc(false); + QPtrListIterator<KoDocumentChild> it( children() ); + for (; it.current(); ++it ) + { + if ( !it.current()->isDeleted() ) + { + KoDocument *doc = it.current()->document(); + if ( doc ) + { + bool foo = doc->isStoredExtern(); + kdDebug(36001) << "========== isStoredExtern() returned " + << foo << " ==========" << endl; + + if ( foo ) //###TODO: Handle non-native mimetype docs + { + { + kdDebug(30003)<<k_funcinfo<<" found modified child: "<<doc->url().url()<<" extern="<<doc->isStoredExtern()<<endl; + if ( doc->queryCloseDia() == KMessageBox::Cancel ) + return KMessageBox::Cancel; + } + } + if ( doc->queryCloseExternalChildren() == KMessageBox::Cancel ) + return KMessageBox::Cancel; + } + } + } + return KMessageBox::Ok; +} + +void KoDocument::setTitleModified( const QString caption, bool mod ) +{ + //kdDebug(30003)<<k_funcinfo<<" url: "<<url().url()<<" caption: "<<caption<<" mod: "<<mod<<endl; + KoDocument *doc = dynamic_cast<KoDocument *>( parent() ); + if ( doc ) + { + doc->setTitleModified( caption, mod ); + return; + } + // we must be root doc so update caption in all related windows + QPtrListIterator<KoMainWindow> it( d->m_shells ); + for (; it.current(); ++it ) + { + it.current()->updateCaption(); + it.current()->updateReloadFileAction(this); + it.current()->updateVersionsFileAction(this); + } +} + +void KoDocument::setTitleModified() +{ + //kdDebug(30003)<<k_funcinfo<<" url: "<<url().url()<<" extern: "<<isStoredExtern()<<" current: "<<d->m_current<<endl; + KoDocument *doc = dynamic_cast<KoDocument *>( parent() ); + QString caption; + if ( (url().isEmpty() || isStoredExtern()) && d->m_current ) + { + // Get caption from document info (title(), in about page) + if ( documentInfo() ) + { + KoDocumentInfoPage * page = documentInfo()->page( QString::fromLatin1("about") ); + if (page) + caption = static_cast<KoDocumentInfoAbout *>(page)->title(); + } + if ( caption.isEmpty() ) + caption = url().prettyURL( 0, KURL::StripFileProtocol ); // Fall back to document URL + + //kdDebug(30003)<<k_funcinfo<<" url: "<<url().url()<<" caption: "<<caption<<endl; + if ( doc ) + { + doc->setTitleModified( caption, isModified() ); + return; + } + else + { + // we must be root doc so update caption in all related windows + setTitleModified( caption, isModified() ); + return; + } + } + if ( doc ) + { + // internal doc or not current doc, so pass on the buck + doc->setTitleModified(); + } +} + +bool KoDocument::loadChildren( KoStore* ) +{ + return true; +} + +bool KoDocument::completeLoading( KoStore* ) +{ + return true; +} + +bool KoDocument::completeSaving( KoStore* ) +{ + return true; +} + +QDomDocument KoDocument::createDomDocument( const QString& tagName, const QString& version ) const +{ + return createDomDocument( instance()->instanceName(), tagName, version ); +} + +//static +QDomDocument KoDocument::createDomDocument( const QString& appName, const QString& tagName, const QString& version ) +{ + QDomImplementation impl; + QString url = QString("http://www.koffice.org/DTD/%1-%1.dtd").arg(appName).arg(version); + QDomDocumentType dtype = impl.createDocumentType( tagName, + QString("-//KDE//DTD %1 %1//EN").arg(appName).arg(version), + url ); + // The namespace URN doesn't need to include the version number. + QString namespaceURN = QString("http://www.koffice.org/DTD/%1").arg(appName); + QDomDocument doc = impl.createDocument( namespaceURN, tagName, dtype ); + doc.insertBefore( doc.createProcessingInstruction( "xml", "version=\"1.0\" encoding=\"UTF-8\"" ), doc.documentElement() ); + return doc; +} + +KoXmlWriter* KoDocument::createOasisXmlWriter( QIODevice* dev, const char* rootElementName ) +{ + KoXmlWriter* writer = new KoXmlWriter( dev ); + writer->startDocument( rootElementName ); + writer->startElement( rootElementName ); + writer->addAttribute( "xmlns:office", KoXmlNS::office ); + writer->addAttribute( "xmlns:meta", KoXmlNS::meta ); + + if ( qstrcmp( rootElementName, "office:document-meta" ) != 0 ) { + writer->addAttribute( "xmlns:config", KoXmlNS::config ); + writer->addAttribute( "xmlns:text", KoXmlNS::text ); + writer->addAttribute( "xmlns:table", KoXmlNS::table ); + writer->addAttribute( "xmlns:draw", KoXmlNS::draw ); + writer->addAttribute( "xmlns:presentation", KoXmlNS::presentation ); + writer->addAttribute( "xmlns:dr3d", KoXmlNS::dr3d ); + writer->addAttribute( "xmlns:chart", KoXmlNS::chart ); + writer->addAttribute( "xmlns:form", KoXmlNS::form ); + writer->addAttribute( "xmlns:script", KoXmlNS::script ); + writer->addAttribute( "xmlns:style", KoXmlNS::style ); + writer->addAttribute( "xmlns:number", KoXmlNS::number ); + writer->addAttribute( "xmlns:math", KoXmlNS::math ); + writer->addAttribute( "xmlns:svg", KoXmlNS::svg ); + writer->addAttribute( "xmlns:fo", KoXmlNS::fo ); + writer->addAttribute( "xmlns:koffice", KoXmlNS::koffice ); + } + // missing: office:version="1.0" + + writer->addAttribute( "xmlns:dc", KoXmlNS::dc ); + writer->addAttribute( "xmlns:xlink", KoXmlNS::xlink ); + return writer; +} + +QDomDocument KoDocument::saveXML() +{ + kdError(30003) << "KoDocument::saveXML not implemented" << endl; + d->lastErrorMessage = i18n( "Internal error: saveXML not implemented" ); + return QDomDocument(); +} + +KService::Ptr KoDocument::nativeService() +{ + if ( !m_nativeService ) + m_nativeService = readNativeService( instance() ); + + return m_nativeService; +} + +QCString KoDocument::nativeFormatMimeType() const +{ + KService::Ptr service = const_cast<KoDocument *>(this)->nativeService(); + if ( !service ) + return QCString(); + QCString nativeMimeType = service->property( "X-KDE-NativeMimeType" ).toString().latin1(); + if ( nativeMimeType.isEmpty() ) { + // shouldn't happen, let's find out why it happened + if ( !service->serviceTypes().contains( "KOfficePart" ) ) + kdWarning(30003) << "Wrong desktop file, KOfficePart isn't mentionned" << endl; + else if ( !KServiceType::serviceType( "KOfficePart" ) ) + kdWarning(30003) << "The KOfficePart service type isn't installed!" << endl; + } + return nativeMimeType; +} + +QCString KoDocument::nativeOasisMimeType() const +{ + KService::Ptr service = const_cast<KoDocument *>(this)->nativeService(); + if ( !service ) + return QCString(); + return service->property( "X-KDE-NativeOasisMimeType" ).toString().latin1(); +} + + +//static +KService::Ptr KoDocument::readNativeService( KInstance *instance ) +{ + QString instname = instance ? instance->instanceName() : kapp->instanceName(); + + // The new way is: we look for a foopart.desktop in the kde_services dir. + QString servicepartname = instname + "part.desktop"; + KService::Ptr service = KService::serviceByDesktopPath( servicepartname ); + if ( service ) + kdDebug(30003) << servicepartname << " found." << endl; + if ( !service ) + { + // The old way is kept as fallback for compatibility, but in theory this is really never used anymore. + + // Try by path first, so that we find the global one (which has the native mimetype) + // even if the user created a kword.desktop in ~/.kde/share/applnk or any subdir of it. + // If he created it under ~/.kde/share/applnk/Office/ then no problem anyway. + service = KService::serviceByDesktopPath( QString::fromLatin1("Office/%1.desktop").arg(instname) ); + } + if ( !service ) + service = KService::serviceByDesktopName( instname ); + + return service; +} + +QCString KoDocument::readNativeFormatMimeType( KInstance *instance ) //static +{ + KService::Ptr service = readNativeService( instance ); + if ( !service ) + return QCString(); + + if ( service->property( "X-KDE-NativeMimeType" ).toString().isEmpty() ) + { + // It may be that the servicetype "KOfficePart" is missing, which leads to this property not being known + if ( KServiceType::serviceType( "KOfficePart" ) == 0L ) + kdError(30003) << "The serviceType KOfficePart is missing. Check that you have a kofficepart.desktop file in the share/servicetypes directory." << endl; + else { + QString instname = instance ? instance->instanceName() : kapp->instanceName(); + if ( instname != "koshell" ) // hack for koshell + kdWarning(30003) << service->desktopEntryPath() << ": no X-KDE-NativeMimeType entry!" << endl; + } + } + + return service->property( "X-KDE-NativeMimeType" ).toString().latin1(); +} + +QStringList KoDocument::readExtraNativeMimeTypes( KInstance *instance ) //static +{ + KService::Ptr service = readNativeService( instance ); + if ( !service ) + return QStringList(); + return service->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); +} + +void KoDocument::setupXmlReader( QXmlSimpleReader& reader, bool namespaceProcessing ) +{ + if ( namespaceProcessing ) + { + reader.setFeature( "http://xml.org/sax/features/namespaces", TRUE ); + reader.setFeature( "http://xml.org/sax/features/namespace-prefixes", FALSE ); + } + else + { + reader.setFeature( "http://xml.org/sax/features/namespaces", FALSE ); + reader.setFeature( "http://xml.org/sax/features/namespace-prefixes", TRUE ); + } + reader.setFeature( "http://trolltech.com/xml/features/report-whitespace-only-CharData", TRUE ); +} + + +bool KoDocument::isNativeFormat( const QCString& mimetype ) const +{ + if ( mimetype == nativeFormatMimeType() ) + return true; + return extraNativeMimeTypes().contains( mimetype ); +} + +QStringList KoDocument::extraNativeMimeTypes() const +{ + QStringList lst; + // This implementation is temporary while we treat both koffice-1.3 and OASIS formats as native. + // But it's good to have this virtual method, in case some app want to + // support more than one native format. + KService::Ptr service = const_cast<KoDocument *>(this)->nativeService(); + if ( !service ) // can't happen + return lst; + return service->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); +} + +int KoDocument::supportedSpecialFormats() const +{ + // Apps which support special output flags can add reimplement and add to this. + // E.g. this is how did "saving in the 1.1 format". + // SaveAsDirectoryStore is a given since it's implemented by KoDocument itself. + return SaveAsDirectoryStore; +} + +void KoDocument::addShell( KoMainWindow *shell ) +{ + if ( d->m_shells.findRef( shell ) == -1 ) + { + //kdDebug(30003) << "addShell: shell " << (void*)shell << " added to doc " << this << endl; + d->m_shells.append( shell ); + } +} + +void KoDocument::removeShell( KoMainWindow *shell ) +{ + //kdDebug(30003) << "removeShell: shell " << (void*)shell << " removed from doc " << this << endl; + d->m_shells.removeRef( shell ); +} + +const QPtrList<KoMainWindow>& KoDocument::shells() const +{ + return d->m_shells; +} + +int KoDocument::shellCount() const +{ + return d->m_shells.count(); +} + +DCOPObject * KoDocument::dcopObject() +{ + if ( !d->m_dcopObject ) + d->m_dcopObject = new KoDocumentIface( this ); + return d->m_dcopObject; +} + +QCString KoDocument::dcopObjectId() const +{ + return const_cast<KoDocument *>(this)->dcopObject()->objId(); +} + +void KoDocument::setErrorMessage( const QString& errMsg ) +{ + d->lastErrorMessage = errMsg; +} + +QString KoDocument::errorMessage() const +{ + return d->lastErrorMessage; +} + +void KoDocument::showSavingErrorDialog() +{ + if ( d->lastErrorMessage.isEmpty() ) + { + KMessageBox::error( 0L, i18n( "Could not save\n%1" ).arg( m_file ) ); + } + else if ( d->lastErrorMessage != "USER_CANCELED" ) + { + KMessageBox::error( 0L, i18n( "Could not save %1\nReason: %2" ).arg( m_file, d->lastErrorMessage ) ); + } +} + +void KoDocument::showLoadingErrorDialog() +{ + if ( d->lastErrorMessage.isEmpty() ) + { + KMessageBox::error( 0L, i18n( "Could not open\n%1" ).arg( url().prettyURL( 0, KURL::StripFileProtocol ) ) ); + } + else if ( d->lastErrorMessage != "USER_CANCELED" ) + { + KMessageBox::error( 0L, i18n( "Could not open %1\nReason: %2" ).arg( url().prettyURL( 0, KURL::StripFileProtocol ), d->lastErrorMessage ) ); + } +} + +bool KoDocument::isAutosaving() const +{ + return d->m_autosaving; +} + +bool KoDocument::isLoading() const +{ + return d->m_bLoading; +} + +void KoDocument::removeAutoSaveFiles() +{ + // Eliminate any auto-save file + QString asf = autoSaveFile( m_file ); // the one in the current dir + if ( QFile::exists( asf ) ) + QFile::remove( asf ); + asf = autoSaveFile( QString::null ); // and the one in $HOME + if ( QFile::exists( asf ) ) + QFile::remove( asf ); +} + +void KoDocument::setBackupFile( bool _b ) +{ + d->m_backupFile = _b; +} + +bool KoDocument::backupFile()const +{ + return d->m_backupFile; +} + + +void KoDocument::setBackupPath( const QString & _path) +{ + d->m_backupPath = _path; +} + +QString KoDocument::backupPath()const +{ + return d->m_backupPath; +} + +void KoDocument::setCurrent( bool on ) +{ + //kdDebug(30003)<<k_funcinfo<<" url: "<<url().url()<<" set to: "<<on<<endl; + KoDocument *doc = dynamic_cast<KoDocument *>( parent() ); + if ( doc ) + { + if ( !isStoredExtern() ) + { + // internal doc so set next external to current (for safety) + doc->setCurrent( true ); + return; + } + // only externally stored docs shall have file name in title + d->m_current = on; + if ( !on ) + { + doc->setCurrent( true ); // let my next external parent take over + return; + } + doc->forceCurrent( false ); // everybody else should keep off + } + else + d->m_current = on; + + setTitleModified(); +} + +void KoDocument::forceCurrent( bool on ) +{ + //kdDebug(30003)<<k_funcinfo<<" url: "<<url().url()<<" force to: "<<on<<endl; + d->m_current = on; + KoDocument *doc = dynamic_cast<KoDocument *>( parent() ); + if ( doc ) + { + doc->forceCurrent( false ); + } +} + +bool KoDocument::isCurrent() const +{ + return d->m_current; +} + +bool KoDocument::storeInternal() const +{ + return d->m_storeInternal; +} + +void KoDocument::setStoreInternal( bool i ) +{ + d->m_storeInternal = i; + //kdDebug(30003)<<k_funcinfo<<"="<<d->m_storeInternal<<" doc: "<<url().url()<<endl; +} + +bool KoDocument::hasExternURL() const +{ + return !url().protocol().isEmpty() && url().protocol() != STORE_PROTOCOL && url().protocol() != INTERNAL_PROTOCOL; +} + +void KoDocument::slotStarted( KIO::Job* job ) +{ + if ( job ) + { + job->setWindow( d->m_shells.current() ); + } +} + +static const struct { + const char* localName; + const char* documentType; +} TN2DTArray[] = { + { "text", I18N_NOOP( "a word processing" ) }, + { "spreadsheet", I18N_NOOP( "a spreadsheet" ) }, + { "presentation", I18N_NOOP( "a presentation" ) }, + { "chart", I18N_NOOP( "a chart" ) }, + { "drawing", I18N_NOOP( "a drawing" ) } +}; +static const unsigned int numTN2DT = sizeof( TN2DTArray ) / sizeof( *TN2DTArray ); + +QString KoDocument::tagNameToDocumentType( const QString& localName ) +{ + for ( unsigned int i = 0 ; i < numTN2DT ; ++i ) + if ( localName == TN2DTArray[i].localName ) + return i18n( TN2DTArray[i].documentType ); + return localName; +} + +QValueList<KoTextDocument *> KoDocument::allTextDocuments() const +{ + return QValueList<KoTextDocument *>(); +} + +KoPageLayout KoDocument::pageLayout(int /*pageNumber*/) const +{ + return m_pageLayout; +} + +KoUnit::Unit KoDocument::unit() const +{ + return d->m_unit; +} + +void KoDocument::setUnit( KoUnit::Unit unit ) +{ + if ( d->m_unit != unit ) + { + d->m_unit = unit; + emit unitChanged( unit ); + } +} + +QString KoDocument::unitName() const +{ + return KoUnit::unitName( unit() ); +} + +void KoDocument::showStartUpWidget( KoMainWindow* parent, bool alwaysShow ) +{ + if(!alwaysShow) { + KConfigGroup cfgGrp( instance()->config(), "TemplateChooserDialog" ); + QString fullTemplateName = cfgGrp.readPathEntry( "AlwaysUseTemplate" ); + + if( !fullTemplateName.isEmpty() ) { + openTemplate( fullTemplateName ); + shells().getFirst()->setRootDocument( this ); + return; + } + } + + if(d->m_startUpWidget){ + d->m_startUpWidget->show(); + } else { + d->m_startUpWidget = createOpenPane( parent->centralWidget(), instance(), templateType() ); + } + + parent->setDocToOpen( this ); + parent->factory()->container("mainToolBar", parent)->hide(); +} + +void KoDocument::openExistingFile( const QString& file ) +{ + KURL url( file ); + bool ok = openURL( url ); + setModified( false ); + + if( ok ) + QTimer::singleShot( 0, this, SLOT( deleteOpenPane() ) ); +} + +void KoDocument::openTemplate( const QString& file ) +{ + bool ok = loadNativeFormat( file ); + setModified( false ); + + if ( ok ) { + deleteOpenPane(); + resetURL(); + setEmpty(); + } else { + showLoadingErrorDialog(); + initEmpty(); + } +} + +void KoDocument::initEmpty() +{ + setEmpty(); + setModified(false); +} + +void KoDocument::startCustomDocument() { + deleteOpenPane(); +} + +KoOpenPane* KoDocument::createOpenPane( QWidget* parent, KInstance* instance, + const QString& templateType ) +{ + KoOpenPane* openPane = new KoOpenPane( parent, instance, templateType ); + QWidget *customDoc = createCustomDocumentWidget(openPane); + if(customDoc) { + openPane->setCustomDocumentWidget( customDoc ); + connect( customDoc, SIGNAL( documentSelected() ), this, SLOT( startCustomDocument() ) ); + } + openPane->show(); + + connect( openPane, SIGNAL( openExistingFile( const QString& ) ), + this, SLOT( openExistingFile( const QString& ) ) ); + connect( openPane, SIGNAL( openTemplate( const QString& ) ), + this, SLOT( openTemplate( const QString& ) ) ); + + return openPane; +} + +void KoDocument::setTemplateType( const QString& _templateType ) +{ + d->m_templateType = _templateType; +} + +QString KoDocument::templateType() const +{ + return d->m_templateType; +} + +void KoDocument::deleteOpenPane() +{ + if( d->m_startUpWidget ) { + d->m_startUpWidget->hide(); + QTimer::singleShot(1000, this, SLOT(deleteOpenPaneDelayed())); + + shells().getFirst()->factory()->container("mainToolBar", shells().getFirst())->show(); + shells().getFirst()->setRootDocument( this ); + } else { + emit closeEmbedInitDialog(); + } +} + +void KoDocument::deleteOpenPaneDelayed() +{ + delete d->m_startUpWidget; + d->m_startUpWidget = 0; +} + +QWidget* KoDocument::createCustomDocumentWidget(QWidget */*parent*/) { + return 0; +} + +bool KoDocument::showEmbedInitDialog(QWidget* parent) +{ + KDialogBase dlg(parent, "EmbedInitDialog", true, i18n("Embedding Object"), 0, KDialogBase::NoDefault); + KoOpenPane* pane = createOpenPane(&dlg, instance(), templateType()); + pane->layout()->setMargin(0); + dlg.setMainWidget(pane); + dlg.setInitialSize(dlg.configDialogSize("EmbedInitDialog")); + connect(this, SIGNAL(closeEmbedInitDialog()), &dlg, SLOT(slotOk())); + + bool ok = dlg.exec() == QDialog::Accepted; + + dlg.saveDialogSize("EmbedInitDialog"); + + return ok; +} + +#include "KoDocument_p.moc" +#include "KoDocument.moc" diff --git a/lib/kofficecore/KoDocument.h b/lib/kofficecore/KoDocument.h new file mode 100644 index 00000000..1e8f1f83 --- /dev/null +++ b/lib/kofficecore/KoDocument.h @@ -0,0 +1,1158 @@ +// -*- c-basic-offset: 4 -*- +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright (C) 2000-2005 David Faure <faure@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. +*/ + +#ifndef __ko_document_h__ +#define __ko_document_h__ + +class KoTextDocument; +namespace std { } +using namespace std; +#include <qwmatrix.h> + +#include <kparts/part.h> +#include <kurl.h> +#include <kservice.h> +#include <KoGlobal.h> +#include <KoUnit.h> +#include <KoPageLayout.h> +#include <koffice_export.h> + +class QDomElement; +class QDomDocument; +class QXmlSimpleReader; + +class KoStore; +class KoMainWindow; + +class KoChild; +class KoDocumentChild; +class KoView; +class KoDocumentInfo; +class DCOPObject; +class KoOasisStyles; +class KoXmlWriter; +class KoOpenPane; + +/** + * The %KOffice document class + * + * This class provides some functionality each %KOffice document should have. + * + * @short The %KOffice document class + */ +class KOFFICECORE_EXPORT KoDocument : public KParts::ReadWritePart +{ + Q_OBJECT + Q_PROPERTY( QCString dcopObjectId READ dcopObjectId) + Q_PROPERTY( bool backupFile READ backupFile WRITE setBackupFile ) + +public: + + /** + * Constructor. + * The first 4 arguments are the same as the ones passed to KParts::Factory::createPart. + * + * @param parentWidget the parent widget, in case we create a wrapper widget + * (in single view mode). + * Usually the first argument passed by KParts::Factory::createPart. + * @param widgetName name of the widget. + * @param parent may be another KoDocument, or anything else. + * Usually the third argument of KParts::Factory::createPart. + * @param name is used to identify this document via DCOP so you may want to + * pass a meaningful name here which matches the pattern [A-Za-z_][A-Za-z_0-9]*. + * @param singleViewMode determines whether the document may only have one view. In this case + * the @p parent must be a QWidget derived class. KoDocument will then create a wrapper widget + * (KoViewWrapperWidget) which is a child of @p parentWidget. + * This widget can be retrieved by calling widget(). + * + * @todo explain what the purpose of widgetName is. + */ + KoDocument( QWidget* parentWidget, + const char* widgetName, + QObject* parent, + const char* name, + bool singleViewMode = false ); + + /** + * Destructor. + * + * The destructor does not delete any attached KoView objects and it does not + * delete the attached widget as returned by widget(). + */ + virtual ~KoDocument(); + + /** + * Tells whether this document is in singleview mode. This mode can only be set + * in the constructor. + */ + bool isSingleViewMode() const; + + /** + * Is the document embedded? + */ + bool isEmbedded() const; + + /** + * Returns the action described action object. In fact only the "name" attribute + * of @p element is of interest here. The method searches first in the + * KActionCollection of the first view and then in the KActionCollection of this + * document. + * This allows %KOffice applications to define actions in both the view and the document. + * They should only define view-actions (like zooming and stuff) in the view. + * Every action which changes the document should be defined in the document. + * + * Please notice that KoDocument indirectly inherits KXMLGUIClient. + * + * @see KXMLGUIClient + * @see KXMLGUIClient::actionCollection + * @see KoView::action + */ + virtual KAction *action( const QDomElement &element ) const; + + /** + * Returns the DOM document which describes the GUI of the + * first view. + */ + virtual QDomDocument domDocument() const; + + /** + * @internal + */ + virtual void setManager( KParts::PartManager *manager ); + + /** + * Reimplemented from KParts::ReadWritePart for internal reasons + * (for the autosave functionality) + */ + virtual bool openURL( const KURL & url ); + + /** + * Opens the document given by @p url, without storing the URL + * in the KoDocument. + * Call this instead of openURL() to implement KoMainWindow's + * File --> Import feature. + * + * @note This will call openURL(). To differentiate this from an ordinary + * Open operation (in any reimplementation of openURL() or openFile()) + * call isImporting(). + */ + bool import( const KURL &url ); + + /** + * Saves the document as @p url without changing the state of the + * KoDocument (URL, modified flag etc.). Call this instead of + * KParts::ReadWritePart::saveAs() to implement KoMainWindow's + * File --> Export feature. + * + * @note This will call KoDocument::saveAs(). To differentiate this + * from an ordinary Save operation (in any reimplementation of + * saveFile()) call isExporting(). + */ + bool exp0rt( const KURL &url ); + + /** + * @brief Sets whether the document can be edited or is read only. + * + * This recursively applied to all child documents and + * KoView::updateReadWrite is called for every attached + * view. + */ + virtual void setReadWrite( bool readwrite = true ); + + /** + * @brief Used by KoApplication, and by KoMainWindow, when no document exists yet. + * + * With the help of @p instance or KApplication::instance() this + * method figures out which .desktop file matches this application. In this + * file it searches for the "X-KDE-NativeMimeType" entry and returns it. + * + * @see KService + * @see KDesktopFile + */ + static QCString readNativeFormatMimeType( KInstance *instance = 0 ); + + /** + * Used by KoMainWindow, when no document exists yet. + * + * With the help of @p instance or KApplication::instance() this + * method figures out which .desktop file matches this application. In this + * file it searches for the "X-KDE-ExtraNativeMimeTypes" entry and returns it. + * + * @see KService + * @see KDesktopFile + */ + static QStringList readExtraNativeMimeTypes( KInstance *instance = 0 ); + + /** + * With the help of @p instance or KApplication::instance() this + * method figures out which .desktop file matches this application, + * and returns the KService instance for it. + */ + static KService::Ptr readNativeService( KInstance *instance = 0 ); + + /** + * setup the XML reader, so that we don't have to duplicate the code. + */ + static void setupXmlReader( QXmlSimpleReader& reader, bool namespaceProcessing = false ); + + /** + * To be preferred when a document exists. It is fast when calling + * it multiple times since it caches the result that readNativeFormatMimeType() + * delivers. + * This comes from the X-KDE-NativeMimeType key in the .desktop file + * You do NOT have to reimplement this (it is only virtual for kounavail). + */ + virtual QCString nativeFormatMimeType() const; + + /** + * Returns the OASIS OpenDocument mimetype of the document, if supported + * This comes from the X-KDE-NativeOasisMimeType key in the .desktop file + */ + QCString nativeOasisMimeType() const; + + /// Checks whether a given mimetype can be handled natively. + bool isNativeFormat( const QCString& mimetype ) const; + + /// Returns a list of the mimetypes considered "native", i.e. which can + /// be saved by KoDocument without a filter, in *addition* to the main one + virtual QStringList extraNativeMimeTypes() const; + + /// Enum values used by specialOutputFlag - note that it's a bitfield for supportedSpecialFormats + enum { /*SaveAsKOffice1dot1 = 1,*/ // old and removed + SaveAsDirectoryStore = 2, + SaveAsFlatXML = 4 + // bitfield! next value is 8 + }; + + /** + * Return the set of SupportedSpecialFormats that the application wants to + * offer in the "Save" file dialog. + */ + virtual int supportedSpecialFormats() const; + + /** + * Returns the actual mimetype of the document + */ + QCString mimeType() const; + + /** + * @brief Sets the mime type for the document. + * + * When choosing "save as" this is also the mime type + * selected by default. + */ + void setMimeType( const QCString & mimeType ); + + /** + * @brief Set the format in which the document should be saved. + * + * This is called on loading, and in "save as", so you shouldn't + * have to call it. + * + * @param mimeType the mime type (format) to use. + * @param specialOutputFlag is for "save as older version" etc. + */ + void setOutputMimeType( const QCString & mimeType, int specialOutputFlag = 0 ); + QCString outputMimeType() const; + int specialOutputFlag() const; + + /** + * Returns true if this document was the result of opening a foreign + * file format and if the user hasn't yet saved the document (in any + * format). + * + * Used by KoMainWindow to warn the user when s/he lazily presses + * CTRL+S to save in the same foreign format, putting all his/her + * formatting at risk (normally an export confirmation only comes up + * with Save As). + * + * @param exporting specifies whether this is the setting for a + * File --> Export or File --> Save/Save As operation. + */ + bool confirmNonNativeSave( const bool exporting ) const; + void setConfirmNonNativeSave( const bool exporting, const bool on ); + + virtual bool wantExportConfirmation() const; + + /** + * Sets the error message to be shown to the user (use i18n()!) + * when loading or saving fails. + * If you asked the user about something and he chose "Cancel", + * set the message to the magic string "USER_CANCELED", to skip the error dialog. + */ + void setErrorMessage( const QString& errMsg ); + + /** + * Return the last error message. Usually KoDocument takes care of + * showing it; this method is mostly provided for non-interactive use. + * @since 1.4 + */ + QString errorMessage() const; + + /** + * Show the last error message in a message box. + * The dialog box will mention a saving problem. + * Note that save/saveFile takes care of doing it. + * @since 1.4 + */ + void showSavingErrorDialog(); + + /** + * Show the last error message in a message box. + * The dialog box will mention a loading problem. + * openURL/openFile takes care of doing it, but not loadNativeFormat itself, + * so this is often called after loadNativeFormat returned false. + * @since 1.4 + */ + void showLoadingErrorDialog(); + + /** + * Create a new view for the document. + */ + KoView *createView( QWidget *parent = 0, const char *name = 0 ); + + /** + * Adds a view to the document. + * + * This calls KoView::updateReadWrite to tell the new view + * whether the document is readonly or not. + */ + virtual void addView( KoView *view ); + + /** + * Removes a view of the document. + */ + virtual void removeView( KoView *view ); + + /** + * @return a list of views this document is displayed in + */ + const QPtrList<KoView> & views() const; + + /** + * @return number of views this document is displayed in + */ + int viewCount() const; + + /** + * Reimplemented from KParts::Part + */ + virtual KParts::Part *hitTest( QWidget *widget, const QPoint &globalPos ); + + /** + * Find the most nested child document which contains the + * requested point. The point is in the coordinate system + * of this part. If no child document contains this point, then + * a pointer to this document is returned. + * + * This function has to be overloaded if the document features child documents. + * + * @param pos is in (unzoomed) document coordinates + * @param matrix transforms points from the documents coordinate system + * to the coordinate system of the requested point. This is used by + * transformed child documents, see KoDocumentChild/KoChild. + * + * @return Pointer to the document under the mouse at that position + */ + virtual KoDocument *hitTest( const QPoint &pos, const QWMatrix& matrix = QWMatrix() ); + /// Temporary API for accessing the view that calls hitTest. + /// Will be passed to hitTest() in 2.x. + /// Only call this from within hitTest()! + KoView* hitTestView(); + + /** + * Paints the whole document into the given painter object. + * + * @param painter The painter object onto which will be drawn. + * @param rect The rect that should be used in the painter object. + * @param transparent If true then the entire rectangle is erased before painting. + * @param view The KoView is needed to fiddle about with the active widget, when painting children. + * @param zoomX The zoom value to be applied to X coordinates when painting. + * @param zoomY The zoom value to be applied to Y coordinates when painting. + */ + virtual void paintEverything( QPainter &painter, const QRect &rect, bool transparent = false, + KoView *view = 0L, double zoomX = 1.0, double zoomY = 1.0 ); + + /** + * @brief Generates a preview picture of the document + * @note The preview is used in the File Dialog and also to create the Thumbnail + */ + virtual QPixmap generatePreview( const QSize& size ); + + /** + * Paints all of the documents children into the given painter object. + * + * @param painter The painter object onto which will be drawn. + * @param rect The rect that should be used in the painter object. + * @param view The KoView is needed to fiddle about with the active widget. + * @param zoomX The zoom value to be applied to X coordinates when painting. + * @param zoomY The zoom value to be applied to Y coordinates when painting. + * + * @see #paintChild #paintEverything #paintContent + */ + virtual void paintChildren( QPainter &painter, const QRect &rect, KoView *view, double zoomX = 1.0, double zoomY = 1.0 ); + + /** + * Paint a given child. Normally called by paintChildren(). + * + * @param child The child to be painted. + * @param painter The painter object onto which will be drawn. + * @param view The KoView is needed to fiddle about with the active widget. + * @param zoomX The zoom value to be applied to X coordinates when painting. + * @param zoomY The zoom value to be applied to Y coordinates when painting. + * + * @see #paintEverything #paintChildren #paintContent + */ + virtual void paintChild( KoDocumentChild *child, QPainter &painter, KoView *view, + double zoomX = 1.0, double zoomY = 1.0 ); + + /** + * Paints the data itself. Normally called by paintEverything(). It does not + * paint the children. + * It's this method that %KOffice Parts have to implement. + * + * @param painter The painter object onto which will be drawn. + * @param rect The rect that should be used in the painter object. + * @param transparent If false the implementing method should fill the background. + * @param zoomX The zoom value to be applied to X coordinates when painting. + * @param zoomY The zoom value to be applied to Y coordinates when painting. + * + * @see #paintEverything + */ + virtual void paintContent( QPainter &painter, const QRect &rect, bool transparent = false, + double zoomX = 1.0, double zoomY = 1.0 ) = 0; + + /** + * Called by koApplication to check for an autosave file in $HOME + */ + bool checkAutoSaveFile(); + + /** + * This setting indicates who is calling initDoc. + * Usually the app will want to + * - show the template dialog with 'everything' if InitDocAppStarting, InitDocFileClose or InitDocEmbedded + * - show the template dialog with 'templates only' if InitDocFileNew + * - create an empty document with default settings if InitDocEmpty + */ + enum InitDocFlags { InitDocAppStarting, InitDocFileNew, InitDocFileClose, InitDocEmbedded, InitDocEmpty }; + + /** + * Initializes an empty document (display the template dialog!). + * You have to overload this method to initialize all your document variables. + * @param flags see InitDocFlags + * @param parentWidget the widget this document belongs with + */ + virtual bool initDoc(InitDocFlags flags, QWidget* parentWidget=0) = 0; + + /** + * Creates and shows the start up widget. + * @param parent the KoMainWindow used as parent for the widget. + * @param alwaysShow always show the widget even if the user has configured it to not show. + * @since 1.5 + */ + virtual void showStartUpWidget(KoMainWindow* parent, bool alwaysShow = false); + + /** + * Sets the modified flag on the document. This means that it has + * to be saved or not before deleting it. + */ + virtual void setModified( bool _mod ); + + /** + * Tells the document that its title has been modified, either because + * the modified status changes (this is done by setModified() ) or + * because the URL or the document-info's title changed. + */ + virtual void setTitleModified(); + + /** + * @return true if the document is empty. + */ + virtual bool isEmpty() const { return m_bEmpty; } + + /** + * @brief Sets the document to empty. + * + * Used after loading a template + * (which is not empty, but not the user's input). + * + * @see isEmpty() + */ + virtual void setEmpty() { m_bEmpty = true; } + + /** + * @brief Loads a document from a store. + * + * You should never have to reimplement. + * + * @param store The store to load from + * @param url An internal url, like tar:/1/2 + */ + virtual bool loadFromStore( KoStore* store, const QString& url ); + + /** + * @brief Loads an OASIS document from a store. + * This is used for both the main document and embedded objects. + */ + virtual bool loadOasisFromStore( KoStore* store ); + + /** + * @brief Saves a document to a store. + * + * You should not have to reimplement this - but call it in saveChildren(). + */ + virtual bool saveToStore( KoStore* store, const QString& path ); + + /** + * Reimplement this method to load the contents of your %KOffice document, + * from the XML document. This is for the pre-Oasis file format (maindoc.xml). + * + * You are supposed to use the QDomDocument. The QIODevice is provided only + * for the cases where some pre-processing is needed, like kpresenter's kprconverter. + * Note that the QIODevice could be 0L, when called from an import filter. + */ + virtual bool loadXML( QIODevice *, const QDomDocument & doc ) = 0; + + /** + * Reimplement this method to load the contents of your %KOffice document, + * from the XML document ("content.xml"). The styles have been parsed already, + * you can find them in the oasisStyles parameter. The store can be used + * to load images and embedded documents. + */ + virtual bool loadOasis( const QDomDocument & doc, KoOasisStyles& oasisStyles, + const QDomDocument & settings, KoStore* store ) = 0; + + /** + * Reimplement this method to save the contents of your %KOffice document, + * using the OASIS format. + */ + virtual bool saveOasis( KoStore* store, KoXmlWriter* manifestWriter ) = 0; + + /** + * Reimplement this to save the contents of the %KOffice document into + * a QDomDocument. The framework takes care of saving it to the store. + */ + virtual QDomDocument saveXML(); + + /** + * Return a correctly created QDomDocument for this KoDocument, + * including processing instruction, complete DOCTYPE tag (with systemId and publicId), and root element. + * @param tagName the name of the tag for the root element + * @param version the DTD version (usually the application's version). + * @deprecated use createOasisXmlWriter instead + */ + QDomDocument createDomDocument( const QString& tagName, const QString& version ) const; + + /** + * Return an XML writer for saving Oasis XML into the device @p dev, + * including the XML processing instruction, + * and the root element with all its namespaces. + * You can add more namespaces afterwards with addAttribute. + * + * @param dev the device into which the XML will be written. + * @param rootElementName the tag name of the root element. + * This is either office:document, office:document-content, + * office:document-styles, office:document-meta or office:document-settings + * @return the KoXmlWriter instance. It becomes owned by the caller, which + * must delete it at some point. + * + * Once done with writing the contents of the root element, you + * will need to call endElement(); endDocument(); before destroying the KoXmlWriter. + * @note OASIS-specific + */ + static KoXmlWriter* createOasisXmlWriter( QIODevice* dev, const char* rootElementName ); + + /** + * Return a correctly created QDomDocument for an old (1.3-style) %KOffice document, + * including processing instruction, complete DOCTYPE tag (with systemId and publicId), and root element. + * This static method can be used e.g. by filters. + * @param appName the app's instance name, e.g. kword, kspread, kpresenter etc. + * @param tagName the name of the tag for the root element, e.g. DOC for kword/kpresenter. + * @param version the DTD version (usually the application's version). + */ + static QDomDocument createDomDocument( const QString& appName, const QString& tagName, const QString& version ); + + /** + * The first thing to do in loadOasis is get hold of the office:body tag, then its child. + * If the child isn't the expected one, the error message can indicate what it is instead. + * This method returns a translated name for the type of document, + * e.g. i18n("Word Processing") for office:text. + */ + static QString tagNameToDocumentType( const QString& localName ); + + /** + * Save the document. The default implementation is to call + * saveXML(). This method exists only for applications that + * don't use QDomDocument for saving, i.e. kword and kpresenter. + */ + virtual bool saveToStream( QIODevice * dev ); + + /** + * Loads a document in the native format from a given URL. + * Reimplement if your native format isn't XML. + * + * @param file the file to load - usually KReadOnlyPart::m_file or the result of a filter + */ + virtual bool loadNativeFormat( const QString & file ); + + /** + * Saves the document in native format, to a given file + * You should never have to reimplement. + * Made public for writing templates. + */ + virtual bool saveNativeFormat( const QString & file ); + + /** + * Activate/deactivate/configure the autosave feature. + * @param delay in seconds, 0 to disable + */ + void setAutoSave( int delay ); + + /** + * Checks whether the document is currently in the process of autosaving + */ + bool isAutosaving() const; + + /** + * Set whether the next openURL call should check for an auto-saved file + * and offer to open it. This is usually true, but can be turned off + * (e.g. for the preview module). + */ + void setCheckAutoSaveFile( bool b ); + + /** + * Set whether the next openURL call should show error message boxes in case + * of errors. This is usually the case, but e.g. not when generating thumbnail + * previews. + */ + void setAutoErrorHandlingEnabled( bool b ); + + /** + * Checks whether error message boxes should be shown. + * @since 1.3.1 + */ + bool isAutoErrorHandlingEnabled() const; + + /** + * Retrieve the default value for autosave in seconds. + * Called by the applications to use the correct default in their config + */ + static int defaultAutoSave() { return s_defaultAutoSave; } + + /** + * @return the list of all children. Do not modify the + * returned list. + */ + const QPtrList<KoDocumentChild>& children() const; + + /** + * @return the KoDocumentChild associated with the given Document, but only if + * @p doc is a direct child of this document. + * + * This is a convenience function. You could get the same result + * by traversing the list returned by children(). + */ + KoDocumentChild *child( KoDocument *doc ); + + /** + * @return the information concerning this document. + * @see KoDocumentInfo + */ + KoDocumentInfo *documentInfo() const; + + void setViewBuildDocument( KoView *view, const QDomDocument &doc ); + QDomDocument viewBuildDocument( KoView *view ); + + /** + * Appends the shell to the list of shells which show this + * document as their root document. + * + * This method is automatically called from KoMainWindow::setRootDocument, + * so you dont need to call it. + */ + virtual void addShell( KoMainWindow *shell ); + + /** + * Removes the shell from the list. That happens automatically if the shell changes its + * root document. Usually you dont need to call this method. + */ + virtual void removeShell( KoMainWindow *shell ); + + /** + * @return the list of shells for the main window + */ + const QPtrList<KoMainWindow>& shells() const; + + /** + * @return the number of shells for the main window + */ + int shellCount() const; + + /** + * @return the list of all the currently opened documents + */ + static QPtrList<KoDocument> *documentList() { return s_documentList; } + + /** + * @brief Return a DCOP interface for this document + * + * %KOffice parts are strongly recommended to reimplement this method, + * so that their DCOP interface provides more functionality than the basic KoDocumentIface + */ + virtual DCOPObject * dcopObject(); + + /** + * @return the ID of the DCOP interface for this document. + **/ + QCString dcopObjectId() const; + + /** + * Signal the progress of operations such as loading or saving. + */ + void emitProgress( int value ) { emit sigProgress( value ); } + + bool isInOperation() const; + virtual void emitBeginOperation(); + virtual void emitEndOperation(); + + /** + * Return true if url() is a real filename, false if url() is + * an internal url in the store, like "tar:/..." + */ + virtual bool isStoredExtern() const; + + /** + * @return the page layout associated with this document (margins, pageSize, etc). + * Override this if you want to provide different sized pages. + * + * @see KoPageLayout + */ + virtual KoPageLayout pageLayout(int pageNumber = 0) const; + + /** + * Performs a cleanup of unneeded backup files + */ + void removeAutoSaveFiles(); + + void setBackupFile( bool _b ); + + bool backupFile()const; + + /** + * Returns true if this document or any of its internal child documents are modified. + */ + bool isModified() const; + + /** + * Returns true during loading (openURL can be asynchronous) + */ + bool isLoading() const; + + int queryCloseExternalChildren(); + int queryCloseDia(); + + /** + * @brief Set when we do not want to save external children when saving our 'main' doc. + * + * This makes it possible to save 'main' doc + all its internal children first, then + * go on to save external children. Typically used by query close. + * Use: + * @code + * doc->setDoNotSaveExtDoc(); + * doc->save(); // saves doc and its internal children, + * //also calls saveExternalChildren() which sets setDoNotSaveExtDoc(false) + * doc->saveExternalChildren(); + * @endcode + */ + void setDoNotSaveExtDoc( bool on = true ); + + /** + * Sets the backup path of the document + */ + void setBackupPath( const QString & _path ); + + /** + * @return path to the backup document + */ + QString backupPath()const; + + /** + * Indicates that this document is currently viewed + * and thus should control the title caption. + * Also resets current flag for all parents. + */ + void setCurrent( bool on = true ); + + /** + * Sets current flag for this document and all its parents + */ + void forceCurrent( bool on ); + bool isCurrent() const; + + void setTitleModified( const QString caption, bool mod ); + + /** + * Sets the document URL to empty URL + * KParts doesn't allow this, but %KOffice apps have e.g. templates + * After using loadNativeFormat on a template, one wants + * to set the url to KURL() + */ + void resetURL() { m_url = KURL(); m_file = QString::null; } + + /** + * Set when you want an external embedded document to be stored internally + */ + void setStoreInternal( bool i ); + + /** + * @return true when external embedded documents are stored internally + */ + bool storeInternal() const; + + bool hasExternURL() const; + + /** + * Sets the document URL to @p url + * KParts doesn't really allow this, but it is needed for undo of setStoreInternal() + */ + void setURL( const KURL& url ) { m_url = url; } + + /** + * _Only_ use these functions to restore m_file (in KoMainWindow) after a + * failed save (remember to use setURL() to restore the URL as well). + * + * @warning Do _not_ use these functions for any other purpose. + * + * @internal + */ + QString &file() { return m_file; } + + /** + * _Only_ use these functions to restore m_file (in KoMainWindow) after a + * failed save (remember to use setURL() to restore the URL as well). + * + * @warning Do _not_ use these functions for any other purpose. + * + * @internal + */ + void setFile( const QString &file ) { m_file = file; } + + /** + * @internal (public for KoMainWindow) + */ + void setMimeTypeAfterLoading( const QString& mimeType ); + + /** + * @return returns the number of pages in the document. + */ + virtual int pageCount() const { return 1; } + + /** + * @return all kotext-based text objects in the document + * This is used by some text-analyzer plugins. + * @since 1.5 + */ + virtual QValueList<KoTextDocument *> allTextDocuments() const; + + /** + * Returns the unit used to display all measures/distances. + * @since 1.5 + */ + KoUnit::Unit unit() const; + + /** + * Sets the unit used to display all measures/distances. + * @since 1.5 + */ + void setUnit( KoUnit::Unit u ); + + /** + * Returns the name of the unit used to display all measures/distances. + * Use this method for displaying it in the user interface, but use + * unit() for everything else (conversions etc.) + * @since 1.5 + */ + QString unitName() const; + + /** + * Set the template type used. This is used by the start up widget to show + * the correct templates. + * @since 1.5 + */ + void setTemplateType(const QString& _templateType); + /** + * Template type used. This is used by the start up widget to show + * the correct templates. + * @since 1.5 + */ + QString templateType() const; + + /** + * Shows the init dialog when embeding + * @param parent the parent widget + * @since 1.5 + */ + virtual bool showEmbedInitDialog(QWidget* parent); + +public slots: + /** + * Initialize an empty document using default values + * @since 1.5 + */ + virtual void initEmpty(); + +signals: + + /** + * This signal is emitted when the unit is changed by setUnit() + * It is common to connect views to it, in order to change the displayed units + * (e.g. in the rulers) + */ + void unitChanged(KoUnit::Unit); + + /** + * This signal is emitted when a direct or indirect child document changes + * and needs to be updated in all views. + * + * If one of your child documents emits the childChanged signal, then you may + * usually just want to redraw this child. In this case you can ignore the parameter + * passed by the signal. + */ + void childChanged( KoDocumentChild *child ); + + /** + * Progress info while loading or saving. The value is in percents (i.e. a number between 0 and 100) + * Your KoDocument-derived class should emit the signal now and then during load/save. + * KoMainWindow will take care of displaying a progress bar automatically. + */ + void sigProgress( int value ); + + /** + * Emitted e.g. at the beginning of a save operation + * This is emitted by KoDocument and used by KoView to display a statusbar message + */ + void sigStatusBarMessage( const QString& text ); + + /** + * Emitted e.g. at the end of a save operation + * This is emitted by KoDocument and used by KoView to clear the statusbar message + */ + void sigClearStatusBarMessage(); + + void sigBeginOperation(); + void sigEndOperation(); + + /** + * Emitted when the document is modified + */ + void modified( bool ); + + void closeEmbedInitDialog(); + +protected slots: + /** + * This slot loads an existing file and deletes the start up widget. + * @param file the file to load (including path) + * @since 1.5 + */ + virtual void openExistingFile( const QString& file ); + /** + * This slot loads a template and deletes the start up widget. + * @param file the template to load + * @since 1.5 + */ + virtual void openTemplate( const QString& file ); + + void deleteOpenPaneDelayed(); + +protected: + + QString autoSaveFile( const QString & path ) const; + + virtual KoView *createViewInstance( QWidget *parent, const char *name ) = 0; + + /** + * Loads a document from KReadOnlyPart::m_file (KParts takes care of downloading + * remote documents). + * Applies a filter if necessary, and calls loadNativeFormat in any case + * You should not have to reimplement, except for very special cases. + * + * This method is called from the KReadOnlyPart::openURL method. + */ + virtual bool openFile(); + + /** + * Saves a document to KReadOnlyPart::m_file (KParts takes care of uploading + * remote documents) + * Applies a filter if necessary, and calls saveNativeFormat in any case + * You should not have to reimplement, except for very special cases. + */ + virtual bool saveFile(); + + /** + * Override this method in your derived class to show a widget in the startup 'dialog'. + * This widget should allow the user to set settings for a custom document (i.e. one + * not based on a template). + * The returned widget should provide its own button (preferrably 'Create') and + * implement the logic to implement the document instance correctly. + * After initializing the widget should emit a signal called 'documentSelected()' which + * will remove the startupWidget and show the document. + * @param parent the parent of the to be created widget. + */ + virtual QWidget* createCustomDocumentWidget(QWidget *parent); + + /** + * OLD XML method. For OASIS just call KoDocumentChild::loadOasisDocument + * after KoDocumentChild::loadOasis. + * + * You need to overload this function if your document may contain + * embedded documents. This function is called to load embedded documents. + * + * An example implementation may look like this: + * @code + * QPtrListIterator<KoDocumentChild> it( children() ); + * for( ; it.current(); ++it ) + * { + * if ( !it.current()->loadDocument( _store ) ) + * { + * return false; + * } + * } + * return true; + * @endcode + */ + virtual bool loadChildren( KoStore* ); + + /** + * Saves all internal children (only!). + * @see saveExternalChildren if you have external children. + * Returns true on success. + */ + virtual bool saveChildren( KoStore* store ); + + /** + * Saves all internal children (only!), to the store, using the OASIS format. + * This is called automatically during saveNativeFormat. + * @see saveExternalChildren if you have external children. + * Returns true on success. + */ + virtual bool saveChildrenOasis( KoStore* store, KoXmlWriter* manifestWriter ); + + /** + * Overload this function if you have to load additional files + * from a store. This function is called after loadXML() + * and after loadChildren() have been called. + */ + virtual bool completeLoading( KoStore* store ); + + /** + * If you want to write additional files to a store, + * then you must do it here. + * In the implementation, you should prepend the document + * url (using url().url()) before the filename, so that everything is kept relative + * to this document. For instance it will produce urls such as + * tar:/1/pictures/picture0.png, if the doc url is tar:/1 + * But do this ONLY if the document is not stored extern (see isStoredExtern() ). + * If it is, then the pictures should be saved to tar:/pictures. + */ + virtual bool completeSaving( KoStore* store ); + + /** + * Inserts the new child in the list of children and emits the + * childChanged() signal. + * + * At the same time this method marks this document as modified. + * + * To remove a child, just delete it. KoDocument will detect this + * and remove the child from its lists. + * + * @see #isModified + */ + virtual void insertChild( KoDocumentChild *child ); + + /** @internal */ + virtual void setModified() { KParts::ReadWritePart::setModified(); } + + /** @internal */ + virtual void insertChild(QObject *o) { QObject::insertChild(o); } + + KoPageLayout m_pageLayout; + + /** + * Saves all externally stored children. + * Returns true on success. + * @see #saveChildren for internal children + */ + virtual bool saveExternalChildren(); + + /** + * Returns whether or not the current openURL() or openFile() call is + * actually an import operation (like File --> Import). + * This is for informational purposes only. + */ + bool isImporting() const; + + /** + * Returns whether or not the current saveFile() call is actually an export + * operation (like File --> Export). + * If this function returns true during saveFile() and you are changing + * some sort of state, you _must_ restore it before the end of saveFile(); + * otherwise, File --> Export will not work properly. + */ + bool isExporting() const; + + /** + * Creates the open widget showed at application start up. + * @param parent the parent widget + * @param instance the KInstance to be used for KConfig data + * @param templateType the template-type (group) that should be selected on creation. + * @since 1.5 + */ + KoOpenPane* createOpenPane( QWidget* parent, KInstance* instance, + const QString& templateType = QString::null); + +private slots: + void slotChildChanged( KoChild *c ); + void slotChildDestroyed(); + void slotAutoSave(); + void slotStarted( KIO::Job* ); + void startCustomDocument(); + /** + * Removes the open widget showed at application start up. + * @since 1.5 + */ + void deleteOpenPane(); + +private: + KService::Ptr nativeService(); + bool oldLoadAndParse( KoStore* store, const QString& filename, QDomDocument& doc ); + bool loadNativeFormatFromStore( const QString& file ); + bool savePreview( KoStore* store ); + bool saveOasisPreview( KoStore* store, KoXmlWriter* manifestWriter ); + class Private; + Private *d; + KService::Ptr m_nativeService; + bool m_bEmpty; + static QPtrList<KoDocument> *s_documentList; + static const int s_defaultAutoSave; +}; + +#endif diff --git a/lib/kofficecore/KoDocumentChild.cpp b/lib/kofficecore/KoDocumentChild.cpp new file mode 100644 index 00000000..8157b95a --- /dev/null +++ b/lib/kofficecore/KoDocumentChild.cpp @@ -0,0 +1,523 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright (C) 2004-2006 David Faure <faure@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 "KoDocumentChild.h" +#include "KoOasisStore.h" +#include <KoDocument.h> +#include <KoQueryTrader.h> +#include <KoXmlWriter.h> +#include <KoXmlNS.h> +#include <KoUnit.h> +#include <KoStore.h> +#include <KoStoreDevice.h> + +#include <kparts/partmanager.h> + +#include <kmimetype.h> +#include <klocale.h> +#include <kmessagebox.h> +#include <kdebug.h> + +#include <qapplication.h> + +#include <assert.h> + +// Define the protocol used here for embedded documents' URL +// This used to "store" but KURL didn't like it, +// so let's simply make it "tar" ! +#define STORE_PROTOCOL "tar" +#define INTERNAL_PROTOCOL "intern" +// Warning, keep it sync in koStore.cc and koDocument.cc + +/********************************************************** + * + * KoDocumentChild + * + **********************************************************/ + +class KoDocumentChildPrivate +{ +public: + KoDocumentChildPrivate() + { + } + ~KoDocumentChildPrivate() + { + } + + KoDocument *m_parent; + KoDocument *m_doc; + bool m_deleted; +}; + +KoDocumentChild::KoDocumentChild( KoDocument* parent, KoDocument* doc, const QRect& geometry ) + : KoChild( parent ) +{ + d = new KoDocumentChildPrivate; + d->m_parent = parent; + d->m_doc = doc; + setGeometry( geometry ); + d->m_deleted = false; + if ( doc ) + doc->setStoreInternal( !doc->hasExternURL() ); +} + +KoDocumentChild::KoDocumentChild( KoDocument* parent ) + : KoChild( parent ) +{ + d = new KoDocumentChildPrivate; + d->m_parent = parent; + d->m_doc = 0L; + d->m_deleted = false; +} + +void KoDocumentChild::setDocument( KoDocument *doc, const QRect &geometry ) +{ + kdDebug()<<k_funcinfo<<"doc: "<<doc->url().url()<<endl; + d->m_doc = doc; + setGeometry( geometry ); + + updateMatrix(); +} + +KoDocument *KoDocumentChild::document() const +{ + return d->m_doc; +} + +KoDocument* KoDocumentChild::parentDocument() const +{ + return d->m_parent; +} + +KoDocument* KoDocumentChild::hitTest( const QPoint &p, const QWMatrix &_matrix ) +{ + if ( !region( _matrix ).contains( p ) || !d->m_doc ) + return 0L; + + QWMatrix m( _matrix ); + m = matrix() * m; + m.scale( xScaling(), yScaling() ); + + return d->m_doc->hitTest( p, m ); +} + +void KoDocumentChild::loadOasis( const QDomElement &frameElement, const QDomElement& objectElement ) +{ + double x, y, w, h; + x = KoUnit::parseValue( frameElement.attributeNS( KoXmlNS::svg, "x", QString::null ) ); + y = KoUnit::parseValue( frameElement.attributeNS( KoXmlNS::svg, "y", QString::null ) ); + w = KoUnit::parseValue( frameElement.attributeNS( KoXmlNS::svg, "width", QString::null ) ); + h = KoUnit::parseValue( frameElement.attributeNS( KoXmlNS::svg, "height", QString::null ) ); + m_tmpGeometry = QRect((int)x, (int)y, (int)w, (int)h); // #### double->int conversion + setGeometry(m_tmpGeometry); + + QString url = objectElement.attributeNS( KoXmlNS::xlink, "href", QString::null ); + if ( url[0] == '#' ) + url = url.mid( 1 ); + if ( url.startsWith( "./" ) ) + m_tmpURL = QString( INTERNAL_PROTOCOL ) + ":/" + url.mid( 2 ); + else + m_tmpURL = url; + kdDebug() << k_funcinfo << m_tmpURL << endl; +} + + +bool KoDocumentChild::load( const QDomElement& element, bool uppercase ) +{ + if ( element.hasAttribute( "url" ) ) + m_tmpURL = element.attribute("url"); + if ( element.hasAttribute("mime") ) + m_tmpMimeType = element.attribute("mime"); + + if ( m_tmpURL.isEmpty() ) + { + kdDebug(30003) << "Empty 'url' attribute in OBJECT" << endl; + return false; + } + if ( m_tmpMimeType.isEmpty() ) + { + kdDebug(30003) << "Empty 'mime' attribute in OBJECT" << endl; + return false; + } + + bool brect = FALSE; + QDomNode n = element.firstChild(); + for( ; !n.isNull(); n = n.nextSibling() ) + { + QDomElement e = n.toElement(); + if ( e.isNull() ) continue; + if ( e.tagName() == "rect" || ( uppercase && e.tagName() == "RECT" ) ) + { + brect = true; + int x, y, w, h; + x=y=w=h=0; + if ( e.hasAttribute( "x" ) ) + x = e.attribute( "x" ).toInt(&brect); + if ( e.hasAttribute( "y" ) ) + y = e.attribute( "y" ).toInt(&brect); + if ( e.hasAttribute( "w" ) ) + w = e.attribute( "w" ).toInt(&brect); + if ( e.hasAttribute( "h" ) ) + h = e.attribute( "h" ).toInt(&brect); + m_tmpGeometry = QRect(x, y, w, h); + setGeometry(m_tmpGeometry); + } + } + + if ( !brect ) + { + kdDebug(30003) << "Missing RECT in OBJECT" << endl; + return false; + } + + return true; +} + +bool KoDocumentChild::loadDocument( KoStore* store ) +{ + assert( !m_tmpURL.isEmpty() ); + + kdDebug(30003) << "KoDocumentChild::loadDocument: trying to load " << m_tmpURL << endl; + + // Backwards compatibility + if ( m_tmpMimeType == "application/x-killustrator" ) + m_tmpMimeType = "application/x-kontour"; + + KoDocumentEntry e = KoDocumentEntry::queryByMimeType( m_tmpMimeType ); + if ( e.isEmpty() ) + { + kdWarning(30003) << "Could not create child document with type " << m_tmpMimeType << endl; + bool res = createUnavailDocument( store, true, m_tmpMimeType ); + if ( res ) + { + // Try to turn the mimetype name into its comment + QString mimeName = m_tmpMimeType; + KMimeType::Ptr mime = KMimeType::mimeType( m_tmpMimeType ); + if ( mime->name() != KMimeType::defaultMimeType() ) + mimeName = mime->comment(); + d->m_doc->setProperty( "unavailReason", i18n( "No handler found for %1" ).arg( mimeName ) ); + } + return res; + } + + return loadDocumentInternal( store, e, true /*open url*/, false /*not oasis*/ ); +} + +bool KoDocumentChild::loadOasisDocument( KoStore* store, const QDomDocument& manifestDoc ) +{ + QString path = m_tmpURL; + if ( m_tmpURL.startsWith( INTERNAL_PROTOCOL ) ) { + path = store->currentDirectory(); + if ( !path.isEmpty() ) + path += '/'; + QString relPath = KURL( m_tmpURL ).path(); + path += relPath.mid( 1 ); // remove leading '/' + } + if ( !path.endsWith( "/" ) ) + path += '/'; + const QString mimeType = KoOasisStore::mimeForPath( manifestDoc, path ); + kdDebug() << k_funcinfo << "path for manifest file=" << path << " mimeType=" << mimeType << endl; + if ( mimeType.isEmpty() ) { + kdError(30003) << "Manifest doesn't have media-type for " << path << endl; + return false; + } + + KoDocumentEntry e = KoDocumentEntry::queryByMimeType( mimeType ); + if ( e.isEmpty() ) + { + kdWarning(30003) << "Could not create child document with type " << mimeType << endl; + bool res = createUnavailDocument( store, true, mimeType ); + if ( res ) + { + // Try to turn the mimetype name into its comment + QString mimeName = mimeType; + KMimeType::Ptr mime = KMimeType::mimeType( mimeType ); + if ( mime->name() != KMimeType::defaultMimeType() ) + mimeName = mime->comment(); + d->m_doc->setProperty( "unavailReason", i18n( "No handler found for %1" ).arg( mimeName ) ); + } + return res; + } + + const bool oasis = mimeType.startsWith( "application/vnd.oasis.opendocument" ); + if ( !oasis ) { + m_tmpURL += "/maindoc.xml"; + kdDebug() << " m_tmpURL adjusted to " << m_tmpURL << endl; + } + return loadDocumentInternal( store, e, true /*open url*/, oasis ); +} + +bool KoDocumentChild::loadDocumentInternal( KoStore* store, const KoDocumentEntry& e, bool doOpenURL, bool oasis ) +{ + kdDebug(30003) << "KoDocumentChild::loadDocumentInternal doOpenURL=" << doOpenURL << " m_tmpURL=" << m_tmpURL << endl; + KoDocument * doc = e.createDoc( d->m_parent ); + if (!doc) { + kdWarning(30003) << "createDoc failed" << endl; + return false; + } + setDocument( doc, m_tmpGeometry ); + + bool res = true; + if ( doOpenURL ) + { + bool internalURL = false; + if ( m_tmpURL.startsWith( STORE_PROTOCOL ) || m_tmpURL.startsWith( INTERNAL_PROTOCOL ) || KURL::isRelativeURL( m_tmpURL ) ) + { + if ( oasis ) { + store->pushDirectory(); + assert( m_tmpURL.startsWith( INTERNAL_PROTOCOL ) ); + QString relPath = KURL( m_tmpURL ).path().mid( 1 ); + store->enterDirectory( relPath ); + res = d->m_doc->loadOasisFromStore( store ); + store->popDirectory(); + } else { + if ( m_tmpURL.startsWith( INTERNAL_PROTOCOL ) ) + m_tmpURL = KURL( m_tmpURL ).path().mid( 1 ); + res = d->m_doc->loadFromStore( store, m_tmpURL ); + } + internalURL = true; + d->m_doc->setStoreInternal( true ); + } + else + { + // Reference to an external document. Hmmm... + d->m_doc->setStoreInternal( false ); + KURL url( m_tmpURL ); + if ( !url.isLocalFile() ) + { + QApplication::restoreOverrideCursor(); + // For security reasons we need to ask confirmation if the url is remote + int result = KMessageBox::warningYesNoCancel( + 0, i18n( "This document contains an external link to a remote document\n%1").arg(m_tmpURL), + i18n( "Confirmation Required" ), i18n( "Download" ), i18n( "Skip" ) ); + + if ( result == KMessageBox::Cancel ) + { + d->m_parent->setErrorMessage("USER_CANCELED"); + return false; + } + if ( result == KMessageBox::Yes ) + res = d->m_doc->openURL( url ); + // and if == No, res will still be false so we'll use a kounavail below + } + else + res = d->m_doc->openURL( url ); + } + if ( !res ) + { + // Keep the error message from the attempted loading + QString errorMessage = d->m_doc->errorMessage(); + delete d->m_doc; + d->m_doc = 0; + QString tmpURL = m_tmpURL; // keep a copy, createUnavailDocument will erase it + // Not found -> use a kounavail instead + res = createUnavailDocument( store, false /* the URL doesn't exist, don't try to open it */, m_tmpMimeType ); + if ( res ) + { + d->m_doc->setProperty( "realURL", tmpURL ); // so that it gets saved correctly + d->m_doc->setStoreInternal( true ); + if ( internalURL ) + d->m_doc->setProperty( "unavailReason", i18n( "Could not load embedded object:\n%1" ).arg( errorMessage ) ); + else + d->m_doc->setProperty( "unavailReason", i18n( "Could not load external document %1:\n%2" ).arg( tmpURL, errorMessage ) ); + } + return res; + } + // Still waiting... + QApplication::setOverrideCursor( waitCursor ); + } + + m_tmpURL = QString::null; + + // see KoDocument::insertChild for an explanation what's going on + // now :-) + if ( parentDocument() ) + { + KoDocument *parent = parentDocument(); + + if ( parent->manager() && parent->manager()->parts() ) + { + KParts::PartManager *manager = parent->manager(); + + if ( !manager->parts()->containsRef( d->m_doc ) && + !parent->isSingleViewMode() ) + manager->addPart( d->m_doc, false ); + } + } + + QApplication::restoreOverrideCursor(); + + return res; +} + +bool KoDocumentChild::createUnavailDocument( KoStore* store, bool doOpenURL, const QString& mimeType ) +{ + // We don't need a trader query here. We're looking for a very specific component. + KService::Ptr serv = KService::serviceByDesktopName( "kounavail" ); + if ( serv == 0L ) + { + kdWarning(30003) << "ERROR: service kounavail not found " << endl; + return false; + } + KoDocumentEntry e( serv ); + if ( !loadDocumentInternal( store, e, doOpenURL, false ) ) + return false; + d->m_doc->setProperty( "mimetype", mimeType ); + return true; +} + +bool KoDocumentChild::saveOasis( KoStore* store, KoXmlWriter* manifestWriter ) +{ + QString path; + if ( d->m_doc->isStoredExtern() ) + { + kdDebug(30003)<<k_funcinfo<<" external (don't save) url:" << d->m_doc->url().url()<<endl; + path = d->m_doc->url().url(); + } + else + { + // The name comes from KoDocumentChild (which was set while saving the + // parent document). + assert( d->m_doc->url().protocol() == INTERNAL_PROTOCOL ); + const QString name = d->m_doc->url().path(); + kdDebug() << k_funcinfo << "saving " << name << endl; + + if ( d->m_doc->nativeOasisMimeType().isEmpty() ) { + // Embedded object doesn't support OASIS OpenDocument, save in the old format. + kdDebug() << k_funcinfo << "Embedded object doesn't support OASIS OpenDocument, save in the old format." << endl; + + if ( !d->m_doc->saveToStore( store, name ) ) + return false; + } + else + { + // To make the children happy cd to the correct directory + store->pushDirectory(); + store->enterDirectory( name ); + + if ( !d->m_doc->saveOasis( store, manifestWriter ) ) { + kdWarning(30003) << "KoDocumentChild::saveOasis failed" << endl; + return false; + } + // Now that we're done leave the directory again + store->popDirectory(); + } + + assert( d->m_doc->url().protocol() == INTERNAL_PROTOCOL ); + path = store->currentDirectory(); + if ( !path.isEmpty() ) + path += '/'; + path += d->m_doc->url().path(); + if ( path.startsWith( "/" ) ) + path = path.mid( 1 ); // remove leading '/', no wanted in manifest + } + + // OOo uses a trailing slash for the path to embedded objects (== directories) + if ( !path.endsWith( "/" ) ) + path += '/'; + QCString mimetype = d->m_doc->nativeOasisMimeType(); + if ( mimetype.isEmpty() ) + mimetype = d->m_doc->nativeFormatMimeType(); + manifestWriter->addManifestEntry( path, mimetype ); + + return true; +} + +void KoDocumentChild::saveOasisAttributes( KoXmlWriter &xmlWriter, const QString& name ) +{ + if ( !d->m_doc->isStoredExtern() ) + { + // set URL in document so that KoDocument::saveChildrenOasis will save + // the actual embedded object with the right name in the store. + KURL u; + u.setProtocol( INTERNAL_PROTOCOL ); + u.setPath( name ); + kdDebug() << k_funcinfo << u << endl; + d->m_doc->setURL( u ); + } + + //<draw:object draw:style-name="standard" draw:id="1" draw:layer="layout" svg:width="14.973cm" svg:height="4.478cm" svg:x="11.641cm" svg:y="14.613cm" xlink:href="#./Object 1" xlink:type="simple" xlink:show="embed" xlink:actuate="onLoad"/> + xmlWriter.addAttribute( "xlink:type", "simple" ); + xmlWriter.addAttribute( "xlink:show", "embed" ); + xmlWriter.addAttribute( "xlink:actuate", "onLoad" ); + + const QString ref = d->m_doc->isStoredExtern() ? d->m_doc->url().url() : "./"+ name; + kdDebug(30003) << "KoDocumentChild::saveOasis saving reference to embedded document as " << ref << endl; + xmlWriter.addAttribute( "xlink:href", /*"#" + */ref ); +} + +QDomElement KoDocumentChild::save( QDomDocument& doc, bool uppercase ) +{ + if( d->m_doc ) + { + QDomElement e = doc.createElement( ( uppercase ? "OBJECT" : "object" ) ); + if ( d->m_doc->url().protocol() != INTERNAL_PROTOCOL ) { + e.setAttribute( "url", d->m_doc->url().url() ); + kdDebug() << "KoDocumentChild::save url=" << d->m_doc->url().url() << endl; + } + else { + e.setAttribute( "url", d->m_doc->url().path().mid( 1 ) ); + kdDebug() << "KoDocumentChild::save url=" << d->m_doc->url().path().mid( 1 ) << endl; + } + e.setAttribute( "mime", d->m_doc->nativeFormatMimeType() ); + kdDebug() << "KoDocumentChild::save mime=" << d->m_doc->nativeFormatMimeType() << endl; + QDomElement rect = doc.createElement( ( uppercase ? "RECT" : "rect" ) ); + rect.setAttribute( "x", geometry().left() ); + rect.setAttribute( "y", geometry().top() ); + rect.setAttribute( "w", geometry().width() ); + rect.setAttribute( "h", geometry().height() ); + e.appendChild(rect); + return e; + } + return QDomElement(); +} + +bool KoDocumentChild::isStoredExtern() const +{ + assert( d->m_doc ); + return d->m_doc->isStoredExtern(); +} + +KURL KoDocumentChild::url() const +{ + return ( d->m_doc ? d->m_doc->url() : KURL() ); +} + +KoDocumentChild::~KoDocumentChild() +{ + if ( d->m_doc ) { + delete d->m_doc; + d->m_doc=0L; + } + delete d; +} + +bool KoDocumentChild::isDeleted() const +{ + return d->m_deleted; +} + +void KoDocumentChild::setDeleted( bool on ) +{ + d->m_deleted = on; +} + +#include "KoDocumentChild.moc" diff --git a/lib/kofficecore/KoDocumentChild.h b/lib/kofficecore/KoDocumentChild.h new file mode 100644 index 00000000..349961ed --- /dev/null +++ b/lib/kofficecore/KoDocumentChild.h @@ -0,0 +1,180 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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. +*/ +#ifndef __koDocumentChild_h__ +#define __koDocumentChild_h__ + +#include <KoChild.h> +#include <koffice_export.h> +class QDomDocument; +class QDomElement; +class KURL; +class KoStore; +class KoDocument; +class KoDocumentChildPrivate; +class KoDocumentEntry; +class KoXmlWriter; + +/** + * @brief Holds an embedded object. + * + * In addition to its base class @ref KoChild it cares about the content + * of an embedded document. That means it supports operations like + * loading and saving. + * + * If you need a representation for embedded documents in your KOffice + * component then you should inherit from this class. + */ +class KOFFICECORE_EXPORT KoDocumentChild : public KoChild +{ + Q_OBJECT +public: + KoDocumentChild( KoDocument* parent, KoDocument* doc, const QRect& geometry ); + + /** + * When using this constructor you must call @ref #setDocument before + * you can call any other function of this class. + */ + KoDocumentChild( KoDocument* parent ); + + virtual ~KoDocumentChild(); + + /** + * Call this function only directly after calling the constructor + * that takes only a parent as argument. + */ + virtual void setDocument( KoDocument *doc, const QRect &geometry ); + + /** + * @return document contained in this child + * + * @see KoDocument + */ + KoDocument *document() const; + + /** + * @return parent document of this child + * + * @see KoDocument + */ + virtual KoDocument *parentDocument() const; // TODO: remove virtual, makes no sense + + virtual KoDocument* hitTest( const QPoint& p, const QWMatrix& _matrix = QWMatrix() ); + + /** + * @note Can be empty (which is why it doesn't return a const KURL &) + */ + KURL url() const; + + /** + * Writes the "object" tag, but does NOT write the content of the + * embedded documents. Saving the embedded documents themselves + * is done in @ref KoDocument::saveChildren. This function just stores information + * about the position and id of the embedded document and should be + * called from within KoDocument::saveXML. + * + * The "object" element is not added to the document. It is just created + * and returned. + * + * @return the element containing the "object" tag. + */ + virtual QDomElement save( QDomDocument& doc, bool uppercase=false ); + + /** + * Save an embedded object to OASIS. + * This method sets the attributes for the draw:object element in the parent XML document. + * It also prepares the embedded object for being saved into the store at + * the end of saving (see saveOasisToStore). + * Note that @p name is only used for "internal" documents (not extern). + */ + void saveOasisAttributes( KoXmlWriter &xmlWriter, const QString& name ); + + /** + * Save an embedded object to an OASIS store. + * This is called automatically by the parent KoDocument's saveOasis + */ + virtual bool saveOasis( KoStore* store, KoXmlWriter* manifestWriter ); + + /** + * Parses the "object" tag. This does NOT mean creating the child documents. + * AFTER the 'parser' has finished parsing, you must use @ref #loadDocument + * to actually load the embedded documents. + * + * What you should have in mind is that this method is called from within + * @ref KoDocument::loadXML while @ref #loadDocument is called from within + * @ref KoDocument::loadChildren, respectively from your implementation + * of these methods. + */ + virtual bool load( const QDomElement& element, bool uppercase=false ); + + void loadOasis( const QDomElement &frameElement, const QDomElement& objectElement ); + + /** + * Actually loads the document from the disk/net or from the store, + * depending on @ref #url + */ + virtual bool loadDocument( KoStore* ); + + /** + * Actually loads the document from the disk/net or from the store + * depending on the document's url + */ + virtual bool loadOasisDocument( KoStore* store, const QDomDocument& manifestDoc ); + + virtual bool isStoredExtern() const; + + /** + * This document (child) is deleted. + */ + bool isDeleted() const; + void setDeleted( bool on = true ); + +protected: // Should be private, but KWord needs access to the variables + // because it reimplements load/save (for uppercase tags) + + /** + * Holds the source of this object, for example "file:/home/weis/image.gif" + * or "tar:/table1/2" if it is stored in a koffice store. This variable is + * set after parsing the OBJECT tag in @ref #load and is reset after + * calling @ref #loadDocument. + */ + QString m_tmpURL; + + /** + * This variable is + * set after parsing the OBJECT tag in @ref #load and is reset after + * calling @ref #loadDocument. + */ + QRect m_tmpGeometry; + + /** + * This variable is + * set after parsing the OBJECT tag in @ref #load and is reset after + * calling @ref #loadDocument. + */ + QString m_tmpMimeType; + +private: + bool createUnavailDocument( KoStore* store, bool doOpenURL, const QString& mimeType ); + bool loadDocumentInternal( KoStore* _store, const KoDocumentEntry& e, bool doOpenURL, bool oasis ); + +private: + KoDocumentChildPrivate *d; +}; + +#endif diff --git a/lib/kofficecore/KoDocumentIface.cc b/lib/kofficecore/KoDocumentIface.cc new file mode 100644 index 00000000..2ff2f5fb --- /dev/null +++ b/lib/kofficecore/KoDocumentIface.cc @@ -0,0 +1,568 @@ +/* This file is part of the KDE project + Copyright (C) 2000 David Faure <faure@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. + + 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 "KoDocument.h" +#include "KoDocumentIface.h" +#include "KoDocumentInfoDlg.h" +#include "KoDocumentInfo.h" +#include "KoView.h" +#include <kapplication.h> +#include <dcopclient.h> +#include <kdcopactionproxy.h> +#include <kaction.h> +#include <kdebug.h> +#include <kdcoppropertyproxy.h> + +//static +QCString KoDocumentIface::newIfaceName() +{ + static int s_docIFNumber = 0; + QCString name; name.setNum( s_docIFNumber++ ); name.prepend("Document-"); + return name; +} + +KoDocumentIface::KoDocumentIface( KoDocument * doc, const char * name ) + : DCOPObject( name ? QCString(name) : newIfaceName() ) +{ + m_pDoc = doc; + m_actionProxy = new KDCOPActionProxy( doc->actionCollection(), this ); +} + +KoDocumentIface::~KoDocumentIface() +{ + delete m_actionProxy; +} + +void KoDocumentIface::openURL( QString url ) +{ + m_pDoc->openURL( KURL( url ) ); +} + +bool KoDocumentIface::isLoading() +{ + return m_pDoc->isLoading(); +} + +QString KoDocumentIface::url() +{ + return m_pDoc->url().url(); +} + +bool KoDocumentIface::isModified() +{ + return m_pDoc->isModified(); +} + +int KoDocumentIface::viewCount() +{ + return m_pDoc->viewCount(); +} + +DCOPRef KoDocumentIface::view( int idx ) +{ + QPtrList<KoView> views = m_pDoc->views(); + KoView *v = views.at( idx ); + if ( !v ) + return DCOPRef(); + + DCOPObject *obj = v->dcopObject(); + + if ( !obj ) + return DCOPRef(); + + return DCOPRef( kapp->dcopClient()->appId(), obj->objId() ); +} + +DCOPRef KoDocumentIface::action( const QCString &name ) +{ + return DCOPRef( kapp->dcopClient()->appId(), m_actionProxy->actionObjectId( name ) ); +} + +QCStringList KoDocumentIface::actions() +{ + QCStringList res; + QValueList<KAction *> lst = m_actionProxy->actions(); + QValueList<KAction *>::ConstIterator it = lst.begin(); + QValueList<KAction *>::ConstIterator end = lst.end(); + for (; it != end; ++it ) + res.append( (*it)->name() ); + + return res; +} + +QMap<QCString,DCOPRef> KoDocumentIface::actionMap() +{ + return m_actionProxy->actionMap(); +} + +void KoDocumentIface::save() +{ + m_pDoc->save(); +} + +void KoDocumentIface::saveAs( const QString & url ) +{ + m_pDoc->saveAs( KURL( url ) ); + m_pDoc->waitSaveComplete(); // see ReadWritePart +} + +void KoDocumentIface::setOutputMimeType( const QCString & mimetype ) +{ + m_pDoc->setOutputMimeType( mimetype ); +} + +QString KoDocumentIface::documentInfoAuthorName() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->fullName(); +} + +QString KoDocumentIface::documentInfoEmail() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->email(); +} + +QString KoDocumentIface::documentInfoCompanyName() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->company(); +} + +QString KoDocumentIface::documentInfoTelephone() const +{ + kdDebug()<<" Keep compatibility with koffice <= 1.3 : use documentInfoTelephoneWork\n"; + return documentInfoTelephoneWork(); +} + +QString KoDocumentIface::documentInfoTelephoneWork() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->telephoneWork(); +} + +QString KoDocumentIface::documentInfoTelephoneHome() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->telephoneHome(); +} + + +QString KoDocumentIface::documentInfoFax() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->fax(); + +} +QString KoDocumentIface::documentInfoCountry() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->country(); + +} +QString KoDocumentIface::documentInfoPostalCode() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->postalCode(); + +} +QString KoDocumentIface::documentInfoCity() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->city(); + +} + +QString KoDocumentIface::documentInfoInitial() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->initial(); +} + +QString KoDocumentIface::documentInfoAuthorPostion() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->position(); +} + + +QString KoDocumentIface::documentInfoStreet() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->street(); + +} + +QString KoDocumentIface::documentInfoTitle() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + return QString::null; + } + else + return aboutPage->title(); + +} + +QString KoDocumentIface::documentInfoAbstract() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + return QString::null; + } + else + return aboutPage->abstract(); +} + +QString KoDocumentIface::documentInfoKeywords() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + return QString::null; + } + else + return aboutPage->keywords(); +} + +QString KoDocumentIface::documentInfoSubject() const +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + return QString::null; + } + else + return aboutPage->subject(); +} +void KoDocumentIface::setDocumentInfoKeywords(const QString & text ) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + } + else + aboutPage->setKeywords(text); +} + +void KoDocumentIface::setDocumentInfoSubject(const QString & text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + } + else + aboutPage->setSubject(text); +} + +void KoDocumentIface::setDocumentInfoAuthorName(const QString & text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setFullName(text); + +} + +void KoDocumentIface::setDocumentInfoEmail(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setEmail(text); +} + +void KoDocumentIface::setDocumentInfoCompanyName(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setCompany(text); +} + +void KoDocumentIface::setDocumentInfoAuthorPosition(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setPosition(text); +} + + +void KoDocumentIface::setDocumentInfoTelephone(const QString &text) +{ + kdDebug()<<"Keep compatibility with koffice <= 1.3 : use setDocumentInfoTelephoneWork\n"; + setDocumentInfoTelephoneWork(text); +} + +void KoDocumentIface::setDocumentInfoTelephoneWork(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setTelephoneWork(text); +} + +void KoDocumentIface::setDocumentInfoTelephoneHome(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setTelephoneHome(text); +} + + +void KoDocumentIface::setDocumentInfoFax(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setFax(text); +} + +void KoDocumentIface::setDocumentInfoCountry(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setCountry(text); + +} + +void KoDocumentIface::setDocumentInfoTitle(const QString & text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + } + else + aboutPage->setTitle(text); +} + +void KoDocumentIface::setDocumentInfoPostalCode(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setPostalCode(text); + +} + + +void KoDocumentIface::setDocumentInfoCity(const QString & text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setCity(text); +} + +void KoDocumentIface::setDocumentInfoInitial(const QString & text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setInitial(text); +} + + +void KoDocumentIface::setDocumentInfoStreet(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(info->page( "author" )); + if ( !authorPage ) + { + kdWarning() << "Author information not found in documentInfo !" << endl; + } + else + authorPage->setStreet(text); + +} + + +void KoDocumentIface::setDocumentInfoAbstract(const QString &text) +{ + KoDocumentInfo * info = m_pDoc->documentInfo(); + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(info->page( "about" )); + if ( !aboutPage ) + { + kdWarning() << "'About' page not found in documentInfo !" << endl; + } + else + aboutPage->setAbstract(text); +} + +QCStringList KoDocumentIface::functionsDynamic() +{ + return DCOPObject::functionsDynamic() + KDCOPPropertyProxy::functions( m_pDoc ); +} + +bool KoDocumentIface::processDynamic( const QCString &fun, const QByteArray &data, + QCString& replyType, QByteArray &replyData ) +{ + if ( KDCOPPropertyProxy::isPropertyRequest( fun, m_pDoc ) ) + return KDCOPPropertyProxy::processPropertyRequest( fun, data, replyType, replyData, m_pDoc ); + + return DCOPObject::processDynamic( fun, data, replyType, replyData ); +} + diff --git a/lib/kofficecore/KoDocumentIface.h b/lib/kofficecore/KoDocumentIface.h new file mode 100644 index 00000000..2713cb43 --- /dev/null +++ b/lib/kofficecore/KoDocumentIface.h @@ -0,0 +1,157 @@ +/* This file is part of the KDE project + Copyright (C) 2000 David Faure <faure@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. + + 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. +*/ + +#ifndef __KoDocumentIface_h__ +#define __KoDocumentIface_h__ + +#include <dcopobject.h> +#include <qvaluelist.h> +#include <dcopref.h> +#include <koffice_export.h> +class KoDocument; +class KDCOPActionProxy; + +/** + * DCOP interface for any KOffice document + * Use KoApplicationIface to get hold of an existing document's interface, + * or to create a document. + * + * Note: KOffice Applications may (and should) reimplement KoDocument::dcopObject() + * In this case, don't look here... (unless the DCOP interface for the document + * inherits KoDocumentIface, which is a good thing to do) + */ +class KOFFICECORE_EXPORT KoDocumentIface : public DCOPObject +{ + K_DCOP +public: + + KoDocumentIface( KoDocument * doc, const char * name = 0 ); + ~KoDocumentIface(); + + /** + * Generate a name for this interface. Automatically used if name=0 is + * passed to the constructor + */ + static QCString newIfaceName(); + +k_dcop: + /** + * Returns the URL for this document (empty, real URL, or internal one) + */ + QString url(); + + /** + * Opens a document stored in @p url + * Warning: this is asynchronous. The document might not be loaded yet when + * this call returns. See isLoading. + */ + void openURL( QString url ); + + /** + * @return TRUE is the document is still loading + */ + bool isLoading(); + + /** + * @return TRUE is the document has been modified + */ + bool isModified(); + + /** + * @return the number of views this document is displayed in + */ + int viewCount(); + + /** + * @return a DCOP reference (DCOPRef) to the view with index @p idx + */ + DCOPRef view( int idx ); + + /** + * DCOP-action proxy + */ + DCOPRef action( const QCString &name ); + /** + * @return list of actions + */ + QCStringList actions(); + /** + * @return a map of (action name, DCOP reference) + */ + QMap<QCString,DCOPRef> actionMap(); + + /** + * Saves the document under its existing filename + */ + void save(); + + /** + * Saves the document under a new name + */ + void saveAs( const QString & url ); + + void setOutputMimeType( const QCString & mimetype ); + + QString documentInfoAuthorName() const; + QString documentInfoEmail() const; + QString documentInfoCompanyName() const; + QString documentInfoTitle() const; + QString documentInfoAbstract() const; + QString documentInfoKeywords() const; + QString documentInfoSubject() const; + QString documentInfoTelephone() const; + QString documentInfoTelephoneWork() const; + QString documentInfoTelephoneHome() const; + QString documentInfoFax() const; + QString documentInfoCountry() const; + QString documentInfoPostalCode() const; + QString documentInfoCity() const; + QString documentInfoStreet() const; + QString documentInfoInitial() const; + QString documentInfoAuthorPostion() const; + void setDocumentInfoAuthorName(const QString & text); + void setDocumentInfoEmail(const QString &text); + void setDocumentInfoCompanyName(const QString &text); + void setDocumentInfoTelephone(const QString &text); + void setDocumentInfoTelephoneWork(const QString &text); + void setDocumentInfoTelephoneHome(const QString &text); + void setDocumentInfoFax(const QString &text); + void setDocumentInfoCountry(const QString &text); + void setDocumentInfoTitle(const QString & text); + void setDocumentInfoPostalCode(const QString &text); + void setDocumentInfoCity(const QString & text); + void setDocumentInfoStreet(const QString &text); + void setDocumentInfoAbstract(const QString &text); + void setDocumentInfoInitial(const QString & text); + void setDocumentInfoKeywords(const QString & text ); + void setDocumentInfoSubject(const QString & text); + void setDocumentInfoAuthorPosition(const QString & text); + +public: + virtual QCStringList functionsDynamic(); + virtual bool processDynamic( const QCString &fun, const QByteArray &data, + QCString& replyType, QByteArray &replyData ); + +protected: + KoDocument * m_pDoc; + KDCOPActionProxy *m_actionProxy; +}; + +#endif + diff --git a/lib/kofficecore/KoDocumentInfo.cpp b/lib/kofficecore/KoDocumentInfo.cpp new file mode 100644 index 00000000..3d4e0505 --- /dev/null +++ b/lib/kofficecore/KoDocumentInfo.cpp @@ -0,0 +1,943 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@kde.org> + Copyright (C) 2004 David Faure <faure@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 "KoDocumentInfo.h" +#include "KoDom.h" +#include "KoDocument.h" +#include "kofficeversion.h" +#include "KoApplication.h" + +#include <KoStoreDevice.h> +#include <KoXmlWriter.h> + +#include <kconfig.h> +#include <kdebug.h> +#include <kglobal.h> +#include <klocale.h> + +#include <qobjectlist.h> +#include <qdatetime.h> +#include "KoXmlNS.h" + +/***************************************** + * + * KoDocumentInfo + * + *****************************************/ + +KoDocumentInfo::KoDocumentInfo( QObject* parent, const char* name ) + : QObject( parent, name ) +{ + (void)new KoDocumentInfoUserMetadata( this ); + (void)new KoDocumentInfoAuthor( this ); + (void)new KoDocumentInfoAbout( this ); +} + +KoDocumentInfo::~KoDocumentInfo() +{ +} + +// KOffice-1.3 format +bool KoDocumentInfo::load( const QDomDocument& doc ) +{ + QStringList lst = pages(); + QStringList::ConstIterator it = lst.begin(); + for( ; it != lst.end(); ++it ) + { + KoDocumentInfoPage* p = page( *it ); + Q_ASSERT( p ); + if ( !p->load( doc.documentElement() ) ) + return false; + } + + return true; +} + +bool KoDocumentInfo::loadOasis( const QDomDocument& metaDoc ) +{ + //kdDebug()<<" metaDoc.toString() :"<<metaDoc.toString()<<endl; + QStringList lst = pages(); + QStringList::ConstIterator it = lst.begin(); + for( ; it != lst.end(); ++it ) + { + KoDocumentInfoPage* p = page( *it ); + Q_ASSERT( p ); + + QDomNode meta = KoDom::namedItemNS( metaDoc, KoXmlNS::office, "document-meta" ); + QDomNode office = KoDom::namedItemNS( meta, KoXmlNS::office, "meta" ); + + if ( office.isNull() ) + return false; + + if ( !p->loadOasis( office ) ) + return false; + } + return true; +} + +// KOffice-1.3 format +QDomDocument KoDocumentInfo::save() +{ + QDomDocument doc = KoDocument::createDomDocument( "document-info" /*DTD name*/, "document-info" /*tag name*/, "1.1" ); + QDomElement e = doc.documentElement(); + + const QStringList lst = pages(); + QStringList::ConstIterator it = lst.begin(); + for( ; it != lst.end(); ++it ) + { + KoDocumentInfoPage* p = page( *it ); + Q_ASSERT( p ); + QDomElement s = p->save( doc ); + if ( s.isNull() ) + continue; + e.appendChild( s ); + } + if ( e.isNull() ) + return QDomDocument(); + + return doc; +} + +bool KoDocumentInfo::saveOasis( KoStore* store ) +{ + KoStoreDevice dev( store ); + KoXmlWriter* xmlWriter = KoDocument::createOasisXmlWriter( &dev, "office:document-meta" ); + xmlWriter->startElement( "office:meta" ); + + xmlWriter->startElement( "meta:generator"); + xmlWriter->addTextNode( QString( "KOffice/%1" ).arg( KOFFICE_VERSION_STRING ) ); + xmlWriter->endElement(); + QStringList lst = pages(); + QStringList::ConstIterator it = lst.begin(); + for( ; it != lst.end(); ++it ) + { + KoDocumentInfoPage* p = page( *it ); + Q_ASSERT( p ); + if ( !p->saveOasis( *xmlWriter ) ) + return false; + } + xmlWriter->endElement(); + xmlWriter->endElement(); // root element + xmlWriter->endDocument(); + delete xmlWriter; + return true; +} + +KoDocumentInfoPage* KoDocumentInfo::page( const QString& name ) const +{ + QObject* obj = const_cast<KoDocumentInfo*>(this)->child( name.latin1() ); + + return (KoDocumentInfoPage*)obj; +} + +QStringList KoDocumentInfo::pages() const +{ + QStringList ret; + + const QObjectList *list = children(); + if ( list ) + { + QObjectListIt it( *list ); + QObject *obj; + while ( ( obj = it.current() ) ) + { + ret.prepend( obj->name() ); + ++it; + } + } + + return ret; +} + +QString KoDocumentInfo::title() const +{ + KoDocumentInfoAbout * aboutPage = static_cast<KoDocumentInfoAbout *>(page( "about" )); + if ( !aboutPage ) { + kdWarning() << "'About' page not found in documentInfo !" << endl; + return QString::null; + } + else + return aboutPage->title(); +} + +QString KoDocumentInfo::creator() const +{ + KoDocumentInfoAuthor * authorPage = static_cast<KoDocumentInfoAuthor *>(page( "author" )); + if ( !authorPage ) { + kdWarning() << "'Author' page not found in documentInfo !" << endl; + return QString::null; + } + else + return authorPage->fullName(); +} + +/***************************************** + * + * KoDocumentInfoPage + * + *****************************************/ + +KoDocumentInfoPage::KoDocumentInfoPage( QObject* parent, const char* name ) + : QObject( parent, name ) +{ +} + +/***************************************** + * + * KoDocumentInfoAuthor + * + *****************************************/ + +KoDocumentInfoAuthor::KoDocumentInfoAuthor( KoDocumentInfo* info ) + : KoDocumentInfoPage( info, "author" ) +{ + initParameters(); +} + +KoDocumentInfoAuthor::~KoDocumentInfoAuthor() +{ + delete m_emailCfg; +} +void KoDocumentInfoAuthor::initParameters() +{ + KConfig* config = KoGlobal::kofficeConfig(); + if ( config->hasGroup( "Author" ) ) { + KConfigGroupSaver cgs( config, "Author" ); + m_telephoneHome=config->readEntry( "telephone" ); + m_telephoneWork=config->readEntry( "telephone-work" ); + m_fax=config->readEntry( "fax" ); + m_country=config->readEntry( "country" ); + m_postalCode=config->readEntry( "postal-code" ); + m_city=config->readEntry( "city" ); + m_street=config->readEntry( "street" ); + } + + m_emailCfg = new KConfig( "emaildefaults", true ); + m_emailCfg->setGroup( "Defaults" ); + QString group = m_emailCfg->readEntry("Profile","Default"); + m_emailCfg->setGroup(QString("PROFILE_%1").arg(group)); + + if ( m_fullName.isNull() ) // only if null. Empty means the user made it explicitly empty. + { + QString name = m_emailCfg->readEntry( "FullName" ); + if ( !name.isEmpty() ) + m_fullName = name; + } + if ( m_company.isNull() ) + { + QString name = m_emailCfg->readEntry( "Organization" ); + if ( !name.isEmpty() ) + m_company = name; + } +} + +bool KoDocumentInfoAuthor::saveOasis( KoXmlWriter &xmlWriter ) +{ + if ( !m_fullName.isEmpty() ) + { + xmlWriter.startElement( "dc:creator"); + xmlWriter.addTextNode( m_fullName ); + xmlWriter.endElement(); + } + if ( !m_initial.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "initial" ); + xmlWriter.addTextNode( m_initial ); + xmlWriter.endElement(); + } + if ( !m_title.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "author-title" ); + xmlWriter.addTextNode( m_title ); + xmlWriter.endElement(); + } + if ( !m_company.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "company" ); + xmlWriter.addTextNode( m_company ); + xmlWriter.endElement(); + } + if ( !m_email.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "email" ); + xmlWriter.addTextNode( m_email ); + xmlWriter.endElement(); + } + if ( !m_telephoneHome.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "telephone" ); + xmlWriter.addTextNode( m_telephoneHome ); + xmlWriter.endElement(); + } + if ( !m_telephoneWork.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "telephone-work" ); + xmlWriter.addTextNode( m_telephoneWork ); + xmlWriter.endElement(); + } + if ( !m_fax.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "fax" ); + xmlWriter.addTextNode( m_fax ); + xmlWriter.endElement(); + } + if ( !m_country.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "country" ); + xmlWriter.addTextNode( m_country ); + xmlWriter.endElement(); + } + if ( !m_postalCode.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "postal-code" ); + xmlWriter.addTextNode( m_postalCode ); + xmlWriter.endElement(); + } + if ( !m_city.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "city" ); + xmlWriter.addTextNode( m_city ); + xmlWriter.endElement(); + } + if ( !m_street.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "street" ); + xmlWriter.addTextNode( m_street ); + xmlWriter.endElement(); + } + if ( !m_position.isEmpty() ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", "position" ); + xmlWriter.addTextNode( m_position ); + xmlWriter.endElement(); + } + return true; +} + +bool KoDocumentInfoAuthor::loadOasis( const QDomNode& metaDoc ) +{ + QDomElement e = KoDom::namedItemNS( metaDoc, KoXmlNS::dc, "creator" ); + if ( !e.isNull() && !e.text().isEmpty() ) + m_fullName = e.text(); + QDomNode n = metaDoc.firstChild(); + for ( ; !n.isNull(); n = n.nextSibling() ) + { + if (n.isElement()) + { + QDomElement e = n.toElement(); + if ( e.namespaceURI() == KoXmlNS::meta && e.localName() == "user-defined" && !e.text().isEmpty() ) + { + QString name = e.attributeNS( KoXmlNS::meta, "name", QString::null ); + if ( name == "initial" ) + m_initial = e.text(); + else if ( name == "author-title" ) + m_title = e.text(); + else if ( name == "company" ) + m_company = e.text(); + else if ( name == "email" ) + m_email = e.text(); + else if ( name == "telephone" ) + m_telephoneHome = e.text(); + else if ( name == "telephone-work" ) + m_telephoneWork = e.text(); + else if ( name == "fax" ) + m_fax = e.text(); + else if ( name == "country" ) + m_country = e.text(); + else if ( name == "postal-code" ) + m_postalCode = e.text(); + else if ( name == "city" ) + m_city = e.text(); + else if ( name == "street" ) + m_street = e.text(); + else if ( name == "position" ) + m_position = e.text(); + } + } + } + return true; +} + +// KOffice-1.3 format +bool KoDocumentInfoAuthor::load( const QDomElement& e ) +{ + QDomNode n = e.namedItem( "author" ).firstChild(); + for( ; !n.isNull(); n = n.nextSibling() ) + { + QDomElement e = n.toElement(); + if ( e.isNull() ) continue; + if ( e.tagName() == "full-name" ) + m_fullName = e.text(); + else if ( e.tagName() == "initial" ) + m_initial = e.text(); + else if ( e.tagName() == "title" ) + m_title = e.text(); + else if ( e.tagName() == "company" ) + m_company = e.text(); + else if ( e.tagName() == "email" ) + m_email = e.text(); + else if ( e.tagName() == "telephone" ) + m_telephoneHome = e.text(); + else if ( e.tagName() == "telephone-work" ) + m_telephoneWork = e.text(); + else if ( e.tagName() == "fax" ) + m_fax = e.text(); + else if ( e.tagName() == "country" ) + m_country = e.text(); + else if ( e.tagName() == "postal-code" ) + m_postalCode = e.text(); + else if ( e.tagName() == "city" ) + m_city = e.text(); + else if ( e.tagName() == "street" ) + m_street = e.text(); + else if ( e.tagName() == "position" ) + m_position = e.text(); + } + return true; +} + +QDomElement KoDocumentInfoAuthor::save( QDomDocument& doc ) +{ + QDomElement e = doc.createElement( "author" ); + + QDomElement t = doc.createElement( "full-name" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_fullName ) ); + + t = doc.createElement( "initial" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_initial ) ); + + + t = doc.createElement( "title" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_title ) ); + + t = doc.createElement( "company" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_company ) ); + + t = doc.createElement( "email" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_email ) ); + + t = doc.createElement( "telephone" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_telephoneHome ) ); + + t = doc.createElement( "telephone-work" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_telephoneWork ) ); + + t = doc.createElement( "fax" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_fax ) ); + + t = doc.createElement( "country" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_country ) ); + + t = doc.createElement( "postal-code" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_postalCode ) ); + + t = doc.createElement( "city" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_city ) ); + + t = doc.createElement( "street" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_street ) ); + + t = doc.createElement( "position" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_position ) ); + + return e; +} + +QString KoDocumentInfoAuthor::fullName() const +{ + return m_fullName; +} + +QString KoDocumentInfoAuthor::initial() const +{ + return m_initial; +} + +QString KoDocumentInfoAuthor::title() const +{ + return m_title; +} + +QString KoDocumentInfoAuthor::company() const +{ + return m_company; +} + +QString KoDocumentInfoAuthor::email() const +{ + return m_email; +} + +QString KoDocumentInfoAuthor::telephoneHome() const +{ + return m_telephoneHome; +} + +QString KoDocumentInfoAuthor::telephoneWork() const +{ + return m_telephoneWork; +} + +QString KoDocumentInfoAuthor::fax() const +{ + return m_fax; +} + +QString KoDocumentInfoAuthor::country() const +{ + return m_country; +} + +QString KoDocumentInfoAuthor::postalCode() const +{ + return m_postalCode; +} + +QString KoDocumentInfoAuthor::city() const +{ + return m_city; +} + +QString KoDocumentInfoAuthor::street() const +{ + return m_street; +} + +QString KoDocumentInfoAuthor::position() const +{ + return m_position; +} + +void KoDocumentInfoAuthor::setFullName( const QString& n ) +{ + m_fullName = n; +} + +void KoDocumentInfoAuthor::setInitial( const QString& n ) +{ + m_initial = n; +} + +void KoDocumentInfoAuthor::setTitle( const QString& n ) +{ + m_title = n; +} + +void KoDocumentInfoAuthor::setCompany( const QString& n ) +{ + m_company = n; +} + +void KoDocumentInfoAuthor::setEmail( const QString& n ) +{ + m_email = n; +} + +void KoDocumentInfoAuthor::setTelephoneHome( const QString& n ) +{ + m_telephoneHome = n; +} + +void KoDocumentInfoAuthor::setTelephoneWork( const QString& n ) +{ + m_telephoneWork = n; +} + +void KoDocumentInfoAuthor::setFax( const QString& n ) +{ + m_fax = n; +} + +void KoDocumentInfoAuthor::setCountry( const QString& n ) +{ + m_country = n; +} + +void KoDocumentInfoAuthor::setPostalCode( const QString& n ) +{ + m_postalCode = n; +} + +void KoDocumentInfoAuthor::setCity( const QString& n ) +{ + m_city = n; +} + +void KoDocumentInfoAuthor::setStreet( const QString& n ) +{ + m_street = n; +} + +void KoDocumentInfoAuthor::setPosition( const QString& n ) +{ + m_position = n; +} + + +/***************************************** + * + * KoDocumentInfoAbout + * + *****************************************/ + +KoDocumentInfoAbout::KoDocumentInfoAbout( KoDocumentInfo* info ) + : KoDocumentInfoPage( info, "about" ) +{ + m_firstSave = true; + m_docInfo = info; + m_editingCycles = 0; + m_initialCreator = m_docInfo->creator(); + m_creationDate = QDateTime::currentDateTime(); +} + +void KoDocumentInfoAbout::saveParameters() +{ + KoDocument* doc = dynamic_cast< KoDocument* >( m_docInfo->parent() ); + if ( m_firstSave && doc && !doc->isAutosaving() ) + m_editingCycles++; + m_modificationDate = QDateTime::currentDateTime(); + m_firstSave = false; +} + +bool KoDocumentInfoAbout::saveOasis( KoXmlWriter &xmlWriter ) +{ + saveParameters(); + if ( !m_title.isEmpty() ) + { + xmlWriter.startElement( "dc:title" ); + xmlWriter.addTextNode( m_title ); + xmlWriter.endElement(); + } + if ( !m_abstract.isEmpty() ) + { + xmlWriter.startElement( "dc:description" ); + xmlWriter.addTextNode( m_abstract ); + xmlWriter.endElement(); + } + if ( !m_keywords.isEmpty() ) + { + xmlWriter.startElement( "meta:keyword" ); + xmlWriter.addTextNode( m_keywords ); + xmlWriter.endElement(); + } + if ( !m_subject.isEmpty() ) + { + xmlWriter.startElement( "dc:subject" ); + xmlWriter.addTextNode( m_subject ); + xmlWriter.endElement(); + } + if ( !m_initialCreator.isEmpty() ) + { + xmlWriter.startElement( "meta:initial-creator" ); + xmlWriter.addTextNode( m_initialCreator ); + xmlWriter.endElement(); + } + + xmlWriter.startElement( "meta:editing-cycles" ); + xmlWriter.addTextNode( QString::number( m_editingCycles ) ); + xmlWriter.endElement(); + + if ( m_creationDate.isValid() ) + { + xmlWriter.startElement( "meta:creation-date" ); + xmlWriter.addTextNode( m_creationDate.toString( Qt::ISODate ) ); + xmlWriter.endElement(); + } + + if ( m_modificationDate.isValid() ) + { + xmlWriter.startElement( "dc:date" ); + xmlWriter.addTextNode( m_modificationDate.toString( Qt::ISODate ) ); + xmlWriter.endElement(); + } + return true; +} + +bool KoDocumentInfoAbout::loadOasis( const QDomNode& metaDoc ) +{ + QDomElement e = KoDom::namedItemNS( metaDoc, KoXmlNS::dc, "title" ); + if ( !e.isNull() && !e.text().isEmpty() ) + { + m_title = e.text(); + } + e = KoDom::namedItemNS( metaDoc, KoXmlNS::dc, "description" ); + if ( !e.isNull() && !e.text().isEmpty() ) + { + m_abstract = e.text(); + } + e = KoDom::namedItemNS( metaDoc, KoXmlNS::dc, "subject" ); + if ( !e.isNull() && !e.text().isEmpty() ) + { + m_subject = e.text(); + } + e = KoDom::namedItemNS( metaDoc, KoXmlNS::meta, "keyword" ); + if ( !e.isNull() && !e.text().isEmpty() ) + { + m_keywords = e.text(); + } + e = KoDom::namedItemNS( metaDoc, KoXmlNS::meta, "initial-creator" ); + if ( !e.isNull() && !e.text().isEmpty() ) + m_initialCreator = e.text(); + else + m_initialCreator = i18n( "Unknown" ); + + e = KoDom::namedItemNS( metaDoc, KoXmlNS::meta, "editing-cycles" ); + if ( !e.isNull() && !e.text().isEmpty() ) + m_editingCycles = e.text().toInt(); + + e = KoDom::namedItemNS( metaDoc, KoXmlNS::meta, "creation-date" ); + if ( !e.isNull() && !e.text().isEmpty() ) + m_creationDate = QDateTime::fromString( e.text(), Qt::ISODate ); + else + m_creationDate = QDateTime(); + + e = KoDom::namedItemNS( metaDoc, KoXmlNS::dc, "date" ); + if ( !e.isNull() && !e.text().isEmpty() ) + m_modificationDate = QDateTime::fromString( e.text(), Qt::ISODate ); + return true; +} + +// KOffice-1.3 format +bool KoDocumentInfoAbout::load( const QDomElement& e ) +{ + QDomNode n = e.namedItem( "about" ).firstChild(); + for( ; !n.isNull(); n = n.nextSibling() ) + { + QDomElement e = n.toElement(); + if ( e.isNull() ) continue; + if ( e.tagName() == "abstract" ) + m_abstract = e.text(); + else if ( e.tagName() == "title" ) + m_title = e.text(); + else if ( e.tagName() == "subject" ) + m_subject = e.text(); + else if ( e.tagName() == "keyword" ) + m_keywords = e.text(); + else if ( e.tagName() == "initial-creator" ) + m_initialCreator = e.text(); + else if ( e.tagName() == "editing-cycles" ) + m_editingCycles = e.text().toInt(); + else if ( e.tagName() == "creation-date" ) + m_creationDate = QDateTime::fromString( e.text(), Qt::ISODate ); + else if ( e.tagName() == "date" ) + m_modificationDate = QDateTime::fromString( e.text(), Qt::ISODate ); + } + + return true; +} + +// KOffice-1.3 format +QDomElement KoDocumentInfoAbout::save( QDomDocument& doc ) +{ + saveParameters(); + QDomElement e = doc.createElement( "about" ); + + QDomElement t = doc.createElement( "abstract" ); + e.appendChild( t ); + t.appendChild( doc.createCDATASection( m_abstract ) ); + + t = doc.createElement( "title" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_title ) ); + + t = doc.createElement( "keyword" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_keywords ) ); + + t = doc.createElement( "subject" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_subject ) ); + + t = doc.createElement( "initial-creator" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_initialCreator ) ); + + t = doc.createElement( "editing-cycles" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( QString::number( m_editingCycles ) ) ); + + t = doc.createElement( "creation-date" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_creationDate.toString( Qt::ISODate ) ) ); + + t = doc.createElement( "date" ); + e.appendChild( t ); + t.appendChild( doc.createTextNode( m_modificationDate.toString( Qt::ISODate ) ) ); + return e; +} + +QString KoDocumentInfoAbout::title() const +{ + return m_title; +} + +QString KoDocumentInfoAbout::abstract() const +{ + return m_abstract; +} + +QString KoDocumentInfoAbout::initialCreator() const +{ + return m_initialCreator; +} + +QString KoDocumentInfoAbout::editingCycles() const +{ + return QString::number( m_editingCycles ); +} + +QString KoDocumentInfoAbout::creationDate() const +{ + if ( m_creationDate.isValid() ) + return KGlobal::locale()->formatDateTime( m_creationDate ); + else + return QString::null; +} + +QString KoDocumentInfoAbout::modificationDate() const +{ + if ( m_modificationDate.isValid() ) + return KGlobal::locale()->formatDateTime( m_modificationDate ); + else + return QString::null; +} + +void KoDocumentInfoAbout::setTitle( const QString& n ) +{ + m_title = n; +} + +void KoDocumentInfoAbout::setAbstract( const QString& n ) +{ + m_abstract = n; +} + +QString KoDocumentInfoAbout::keywords() const +{ + return m_keywords; +} + +QString KoDocumentInfoAbout::subject() const +{ + return m_subject; +} + +void KoDocumentInfoAbout::setKeywords( const QString& n ) +{ + m_keywords = n; +} + +void KoDocumentInfoAbout::setSubject( const QString& n ) +{ + m_subject = n; +} + +void KoDocumentInfoAbout::resetMetaData() +{ + m_editingCycles = 0; + m_initialCreator = m_docInfo->creator(); + m_creationDate = QDateTime::currentDateTime(); + m_modificationDate = QDateTime(); +} + +/***************************************** + * + * KoDocumentInfoUserMetadata + * + *****************************************/ + +KoDocumentInfoUserMetadata::KoDocumentInfoUserMetadata( KoDocumentInfo* info ) + : KoDocumentInfoPage( info, "user_metadata" ) +{ + m_reserved << "initial" << "author-title" << "company" << "email" << "telephone" + << "telephone-work" << "fax" << "country" << "postal-code" << "city" << "street" + << "position"; +} + +bool KoDocumentInfoUserMetadata::saveOasis( KoXmlWriter &xmlWriter ) +{ + QMap<QString, QString>::iterator it; + for ( it = m_metaList.begin(); it != m_metaList.end(); ++it ) + { + xmlWriter.startElement( "meta:user-defined"); + xmlWriter.addAttribute( "meta:name", it.key() ); + xmlWriter.addTextNode( it.data() ); + xmlWriter.endElement(); + } + return true; +} + +bool KoDocumentInfoUserMetadata::loadOasis( const QDomNode& metaDoc ) +{ + QDomNode n = metaDoc.firstChild(); + for ( ; !n.isNull(); n = n.nextSibling() ) + { + if (n.isElement()) + { + QDomElement e = n.toElement(); + if ( e.namespaceURI() == KoXmlNS::meta && e.localName() == "user-defined" && !e.text().isEmpty() ) + { + QString name = e.attributeNS( KoXmlNS::meta, "name", QString::null ); + if ( !m_reserved.contains( name ) ) + m_metaList[ name ] = e.text(); + } + } + } + return true; +} + +// KOffice-1.3 format +bool KoDocumentInfoUserMetadata::load( const QDomElement& ) +{ + return true; +} + +// KOffice-1.3 format +QDomElement KoDocumentInfoUserMetadata::save( QDomDocument& ) +{ + return QDomElement(); +} + +#include <KoDocumentInfo.moc> diff --git a/lib/kofficecore/KoDocumentInfo.h b/lib/kofficecore/KoDocumentInfo.h new file mode 100644 index 00000000..30e74cf7 --- /dev/null +++ b/lib/kofficecore/KoDocumentInfo.h @@ -0,0 +1,196 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@kde.org> + Copyright (C) 2004 David Faure <faure@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. +*/ + +#ifndef KO_DOCUMENT_INFO_H +#define KO_DOCUMENT_INFO_H + +#include <qobject.h> +#include <qmap.h> +#include <kconfig.h> +#include <koffice_export.h> + +class QString; +class QStringList; +class QDomDocument; +class QDomElement; +class QDomNode; +class QDateTime; +class KoStore; +class KoDocumentInfoPage; +class KoXmlWriter; + +class KOFFICECORE_EXPORT KoDocumentInfo : public QObject +{ + Q_OBJECT +public: + KoDocumentInfo( QObject* parent = 0, const char* name = 0 ); + virtual ~KoDocumentInfo(); + + bool load( const QDomDocument& doc ); + bool loadOasis( const QDomDocument& metaDoc ); + + QDomDocument save(); + bool saveOasis( KoStore* store ); + + /** + * This info has an accessor because it's the most commonly used. + * Equivalent to page("about")->title() (but checking that the page exists) + */ + QString title() const; + QString creator() const; + + KoDocumentInfoPage* page( const QString& name ) const; + QStringList pages() const; + void documentInfochanged() { emit sigDocumentInfoModifed();} + signals: + void sigDocumentInfoModifed(); +}; + +class KOFFICECORE_EXPORT KoDocumentInfoPage : public QObject +{ +public: + KoDocumentInfoPage( QObject *parent, const char* name ); + + virtual bool load( const QDomElement& e ) = 0; + virtual QDomElement save( QDomDocument& doc ) = 0; + virtual bool loadOasis( const QDomNode& metaDoc ) = 0; + virtual bool saveOasis( KoXmlWriter &xmlWriter ) = 0; +}; + +class KOFFICECORE_EXPORT KoDocumentInfoAuthor : public KoDocumentInfoPage +{ + Q_OBJECT +public: + KoDocumentInfoAuthor( KoDocumentInfo* info ); + ~KoDocumentInfoAuthor(); + + virtual bool load( const QDomElement& e ); + virtual QDomElement save( QDomDocument& doc ); + virtual bool loadOasis( const QDomNode& metaDoc ); + virtual bool saveOasis( KoXmlWriter &xmlWriter ); + + QString fullName() const; + QString initial() const; + QString title() const; + QString company() const; + QString email() const; + QString telephoneHome() const; + QString telephoneWork() const; + QString fax() const; + QString country() const; + QString postalCode() const; + QString city() const; + QString street() const; + QString position() const; + + void setFullName( const QString& n ); + void setTitle( const QString& n ); + void setCompany( const QString& n ); + void setEmail( const QString& n ); + void setTelephoneHome( const QString& n ); + void setTelephoneWork( const QString& n ); + void setFax( const QString& n ); + void setCountry( const QString& n ); + void setPostalCode( const QString& n ); + void setCity( const QString& n ); + void setStreet( const QString& n ); + void setInitial( const QString& n ); + void setPosition( const QString& n ); + void initParameters(); + +private: + QString m_fullName; + QString m_title; + QString m_company; + QString m_email; + QString m_telephoneHome; + QString m_telephoneWork; + QString m_fax; + QString m_country; + QString m_postalCode; + QString m_city; + QString m_street; + QString m_initial; + QString m_position; + KConfig *m_emailCfg; +}; + +class KOFFICECORE_EXPORT KoDocumentInfoAbout : public KoDocumentInfoPage +{ + Q_OBJECT +public: + KoDocumentInfoAbout( KoDocumentInfo* info ); + + virtual bool load( const QDomElement& e ); + virtual QDomElement save( QDomDocument& doc ); + virtual bool loadOasis( const QDomNode& metaDoc ); + virtual bool saveOasis( KoXmlWriter &xmlWriter ); + + QString title() const; + QString abstract() const; + QString subject() const; + QString keywords() const; + QString initialCreator() const; + QString editingCycles() const; + QString creationDate() const; + QString modificationDate() const; + + void setKeywords( const QString &n ); + void setSubject( const QString& n ); + + void setTitle( const QString& n ); + void setAbstract( const QString& n ); + void saveParameters(); + +public slots: + void resetMetaData(); + +private: + KoDocumentInfo* m_docInfo; + QString m_title; + QString m_abstract; + QString m_keywords; + QString m_subject; + QString m_initialCreator; + QDateTime m_creationDate; + QDateTime m_modificationDate; + int m_editingCycles; + bool m_firstSave; +}; + +class KOFFICECORE_EXPORT KoDocumentInfoUserMetadata : public KoDocumentInfoPage +{ + Q_OBJECT +public: + KoDocumentInfoUserMetadata( KoDocumentInfo* info ); + + virtual bool load( const QDomElement& e ); + virtual QDomElement save( QDomDocument& doc ); + virtual bool loadOasis( const QDomNode& metaDoc ); + virtual bool saveOasis( KoXmlWriter &xmlWriter ); + + QMap<QString, QString>* metadataList() { return &m_metaList; } + +private: + QMap<QString, QString> m_metaList; + QStringList m_reserved; +}; + +#endif diff --git a/lib/kofficecore/KoDocumentInfoDlg.cpp b/lib/kofficecore/KoDocumentInfoDlg.cpp new file mode 100644 index 00000000..f9ad8904 --- /dev/null +++ b/lib/kofficecore/KoDocumentInfoDlg.cpp @@ -0,0 +1,555 @@ +/* This file is part of the KDE project + Copyright (c) 2000 Simon Hausmann <hausmann@kde.org> + + $Id: KoDocumentInfoDlg.cpp 512890 2006-02-23 21:37:08Z illissius $ + + 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 "KoDocumentInfoDlg.h" +#include "KoDocumentInfo.h" +#include "koDocumentInfoAboutWidget.h" +#include "koDocumentInfoAuthorWidget.h" +#include "koDocumentInfoUserMetadataWidget.h" +#include "KoDocument.h" + +#include <KoGlobal.h> +#include <KoStore.h> + +#include <sys/stat.h> +#include <unistd.h> +#include <assert.h> + +#include <qlabel.h> +#include <qbuffer.h> +#include <qdom.h> +#include <qdir.h> +#include <qvbox.h> +#include <qdatetime.h> + +#include <kabc/addressee.h> +#include <kabc/stdaddressbook.h> +#include <kdeversion.h> +#include <klocale.h> +#include <kmessagebox.h> +#include <ktar.h> +#include <kdebug.h> +#include <ktempfile.h> +#include <kmimetype.h> +#include <qlayout.h> +#include <klistview.h> +#include <qgrid.h> +#include <qmap.h> +#include <kfilterdev.h> +#include <klineedit.h> +#include <ktextedit.h> +#include <kiconloader.h> +#include <kpushbutton.h> +#include <klocale.h> + +class KoDocumentInfoDlg::KoDocumentInfoDlgPrivate +{ +public: + KoDocumentInfoDlgPrivate() + { + } + ~KoDocumentInfoDlgPrivate() + { + } + + KoDocumentInfo *m_info; + KoDocumentInfoAboutWidget *m_aboutWidget; + KoDocumentInfoAuthorWidget *m_authorWidget; + KoDocumentInfoUserMetadataWidget *m_metaWidget; + + bool m_bDeleteDialog; + KDialogBase *m_dialog; +}; + +KoDocumentInfoDlg::KoDocumentInfoDlg( KoDocumentInfo *docInfo, QWidget *parent, const char *name, + KDialogBase *dialog ) +: QObject( parent, "docinfodlg" ) +{ + d = new KoDocumentInfoDlgPrivate; + d->m_info = docInfo; + + d->m_dialog = dialog; + d->m_bDeleteDialog = false; + + if ( !dialog ) + { + d->m_dialog = new KDialogBase( KDialogBase::Tabbed, + i18n( "Document Information" ), + KDialogBase::Ok | KDialogBase::Cancel, + KDialogBase::Ok, parent, name, true, false ); + d->m_dialog->setInitialSize( QSize( 500, 500 ) ); + d->m_bDeleteDialog = true; + } + + QStringList pages = docInfo->pages(); + QStringList::ConstIterator it = pages.begin(); + QStringList::ConstIterator end = pages.end(); + for (; it != end; ++it ) + { + KoDocumentInfoPage *pg = docInfo->page( *it ); + if ( pg->inherits( "KoDocumentInfoAuthor" ) ) + addAuthorPage( static_cast<KoDocumentInfoAuthor *>( pg ) ); + else if ( pg->inherits( "KoDocumentInfoAbout" ) ) + addAboutPage( static_cast<KoDocumentInfoAbout *>( pg ) ); +/* else if ( pg->inherits( "KoDocumentInfoUserMetadata" ) ) + addUserMetadataPage( static_cast<KoDocumentInfoUserMetadata *>( pg ) );*/ + } +} + +KoDocumentInfoDlg::~KoDocumentInfoDlg() +{ + if ( d->m_bDeleteDialog ) + delete d->m_dialog; + + delete d; +} + +int KoDocumentInfoDlg::exec() +{ + return d->m_dialog->exec(); +} + +KDialogBase *KoDocumentInfoDlg::dialog() const +{ + return d->m_dialog; +} + +void KoDocumentInfoDlg::loadFromKABC() +{ + KABC::StdAddressBook *ab = static_cast<KABC::StdAddressBook*> + ( KABC::StdAddressBook::self() ); + + if ( !ab ) + return; + + KABC::Addressee addr = ab->whoAmI(); + if ( addr.isEmpty() ) + { + KMessageBox::sorry( 0L, i18n( "No personal contact data set, please use the option \ + \"Set as Personal Contact Data\" from the \"Edit\" menu in KAddressbook to set one." ) ); + return; + } + + d->m_authorWidget->leFullName->setText( addr.formattedName() ); + d->m_authorWidget->leInitial->setText( addr.givenName()[ 0 ] + ". " + + addr.familyName()[ 0 ] + "." ); + d->m_authorWidget->leAuthorTitle->setText( addr.title() ); + d->m_authorWidget->leCompany->setText( addr.organization() ); + d->m_authorWidget->leEmail->setText( addr.preferredEmail() ); + + KABC::PhoneNumber phone = addr.phoneNumber( KABC::PhoneNumber::Home ); + d->m_authorWidget->leTelephoneHome->setText( phone.number() ); + phone = addr.phoneNumber( KABC::PhoneNumber::Work ); + d->m_authorWidget->leTelephoneWork->setText( phone.number() ); + + phone = addr.phoneNumber( KABC::PhoneNumber::Fax ); + d->m_authorWidget->leFax->setText( phone.number() ); + + KABC::Address a = addr.address( KABC::Address::Home ); + d->m_authorWidget->leCountry->setText( a.country() ); + d->m_authorWidget->lePostalCode->setText( a.postalCode() ); + d->m_authorWidget->leCity->setText( a.locality() ); + d->m_authorWidget->leStreet->setText( a.street() ); + + emit changed(); +} + +void KoDocumentInfoDlg::deleteInfo() +{ + d->m_authorWidget->leFullName->setText( QString::null ); + d->m_authorWidget->leInitial->setText( QString::null ); + d->m_authorWidget->leAuthorTitle->setText( QString::null ); + d->m_authorWidget->leCompany->setText( QString::null ); + d->m_authorWidget->leEmail->setText( QString::null ); + d->m_authorWidget->leTelephoneHome->setText( QString::null ); + d->m_authorWidget->leTelephoneWork->setText( QString::null ); + d->m_authorWidget->leFax->setText( QString::null ); + d->m_authorWidget->leCountry->setText( QString::null ); + d->m_authorWidget->lePostalCode->setText( QString::null ); + d->m_authorWidget->leCity->setText( QString::null ); + d->m_authorWidget->leStreet->setText( QString::null ); + emit changed(); +} + +void KoDocumentInfoDlg::resetMetaData() +{ + QString s = KGlobal::locale()->formatDateTime( QDateTime::currentDateTime() ); + d->m_aboutWidget->labelCreated->setText( s + ", " + d->m_info->creator() ); + d->m_aboutWidget->labelModified->setText( "" ); + d->m_aboutWidget->labelRevision->setText( "0" ); + emit changed(); +} + +void KoDocumentInfoDlg::addAuthorPage( KoDocumentInfoAuthor *authorInfo ) +{ + QVBox *page = d->m_dialog->addVBoxPage( i18n( "Author" ) ); + d->m_authorWidget = new KoDocumentInfoAuthorWidget( page ); + d->m_authorWidget->labelAuthor->setPixmap( KGlobal::iconLoader()->loadIcon( "personal", KIcon::Desktop, 48 ) ); + d->m_authorWidget->pbLoadKABC->setIconSet( QIconSet( KGlobal::iconLoader()->loadIcon( "kaddressbook", KIcon::Small ) ) ); + d->m_authorWidget->pbDelete->setIconSet( QIconSet( KGlobal::iconLoader()->loadIcon( "eraser", KIcon::Small ) ) ); + + d->m_authorWidget->leFullName->setText( authorInfo->fullName() ); + d->m_authorWidget->leInitial->setText( authorInfo->initial() ); + d->m_authorWidget->leAuthorTitle->setText( authorInfo->title() ); + d->m_authorWidget->leCompany->setText( authorInfo->company() ); + d->m_authorWidget->leEmail->setText( authorInfo->email() ); + d->m_authorWidget->leTelephoneWork->setText( authorInfo->telephoneWork() ); + d->m_authorWidget->leTelephoneHome->setText( authorInfo->telephoneHome() ); + d->m_authorWidget->leFax->setText( authorInfo->fax() ); + d->m_authorWidget->leCountry->setText( authorInfo->country() ); + d->m_authorWidget->lePostalCode->setText( authorInfo->postalCode() ); + d->m_authorWidget->leCity->setText( authorInfo->city() ); + d->m_authorWidget->leStreet->setText( authorInfo->street() ); + d->m_authorWidget->leAuthorPosition->setText( authorInfo->position() ); + + connect( d->m_authorWidget->leFullName, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leInitial, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leAuthorTitle, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leCompany, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leEmail, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leTelephoneWork, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leTelephoneHome, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leFax, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leCountry, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->lePostalCode, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leCity, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leStreet, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->leAuthorPosition, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_authorWidget->pbLoadKABC, SIGNAL( clicked() ), + this, SLOT( loadFromKABC() ) ); + connect( d->m_authorWidget->pbDelete, SIGNAL( clicked() ), + this, SLOT( deleteInfo() ) ); +} + +void KoDocumentInfoDlg::addAboutPage( KoDocumentInfoAbout *aboutInfo ) +{ + QVBox *page = d->m_dialog->addVBoxPage( i18n( "General" ) ); + d->m_aboutWidget = new KoDocumentInfoAboutWidget( page ); + d->m_aboutWidget->pbReset->setIconSet( QIconSet( KGlobal::iconLoader()->loadIcon( "reload", KIcon::Small ) ) ); + KoDocument* doc = dynamic_cast< KoDocument* >( d->m_info->parent() ); + if ( doc ) + { + d->m_aboutWidget->leDocFile->setText( doc->file() ); + d->m_aboutWidget->labelType->setText( KMimeType::mimeType( doc->mimeType() )->comment() ); + d->m_aboutWidget->pixmapLabel->setPixmap( KMimeType::mimeType( doc->mimeType() )->pixmap( KIcon::Desktop, 48 ) ); + } + if ( aboutInfo->creationDate() != QString::null ) + d->m_aboutWidget->labelCreated->setText( aboutInfo->creationDate() + ", " + aboutInfo->initialCreator() ); + if ( aboutInfo->modificationDate() != QString::null ) + d->m_aboutWidget->labelModified->setText( aboutInfo->modificationDate() + ", " + d->m_info->creator() ); + d->m_aboutWidget->labelRevision->setText( aboutInfo->editingCycles() ); + d->m_aboutWidget->leDocTitle->setText( aboutInfo->title() ); + d->m_aboutWidget->leDocSubject->setText( aboutInfo->subject() ); + d->m_aboutWidget->leDocKeywords->setText( aboutInfo->keywords() ); + d->m_aboutWidget->meDocAbstract->setText( aboutInfo->abstract() ); + + connect( d->m_aboutWidget->leDocTitle, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_aboutWidget->meDocAbstract, SIGNAL( textChanged() ), + this, SIGNAL( changed() ) ); + connect( d->m_aboutWidget->leDocSubject, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_aboutWidget->leDocKeywords, SIGNAL( textChanged( const QString & ) ), + this, SIGNAL( changed() ) ); + connect( d->m_aboutWidget->pbReset, SIGNAL( clicked() ), + aboutInfo, SLOT( resetMetaData() ) ); + connect( d->m_aboutWidget->pbReset, SIGNAL( clicked() ), + this, SLOT( resetMetaData() ) ); +} + +void KoDocumentInfoDlg::addUserMetadataPage( KoDocumentInfoUserMetadata *userMetadataInfo ) +{ + QVBox *page = d->m_dialog->addVBoxPage( i18n( "User-Defined Metadata" ) ); + d->m_metaWidget = new KoDocumentInfoUserMetadataWidget( page ); + + d->m_metaWidget->metaListView->addColumn( "Name" ); + d->m_metaWidget->metaListView->setFullWidth( true ); + + QMap<QString, QString>::iterator it; + for ( it = userMetadataInfo->metadataList()->begin(); it != userMetadataInfo->metadataList()->end(); ++it ) + { + QString name = it.key(); + QString value = it.data(); + KListViewItem* it = new KListViewItem( d->m_metaWidget->metaListView, name, value ); + it->setPixmap( 0, KGlobal::iconLoader()->loadIcon( "text", KIcon::Small ) ); + } +} + +void KoDocumentInfoDlg::save() +{ + QStringList pages = d->m_info->pages(); + QStringList::ConstIterator it = pages.begin(); + QStringList::ConstIterator end = pages.end(); + bool saveInfo=false; + for (; it != end; ++it ) + { + KoDocumentInfoPage *pg = d->m_info->page( *it ); + if ( pg->inherits( "KoDocumentInfoAuthor" ) ) + { + saveInfo=true; + save( static_cast<KoDocumentInfoAuthor *>( pg ) ); + } + else if ( pg->inherits( "KoDocumentInfoAbout" ) ) + { + saveInfo=true; + save( static_cast<KoDocumentInfoAbout *>( pg ) ); + } + } + if(saveInfo) + d->m_info->documentInfochanged(); +} + +void KoDocumentInfoDlg::save( KoDocumentInfoAuthor *authorInfo ) +{ + authorInfo->setFullName( d->m_authorWidget->leFullName->text() ); + authorInfo->setInitial( d->m_authorWidget->leInitial->text() ); + authorInfo->setTitle( d->m_authorWidget->leAuthorTitle->text() ); + authorInfo->setCompany( d->m_authorWidget->leCompany->text() ); + authorInfo->setEmail( d->m_authorWidget->leEmail->text() ); + authorInfo->setTelephoneWork( d->m_authorWidget->leTelephoneWork->text() ); + authorInfo->setTelephoneHome( d->m_authorWidget->leTelephoneHome->text() ); + authorInfo->setFax( d->m_authorWidget->leFax->text() ); + authorInfo->setCountry( d->m_authorWidget->leCountry->text() ); + authorInfo->setPostalCode( d->m_authorWidget->lePostalCode->text() ); + authorInfo->setCity( d->m_authorWidget->leCity->text() ); + authorInfo->setStreet( d->m_authorWidget->leStreet->text() ); + authorInfo->setPosition( d->m_authorWidget->leAuthorPosition->text() ); + + KConfig* config = KoGlobal::kofficeConfig(); + KConfigGroupSaver cgs( config, "Author" ); + config->writeEntry("telephone", d->m_authorWidget->leTelephoneHome->text()); + config->writeEntry("telephone-work", d->m_authorWidget->leTelephoneWork->text()); + config->writeEntry("fax", d->m_authorWidget->leFax->text()); + config->writeEntry("country",d->m_authorWidget->leCountry->text()); + config->writeEntry("postal-code",d->m_authorWidget->lePostalCode->text()); + config->writeEntry("city", d->m_authorWidget->leCity->text()); + config->writeEntry("street", d->m_authorWidget->leStreet->text()); + config->sync(); +} + +void KoDocumentInfoDlg::save( KoDocumentInfoAbout *aboutInfo ) +{ + aboutInfo->setTitle( d->m_aboutWidget->leDocTitle->text() ); + aboutInfo->setSubject( d->m_aboutWidget->leDocSubject->text() ); + aboutInfo->setKeywords( d->m_aboutWidget->leDocKeywords->text() ); + aboutInfo->setAbstract( d->m_aboutWidget->meDocAbstract->text() ); +} + +void KoDocumentInfoDlg::save( KoDocumentInfoUserMetadata* ) +{ + // FIXME +} + +class KoDocumentInfoPropsPage::KoDocumentInfoPropsPagePrivate +{ +public: + KoDocumentInfo *m_info; + KoDocumentInfoDlg *m_dlg; + KURL m_url; + KTarGz *m_src; + KTarGz *m_dst; + + const KTarFile *m_docInfoFile; +}; + +KoDocumentInfoPropsPage::KoDocumentInfoPropsPage( KPropertiesDialog *props, + const char *, + const QStringList & ) +: KPropsDlgPlugin( props ) +{ + d = new KoDocumentInfoPropsPagePrivate; + d->m_info = new KoDocumentInfo( this, "docinfo" ); + d->m_url = props->item()->url(); + d->m_dlg = 0; + + if ( !d->m_url.isLocalFile() ) + return; + + d->m_dst = 0; + +#ifdef __GNUC__ +#warning TODO port this to KoStore !!! +#endif + d->m_src = new KTarGz( d->m_url.path(), "application/x-gzip" ); + + if ( !d->m_src->open( IO_ReadOnly ) ) + return; + + const KTarDirectory *root = d->m_src->directory(); + if ( !root ) + return; + + const KTarEntry *entry = root->entry( "documentinfo.xml" ); + + if ( entry && entry->isFile() ) + { + d->m_docInfoFile = static_cast<const KTarFile *>( entry ); + + QBuffer buffer( d->m_docInfoFile->data() ); + buffer.open( IO_ReadOnly ); + + QDomDocument doc; + doc.setContent( &buffer ); + + d->m_info->load( doc ); + } + + d->m_dlg = new KoDocumentInfoDlg( d->m_info, 0, 0, props ); + connect( d->m_dlg, SIGNAL( changed() ), + this, SIGNAL( changed() ) ); +} + +KoDocumentInfoPropsPage::~KoDocumentInfoPropsPage() +{ + delete d->m_info; + delete d->m_src; + delete d->m_dst; + delete d->m_dlg; + delete d; +} + +void KoDocumentInfoPropsPage::applyChanges() +{ + const KTarDirectory *root = d->m_src->directory(); + if ( !root ) + return; + + struct stat statBuff; + + if ( stat( QFile::encodeName( d->m_url.path() ), &statBuff ) != 0 ) + return; + + KTempFile tempFile( d->m_url.path(), QString::null, statBuff.st_mode ); + + tempFile.setAutoDelete( true ); + + if ( tempFile.status() != 0 ) + return; + + if ( !tempFile.close() ) + return; + + d->m_dst = new KTarGz( tempFile.name(), "application/x-gzip" ); + + if ( !d->m_dst->open( IO_WriteOnly ) ) + return; + + KMimeType::Ptr mimeType = KMimeType::findByURL( d->m_url, 0, true ); + if ( mimeType && dynamic_cast<KFilterDev *>( d->m_dst->device() ) != 0 ) + { + QCString appIdentification( "KOffice " ); // We are limited in the number of chars. + appIdentification += mimeType->name().latin1(); + appIdentification += '\004'; // Two magic bytes to make the identification + appIdentification += '\006'; // more reliable (DF) + d->m_dst->setOrigFileName( appIdentification ); + } + + bool docInfoSaved = false; + + QStringList entries = root->entries(); + QStringList::ConstIterator it = entries.begin(); + QStringList::ConstIterator end = entries.end(); + for (; it != end; ++it ) + { + const KTarEntry *entry = root->entry( *it ); + + assert( entry ); + + if ( entry->name() == "documentinfo.xml" || + ( !docInfoSaved && !entries.contains( "documentinfo.xml" ) ) ) + { + d->m_dlg->save(); + + QBuffer buffer; + buffer.open( IO_WriteOnly ); + QTextStream str( &buffer ); + str << d->m_info->save(); + buffer.close(); + + kdDebug( 30003 ) << "writing documentinfo.xml" << endl; + d->m_dst->writeFile( "documentinfo.xml", entry->user(), entry->group(), buffer.buffer().size(), + buffer.buffer().data() ); + + docInfoSaved = true; + } + else + copy( QString::null, entry ); + } + + d->m_dst->close(); + + QDir dir; + dir.rename( tempFile.name(), d->m_url.path() ); + + delete d->m_dst; + d->m_dst = 0; +} + +void KoDocumentInfoPropsPage::copy( const QString &path, const KArchiveEntry *entry ) +{ + kdDebug( 30003 ) << "copy " << entry->name() << endl; + if ( entry->isFile() ) + { + const KTarFile *file = static_cast<const KTarFile *>( entry ); + kdDebug( 30003 ) << "file :" << entry->name() << endl; + kdDebug( 30003 ) << "full path is: " << path << entry->name() << endl; + d->m_dst->writeFile( path + entry->name(), entry->user(), entry->group(), + file->size(), + file->data().data() ); + } + else + { + const KTarDirectory *dir = static_cast<const KTarDirectory *>( entry ); + kdDebug( 30003 ) << "dir : " << entry->name() << endl; + kdDebug( 30003 ) << "full path is: " << path << entry->name() << endl; + + QString p = path + entry->name(); + if ( p != "/" ) + { + d->m_dst->writeDir( p, entry->user(), entry->group() ); + p.append( "/" ); + } + + QStringList entries = dir->entries(); + QStringList::ConstIterator it = entries.begin(); + QStringList::ConstIterator end = entries.end(); + for (; it != end; ++it ) + copy( p, dir->entry( *it ) ); + } +} + +/* vim: sw=2 et + */ + +#include "KoDocumentInfoDlg.moc" diff --git a/lib/kofficecore/KoDocumentInfoDlg.h b/lib/kofficecore/KoDocumentInfoDlg.h new file mode 100644 index 00000000..d3e1e9a3 --- /dev/null +++ b/lib/kofficecore/KoDocumentInfoDlg.h @@ -0,0 +1,85 @@ +/* This file is part of the KDE project + Copyright (c) 2000 Simon Hausmann <hausmann@kde.org> + + $Id: KoDocumentInfoDlg.h 508787 2006-02-12 18:28:12Z ingwa $ + + 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. +*/ + +#ifndef __koDocumentInfoDlg_h__ +#define __koDocumentInfoDlg_h__ + +#include <kpropertiesdialog.h> +#include <koffice_export.h> + +class KDialogBase; +class KoDocumentInfo; +class KoDocumentInfoAuthor; +class KoDocumentInfoAbout; +class KoDocumentInfoUserMetadata; +class KArchiveEntry; + +class KOFFICECORE_EXPORT KoDocumentInfoDlg : public QObject +{ + Q_OBJECT +public: + KoDocumentInfoDlg( KoDocumentInfo *docInfo, QWidget *parent = 0, const char *name = 0, + KDialogBase *dialog = 0 ); + virtual ~KoDocumentInfoDlg(); + + int exec(); + KDialogBase *dialog() const; + + void save(); + +signals: + void changed(); + +private slots: + void loadFromKABC(); + void deleteInfo(); + void resetMetaData(); + +private: + void addAuthorPage( KoDocumentInfoAuthor *authorInfo ); + void addAboutPage( KoDocumentInfoAbout *aboutInfo ); + void addUserMetadataPage( KoDocumentInfoUserMetadata *userMetadataInfo ); + + void save( KoDocumentInfoAuthor *authorInfo ); + void save( KoDocumentInfoAbout *aboutInfo ); + void save( KoDocumentInfoUserMetadata *userMetadataInfo ); + + class KoDocumentInfoDlgPrivate; + KoDocumentInfoDlgPrivate *d; +}; + +class KOFFICECORE_EXPORT KoDocumentInfoPropsPage : public KPropsDlgPlugin +{ + Q_OBJECT +public: + KoDocumentInfoPropsPage( KPropertiesDialog *props, const char *name = 0, + const QStringList & = QStringList() ); + virtual ~KoDocumentInfoPropsPage(); + + virtual void applyChanges(); + +private: + void copy( const QString &path, const KArchiveEntry *entry ); + class KoDocumentInfoPropsPagePrivate; + KoDocumentInfoPropsPagePrivate *d; +}; + +#endif diff --git a/lib/kofficecore/KoDocument_p.h b/lib/kofficecore/KoDocument_p.h new file mode 100644 index 00000000..1b21f69a --- /dev/null +++ b/lib/kofficecore/KoDocument_p.h @@ -0,0 +1,41 @@ +/* This file is part of the KDE project + Copyright (C) 2001 David Faure <faure@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. + + 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. +*/ +#ifndef __koDocument_p_h__ +#define __koDocument_p_h__ + +#include <kparts/browserextension.h> + +class KoDocument; + +/** + * Used in singleViewMode, when embedded into a browser + * @internal + */ +class KoBrowserExtension : public KParts::BrowserExtension +{ + Q_OBJECT +public: + KoBrowserExtension( KoDocument * doc, const char * name = 0 ); + +public slots: + /// Automatically detected by konqueror + void print(); +}; + +#endif diff --git a/lib/kofficecore/KoDom.cpp b/lib/kofficecore/KoDom.cpp new file mode 100644 index 00000000..45b00acd --- /dev/null +++ b/lib/kofficecore/KoDom.cpp @@ -0,0 +1,30 @@ +/* This file is part of the KDE project + Copyright (C) 2004 David Faure <faure@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 "KoDom.h" + +QDomElement KoDom::namedItemNS( const QDomNode& node, const char* nsURI, const char* localName ) +{ + QDomNode n = node.firstChild(); + for ( ; !n.isNull(); n = n.nextSibling() ) { + if ( n.isElement() && n.localName() == localName && n.namespaceURI() == nsURI ) + return n.toElement(); + } + return QDomElement(); +} diff --git a/lib/kofficecore/KoDom.h b/lib/kofficecore/KoDom.h new file mode 100644 index 00000000..4d95a42b --- /dev/null +++ b/lib/kofficecore/KoDom.h @@ -0,0 +1,70 @@ +/* This file is part of the KDE project + Copyright (C) 2004 David Faure <faure@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. +*/ + +#ifndef KODOM_H +#define KODOM_H + +#include <qdom.h> +#include <koffice_export.h> +/** + * This namespace contains a few convenience functions to simplify code using QDom + * (when loading OASIS documents, in particular). + * + * To find the child element with a given name, use KoDom::namedItemNS. + * + * To find all child elements with a given name, use + * QDomElement e; + * forEachElement( e, parent ) + * { + * if ( e.localName() == "..." && e.namespaceURI() == KoXmlNS::... ) + * { + * ... + * } + * } + * Note that this means you don't ever need to use QDomNode nor toElement anymore! + * Also note that localName is the part without the prefix, this is the whole point + * of namespace-aware methods. + * + * To find the attribute with a given name, use QDomElement::attributeNS. + * + * Do not use getElementsByTagNameNS, it's recursive (which is never needed in KOffice). + * Do not use tagName() or nodeName() or prefix(), since the prefix isn't fixed. + * + * @author David Faure <faure@kde.org> + */ +namespace KoDom { + + /** + * A namespace-aware version of QDomNode::namedItem(), + * which also takes care of casting to a QDomElement. + * Use this when a domelement is known to have only *one* child element + * with a given tagname. + * + * Note: do *NOT* use getElementsByTagNameNS, it's recursive! + */ + KOFFICECORE_EXPORT QDomElement namedItemNS( const QDomNode& node, const char* nsURI, const char* localName ); + +} + +#define forEachElement( elem, parent ) \ + for ( QDomNode _node = parent.firstChild(); !_node.isNull(); _node = _node.nextSibling() ) \ + if ( !( elem = _node.toElement() ).isNull() ) + +#endif /* KODOM_H */ + diff --git a/lib/kofficecore/KoFactory.cpp b/lib/kofficecore/KoFactory.cpp new file mode 100644 index 00000000..d858e8e9 --- /dev/null +++ b/lib/kofficecore/KoFactory.cpp @@ -0,0 +1,45 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@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 <KoFactory.h> + + +class KoFactory::KoFactoryPrivate +{ +public: + KoFactoryPrivate() + { + } + ~KoFactoryPrivate() + { + } +}; + +KoFactory::KoFactory( QObject *parent, const char *name ) + : KParts::Factory( parent, name ) +{ + //d = new KoFactoryPrivate; +} + +KoFactory::~KoFactory() +{ + //delete d; +} + +#include <KoFactory.moc> diff --git a/lib/kofficecore/KoFactory.h b/lib/kofficecore/KoFactory.h new file mode 100644 index 00000000..bcc77439 --- /dev/null +++ b/lib/kofficecore/KoFactory.h @@ -0,0 +1,39 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@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. +*/ + +#ifndef __koFactory_h__ +#define __koFactory_h__ + +#include <kparts/factory.h> +#include <koffice_export.h> +class KInstance; + +class KOFFICECORE_EXPORT KoFactory : public KParts::Factory +{ + Q_OBJECT +public: + KoFactory( QObject *parent = 0, const char *name = 0 ); + virtual ~KoFactory(); + +private: + class KoFactoryPrivate; + KoFactoryPrivate *d; +}; + +#endif diff --git a/lib/kofficecore/KoFileDialog.cpp b/lib/kofficecore/KoFileDialog.cpp new file mode 100644 index 00000000..b80c23e0 --- /dev/null +++ b/lib/kofficecore/KoFileDialog.cpp @@ -0,0 +1,125 @@ +/* This file is part of the KDE project + Copyright (C) 2002-2005 David Faure <faure@kde.org> + Copyright (C) 2002-2004 Clarence Dang <dang@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 "KoFileDialog.h" +#include "KoDocument.h" +#include <kfilefiltercombo.h> +#include <klocale.h> +#include <kdiroperator.h> +#include <kdebug.h> + +KoFileDialog::KoFileDialog(const QString& startDir, const QString& filter, + QWidget *parent, const char *name, + bool modal) + : KFileDialog( startDir, filter, parent, name, modal ) +{ + connect( filterWidget, SIGNAL( activated( int) ), + this, SLOT( slotChangedfilter( int ) ) ); +} + +void KoFileDialog::slotChangedfilter( int index ) +{ + // Switch to "directory selection" mode for SaveAsDirectoryStore, + // switch back to "file selection" mode otherwise. + KFile::Mode newMode = KFile::File; + if ( index >= 1 && index <= (int)m_specialFormats.count() + && m_specialFormats[index-1] == KoDocument::SaveAsDirectoryStore ) { + newMode = KFile::Directory; + } + if ( newMode != mode() ) + { + ops->setMode( newMode ); + updateAutoSelectExtension(); + } +} + +void KoFileDialog::setSpecialMimeFilter( QStringList& mimeFilter, + const QString& currentFormat, const int specialOutputFlag, + const QString& nativeFormat, + int supportedSpecialFormats ) +{ + Q_ASSERT( !mimeFilter.isEmpty() ); + Q_ASSERT( mimeFilter[0] == nativeFormat ); + + bool addUncompressed = supportedSpecialFormats & KoDocument::SaveAsDirectoryStore; + bool addFlatXML = supportedSpecialFormats & KoDocument::SaveAsFlatXML; + + int idxSpecialOutputFlag = 0; + int numSpecialEntries = 0; + if ( addUncompressed ) { + ++numSpecialEntries; + m_specialFormats.append( KoDocument::SaveAsDirectoryStore ); + if ( specialOutputFlag == KoDocument::SaveAsDirectoryStore ) + idxSpecialOutputFlag = numSpecialEntries; + } + if ( addFlatXML ) { + ++numSpecialEntries; + m_specialFormats.append( KoDocument::SaveAsFlatXML ); + if ( specialOutputFlag == KoDocument::SaveAsFlatXML ) + idxSpecialOutputFlag = numSpecialEntries; + } + + // Insert numSpecialEntries entries with native mimetypes, for the special entries. + QStringList::Iterator mimeFilterIt = mimeFilter.at( 1 ); + mimeFilter.insert( mimeFilterIt /* before 1 -> after 0 */, numSpecialEntries, nativeFormat ); + + // Fill in filter combo + // Note: if currentFormat doesn't exist in mimeFilter, filterWidget + // will default to the first item (native format) + setMimeFilter( mimeFilter, currentFormat.isEmpty() ? nativeFormat : currentFormat ); + + // To get a different description in the combo, we need to change its entries afterwards + KMimeType::Ptr type = KMimeType::mimeType( nativeFormat ); + int idx = 1; // 0 is the native format + + if ( addUncompressed ) + filterWidget->changeItem( i18n("%1 (Uncompressed XML Files)").arg( type->comment() ), idx++ ); + if ( addFlatXML ) + filterWidget->changeItem( i18n("%1 (Flat XML File)").arg( type->comment() ), idx++ ); + // if you add an entry here, update numSpecialEntries above and specialEntrySelected() below + + // For native format... + if (currentFormat == nativeFormat || currentFormat.isEmpty()) + { + // KFileFilterCombo selected the _last_ "native mimetype" entry, select the correct one + filterWidget->setCurrentItem( idxSpecialOutputFlag ); + slotChangedfilter( filterWidget->currentItem() ); + } + // [Mainly KWord] Tell MS Office users that they can save in RTF! + int i = 0; + for (mimeFilterIt = mimeFilter.begin (); mimeFilterIt != mimeFilter.end (); ++mimeFilterIt, i++) + { + KMimeType::Ptr mime = KMimeType::mimeType (*mimeFilterIt); + QString compatString = mime->property ("X-KDE-CompatibleApplication").toString (); + if (!compatString.isEmpty ()) + filterWidget->changeItem (i18n ("%1 (%2 Compatible)").arg (mime->comment ()).arg (compatString), i); + } +} + +int KoFileDialog::specialEntrySelected() +{ + int i = filterWidget->currentItem(); + // Item 0 is the native format, the following ones are the special formats + if ( i >= 1 && i <= (int)m_specialFormats.count() ) + return m_specialFormats[i-1]; + return 0; +} + +#include "KoFileDialog.moc" diff --git a/lib/kofficecore/KoFileDialog.h b/lib/kofficecore/KoFileDialog.h new file mode 100644 index 00000000..fc3b61e2 --- /dev/null +++ b/lib/kofficecore/KoFileDialog.h @@ -0,0 +1,55 @@ +/* This file is part of the KDE project + Copyright (C) 2002-2004 David Faure <faure@kde.org> + Copyright (C) 2002-2004 Clarence Dang <dang@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. +*/ + +#ifndef KOFILEDIALOG_H +#define KOFILEDIALOG_H + +#include <kfiledialog.h> +#include <qvaluevector.h> + +/** + * Extension to KFileDialog in order to add special entries to the filter combo, + * like "save as koffice-1.1", "save as dir" etc. + * @note Used only when saving! + * + * This class is INTERNAL to KoMainWindow, do NOT use in your application. + * Binary compatibility is not guaranteed. + */ +class KoFileDialog : public KFileDialog +{ + Q_OBJECT +public: + KoFileDialog( const QString& startDir, const QString& filter, + QWidget *parent, const char *name, + bool modal ); + void setSpecialMimeFilter( QStringList& mimeFilter, + const QString& currentFormat, const int specialOutputFlag, + const QString& nativeFormat, + int supportedSpecialFormats ); + int specialEntrySelected(); + +private slots: + void slotChangedfilter( int index ); + +private: + QValueVector<int> m_specialFormats; +}; + +#endif /* KOFILEDIALOG_H */ diff --git a/lib/kofficecore/KoFilter.cpp b/lib/kofficecore/KoFilter.cpp new file mode 100644 index 00000000..5b1d4898 --- /dev/null +++ b/lib/kofficecore/KoFilter.cpp @@ -0,0 +1,151 @@ +/* This file is part of the KDE libraries + Copyright (C) 2001 Werner Trobin <trobin@kde.org> + 2002 Werner Trobin <trobin@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 version 2 as published by the Free Software Foundation. + + 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 <KoFilter.h> + +#include <qfile.h> + +#include <kurl.h> +#include <kmimetype.h> +#include <ktempfile.h> +#include <kdebug.h> +#include <KoFilterManager.h> + + +KoFilter::KoFilter() : QObject( 0, 0 ), m_chain( 0 ) +{ +} + +KoFilter::~KoFilter() +{ +} + + +KoEmbeddingFilter::~KoEmbeddingFilter() +{ + if ( m_partStack.count() != 1 ) + kdWarning() << "Someone messed with the part stack" << endl; + delete m_partStack.pop(); +} + +int KoEmbeddingFilter::lruPartIndex() const +{ + return m_partStack.top()->m_lruPartIndex; +} + +QString KoEmbeddingFilter::mimeTypeByExtension( const QString& extension ) +{ + // We need to resort to an ugly hack to determine the mimetype + // from the extension, as kservicetypefactory.h isn't installed + KURL url; + url.setPath( QString( "dummy.%1" ).arg( extension ) ); + KMimeType::Ptr m( KMimeType::findByURL( url, 0, true, true ) ); + return m->name(); +} + +KoEmbeddingFilter::KoEmbeddingFilter() : KoFilter() +{ + m_partStack.push( new PartState() ); +} + +int KoEmbeddingFilter::embedPart( const QCString& from, QCString& to, + KoFilter::ConversionStatus& status, const QString& key ) +{ + ++( m_partStack.top()->m_lruPartIndex ); + + KTempFile tempIn; + tempIn.setAutoDelete( true ); + savePartContents( tempIn.file() ); + tempIn.file()->close(); + + KoFilterManager *manager = new KoFilterManager( tempIn.name(), from, m_chain ); + status = manager->exp0rt( QString::null, to ); + delete manager; + + // Add the part to the current "stack frame", using the number as key + // if the key string is empty + PartReference ref( lruPartIndex(), to ); + m_partStack.top()->m_partReferences.insert( key.isEmpty() ? QString::number( lruPartIndex() ) : key, ref ); + + return lruPartIndex(); +} + +void KoEmbeddingFilter::startInternalEmbedding( const QString& key, const QCString& mimeType ) +{ + filterChainEnterDirectory( QString::number( ++( m_partStack.top()->m_lruPartIndex ) ) ); + PartReference ref( lruPartIndex(), mimeType ); + m_partStack.top()->m_partReferences.insert( key, ref ); + m_partStack.push( new PartState() ); +} + +void KoEmbeddingFilter::endInternalEmbedding() +{ + if ( m_partStack.count() == 1 ) { + kdError( 30500 ) << "You're trying to endInternalEmbedding more often than you started it" << endl; + return; + } + delete m_partStack.pop(); + filterChainLeaveDirectory(); +} + +int KoEmbeddingFilter::internalPartReference( const QString& key ) const +{ + QMapConstIterator<QString, PartReference> it = m_partStack.top()->m_partReferences.find( key ); + if ( it == m_partStack.top()->m_partReferences.end() ) + return -1; + return it.data().m_index; +} + +QCString KoEmbeddingFilter::internalPartMimeType( const QString& key ) const +{ + QMapConstIterator<QString, PartReference> it = m_partStack.top()->m_partReferences.find( key ); + if ( it == m_partStack.top()->m_partReferences.end() ) + return QCString(); + return it.data().m_mimeType; +} + +KoEmbeddingFilter::PartReference::PartReference( int index, const QCString& mimeType ) : + m_index( index ), m_mimeType( mimeType ) +{ +} + +bool KoEmbeddingFilter::PartReference::isValid() const +{ + return m_index != 1 && !m_mimeType.isEmpty(); +} + +KoEmbeddingFilter::PartState::PartState() : m_lruPartIndex( 0 ) +{ +} + +void KoEmbeddingFilter::savePartContents( QIODevice* ) +{ +} + +void KoEmbeddingFilter::filterChainEnterDirectory( const QString& directory ) const +{ + m_chain->enterDirectory( directory ); +} + +void KoEmbeddingFilter::filterChainLeaveDirectory() const +{ + m_chain->leaveDirectory(); +} + +#include <KoFilter.moc> diff --git a/lib/kofficecore/KoFilter.h b/lib/kofficecore/KoFilter.h new file mode 100644 index 00000000..9d82a34f --- /dev/null +++ b/lib/kofficecore/KoFilter.h @@ -0,0 +1,315 @@ +/* This file is part of the KOffice libraries + Copyright (C) 2001 Werner Trobin <trobin@kde.org> + 2002 Werner Trobin <trobin@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 version 2 as published by the Free Software Foundation. + + 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. +*/ + +#ifndef __koffice_filter_h__ +#define __koffice_filter_h__ + +#include <qobject.h> +#include <qmap.h> +#include <qptrstack.h> +#include <koffice_export.h> +class QIODevice; +class KoFilterChain; + +/** + * @brief The base class for import and export filters. + * + * Derive your filter class from this base class and implement + * the @ref convert() method. Don't forget to specify the Q_OBJECT + * macro in your class even if you don't use signals or slots. + * This is needed as filters are created on the fly. + * The m_chain member allows access to the @ref KoFilterChain + * which invokes the filter to query for input/output. + * + * @note Take care: The m_chain pointer is invalid while the constructor + * runs due to the implementation -- @em don't use it in the constructor. + * After the constructor, when running the @ref convert() method it's + * guaranteed to be valid, so no need to check against 0. + * + * @author Werner Trobin <trobin@kde.org> + * @todo the class has no constructor and therefore cannot initialize its private class + */ +class KOFFICECORE_EXPORT KoFilter : public QObject +{ + Q_OBJECT + + friend class KoFilterEntry; // needed for the filter chain pointer :( + friend class KoFilterChain; + +public: + /** + * This enum is used to signal the return state of your filter. + * Return OK in @ref convert() in case everything worked as expected. + * Feel free to add some more error conditions @em before the last item + * if it's needed. + */ + enum ConversionStatus { OK, StupidError, UsageError, CreationError, FileNotFound, + StorageCreationError, BadMimeType, BadConversionGraph, + EmbeddedDocError, WrongFormat, NotImplemented, + ParsingError, InternalError, UnexpectedEOF, + UnexpectedOpcode, UserCancelled, OutOfMemory, + PasswordProtected, + JustInCaseSomeBrokenCompilerUsesLessThanAByte = 255 }; + + virtual ~KoFilter(); + + /** + * The filter chain calls this method to perform the actual conversion. + * The passed mimetypes should be a pair of those you specified in your + * .desktop file. + * You @em have to implement this method to make the filter work. + * + * @param from The mimetype of the source file/document + * @param to The mimetype of the destination file/document + * @return The error status, see the @ref #ConversionStatus enum. + * KoFilter::OK means that everything is alright. + */ + virtual ConversionStatus convert( const QCString& from, const QCString& to ) = 0; + +signals: + /** + * Emit this signal with a value in the range of 1...100 to have some + * progress feedback for the user in the statusbar of the application. + * + * @param value The actual progress state. Should always remain in + * the range 1..100. + */ + void sigProgress( int value ); + +protected: + /** + * This is the constructor your filter has to call, obviously. + */ + KoFilter(); + + /** + * Use this pointer to access all information about input/output + * during the conversion. @em Don't use it in the constructor - + * it's invalid while constructing the object! + */ + KoFilterChain* m_chain; + +private: + KoFilter( const KoFilter& rhs ); + KoFilter& operator=( const KoFilter& rhs ); + + class Private; + Private* d; +}; + + +/** + * The base class for all @em import filters embedding other filters. Right + * now we don't support embedding for export filters, but if there's a + * request for that feature please don't hesitate to contact Werner Trobin + * <trobin@kde.org>. + * + * To make use of embedding features you have to know that there are two kinds + * of embedding for filters: embedding the output of a different filter (library) + * or embedding the output of several internal filters (no separate library). + * The first case is the simpler one. You just have to override savePartContents() + * and call @ref #embedPart to trigger the embedding process. One example for such + * a filter is Kontour's MSOD (MS Office Drawing) filter. + * + * The more complex case is embedding various streams from within the same filter + * library. This is neccesary for OLE like files (at least with the current design + * of the OLEFilter). In this case you have to use @ref #startInternalEmbedding and + * @ref #endInternalEmbedding accordingly. Try to use the previous method if possible. + * + * If you're using this class you can also setup a signal/slot communication + * between parent and child filter. To make that work you simply have to define + * signals and slots along the following rules: + * Signals should be named "commSignal\<name\>" where \<name\> is the name of the signal, + * slots should be named "commSlot\<name\>". The connection will be done automatically + * if names and signatures are matching. + * + * @author Werner Trobin + * @todo the class has no constructor and therefore cannot initialize its private class + */ +class KOFFICECORE_EXPORT KoEmbeddingFilter : public KoFilter +{ + Q_OBJECT + + friend class KoFilterChain; + +public: + virtual ~KoEmbeddingFilter(); + + /** + * @internal + * This method returns the last recently used part index at the + * current directory level. It can be (and is ;) used to generate + * the per-directory-unique address for the next part we have to save. + * It will get updated automatically, you most likely don't have to + * care about that one at all. + */ + int lruPartIndex() const; + + /** + * A static helper method to determine the mimetype via the + * file extension. It allows to go from "wmf" to image/x-wmf + * and so on. Note that you should only pass the pure extension + * and not a whole pattern like "*.doc" or so. + */ + static QString mimeTypeByExtension( const QString& extension ); + +protected: + /** + * Constructs a filter. Note that the m_chain pointer is 0 inside + * the constructor. Most likely your constructor will be empty. + */ + KoEmbeddingFilter(); + + /** + * Embed some document using an external filter (i.e. a different + * filter library). This method works according to the template method + * pattern and calls savePartContents() during execution. + * Call this method when you want to convert some data using one or more + * KOffice filters selected via the filter manager. + * This is the way to go when it comes to embedding unless you have very + * special requirements. + * + * @param from The mimetype of the source data + * @param to The mimetype of the destination part. If this field is set + * to "" the filter manager will try to find the best native + * KOffice mimetype. When the method returns this parameter will + * hold the string of the used mimetype. + * @param status Returns the error status of the filter + * @param key Optional key field to allow custom keys inside the part + * map (see @ref #internalPartReference). If this field is left + * empty we generate a key from the part number (e.g. 1 -> "1") + * @return The number of the part (can be used to refer to the part from + * within the embedding filter). + */ + int embedPart( const QCString& from, QCString& to, + KoFilter::ConversionStatus& status, + const QString& key = QString::null ); + + /** + * Method to perform "internal" embedding of parts in olefilter-style. + * This method can be used to signal the start of a new embedding + * level within your filter. Very evil, but what shall I say ;) + * Unless you really have to you should always use @ref #embedPart as + * it's easier to use and not as hacky. + * + * @param key The key we use to store reference/mimetype of your new part + * @param mimeType The mimetype of the part you're about to embed + */ + void startInternalEmbedding( const QString& key, const QCString& mimeType ); + + /** + * This method signals the end of an internal embedding session. + * You have to call that exactly as often as you call @ref #startInternalEmbedding + * or you'll mess up the internal stack and your file will be invalid. + * Again: use @ref #embedPart if you can :-) + */ + void endInternalEmbedding(); + + /** + * Query the internal part map for the reference of the part + * matching the given key. Note that you can use that plain + * simple int to refer to the respective part (when used as string). + * + * @param key The key you would like to look up + * @return The reference or -1 if we didn't find a part with the + * given key + */ + int internalPartReference( const QString& key ) const; + + /** + * Query the internal part map for the mimetype of the part + * matching the given key. + * + * @param key The key you would like to look up + * @return The mimetype, might be empty if the part matching + * the given key doesn't exist. + */ + QCString internalPartMimeType( const QString& key ) const; + +private: + /** + * Holds the directory's number and the mimetype of the part + * for internal parts. This is all we need to locate a part. + * @internal + */ + struct PartReference + { + PartReference( int index = -1, const QCString& mimeType = "" ); + bool isValid() const; + + int m_index; + QCString m_mimeType; + }; + + /** + * This struct keeps track of the last used index for a + * child part and all references to existing children + * We use it to build a whole stack, one PartState per + * embedding level. + * @internal + */ + struct PartState + { + PartState(); + + int m_lruPartIndex; + QMap<QString, PartReference> m_partReferences; + }; + + /// Better do not copy the filters + KoEmbeddingFilter( const KoEmbeddingFilter& rhs ); + /// Better do not assign the filters + KoEmbeddingFilter& operator=( const KoEmbeddingFilter& rhs ); + + /** + * This method will be called by @ref #embedPart as soon as it + * needs the data of the part (template method pattern). You + * have to override that and simply save the part data to the + * (already opened) file. + * No need to override that when you're not using @ref #embedPart + * (as you should ;) + * + * @param file An already opened file + */ + virtual void savePartContents( QIODevice* file ); + + /** + * Internal methods to support the start/endInternalEmbedding + * methods (we have to change directories and stuff). + * These methods are declared friends of the KoFilterChain + */ + void filterChainEnterDirectory( const QString& directory ) const; + /** + * Internal methods to support the start/endInternalEmbedding + * methods (we have to change directories and stuff). + * These methods are declared friends of the KoFilterChain + */ + void filterChainLeaveDirectory() const; + + /** + * A stack which keeps track of the current part references. + * We push one PartState structure for every embedding level. + */ + QPtrStack<PartState> m_partStack; + + class Private; + Private* d; +}; + +#endif diff --git a/lib/kofficecore/KoFilterChain.cpp b/lib/kofficecore/KoFilterChain.cpp new file mode 100644 index 00000000..2ab53780 --- /dev/null +++ b/lib/kofficecore/KoFilterChain.cpp @@ -0,0 +1,936 @@ +/* This file is part of the KOffice libraries + Copyright (C) 2001 Werner Trobin <trobin@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 version 2 as published by the Free Software Foundation. + + 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 <qmetaobject.h> +#include <ktempfile.h> +#include <kmimetype.h> +#include <KoFilterChain.h> +#include <KoQueryTrader.h> +#include <KoFilterManager.h> // KoFilterManager::filterAvailable, private API +#include <KoDocument.h> +#include <kdebug.h> + +#include <priorityqueue.h> + +#include <limits.h> // UINT_MAX + +// Those "defines" are needed in the setupConnections method below. +// Please always keep the strings and the length in sync! +namespace { + const char* const SIGNAL_PREFIX = "commSignal"; + const int SIGNAL_PREFIX_LEN = 10; + const char* const SLOT_PREFIX = "commSlot"; + const int SLOT_PREFIX_LEN = 8; +} + + +KoFilterChain::ChainLink::ChainLink( KoFilterChain* chain, KoFilterEntry::Ptr filterEntry, + const QCString& from, const QCString& to ) : + m_chain( chain ), m_filterEntry( filterEntry ), m_from( from ), m_to( to ), + m_filter( 0 ), d( 0 ) +{ +} + +KoFilter::ConversionStatus KoFilterChain::ChainLink::invokeFilter( const ChainLink* const parentChainLink ) +{ + if ( !m_filterEntry ) { + kdError( 30500 ) << "This filter entry is null. Strange stuff going on." << endl; + return KoFilter::CreationError; + } + + m_filter = m_filterEntry->createFilter( m_chain, 0, 0 ); + + if ( !m_filter ) { + kdError( 30500 ) << "Couldn't create the filter." << endl; + return KoFilter::CreationError; + } + + if ( parentChainLink ) + setupCommunication( parentChainLink->m_filter ); + + KoFilter::ConversionStatus status = m_filter->convert( m_from, m_to ); + delete m_filter; + m_filter=0; + return status; +} + +void KoFilterChain::ChainLink::dump() const +{ + kdDebug( 30500 ) << " Link: " << m_filterEntry->service()->name() << endl; +} + +int KoFilterChain::ChainLink::lruPartIndex() const +{ + if ( m_filter && m_filter->inherits( "KoEmbeddingFilter" ) ) + return static_cast<KoEmbeddingFilter*>( m_filter )->lruPartIndex(); + return -1; +} + +void KoFilterChain::ChainLink::setupCommunication( const KoFilter* const parentFilter ) const +{ + // progress information + QObject::connect( m_filter, SIGNAL( sigProgress( int ) ), + m_chain->manager(), SIGNAL( sigProgress( int ) ) ); + + if ( !parentFilter ) + return; + + const QMetaObject* const parent = parentFilter->metaObject(); + const QMetaObject* const child = m_filter->metaObject(); + if ( !parent || !child ) + return; + + setupConnections( parentFilter, parent->signalNames(), m_filter, child->slotNames() ); + setupConnections( m_filter, child->signalNames(), parentFilter, parent->slotNames() ); +} + +void KoFilterChain::ChainLink::setupConnections( const KoFilter* sender, const QStrList& sigs, + const KoFilter* receiver, const QStrList& sl0ts ) const +{ + QStrListIterator signalIt( sigs ); + for ( ; signalIt.current(); ++signalIt ) { + if ( strncmp( signalIt.current(), SIGNAL_PREFIX, SIGNAL_PREFIX_LEN ) == 0 ) { + QStrListIterator slotIt( sl0ts ); + for ( ; slotIt.current(); ++slotIt ) { + if ( strncmp( slotIt.current(), SLOT_PREFIX, SLOT_PREFIX_LEN ) == 0 ) { + if ( strcmp( signalIt.current() + SIGNAL_PREFIX_LEN, slotIt.current() + SLOT_PREFIX_LEN ) == 0 ) { + QCString signalString; + signalString.setNum( QSIGNAL_CODE ); + signalString += signalIt.current(); + QCString slotString; + slotString.setNum( QSLOT_CODE ); + slotString += slotIt.current(); + QObject::connect( sender, signalString, receiver, slotString ); + } + } + } + } + } +} + + +KoFilterChain::~KoFilterChain() +{ + if ( filterManagerParentChain() && filterManagerParentChain()->m_outputStorage ) + filterManagerParentChain()->m_outputStorage->leaveDirectory(); + manageIO(); // Called for the 2nd time in a row -> clean up +} + +KoFilter::ConversionStatus KoFilterChain::invokeChain() +{ + KoFilter::ConversionStatus status = KoFilter::OK; + + m_state = Beginning; + int count = m_chainLinks.count(); + + // This is needed due to nasty Microsoft design + const ChainLink* parentChainLink = 0; + if ( filterManagerParentChain() ) + parentChainLink = filterManagerParentChain()->m_chainLinks.current(); + + // No iterator here, as we need m_chainLinks.current() in outputDocument() + m_chainLinks.first(); + for ( ; count > 1 && m_chainLinks.current() && status == KoFilter::OK; + m_chainLinks.next(), --count ) { + status = m_chainLinks.current()->invokeFilter( parentChainLink ); + m_state = Middle; + manageIO(); + } + + if ( !m_chainLinks.current() ) { + kdWarning( 30500 ) << "Huh?? Found a null pointer in the chain" << endl; + return KoFilter::StupidError; + } + + if ( status == KoFilter::OK ) { + if ( m_state & Beginning ) + m_state |= End; + else + m_state = End; + status = m_chainLinks.current()->invokeFilter( parentChainLink ); + manageIO(); + } + + m_state = Done; + if (status == KoFilter::OK) + finalizeIO(); + return status; +} + +QString KoFilterChain::chainOutput() const +{ + if ( m_state == Done ) + return m_inputFile; // as we already called manageIO() + return QString::null; +} + +QString KoFilterChain::inputFile() +{ + if ( m_inputQueried == File ) + return m_inputFile; + else if ( m_inputQueried != Nil ) { + kdWarning( 30500 ) << "You already asked for some different source." << endl; + return QString::null; + } + m_inputQueried = File; + + if ( m_state & Beginning ) { + if ( static_cast<KoFilterManager::Direction>( filterManagerDirection() ) == + KoFilterManager::Import ) + m_inputFile = filterManagerImportFile(); + else + inputFileHelper( filterManagerKoDocument(), filterManagerImportFile() ); + } + else + if ( m_inputFile.isEmpty() ) + inputFileHelper( m_inputDocument, QString::null ); + + return m_inputFile; +} + +QString KoFilterChain::outputFile() +{ + // sanity check: No embedded filter should ask for a plain file + // ###### CHECK: This will break as soon as we support exporting embedding filters + if ( filterManagerParentChain() ) + kdWarning( 30500 )<< "An embedded filter has to use storageFile()!" << endl; + + if ( m_outputQueried == File ) + return m_outputFile; + else if ( m_outputQueried != Nil ) { + kdWarning( 30500 ) << "You already asked for some different destination." << endl; + return QString::null; + } + m_outputQueried = File; + + if ( m_state & End ) { + if ( static_cast<KoFilterManager::Direction>( filterManagerDirection() ) == + KoFilterManager::Import ) + outputFileHelper( false ); // This (last) one gets deleted by the caller + else + m_outputFile = filterManagerExportFile(); + } + else + outputFileHelper( true ); + + return m_outputFile; +} + +KoStoreDevice* KoFilterChain::storageFile( const QString& name, KoStore::Mode mode ) +{ + // ###### CHECK: This works only for import filters. Do we want something like + // that for export filters too? + if ( m_outputQueried == Nil && mode == KoStore::Write && filterManagerParentChain() ) + return storageInitEmbedding( name ); + + // Plain normal use case + if ( m_inputQueried == Storage && mode == KoStore::Read && + m_inputStorage && m_inputStorage->mode() == KoStore::Read ) + return storageNewStreamHelper( &m_inputStorage, &m_inputStorageDevice, name ); + else if ( m_outputQueried == Storage && mode == KoStore::Write && + m_outputStorage && m_outputStorage->mode() == KoStore::Write ) + return storageNewStreamHelper( &m_outputStorage, &m_outputStorageDevice, name ); + else if ( m_inputQueried == Nil && mode == KoStore::Read ) + return storageHelper( inputFile(), name, KoStore::Read, + &m_inputStorage, &m_inputStorageDevice ); + else if ( m_outputQueried == Nil && mode == KoStore::Write ) + return storageHelper( outputFile(), name, KoStore::Write, + &m_outputStorage, &m_outputStorageDevice ); + else { + kdWarning( 30500 ) << "Oooops, how did we get here? You already asked for a" + << " different source/destination?" << endl; + return 0; + } +} + +KoDocument* KoFilterChain::inputDocument() +{ + if ( m_inputQueried == Document ) + return m_inputDocument; + else if ( m_inputQueried != Nil ) { + kdWarning( 30500 ) << "You already asked for some different source." << endl; + return 0; + } + + if ( ( m_state & Beginning ) && + static_cast<KoFilterManager::Direction>( filterManagerDirection() ) == KoFilterManager::Export && + filterManagerKoDocument() ) + m_inputDocument = filterManagerKoDocument(); + else if ( !m_inputDocument ) + m_inputDocument = createDocument( inputFile() ); + + m_inputQueried = Document; + return m_inputDocument; +} + +KoDocument* KoFilterChain::outputDocument() +{ + // sanity check: No embedded filter should ask for a document + // ###### CHECK: This will break as soon as we support exporting embedding filters + if ( filterManagerParentChain() ) { + kdWarning( 30500 )<< "An embedded filter has to use storageFile()!" << endl; + return 0; + } + + if ( m_outputQueried == Document ) + return m_outputDocument; + else if ( m_outputQueried != Nil ) { + kdWarning( 30500 ) << "You already asked for some different destination." << endl; + return 0; + } + + if ( ( m_state & End ) && + static_cast<KoFilterManager::Direction>( filterManagerDirection() ) == KoFilterManager::Import && + filterManagerKoDocument() ) + m_outputDocument = filterManagerKoDocument(); + else + m_outputDocument = createDocument( m_chainLinks.current()->to() ); + + m_outputQueried = Document; + return m_outputDocument; +} + +void KoFilterChain::dump() const +{ + kdDebug( 30500 ) << "########## KoFilterChain with " << m_chainLinks.count() << " members:" << endl; + QPtrListIterator<ChainLink> it( m_chainLinks ); + for ( ; it.current(); ++it ) + it.current()->dump(); + kdDebug( 30500 ) << "########## KoFilterChain (done) ##########" << endl; +} + +KoFilterChain::KoFilterChain( const KoFilterManager* manager ) : + m_manager( manager ), m_state( Beginning ), m_inputStorage( 0 ), + m_inputStorageDevice( 0 ), m_outputStorage( 0 ), m_outputStorageDevice( 0 ), + m_inputDocument( 0 ), m_outputDocument( 0 ), m_inputTempFile( 0 ), + m_outputTempFile( 0 ), m_inputQueried( Nil ), m_outputQueried( Nil ), d( 0 ) +{ + // We "own" our chain links, the filter entries are implicitly shared + m_chainLinks.setAutoDelete( true ); +} + +void KoFilterChain::appendChainLink( KoFilterEntry::Ptr filterEntry, const QCString& from, const QCString& to ) +{ + m_chainLinks.append( new ChainLink( this, filterEntry, from, to ) ); +} + +void KoFilterChain::prependChainLink( KoFilterEntry::Ptr filterEntry, const QCString& from, const QCString& to ) +{ + m_chainLinks.prepend( new ChainLink( this, filterEntry, from, to ) ); +} + +void KoFilterChain::enterDirectory( const QString& directory ) +{ + // Only a little bit of checking as we (have to :} ) trust KoEmbeddingFilter + // If the output storage isn't initialized yet, we perform that step(s) on init. + if ( m_outputStorage ) + m_outputStorage->enterDirectory( directory ); + m_internalEmbeddingDirectories.append( directory ); +} + +void KoFilterChain::leaveDirectory() +{ + if ( m_outputStorage ) + m_outputStorage->leaveDirectory(); + if ( !m_internalEmbeddingDirectories.isEmpty() ) + m_internalEmbeddingDirectories.pop_back(); +} + +QString KoFilterChain::filterManagerImportFile() const +{ + return m_manager->importFile(); +} + +QString KoFilterChain::filterManagerExportFile() const +{ + return m_manager->exportFile(); +} + +KoDocument* KoFilterChain::filterManagerKoDocument() const +{ + return m_manager->document(); +} + +int KoFilterChain::filterManagerDirection() const +{ + return m_manager->direction(); +} + +KoFilterChain* const KoFilterChain::filterManagerParentChain() const +{ + return m_manager->parentChain(); +} + +void KoFilterChain::manageIO() +{ + m_inputQueried = Nil; + m_outputQueried = Nil; + + delete m_inputStorageDevice; + m_inputStorageDevice = 0; + if ( m_inputStorage ) { + m_inputStorage->close(); + delete m_inputStorage; + m_inputStorage = 0; + } + if ( m_inputTempFile ) { + m_inputTempFile->close(); + delete m_inputTempFile; // autodelete + m_inputTempFile = 0; + } + m_inputFile = QString::null; + + if ( !m_outputFile.isEmpty() ) { + m_inputFile = m_outputFile; + m_outputFile = QString::null; + m_inputTempFile = m_outputTempFile; + m_outputTempFile = 0; + + delete m_outputStorageDevice; + m_outputStorageDevice = 0; + if ( m_outputStorage ) { + m_outputStorage->close(); + // Don't delete the storage if we're just pointing to the + // storage of the parent filter chain + if ( !filterManagerParentChain() || m_outputStorage->mode() != KoStore::Write ) + delete m_outputStorage; + m_outputStorage = 0; + } + } + + if ( m_inputDocument != filterManagerKoDocument() ) + delete m_inputDocument; + m_inputDocument = m_outputDocument; + m_outputDocument = 0; +} + +void KoFilterChain::finalizeIO() +{ + // In case we export (to a file, of course) and the last + // filter chose to output a KoDocument we have to save it. + // Should be very rare, but well... + // Note: m_*input*Document as we already called manageIO() + if ( m_inputDocument && + static_cast<KoFilterManager::Direction>( filterManagerDirection() ) == KoFilterManager::Export ) { + kdDebug( 30500 ) << "Saving the output document to the export file" << endl; + m_inputDocument->saveNativeFormat( filterManagerExportFile() ); + m_inputFile = filterManagerExportFile(); + } +} + +bool KoFilterChain::createTempFile( KTempFile** tempFile, bool autoDelete ) +{ + if ( *tempFile ) { + kdError( 30500 ) << "Ooops, why is there already a temp file???" << endl; + return false; + } + *tempFile = new KTempFile(); + ( *tempFile )->setAutoDelete( autoDelete ); + return ( *tempFile )->status() == 0; +} + +void KoFilterChain::inputFileHelper( KoDocument* document, const QString& alternativeFile ) +{ + if ( document ) { + if ( !createTempFile( &m_inputTempFile ) ) { + delete m_inputTempFile; + m_inputTempFile = 0; + m_inputFile = QString::null; + return; + } + if ( !document->saveNativeFormat( m_inputTempFile->name() ) ) { + delete m_inputTempFile; + m_inputTempFile = 0; + m_inputFile = QString::null; + return; + } + m_inputFile = m_inputTempFile->name(); + } + else + m_inputFile = alternativeFile; +} + +void KoFilterChain::outputFileHelper( bool autoDelete ) +{ + if ( !createTempFile( &m_outputTempFile, autoDelete ) ) { + delete m_outputTempFile; + m_outputTempFile = 0; + m_outputFile = QString::null; + } + else + m_outputFile = m_outputTempFile->name(); +} + +KoStoreDevice* KoFilterChain::storageNewStreamHelper( KoStore** storage, KoStoreDevice** device, + const QString& name ) +{ + delete *device; + *device = 0; + if ( ( *storage )->isOpen() ) + ( *storage )->close(); + if ( ( *storage )->bad() ) + return storageCleanupHelper( storage ); + if ( !( *storage )->open( name ) ) + return 0; + + *device = new KoStoreDevice( *storage ); + return *device; +} + +KoStoreDevice* KoFilterChain::storageHelper( const QString& file, const QString& streamName, + KoStore::Mode mode, KoStore** storage, + KoStoreDevice** device ) +{ + if ( file.isEmpty() ) + return 0; + if ( *storage ) { + kdDebug( 30500 ) << "Uh-oh, we forgot to clean up..." << endl; + return 0; + } + + storageInit( file, mode, storage ); + + if ( ( *storage )->bad() ) + return storageCleanupHelper( storage ); + + // Seems that we got a valid storage, at least. Even if we can't open + // the stream the "user" asked us to open, we nontheless change the + // IOState from File to Storage, as it might be possible to open other streams + if ( mode == KoStore::Read ) + m_inputQueried = Storage; + else // KoStore::Write + m_outputQueried = Storage; + + return storageCreateFirstStream( streamName, storage, device ); +} + +void KoFilterChain::storageInit( const QString& file, KoStore::Mode mode, KoStore** storage ) +{ + QCString appIdentification( "" ); + if ( mode == KoStore::Write ) { + // To create valid storages we also have to add the mimetype + // magic "applicationIndentifier" to the storage. + // As only filters with a KOffice destination should query + // for a storage to write to, we don't check the content of + // the mimetype here. It doesn't do a lot of harm if someome + // "abuses" this method. + appIdentification = m_chainLinks.current()->to(); + } + *storage = KoStore::createStore( file, mode, appIdentification ); +} + +KoStoreDevice* KoFilterChain::storageInitEmbedding( const QString& name ) +{ + if ( m_outputStorage ) { + kdWarning( 30500 ) << "Ooops! Something's really screwed here." << endl; + return 0; + } + + m_outputStorage = filterManagerParentChain()->m_outputStorage; + + if ( !m_outputStorage ) { + // If the storage of the parent hasn't been initialized yet, + // we have to do that here. Quite nasty... + storageInit( filterManagerParentChain()->outputFile(), KoStore::Write, &m_outputStorage ); + + // transfer the ownership + filterManagerParentChain()->m_outputStorage = m_outputStorage; + filterManagerParentChain()->m_outputQueried = Storage; + } + + if ( m_outputStorage->isOpen() ) + m_outputStorage->close(); // to be on the safe side, should never happen + if ( m_outputStorage->bad() ) + return storageCleanupHelper( &m_outputStorage ); + + m_outputQueried = Storage; + + // Now that we have a storage we have to change the directory + // and remember it for later! + const int lruPartIndex = filterManagerParentChain()->m_chainLinks.current()->lruPartIndex(); + if ( lruPartIndex == -1 ) { + kdError( 30500 ) << "Huh! You want to use embedding features w/o inheriting KoEmbeddingFilter?" << endl; + return storageCleanupHelper( &m_outputStorage ); + } + + if ( !m_outputStorage->enterDirectory( QString( "part%1" ).arg( lruPartIndex ) ) ) + return storageCleanupHelper( &m_outputStorage ); + + return storageCreateFirstStream( name, &m_outputStorage, &m_outputStorageDevice ); +} + +KoStoreDevice* KoFilterChain::storageCreateFirstStream( const QString& streamName, KoStore** storage, + KoStoreDevice** device ) +{ + // Before we go and create the first stream in this storage we + // have to perform a little hack in case we're used by any ole-style + // filter which utilizes internal embedding. Ugly, but well... + if ( !m_internalEmbeddingDirectories.isEmpty() ) { + QStringList::ConstIterator it = m_internalEmbeddingDirectories.begin(); + QStringList::ConstIterator end = m_internalEmbeddingDirectories.end(); + for ( ; it != end && ( *storage )->enterDirectory( *it ); ++it ); + } + + if ( !( *storage )->open( streamName ) ) + return 0; + + if ( *device ) { + kdDebug( 30500 ) << "Uh-oh, we forgot to clean up the storage device!" << endl; + ( *storage )->close(); + return storageCleanupHelper( storage ); + } + *device = new KoStoreDevice( *storage ); + return *device; +} + +KoStoreDevice* KoFilterChain::storageCleanupHelper( KoStore** storage ) +{ + // Take care not to delete the storage of the parent chain + if ( *storage != m_outputStorage || !filterManagerParentChain() || + ( *storage )->mode() != KoStore::Write ) + delete *storage; + *storage = 0; + return 0; +} + +KoDocument* KoFilterChain::createDocument( const QString& file ) +{ + KURL url; + url.setPath( file ); + KMimeType::Ptr t = KMimeType::findByURL( url, 0, true ); + if ( t->name() == KMimeType::defaultMimeType() ) { + kdError( 30500 ) << "No mimetype found for " << file << endl; + return 0; + } + + KoDocument *doc = createDocument( QCString( t->name().latin1() ) ); + + if ( !doc || !doc->loadNativeFormat( file ) ) { + kdError( 30500 ) << "Couldn't load from the file" << endl; + delete doc; + return 0; + } + return doc; +} + +KoDocument* KoFilterChain::createDocument( const QCString& mimeType ) +{ + KoDocumentEntry entry = KoDocumentEntry::queryByMimeType(mimeType); + + if (entry.isEmpty()) + { + kdError( 30500 ) << "Couldn't find a part that can handle mimetype " << mimeType << endl; + } + + KoDocument* doc = entry.createDoc(); /*entries.first().createDoc();*/ + if ( !doc ) { + kdError( 30500 ) << "Couldn't create the document" << endl; + return 0; + } + return doc; +} + + +namespace KOffice { + + Edge::Edge( Vertex* vertex, KoFilterEntry::Ptr filterEntry ) : + m_vertex( vertex ), m_filterEntry( filterEntry ), d( 0 ) + { + } + + void Edge::relax( const Vertex* predecessor, PriorityQueue<Vertex>& queue ) + { + if ( !m_vertex || !predecessor || !m_filterEntry ) + return; + if ( m_vertex->setKey( predecessor->key() + m_filterEntry->weight ) ) { + queue.keyDecreased( m_vertex ); // maintain the heap property + m_vertex->setPredecessor( predecessor ); + } + } + + void Edge::dump( const QCString& indent ) const + { + if ( m_vertex ) + kdDebug( 30500 ) << indent << "Edge -> '" << m_vertex->mimeType() + << "' (" << m_filterEntry->weight << ")" << endl; + else + kdDebug( 30500 ) << indent << "Edge -> '(null)' (" + << m_filterEntry->weight << ")" << endl; + } + + + Vertex::Vertex( const QCString& mimeType ) : m_predecessor( 0 ), m_mimeType( mimeType ), + m_weight( UINT_MAX ), m_index( -1 ), d( 0 ) + { + m_edges.setAutoDelete( true ); // we take ownership of added edges + } + + bool Vertex::setKey( unsigned int key ) + { + if ( m_weight > key ) { + m_weight = key; + return true; + } + return false; + } + + void Vertex::reset() + { + m_weight = UINT_MAX; + m_predecessor = 0; + } + + void Vertex::addEdge( const Edge* edge ) + { + if ( !edge || edge->weight() == 0 ) + return; + m_edges.append( edge ); + } + + const Edge* Vertex::findEdge( const Vertex* vertex ) const + { + if ( !vertex ) + return 0; + const Edge* edge = 0; + QPtrListIterator<Edge> it( m_edges ); + + for ( ; it.current(); ++it ) { + if ( it.current()->vertex() == vertex && + ( !edge || it.current()->weight() < edge->weight() ) ) + edge = it.current(); + } + return edge; + } + + void Vertex::relaxVertices( PriorityQueue<Vertex>& queue ) + { + for ( Edge *e = m_edges.first(); e; e = m_edges.next() ) + e->relax( this, queue ); + } + + void Vertex::dump( const QCString& indent ) const + { + kdDebug( 30500 ) << indent << "Vertex: " << m_mimeType << " (" << m_weight << "):" << endl; + const QCString i( indent + " " ); + QPtrListIterator<Edge> it( m_edges ); + for ( ; it.current(); ++it ) + it.current()->dump( i ); + } + + + Graph::Graph( const QCString& from ) : m_vertices( 47 ), m_from( from ), + m_graphValid( false ), d( 0 ) + { + m_vertices.setAutoDelete( true ); + buildGraph(); + shortestPaths(); // Will return after a single lookup if "from" is invalid (->no check here) + } + + void Graph::setSourceMimeType( const QCString& from ) + { + if ( from == m_from ) + return; + m_from = from; + m_graphValid = false; + + // Initialize with "infinity" ... + QAsciiDictIterator<Vertex> it( m_vertices ); + for ( ; it.current(); ++it ) + it.current()->reset(); + + // ...and re-run the shortest path search for the new source mime + shortestPaths(); + } + + KoFilterChain::Ptr Graph::chain( const KoFilterManager* manager, QCString& to ) const + { + if ( !isValid() || !manager ) + return 0; + + if ( to.isEmpty() ) { // if the destination is empty we search the closest KOffice part + to = findKOfficePart(); + if ( to.isEmpty() ) // still empty? strange stuff... + return 0; + } + + const Vertex* vertex = m_vertices[ to ]; + if ( !vertex || vertex->key() == UINT_MAX ) + return 0; + + KoFilterChain::Ptr ret = new KoFilterChain( manager ); + + // Fill the filter chain with all filters on the path + const Vertex* tmp = vertex->predecessor(); + while ( tmp ) { + const Edge* const edge = tmp->findEdge( vertex ); + Q_ASSERT( edge ); + ret->prependChainLink( edge->filterEntry(), tmp->mimeType(), vertex->mimeType() ); + vertex = tmp; + tmp = tmp->predecessor(); + } + return ret; + } + + void Graph::dump() const + { + kdDebug( 30500 ) << "+++++++++ Graph::dump +++++++++" << endl; + kdDebug( 30500 ) << "From: " << m_from << endl; + QAsciiDictIterator<Vertex> it( m_vertices ); + for ( ; it.current(); ++it ) + it.current()->dump( " " ); + kdDebug( 30500 ) << "+++++++++ Graph::dump (done) +++++++++" << endl; + } + + // Query the trader and create the vertices and edges representing + // available mime types and filters. + void Graph::buildGraph() + { + // Make sure that all available parts are added to the graph + QValueList<KoDocumentEntry> parts( KoDocumentEntry::query() ); + QValueList<KoDocumentEntry>::ConstIterator partIt( parts.begin() ); + QValueList<KoDocumentEntry>::ConstIterator partEnd( parts.end() ); + + while ( partIt != partEnd ) { + QStringList nativeMimeTypes = ( *partIt ).service()->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); + nativeMimeTypes += ( *partIt ).service()->property( "X-KDE-NativeMimeType" ).toString(); + QStringList::ConstIterator it = nativeMimeTypes.begin(); + QStringList::ConstIterator end = nativeMimeTypes.end(); + for ( ; it != end; ++it ) + if ( !(*it).isEmpty() ) + m_vertices.insert( (*it).latin1(), new Vertex( (*it).latin1() ) ); + ++partIt; + } + + QValueList<KoFilterEntry::Ptr> filters( KoFilterEntry::query() ); // no constraint here - we want *all* :) + QValueList<KoFilterEntry::Ptr>::ConstIterator it = filters.begin(); + QValueList<KoFilterEntry::Ptr>::ConstIterator end = filters.end(); + + for ( ; it != end; ++it ) { + // First add the "starting points" to the dict + QStringList::ConstIterator importIt = ( *it )->import.begin(); + QStringList::ConstIterator importEnd = ( *it )->import.end(); + for ( ; importIt != importEnd; ++importIt ) { + const QCString key = ( *importIt ).latin1(); // latin1 is okay here (werner) + // already there? + if ( !m_vertices[ key ] ) + m_vertices.insert( key, new Vertex( key ) ); + } + + // Are we allowed to use this filter at all? + if ( KoFilterManager::filterAvailable( *it ) ) { + QStringList::ConstIterator exportIt = ( *it )->export_.begin(); + QStringList::ConstIterator exportEnd = ( *it )->export_.end(); + + for ( ; exportIt != exportEnd; ++exportIt ) { + // First make sure the export vertex is in place + const QCString key = ( *exportIt ).latin1(); // latin1 is okay here + Vertex* exp = m_vertices[ key ]; + if ( !exp ) { + exp = new Vertex( key ); + m_vertices.insert( key, exp ); + } + // Then create the appropriate edges + importIt = ( *it )->import.begin(); + for ( ; importIt != importEnd; ++importIt ) + m_vertices[ ( *importIt ).latin1() ]->addEdge( new Edge( exp, *it ) ); + } + } + else + kdDebug( 30500 ) << "Filter: " << ( *it )->service()->name() << " doesn't apply." << endl; + } + } + + // As all edges (=filters) are required to have a positive weight + // we can use Dijkstra's shortest path algorithm from Cormen's + // "Introduction to Algorithms" (p. 527) + // Note: I did some adaptions as our data structures are slightly + // different from the ones used in the book. Further we simply stop + // the algorithm is we don't find any node with a weight != Infinity + // (==UINT_MAX), as this means that the remaining nodes in the queue + // aren't connected anyway. + void Graph::shortestPaths() + { + // Is the requested start mime type valid? + Vertex* from = m_vertices[ m_from ]; + if ( !from ) + return; + + // Inititalize start vertex + from->setKey( 0 ); + + // Fill the priority queue with all the vertices + PriorityQueue<Vertex> queue( m_vertices ); + + while ( !queue.isEmpty() ) { + Vertex *min = queue.extractMinimum(); + // Did we already relax all connected vertices? + if ( min->key() == UINT_MAX ) + break; + min->relaxVertices( queue ); + } + m_graphValid = true; + } + + QCString Graph::findKOfficePart() const + { + // Here we simply try to find the closest KOffice mimetype + QValueList<KoDocumentEntry> parts( KoDocumentEntry::query() ); + QValueList<KoDocumentEntry>::ConstIterator partIt( parts.begin() ); + QValueList<KoDocumentEntry>::ConstIterator partEnd( parts.end() ); + + const Vertex *v = 0; + + // Be sure that v gets initialized correctly + while ( !v && partIt != partEnd ) { + QStringList nativeMimeTypes = ( *partIt ).service()->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); + nativeMimeTypes += ( *partIt ).service()->property( "X-KDE-NativeMimeType" ).toString(); + QStringList::ConstIterator it = nativeMimeTypes.begin(); + QStringList::ConstIterator end = nativeMimeTypes.end(); + for ( ; !v && it != end; ++it ) + if ( !(*it).isEmpty() ) + v = m_vertices[ ( *it ).latin1() ]; + ++partIt; + } + if ( !v ) + return ""; + + // Now we try to find the "cheapest" KOffice vertex + while ( partIt != partEnd ) { + QStringList nativeMimeTypes = ( *partIt ).service()->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); + nativeMimeTypes += ( *partIt ).service()->property( "X-KDE-NativeMimeType" ).toString(); + QStringList::ConstIterator it = nativeMimeTypes.begin(); + QStringList::ConstIterator end = nativeMimeTypes.end(); + for ( ; !v && it != end; ++it ) { + QString key = *it; + if ( !key.isEmpty() ) { + Vertex* tmp = m_vertices[ key.latin1() ]; + if ( !v || ( tmp && tmp->key() < v->key() ) ) + v = tmp; + } + } + ++partIt; + } + + // It seems it already is a KOffice part + if ( v->key() == 0 ) + return ""; + + return v->mimeType(); + } + +} // namespace KOffice diff --git a/lib/kofficecore/KoFilterChain.h b/lib/kofficecore/KoFilterChain.h new file mode 100644 index 00000000..5b8dd3dd --- /dev/null +++ b/lib/kofficecore/KoFilterChain.h @@ -0,0 +1,410 @@ +/* This file is part of the KOffice libraries + Copyright (C) 2001 Werner Trobin <trobin@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 version 2 as published by the Free Software Foundation. + + 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. +*/ + +#ifndef __koffice_filter_chain_h__ +#define __koffice_filter_chain_h__ + +#include <qcstring.h> +#include <qasciidict.h> +#include <qptrlist.h> +#include <qstringlist.h> + +#include <KoFilter.h> +#include <KoQueryTrader.h> +#include <KoStoreDevice.h> +#include <koffice_export.h> + +class KTempFile; +class KoFilterManager; +class KoDocument; +class QStrList; + +namespace KOffice { + class Graph; +} + +/** + * @brief This class represents a chain of plain @ref KoFilter instances. + * + * Instances of this class are shared, so please just hold + * KoFilterChain::Ptr pointers to it. + * + * @author Werner Trobin <trobin@kde.org> + * @todo the class has no constructor and therefore cannot initialize its private class + */ +class KOFFICECORE_EXPORT KoFilterChain : public KShared +{ + // Only KOffice::Graph is allowed to construct instances and + // add chain links. + friend class KOffice::Graph; + friend class KoFilterManager; + +public: + typedef KSharedPtr<KoFilterChain> Ptr; + + virtual ~KoFilterChain(); + + /** + * The filter manager returned may be 0! + */ + const KoFilterManager* manager() const { return m_manager; } + + /** + * Starts the filtering process. + * @return The return status of the conversion. KoFilter::OK + * if everything is alright. + */ + KoFilter::ConversionStatus invokeChain(); + + /** + * Tells the @ref KoFilterManager the output file of the + * filter chain in case of an import operation. If it's + * QString::null we directly manipulated the document. + */ + QString chainOutput() const; + + /** + * Get the current file to read from. This part of the API + * is for the filters in our chain. + */ + QString inputFile(); + /** + * Get the current file to write to. This part of the API + * is for the filters in our chain. + */ + QString outputFile(); + + /** + * Get a file from a storage. May return 0! + * This part of the API is for the filters in our chain. + * If you call it multiple times with the same stream name + * the stream will be closed and re-opened. + * Note: @em Don't delete that @ref KoStoreDevice we return. + * @param name The name of the stream inside the storage + * @param mode Whether we want to read or write from/to the stream + * @return The storage device to access the stream. May be 0! + */ + KoStoreDevice* storageFile( const QString& name = "root", KoStore::Mode mode = KoStore::Read ); + + /** + * This method allows your filter to work directly on the + * @ref KoDocument of the application. + * This part of the API is for the filters in our chain. + * @return The document containing the data. May return 0 on error. + */ + KoDocument* inputDocument(); + /** + * This method allows your filter to work directly on the + * @ref KoDocument of the application. + * This part of the API is for the filters in our chain. + * @return The document you have to write to. May return 0 on error. + */ + KoDocument* outputDocument(); + + + // debugging + void dump() const; + +private: + // ### API for KOffice::Graph: + // Construct a filter chain belonging to some KoFilterManager. + // The parent filter manager may be 0. + KoFilterChain( const KoFilterManager* manager ); + + void appendChainLink( KoFilterEntry::Ptr filterEntry, const QCString& from, const QCString& to ); + void prependChainLink( KoFilterEntry::Ptr filterEntry, const QCString& from, const QCString& to ); + + // ### API for KoEmbeddingFilter + // This is needed as the embedding filter might have to influence + // the way we change directories (e.g. in the olefilter case) + // The ugly friend methods are needed, but I'd welcome and suggestions for + // better design :} + friend void KoEmbeddingFilter::filterChainEnterDirectory( const QString& directory ) const; + void enterDirectory( const QString& directory ); + friend void KoEmbeddingFilter::filterChainLeaveDirectory() const; + void leaveDirectory(); + + // These methods are friends of KoFilterManager and provide access + // to a private part of its API. As I don't want to include + // koFilterManager.h in this header the direction is "int" here. + QString filterManagerImportFile() const; + QString filterManagerExportFile() const; + KoDocument* filterManagerKoDocument() const; + int filterManagerDirection() const; + KoFilterChain* const filterManagerParentChain() const; + + + // Helper methods which keep track of all the temp files, documents, + // storages,... and properly delete them as soon as they are not + // needed anymore. + void manageIO(); + void finalizeIO(); + + bool createTempFile( KTempFile** tempFile, bool autoDelete = true ); + + void inputFileHelper( KoDocument* document, const QString& alternativeFile ); + void outputFileHelper( bool autoDelete ); + KoStoreDevice* storageNewStreamHelper( KoStore** storage, KoStoreDevice** device, const QString& name ); + KoStoreDevice* storageHelper( const QString& file, const QString& streamName, + KoStore::Mode mode, KoStore** storage, KoStoreDevice** device ); + void storageInit( const QString& file, KoStore::Mode mode, KoStore** storage ); + KoStoreDevice* storageInitEmbedding( const QString& name ); + KoStoreDevice* storageCreateFirstStream( const QString& streamName, KoStore** storage, KoStoreDevice** device ); + KoStoreDevice* storageCleanupHelper( KoStore** storage ); + + KoDocument* createDocument( const QString& file ); + KoDocument* createDocument( const QCString& mimeType ); + + /** + * A small private helper class with represents one single filter + * (one link of the chain) + * @internal + */ + class ChainLink + { + + public: + ChainLink( KoFilterChain* chain, KoFilterEntry::Ptr filterEntry, + const QCString& from, const QCString& to ); + + KoFilter::ConversionStatus invokeFilter( const ChainLink* const parentChainLink ); + + QCString from() const { return m_from; } + QCString to() const { return m_to; } + + // debugging + void dump() const; + + // This hack is only needed due to crappy Microsoft design and + // circular dependencies in their embedded files :} + int lruPartIndex() const; + + private: + ChainLink( const ChainLink& rhs ); + ChainLink& operator=( const ChainLink& rhs ); + + void setupCommunication( const KoFilter* const parentFilter ) const; + void setupConnections( const KoFilter* sender, const QStrList& sigs, + const KoFilter* receiver, const QStrList& sl0ts ) const; + + KoFilterChain* m_chain; + KoFilterEntry::Ptr m_filterEntry; + QCString m_from, m_to; + + // This hack is only needed due to crappy Microsoft design and + // circular dependencies in their embedded files :} + KoFilter* m_filter; + + class Private; + Private* d; + }; + + // "A whole is that which has beginning, middle, and end" - Aristotle + // ...but we also need to signal "Done" state, Mr. Aristotle + enum Whole { Beginning = 1, Middle = 2, End = 4, Done = 8 }; + + // Don't copy or assign filter chains + KoFilterChain( const KoFilterChain& rhs ); + KoFilterChain& operator=( const KoFilterChain& rhs ); + + const KoFilterManager* const m_manager; + QPtrList<ChainLink> m_chainLinks; + + // stuff needed for bookkeeping + int m_state; + + QString m_inputFile; // Did we pass around plain files? + QString m_outputFile; + + KoStore* m_inputStorage; // ...or was it a storage+device? + KoStoreDevice* m_inputStorageDevice; + KoStore* m_outputStorage; + KoStoreDevice* m_outputStorageDevice; + + KoDocument* m_inputDocument; // ...or even documents? + KoDocument* m_outputDocument; + + KTempFile* m_inputTempFile; + KTempFile* m_outputTempFile; + + // These two flags keep track of the input/output the + // filter (=user) asked for + enum IOState { Nil, File, Storage, Document }; + IOState m_inputQueried, m_outputQueried; + + // This stack keeps track of directories we have to enter and + // leave due to internal embedding a la OLE filters. This serves + // as a kind of "memory" even if we didn't initialize the store yet. + // I know that it's ugly, and I'll try to clean up that hack + // sooner or later (Werner) + QStringList m_internalEmbeddingDirectories; + + class Private; + Private* d; +}; + + +// As we use quite generic classnames... +namespace KOffice +{ + class Vertex; + template<class T> class PriorityQueue; + + /** + * An internal class representing a filter (=edge) in the filter graph. + * @internal + */ + class Edge + { + + public: + // creates a new edge to "vertex" with the given weight. + Edge( Vertex* vertex, KoFilterEntry::Ptr filterEntry ); + ~Edge() {} + + unsigned int weight() const { return m_filterEntry ? m_filterEntry->weight : 0; } + KoFilterEntry::Ptr filterEntry() const { return m_filterEntry; } + const Vertex* vertex() const { return m_vertex; } + + // Relaxes the "connected" vertex (i.e. the weight of the + // connected vertex = "predec.->key()" (parameter) + weight of this edge + // As this will only be called once we calculate the weight + // of the edge "on the fly" + // Note: We have to pass the queue as we have to call keyDecreased :} + void relax( const Vertex* predecessor, PriorityQueue<Vertex>& queue ); + + // debugging + void dump( const QCString& indent ) const; + + private: + Edge( const Edge& rhs ); + Edge& operator=( const Edge& rhs ); + + Vertex* m_vertex; + KoFilterEntry::Ptr m_filterEntry; + + class Private; + Private* d; + }; + + + /** + * An internal class representing a mime type (=node, vertex) in the filter graph. + * @internal + */ + class Vertex + { + + public: + Vertex( const QCString& mimeType ); + ~Vertex() {} + + QCString mimeType() const { return m_mimeType; } + + // Current "weight" of the vertex - will be "relaxed" when + // running the shortest path algorithm. Returns true if it + // really has been "relaxed" + bool setKey( unsigned int key ); + unsigned int key() const { return m_weight; } + // Can be used to set the key back to "Infinity" (UINT_MAX) + // and reset the predecessor of this vertex + void reset(); + + // Position in the heap, needed for a fast keyDecreased operation + void setIndex( int index ) { m_index=index; } + int index() const { return m_index; } + + // predecessor on the way from the source to the destination, + // needed for the shortest path algorithm + void setPredecessor( const Vertex* predecessor ) { m_predecessor=predecessor; } + const Vertex* predecessor() const { return m_predecessor; } + + // Adds an outgoing edge to the vertex, transfers ownership + void addEdge( const Edge* edge ); + // Finds the lightest(!) edge pointing to the given vertex, if any (0 if not found) + // This means it will always search the whole list of edges + const Edge* findEdge( const Vertex* vertex ) const; + + // This method is called when we need to relax all "our" edges. + // We need to pass the queue as we have to notify it about key changes - ugly :( + void relaxVertices( PriorityQueue<Vertex>& queue ); + + // debugging + void dump( const QCString& indent ) const; + + private: + Vertex( const Vertex& rhs ); + Vertex& operator=( const Vertex& rhs ); + + QPtrList<Edge> m_edges; + const Vertex* m_predecessor; + QCString m_mimeType; + unsigned int m_weight; // "key" inside the queue + int m_index; // position inside the queue, needed for a fast keyDecreased() + + class Private; + Private* d; + }; + + + /** + * The main worker behind the scenes. Manages the creation of the graph, + * processing the information in it, and creating the filter chains. + * @internal + */ + class Graph + { + + public: + Graph( const QCString& from ); + ~Graph() {} + + bool isValid() const { return m_graphValid; } + + QCString sourceMimeType() const { return m_from; } + void setSourceMimeType( const QCString& from ); + + // Creates a chain from "from" to the "to" mimetype + // If the "to" mimetype isEmpty() then we try to find the + // closest KOffice mimetype and use that as destination. + // After such a search "to" will contain the dest. mimetype (return value) + // if the search was successful. Might return 0! + KoFilterChain::Ptr chain( const KoFilterManager* manager, QCString& to ) const; + + // debugging + void dump() const; + + private: + Graph( const Graph& rhs ); + Graph& operator=( const Graph& rhs ); + + void buildGraph(); + void shortestPaths(); + QCString findKOfficePart() const; + + QAsciiDict<Vertex> m_vertices; + QCString m_from; + bool m_graphValid; + + class Private; + Private* d; + }; + +} // namespace KOffice + +#endif // __koffice_filter_chain_h__ diff --git a/lib/kofficecore/KoFilterManager.cpp b/lib/kofficecore/KoFilterManager.cpp new file mode 100644 index 00000000..f3db5c73 --- /dev/null +++ b/lib/kofficecore/KoFilterManager.cpp @@ -0,0 +1,602 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + 2000, 2001 Werner Trobin <trobin@kde.org> + Copyright (C) 2004 Nicolas Goutte <goutte@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 <KoFilterManager.h> +#include <KoFilterManager_p.h> + +#include <qfile.h> +#include <qlabel.h> +#include <qlayout.h> +#include <qptrlist.h> +#include <qapplication.h> + +#include <klocale.h> +#include <kmessagebox.h> +#include <KoDocument.h> +#include <klibloader.h> +#include <klistbox.h> +#include <kmimetype.h> +#include <kdebug.h> + +#include <queue> + +#include <unistd.h> + +class KoFilterManager::Private +{ +public: + bool m_batch; +}; + + +KoFilterChooser::KoFilterChooser (QWidget *parent, const QStringList &mimeTypes, const QString &nativeFormat) + : KDialogBase (parent, "kofilterchooser", true, i18n ("Choose Filter"), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok, true), + m_mimeTypes (mimeTypes) +{ + setInitialSize (QSize (300, 350)); + + QWidget *page = new QWidget (this); + setMainWidget (page); + + // looks too squashed together without * 2 + QVBoxLayout *layout = new QVBoxLayout (page, marginHint (), spacingHint () * 2); + + QLabel *filterLabel = new QLabel (i18n ("Select a filter:"), page, "filterlabel"); + layout->addWidget (filterLabel); + + m_filterList = new KListBox (page, "filterlist"); + layout->addWidget (m_filterList); + + Q_ASSERT (!m_mimeTypes.isEmpty ()); + for (QStringList::ConstIterator it = m_mimeTypes.begin (); + it != m_mimeTypes.end (); + it++) + { + KMimeType::Ptr mime = KMimeType::mimeType (*it); + m_filterList->insertItem (mime->comment ()); + } + + if (nativeFormat == "application/x-kword") + { + const int index = m_mimeTypes.findIndex ("text/plain"); + if (index > -1) + m_filterList->setCurrentItem (index); + } + + if (m_filterList->currentItem () == -1) + m_filterList->setCurrentItem (0); + + m_filterList->centerCurrentItem (); + m_filterList->setFocus (); + + connect (m_filterList, SIGNAL (selected (int)), this, SLOT (slotOk ())); +} + +KoFilterChooser::~KoFilterChooser () +{ +} + +QString KoFilterChooser::filterSelected () +{ + const int item = m_filterList->currentItem (); + + if (item > -1) + return m_mimeTypes [item]; + else + return QString::null; +} + + +// static cache for filter availability +QMap<QString, bool> KoFilterManager::m_filterAvailable; + +const int KoFilterManager::s_area = 30500; + + +KoFilterManager::KoFilterManager( KoDocument* document ) : + m_document( document ), m_parentChain( 0 ), m_graph( "" ), d( 0 ) +{ + d = new KoFilterManager::Private; + d -> m_batch = false; + if ( document ) + QObject::connect( this, SIGNAL( sigProgress( int ) ), + document, SIGNAL( sigProgress( int ) ) ); +} + + +KoFilterManager::KoFilterManager( const QString& url, const QCString& mimetypeHint, + KoFilterChain* const parentChain ) : + m_document( 0 ), m_parentChain( parentChain ), m_importUrl( url ), m_importUrlMimetypeHint( mimetypeHint ), + m_graph( "" ), d( 0 ) +{ + d = new KoFilterManager::Private; + d -> m_batch = false; +} + +KoFilterManager::~KoFilterManager() +{ + delete d; +} + +QString KoFilterManager::import( const QString& url, KoFilter::ConversionStatus& status ) +{ + // Find the mime type for the file to be imported. + KURL u; + u.setPath( url ); + KMimeType::Ptr t = KMimeType::findByURL( u, 0, true ); + + m_graph.setSourceMimeType( t->name().latin1() ); // .latin1() is okay here (Werner) + if ( !m_graph.isValid() ) { + bool userCancelled = false; + + kdWarning(s_area) << "Can't open " << t->name () << ", trying filter chooser" << endl; + if ( m_document ) + { + if ( !m_document->isAutoErrorHandlingEnabled() ) + { + status = KoFilter::BadConversionGraph; + return QString::null; + } + QCString nativeFormat = m_document->nativeFormatMimeType (); + + QApplication::setOverrideCursor( arrowCursor ); + KoFilterChooser chooser(0, + KoFilterManager::mimeFilter (nativeFormat, KoFilterManager::Import, m_document->extraNativeMimeTypes()), + nativeFormat); + if (chooser.exec ()) + { + QCString f = chooser.filterSelected ().latin1(); + + if (f == nativeFormat) + { + status = KoFilter::OK; + QApplication::restoreOverrideCursor(); + return url; + } + + m_graph.setSourceMimeType (f); + } + else + userCancelled = true; + QApplication::restoreOverrideCursor(); + } + + if (!m_graph.isValid()) + { + kdError(s_area) << "Couldn't create a valid graph for this source mimetype: " + << t->name() << endl; + importErrorHelper( t->name(), userCancelled ); + status = KoFilter::BadConversionGraph; + return QString::null; + } + } + + KoFilterChain::Ptr chain( 0 ); + // Are we owned by a KoDocument? + if ( m_document ) { + QCString mimeType = m_document->nativeFormatMimeType(); + QStringList extraMimes = m_document->extraNativeMimeTypes(); + int i=0, n = extraMimes.count(); + chain = m_graph.chain( this, mimeType ); + while( !chain && i<n) { + mimeType = extraMimes[i].utf8(); + chain = m_graph.chain( this, mimeType ); + ++i; + } + } + else { + kdError(s_area) << "You aren't supposed to use import() from a filter!" << endl; + status = KoFilter::UsageError; + return QString::null; + } + + if ( !chain ) { + kdError(s_area) << "Couldn't create a valid filter chain!" << endl; + importErrorHelper( t->name() ); + status = KoFilter::BadConversionGraph; + return QString::null; + } + + // Okay, let's invoke the filters one after the other + m_direction = Import; // vital information! + m_importUrl = url; // We want to load that file + m_exportUrl = QString::null; // This is null for sure, as embedded stuff isn't + // allowed to use that method + status = chain->invokeChain(); + + m_importUrl = QString::null; // Reset the import URL + + if ( status == KoFilter::OK ) + return chain->chainOutput(); + return QString::null; +} + +KoFilter::ConversionStatus KoFilterManager::exp0rt( const QString& url, QCString& mimeType ) +{ + bool userCancelled = false; + + // The import url should already be set correctly (null if we have a KoDocument + // file manager and to the correct URL if we have an embedded manager) + m_direction = Export; // vital information! + m_exportUrl = url; + + KoFilterChain::Ptr chain = 0; + if ( m_document ) { + // We have to pick the right native mimetype as source. + QStringList nativeMimeTypes; + nativeMimeTypes.append( m_document->nativeFormatMimeType() ); + nativeMimeTypes += m_document->extraNativeMimeTypes(); + QStringList::ConstIterator it = nativeMimeTypes.begin(); + const QStringList::ConstIterator end = nativeMimeTypes.end(); + for ( ; !chain && it != end; ++it ) + { + m_graph.setSourceMimeType( (*it).latin1() ); + if ( m_graph.isValid() ) + chain = m_graph.chain( this, mimeType ); + } + } + else if ( !m_importUrlMimetypeHint.isEmpty() ) { + kdDebug(s_area) << "Using the mimetype hint: '" << m_importUrlMimetypeHint << "'" << endl; + m_graph.setSourceMimeType( m_importUrlMimetypeHint ); + } + else { + KURL u; + u.setPath( m_importUrl ); + KMimeType::Ptr t = KMimeType::findByURL( u, 0, true ); + if ( t->name() == KMimeType::defaultMimeType() ) { + kdError(s_area) << "No mimetype found for " << m_importUrl << endl; + return KoFilter::BadMimeType; + } + m_graph.setSourceMimeType( t->name().latin1() ); + + if ( !m_graph.isValid() ) { + kdWarning(s_area) << "Can't open " << t->name () << ", trying filter chooser" << endl; + + QApplication::setOverrideCursor( arrowCursor ); + KoFilterChooser chooser(0, KoFilterManager::mimeFilter ()); + if (chooser.exec ()) + m_graph.setSourceMimeType (chooser.filterSelected ().latin1 ()); + else + userCancelled = true; + + QApplication::restoreOverrideCursor(); + } + } + + if (!m_graph.isValid ()) + { + kdError(s_area) << "Couldn't create a valid graph for this source mimetype." << endl; + if (!userCancelled) KMessageBox::error( 0L, i18n("Could not export file."), i18n("Missing Export Filter") ); + return KoFilter::BadConversionGraph; + } + + if ( !chain ) // already set when coming from the m_document case + chain = m_graph.chain( this, mimeType ); + + if ( !chain ) { + kdError(s_area) << "Couldn't create a valid filter chain to " << mimeType << " !" << endl; + KMessageBox::error( 0L, i18n("Could not export file."), i18n("Missing Export Filter") ); + return KoFilter::BadConversionGraph; + } + + return chain->invokeChain(); +} + +namespace // in order not to mess with the global namespace ;) +{ + // This class is needed only for the static mimeFilter method + class Vertex + { + public: + Vertex( const QCString& mimeType ) : m_color( White ), m_mimeType( mimeType ) {} + + enum Color { White, Gray, Black }; + Color color() const { return m_color; } + void setColor( Color color ) { m_color = color; } + + QCString mimeType() const { return m_mimeType; } + + void addEdge( Vertex* vertex ) { if ( vertex ) m_edges.append( vertex ); } + QPtrList<Vertex> edges() const { return m_edges; } + + private: + Color m_color; + QCString m_mimeType; + QPtrList<Vertex> m_edges; + }; + + // Some helper methods for the static stuff + // This method builds up the graph in the passed ascii dict + void buildGraph( QAsciiDict<Vertex>& vertices, KoFilterManager::Direction direction ) + { + QStringList stopList; // Lists of mimetypes that are considered end of chains + stopList << "text/plain"; + stopList << "text/csv"; + stopList << "text/x-tex"; + stopList << "text/html"; + + vertices.setAutoDelete( true ); + + // partly copied from build graph, but I don't see any other + // way without crude hacks, as we have to obey the direction here + QValueList<KoDocumentEntry> parts( KoDocumentEntry::query(false, QString::null) ); + QValueList<KoDocumentEntry>::ConstIterator partIt( parts.begin() ); + QValueList<KoDocumentEntry>::ConstIterator partEnd( parts.end() ); + + while ( partIt != partEnd ) { + QStringList nativeMimeTypes = ( *partIt ).service()->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); + nativeMimeTypes += ( *partIt ).service()->property( "X-KDE-NativeMimeType" ).toString(); + QStringList::ConstIterator it = nativeMimeTypes.begin(); + const QStringList::ConstIterator end = nativeMimeTypes.end(); + for ( ; it != end; ++it ) + if ( !(*it).isEmpty() ) + vertices.insert( (*it).latin1(), new Vertex( (*it).latin1() ) ); + ++partIt; + } + + QValueList<KoFilterEntry::Ptr> filters = KoFilterEntry::query(); // no constraint here - we want *all* :) + QValueList<KoFilterEntry::Ptr>::ConstIterator it = filters.begin(); + const QValueList<KoFilterEntry::Ptr>::ConstIterator end = filters.end(); + + for ( ; it != end; ++it ) { + + QStringList impList; // Import list + QStringList expList; // Export list + + const QStringList::Iterator stopEnd = stopList.end(); + // Now we have to exclude the "stop" mimetypes (in the right direction!) + if ( direction == KoFilterManager::Import ) { + // Import: "stop" mime type should not appear in export + QStringList::ConstIterator testIt = ( *it )->export_.begin(); + QStringList::ConstIterator testEnd = ( *it )->export_.end(); + for ( ; testIt != testEnd ; ++testIt ) { + if ( stopList.find( *testIt ) == stopEnd ) { + expList.append( *testIt ); + } + } + impList = ( *it )->import; + } + else { + // Export: "stop" mime type should not appear in import + QStringList::ConstIterator testIt = ( *it )->import.begin(); + const QStringList::ConstIterator testEnd = ( *it )->import.end(); + for ( ; testIt != testEnd ; ++testIt ) { + if ( stopList.find( *testIt ) == stopEnd ) { + impList.append( *testIt ); + } + } + expList = ( *it )->export_; + } + + if ( impList.empty() || expList.empty() ) + { + // This filter cannot be used under these conditions + kdDebug( 30500 ) << "Filter: " << ( *it )->service()->name() << " ruled out" << endl; + continue; + } + + // First add the "starting points" to the dict + QStringList::ConstIterator importIt = impList.begin(); + const QStringList::ConstIterator importEnd = impList.end(); + for ( ; importIt != importEnd; ++importIt ) { + const QCString key = ( *importIt ).latin1(); // latin1 is okay here (werner) + // already there? + if ( !vertices[ key ] ) + vertices.insert( key, new Vertex( key ) ); + } + + // Are we allowed to use this filter at all? + if ( KoFilterManager::filterAvailable( *it ) ) { + QStringList::ConstIterator exportIt = expList.begin(); + const QStringList::ConstIterator exportEnd = expList.end(); + + for ( ; exportIt != exportEnd; ++exportIt ) { + // First make sure the export vertex is in place + const QCString key = ( *exportIt ).latin1(); // latin1 is okay here + Vertex* exp = vertices[ key ]; + if ( !exp ) { + exp = new Vertex( key ); + vertices.insert( key, exp ); + } + // Then create the appropriate edges depending on the + // direction (import/export) + // This is the chunk of code which actually differs from the + // graph stuff (apart from the different vertex class) + importIt = impList.begin(); // ### TODO: why only the first one? + if ( direction == KoFilterManager::Import ) { + for ( ; importIt != importEnd; ++importIt ) + exp->addEdge( vertices[ ( *importIt ).latin1() ] ); + } else { + for ( ; importIt != importEnd; ++importIt ) + vertices[ ( *importIt ).latin1() ]->addEdge( exp ); + } + } + } + else + kdDebug( 30500 ) << "Filter: " << ( *it )->service()->name() << " does not apply." << endl; + } + } + + // This method runs a BFS on the graph to determine the connected + // nodes. Make sure that the graph is "cleared" (the colors of the + // nodes are all white) + QStringList connected( const QAsciiDict<Vertex>& vertices, const QCString& mimetype ) + { + if ( mimetype.isEmpty() ) + return QStringList(); + Vertex *v = vertices[ mimetype ]; + if ( !v ) + return QStringList(); + + v->setColor( Vertex::Gray ); + std::queue<Vertex*> queue; + queue.push( v ); + QStringList connected; + + while ( !queue.empty() ) { + v = queue.front(); + queue.pop(); + QPtrList<Vertex> edges = v->edges(); + QPtrListIterator<Vertex> it( edges ); + for ( ; it.current(); ++it ) { + if ( it.current()->color() == Vertex::White ) { + it.current()->setColor( Vertex::Gray ); + queue.push( it.current() ); + } + } + v->setColor( Vertex::Black ); + connected.append( v->mimeType() ); + } + return connected; + } +} + +// The static method to figure out to which parts of the +// graph this mimetype has a connection to. +QStringList KoFilterManager::mimeFilter( const QCString& mimetype, Direction direction, const QStringList& extraNativeMimeTypes ) +{ + //kdDebug(s_area) << "mimetype=" << mimetype << " extraNativeMimeTypes=" << extraNativeMimeTypes << endl; + QAsciiDict<Vertex> vertices; + buildGraph( vertices, direction ); + + // TODO maybe use the fake vertex trick from the method below, to make the search faster? + + QStringList nativeMimeTypes; + nativeMimeTypes.append( QString::fromLatin1( mimetype ) ); + nativeMimeTypes += extraNativeMimeTypes; + + // Add the native mimetypes first so that they are on top. + QStringList lst = nativeMimeTypes; + + // Now look for filters which output each of those natives mimetypes + for( QStringList::ConstIterator natit = nativeMimeTypes.begin(); natit != nativeMimeTypes.end(); ++natit ) { + const QStringList outMimes = connected( vertices, (*natit).latin1() ); + //kdDebug(s_area) << k_funcinfo << "output formats connected to mime " << *natit << " : " << outMimes << endl; + for ( QStringList::ConstIterator mit = outMimes.begin(); mit != outMimes.end(); ++mit ) + if ( lst.find( *mit ) == lst.end() ) // append only if not there already. Qt4: QSet<QString>? + lst.append( *mit ); + } + return lst; +} + +QStringList KoFilterManager::mimeFilter() +{ + QAsciiDict<Vertex> vertices; + buildGraph( vertices, KoFilterManager::Import ); + + QValueList<KoDocumentEntry> parts( KoDocumentEntry::query(false, QString::null) ); + QValueList<KoDocumentEntry>::ConstIterator partIt( parts.begin() ); + QValueList<KoDocumentEntry>::ConstIterator partEnd( parts.end() ); + + if ( partIt == partEnd ) + return QStringList(); + + // To find *all* reachable mimetypes, we have to resort to + // a small hat trick, in order to avoid multiple searches: + // We introduce a fake vertex, which is connected to every + // single KOffice mimetype. Due to that one BFS is enough :) + // Now we just need an... ehrm.. unique name for our fake mimetype + Vertex *v = new Vertex( "supercalifragilistic/x-pialadocious" ); + vertices.insert( "supercalifragilistic/x-pialadocious", v ); + while ( partIt != partEnd ) { + QStringList nativeMimeTypes = ( *partIt ).service()->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); + nativeMimeTypes += ( *partIt ).service()->property( "X-KDE-NativeMimeType" ).toString(); + QStringList::ConstIterator it = nativeMimeTypes.begin(); + const QStringList::ConstIterator end = nativeMimeTypes.end(); + for ( ; it != end; ++it ) + if ( !(*it).isEmpty() ) + v->addEdge( vertices[ (*it).latin1() ] ); + ++partIt; + } + QStringList result = connected( vertices, "supercalifragilistic/x-pialadocious" ); + + // Finally we have to get rid of our fake mimetype again + result.remove( "supercalifragilistic/x-pialadocious" ); + return result; +} + +// Here we check whether the filter is available. This stuff is quite slow, +// but I don't see any other convenient (for the user) way out :} +bool KoFilterManager::filterAvailable( KoFilterEntry::Ptr entry ) +{ + if ( !entry ) + return false; + if ( entry->available != "check" ) + return true; + + //kdDebug( 30500 ) << "Checking whether " << entry->service()->name() << " applies." << endl; + // generate some "unique" key + QString key( entry->service()->name() ); + key += " - "; + key += entry->service()->library(); + + if ( !m_filterAvailable.contains( key ) ) { + //kdDebug( 30500 ) << "Not cached, checking..." << endl; + + KLibrary* library = KLibLoader::self()->library( QFile::encodeName( entry->service()->library() ) ); + if ( !library ) { + kdWarning( 30500 ) << "Huh?? Couldn't load the lib: " + << KLibLoader::self()->lastErrorMessage() << endl; + m_filterAvailable[ key ] = false; + return false; + } + + // This code is "borrowed" from klibloader ;) + QCString symname; + symname.sprintf("check_%s", library->name().latin1() ); + void* sym = library->symbol( symname ); + if ( !sym ) + { + kdWarning( 30500 ) << "The library " << library->name() + << " does not offer a check_" << library->name() + << " function." << endl; + m_filterAvailable[ key ] = false; + } + else { + typedef int (*t_func)(); + t_func check = (t_func)sym; + m_filterAvailable[ key ] = check() == 1; + } + } + return m_filterAvailable[ key ]; +} + +void KoFilterManager::importErrorHelper( const QString& mimeType, const bool suppressDialog ) +{ + QString tmp = i18n("Could not import file of type\n%1").arg( mimeType ); + // ###### FIXME: use KLibLoader::lastErrorMessage() here + if (!suppressDialog) KMessageBox::error( 0L, tmp, i18n("Missing Import Filter") ); +} + +void KoFilterManager::setBatchMode( const bool batch ) +{ + d->m_batch = batch; +} + +bool KoFilterManager::getBatchMode( void ) const +{ + return d->m_batch; +} + +#include <KoFilterManager.moc> +#include <KoFilterManager_p.moc> diff --git a/lib/kofficecore/KoFilterManager.h b/lib/kofficecore/KoFilterManager.h new file mode 100644 index 00000000..952c8475 --- /dev/null +++ b/lib/kofficecore/KoFilterManager.h @@ -0,0 +1,178 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + 2000, 2001 Werner Trobin <trobin@kde.org> + Copyright (C) 2004 Nicolas Goutte <goutte@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. +*/ + +#ifndef __koffice_filter_manager_h__ +#define __koffice_filter_manager_h__ + +#include <qobject.h> +#include <qmap.h> +#include <KoFilterChain.h> +#include <koffice_export.h> +class KoDocument; + +/** + * @brief The class managing all the filters. + * + * This class manages all filters for a %KOffice application. Normally + * you will not have to use it, since KoMainWindow takes care of loading + * and saving documents. + * + * @ref KoFilter + * + * @author Kalle Dalheimer <kalle@kde.org> + * @author Torben Weis <weis@kde.org> + * @author Werner Trobin <trobin@kde.org> + */ +class KOFFICECORE_EXPORT KoFilterManager : public QObject +{ + Q_OBJECT +public: + /** + * This enum is used to distinguish the import/export cases + */ + enum Direction { Import = 1, Export = 2 }; + + /** + * Create a filter manager for a document + */ + KoFilterManager( KoDocument* document ); + /** + * Create a filter manager for a filter which wants to embed something. + * The url it passes is the file to convert, obviously. You cannot use + * the @ref import() method -- use @ref exp0rt() to convert the file to + * the destination mimetype you prefer. + * + * @param url The file you want to export + * @param mimetypeHint The mimetype of the file you want to export. You have + * to specify this information only if the automatic detection will + * fail because e.g. you saved an embedded stream to a *.tmp file. + * Most likely you do not have to care about that. + * @param parentChain The parent filter chain of this filter manager. Used + * to allow embedding for filters. Most likely you do not have to care. + */ + KoFilterManager( const QString& url, const QCString& mimetypeHint = "", + KoFilterChain* const parentChain = 0 ); + + virtual ~KoFilterManager(); + + /** + * Imports the passed URL and returns the resultant filename + * (most likely some file in /tmp). + * The @p status variable signals the success/error of the conversion + * If the QString which is returned isEmpty() and the status is OK, + * then we imported the file directly into the document. + */ + QString import( const QString& url, KoFilter::ConversionStatus& status ); + /** + * @brief Exports the given file/document to the specified URL/mimetype. + * + * If @p mimeType is empty, then the closest matching KOffice part is searched + * and when the method returns @p mimeType contains this mimetype. + * Oh, well, export is a C++ keyword ;) + */ + KoFilter::ConversionStatus exp0rt( const QString& url, QCString& mimeType ); + + + + ///@name Static API + //@{ + /** + * Suitable for passing to KFileDialog::setMimeFilter. The default mime + * gets set by the "users" of this method, as we do not have enough + * information here. + * Optionally, @p extraNativeMimeTypes are added after the native mimetype. + */ + KOFFICECORE_EXPORT static QStringList mimeFilter( const QCString& mimetype, Direction direction, + const QStringList& extraNativeMimeTypes = QStringList() ); + + /** + * The same method as KoFilterManager::mimeFilter but suited for KoShell. + * We do not need the mimetype, as we will simply use all available + * %KOffice mimetypes. The Direction enum is omitted, as we only + * call this for importing. When saving from KoShell we already + * know the KOffice part we are using. + */ + static QStringList mimeFilter(); + + /** + * Method used to check if that filter is available at all. + * @note Slow, but cached + */ + static bool filterAvailable( KoFilterEntry::Ptr entry ); + + //@} + + /** + * Set the filter manager is batch mode (no dialog shown) + * instead of the interactive mode (dialog shown) + * @since 1.4 + */ + void setBatchMode ( const bool batch ); + + /** + * Get if the filter manager is batch mode (true) + * or in interactive mode (true) + * @since 1.4 + */ + bool getBatchMode ( void ) const; + +signals: + void sigProgress( int ); + +private: + // === API for KoFilterChains === (internal) + // The friend methods are private in KoFilterChain and + // just forward calls to the methods here. Should be + // pretty save. + friend QString KoFilterChain::filterManagerImportFile() const; + QString importFile() const { return m_importUrl; } + friend QString KoFilterChain::filterManagerExportFile() const; + QString exportFile() const { return m_exportUrl; } + friend KoDocument* KoFilterChain::filterManagerKoDocument() const; + KoDocument* document() const { return m_document; } + friend int KoFilterChain::filterManagerDirection() const; + int direction() const { return static_cast<int>( m_direction ); } + friend KoFilterChain* const KoFilterChain::filterManagerParentChain() const; + KoFilterChain* const parentChain() const { return m_parentChain; } + + // Private API + KoFilterManager( const KoFilterManager& rhs ); + KoFilterManager &operator=( const KoFilterManager& rhs ); + + void importErrorHelper( const QString& mimeType, const bool suppressDialog = false ); + + static const int s_area; + + KoDocument* m_document; + KoFilterChain* const m_parentChain; + QString m_importUrl, m_exportUrl; + QCString m_importUrlMimetypeHint; ///< suggested mimetype + KOffice::Graph m_graph; + Direction m_direction; + + /// A static cache for the availability checks of filters + static QMap<QString, bool> m_filterAvailable; + + class Private; + Private *d; +}; + +#endif // __koffice_filter_manager_h__ diff --git a/lib/kofficecore/KoFilterManager_p.h b/lib/kofficecore/KoFilterManager_p.h new file mode 100644 index 00000000..20c6cac8 --- /dev/null +++ b/lib/kofficecore/KoFilterManager_p.h @@ -0,0 +1,46 @@ +/* This file is part of the KDE project + Copyright (C) 2003 Clarence Dang <dang@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 Version 2 as published by the Free Software Foundation. + + 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 Version 2 for more details. + + You should have received a copy of the GNU Library General Public License + Version 2 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. +*/ + +#ifndef __koFilterManager_p_h__ +#define __koFilterManager_p_h__ + +#include <qstring.h> +#include <qstringlist.h> + +#include <kdialogbase.h> + +class KListBox; + + +class KoFilterChooser : public KDialogBase +{ +Q_OBJECT + +public: + KoFilterChooser (QWidget *parent, const QStringList &mimeTypes, + const QString &nativeFormat = QString::null); + ~KoFilterChooser (); + + QString filterSelected (); + +private: + QStringList m_mimeTypes; + KListBox *m_filterList; +}; + +#endif diff --git a/lib/kofficecore/KoFrame.cpp b/lib/kofficecore/KoFrame.cpp new file mode 100644 index 00000000..4c07d5c1 --- /dev/null +++ b/lib/kofficecore/KoFrame.cpp @@ -0,0 +1,393 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@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 <KoFrame.h> +#include <KoView.h> + +#include <qpainter.h> +#include <kparts/event.h> +#include <kcursor.h> +#include <kdebug.h> + +#define QMAX32767(a,b) QMAX(QMAX(32767,a),b) + +class KoFramePrivate +{ +public: + KoFramePrivate() + { + } + ~KoFramePrivate() + { + } + + KoView *m_view; + + QPoint m_mousePressPos; + QPoint m_framePos; + int m_width; + int m_height; + int m_mode; + + KoFrame::State m_state; +}; + +KoFrame::KoFrame( QWidget *parent, const char *name ) + : QWidget( parent, name ) +{ + d = new KoFramePrivate; + d->m_state = Inactive; + d->m_mode = -1; + d->m_view = 0; + + setBackgroundColor( white ); + setMouseTracking( true ); +} + +KoFrame::~KoFrame() +{ + delete d; +} + +void KoFrame::setView( KoView *view ) +{ + if ( view == d->m_view ) + return; + + if ( d->m_view ) + d->m_view->removeEventFilter( this ); + + d->m_view = view; + if ( d->m_view ) + d->m_view->installEventFilter( this ); + + resizeEvent( 0L ); +} + +KoView *KoFrame::view() const +{ + return d->m_view; +} + +void KoFrame::setState( State s ) +{ + if ( d->m_state == s ) + return; + + State old = d->m_state; + d->m_state = s; + + if ( d->m_view ) + { + /* + kdDebug(30003) << "KoFrame::setView setMaximumSize " + << "QMAX32767(" << d->m_view->maximumWidth() + 2 * border() << "," << d->m_view->maximumWidth() << "), " + << "QMAX32767(" << d->m_view->maximumHeight() + 2 * border() << "," << d->m_view->maximumHeight() << ")" + << endl; + */ + setMaximumSize( QMAX32767( d->m_view->maximumWidth() + 2 * border(), d->m_view->maximumWidth() ), + QMAX32767( d->m_view->maximumHeight() + 2 * border(), d->m_view->maximumHeight() ) ); + setMinimumSize( d->m_view->minimumWidth() + leftBorder() + rightBorder(), + d->m_view->minimumHeight() + topBorder() + bottomBorder() ); + } + + if ( d->m_state == Inactive ) + { + d->m_state = old; + int l = leftBorder(); + int r = rightBorder(); + int t = topBorder(); + int b = bottomBorder(); + d->m_state = Inactive; + setGeometry( x() + l, y() + t, width() - l - r, height() - t - b ); + } + else if ( ( d->m_state == Active || d->m_state == Selected ) && old == Inactive ) + setGeometry( x() - leftBorder(), y() - topBorder(), + width() + leftBorder() + rightBorder(), + height() + topBorder() + bottomBorder() ); + else if ( d->m_state == Active && old == Selected ) + { + setGeometry( x() - leftBorder() + border(), y() - topBorder() + border(), + width() + leftBorder() + rightBorder() - 2 * border(), + height() + topBorder() + bottomBorder() - 2 * border() ); + } + + update(); +} + +KoFrame::State KoFrame::state() const +{ + return d->m_state; +} + +int KoFrame::leftBorder() const +{ + if ( d->m_state == Inactive ) + return 0; + if ( d->m_state == Selected || !d->m_view ) + return border(); + + return d->m_view->leftBorder() + border(); +} + +int KoFrame::rightBorder() const +{ + if ( d->m_state == Inactive ) + return 0; + if ( d->m_state == Selected || !d->m_view ) + return border(); + + return d->m_view->rightBorder() + border(); +} + +int KoFrame::topBorder() const +{ + if ( d->m_state == Inactive ) + return 0; + if ( d->m_state == Selected || !d->m_view ) + return border(); + + return d->m_view->topBorder() + border(); +} + +int KoFrame::bottomBorder() const +{ + if ( d->m_state == Inactive ) + return 0; + if ( d->m_state == Selected || !d->m_view ) + return border(); + + return d->m_view->bottomBorder() + border(); +} + +int KoFrame::border() const +{ + if ( d->m_state == Inactive ) + return 0; + return 5; +} + +void KoFrame::paintEvent( QPaintEvent* ) +{ + QPainter painter; + painter.begin( this ); + + painter.setPen( black ); + painter.fillRect( 0, 0, width(), height(), BDiagPattern ); + + if ( d->m_state == Selected ) + { + painter.fillRect( 0, 0, 5, 5, black ); + painter.fillRect( 0, height() - 5, 5, 5, black ); + painter.fillRect( width() - 5, height() - 5, 5, 5, black ); + painter.fillRect( width() - 5, 0, 5, 5, black ); + painter.fillRect( width() / 2 - 3, 0, 5, 5, black ); + painter.fillRect( width() / 2 - 3, height() - 5, 5, 5, black ); + painter.fillRect( 0, height() / 2 - 3, 5, 5, black ); + painter.fillRect( width() - 5, height() / 2 - 3, 5, 5, black ); + } + + painter.end(); +} + +void KoFrame::mousePressEvent( QMouseEvent* ev ) +{ + QRect r1( 0, 0, 5, 5 ); + QRect r2( 0, height() - 5, 5, 5 ); + QRect r3( width() - 5, height() - 5, 5, 5 ); + QRect r4( width() - 5, 0, 5, 5 ); + QRect r5( width() / 2 - 3, 0, 5, 5 ); + QRect r6( width() / 2 - 3, height() - 5, 5, 5 ); + QRect r7( 0, height() / 2 - 3, 5, 5 ); + QRect r8( width()- 5, height() / 2 - 3, 5, 5 ); + + if ( r1.contains( ev->pos() ) ) + d->m_mode = 1; + else if ( r2.contains( ev->pos() ) ) + d->m_mode = 2; + else if ( r3.contains( ev->pos() ) ) + d->m_mode = 3; + else if ( r4.contains( ev->pos() ) ) + d->m_mode = 4; + else if ( r5.contains( ev->pos() ) ) + d->m_mode = 5; + else if ( r6.contains( ev->pos() ) ) + d->m_mode = 6; + else if ( r7.contains( ev->pos() ) ) + d->m_mode = 7; + else if ( r8.contains( ev->pos() ) ) + d->m_mode = 8; + else + d->m_mode = 0; + + // if ( d->m_state == Active ) + // d->m_mode = 0; + + kdDebug(30003) << "---- MODE=" << d->m_mode << endl; + + d->m_mousePressPos = mapToParent( ev->pos() ); + d->m_framePos = mapToParent( QPoint( 0, 0 ) ); + d->m_width = width(); + d->m_height = height(); +} + +void KoFrame::mouseMoveEvent( QMouseEvent* ev ) +{ + if ( d->m_mode == 0 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + move( QPoint( p.x() + d->m_framePos.x() - d->m_mousePressPos.x(), + p.y() + d->m_framePos.y() - d->m_mousePressPos.y() ) ); + // The other modes change the geometry so they call resizeEvent. + // This one doesn't, so it has to emit geometryChangedby itself. + emit geometryChanged(); + //kdDebug() << "KoFrame::mouseMoveEvent koFrame position is " << x() << "," << y() << endl; + } + else if ( d->m_mode == 1 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int w = QMIN( QMAX( d->m_width + d->m_mousePressPos.x() - p.x(), minimumWidth() ), maximumWidth() ); + int h = QMIN( QMAX( d->m_height + d->m_mousePressPos.y() - p.y(), minimumHeight() ), maximumHeight() ); + setGeometry( d->m_framePos.x() - w + d->m_width, + d->m_framePos.y() - h + d->m_height, w, h ); + } + else if ( d->m_mode == 2 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int w = QMIN( QMAX( d->m_width + d->m_mousePressPos.x() - p.x(), minimumWidth() ), maximumWidth() ); + int h = QMIN( QMAX( d->m_height - d->m_mousePressPos.y() + p.y(), minimumHeight() ), maximumHeight() ); + setGeometry( d->m_framePos.x() - w + d->m_width, + d->m_framePos.y(), w, h ); + } + else if ( d->m_mode == 3 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int w = QMIN( QMAX( d->m_width - d->m_mousePressPos.x() + p.x(), minimumWidth() ), maximumWidth() ); + int h = QMIN( QMAX( d->m_height - d->m_mousePressPos.y() + p.y(), minimumHeight() ), maximumHeight() ); + resize( w, h ); + } + else if ( d->m_mode == 4 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int w = QMIN( QMAX( d->m_width - d->m_mousePressPos.x() + p.x(), minimumWidth() ), maximumWidth() ); + int h = QMIN( QMAX( d->m_height + d->m_mousePressPos.y() - p.y(), minimumHeight() ), maximumHeight() ); + setGeometry( d->m_framePos.x(), + d->m_framePos.y() - h + d->m_height, w, h ); + } + else if ( d->m_mode == 5 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int h = QMIN( QMAX( d->m_height + d->m_mousePressPos.y() - p.y(), minimumHeight() ), maximumHeight() ); + setGeometry( d->m_framePos.x(), + d->m_framePos.y() - h + d->m_height, d->m_width, h ); + } + else if ( d->m_mode == 6 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int h = QMIN( QMAX( d->m_height - d->m_mousePressPos.y() + p.y(), minimumHeight() ), maximumHeight() ); + resize( d->m_width, h ); + } + else if ( d->m_mode == 7 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int w = QMIN( QMAX( d->m_width + d->m_mousePressPos.x() - p.x(), minimumWidth() ), maximumWidth() ); + setGeometry( d->m_framePos.x() - w + d->m_width, + d->m_framePos.y(), w, d->m_height ); + } + else if ( d->m_mode == 8 ) + { + QPoint p = parentWidget()->mapFromGlobal( ev->globalPos() ); + int w = QMIN( QMAX( d->m_width - d->m_mousePressPos.x() + p.x(), minimumWidth() ), maximumWidth() ); + resize( w, d->m_height ); + } + else if ( d->m_state == Selected || d->m_state == Active ) + { + QRect r1( 0, 0, 5, 5 ); + QRect r2( 0, height() - 5, 5, 5 ); + QRect r3( width() - 5, height() - 5, 5, 5 ); + QRect r4( width() - 5, 0, 5, 5 ); + QRect r5( width() / 2 - 3, 0, 5, 5 ); + QRect r6( width() / 2 - 3, height() - 5, 5, 5 ); + QRect r7( 0, height() / 2 - 3, 5, 5 ); + QRect r8( width()- 5, height() / 2 - 3, 5, 5 ); + + if ( r1.contains( ev->pos() ) || r3.contains( ev->pos() ) ) + setCursor( sizeFDiagCursor ); + else if ( r2.contains( ev->pos() ) || r4.contains( ev->pos() ) ) + setCursor( sizeBDiagCursor ); + else if ( r5.contains( ev->pos() ) || r6.contains( ev->pos() ) ) + setCursor( sizeVerCursor ); + else if ( r7.contains( ev->pos() ) || r8.contains( ev->pos() ) ) + setCursor( sizeHorCursor ); + else + setCursor( KCursor::handCursor() ); + } + else + setCursor( KCursor::handCursor() ); +} + +void KoFrame::mouseReleaseEvent( QMouseEvent* ) +{ + d->m_mode = -1; +} + +void KoFrame::resizeEvent( QResizeEvent* ) +{ + if ( !d->m_view ) + return; + + if ( d->m_state == Active || d->m_state == Selected ) + d->m_view->setGeometry( 5, 5, width() - 10, height() - 10 ); + else + d->m_view->setGeometry( 0, 0, width(), height() ); + + emit geometryChanged(); +} + +bool KoFrame::eventFilter( QObject* obj, QEvent* ev ) +{ + if ( obj == d->m_view && KParts::PartActivateEvent::test( ev ) ) + { + kdDebug(30003) << "Activate event"<< endl; + KParts::PartActivateEvent* e = (KParts::PartActivateEvent*)ev; + if ( e->part() == (KParts::Part *)d->m_view->koDocument() ) + { + if ( e->activated() ) + setState( Active ); + else + setState( Inactive ); + } + } + else if ( obj == d->m_view && KParts::PartSelectEvent::test( ev ) ) + { + kdDebug(30003) << "Selected event" << endl; + KParts::PartSelectEvent* e = (KParts::PartSelectEvent*)ev; + if ( e->part() == (KParts::Part *)d->m_view->koDocument() ) + { + if ( e->selected() ) + setState( Selected ); + else + setState( Inactive ); + } + } + + return false; +} + +#include <KoFrame.moc> diff --git a/lib/kofficecore/KoFrame.h b/lib/kofficecore/KoFrame.h new file mode 100644 index 00000000..7f729bdb --- /dev/null +++ b/lib/kofficecore/KoFrame.h @@ -0,0 +1,65 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999, 2000 Torben Weis <weis@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. +*/ + +#ifndef __koFrame_h__ +#define __koFrame_h__ + +#include <qwidget.h> + +class KoView; +class KoFramePrivate; + +class KoFrame : public QWidget +{ + Q_OBJECT +public: + enum State { Inactive, Selected, Active }; + + KoFrame( QWidget *parent, const char *name = 0 ); + virtual ~KoFrame(); + + virtual void setView( KoView *view ); + virtual KoView *view() const; + + virtual void setState( State s ); + virtual State state() const; + + virtual int leftBorder() const; + virtual int rightBorder() const; + virtual int topBorder() const; + virtual int bottomBorder() const; + + virtual int border() const; + +signals: + void geometryChanged(); + +protected: + virtual void paintEvent( QPaintEvent* ); + virtual void mousePressEvent( QMouseEvent* ); + virtual void mouseMoveEvent( QMouseEvent* ); + virtual void mouseReleaseEvent( QMouseEvent* ); + virtual void resizeEvent( QResizeEvent* ); + virtual bool eventFilter( QObject*, QEvent* ); + +private: + KoFramePrivate *d; +}; + +#endif diff --git a/lib/kofficecore/KoGenStyles.cpp b/lib/kofficecore/KoGenStyles.cpp new file mode 100644 index 00000000..503b6802 --- /dev/null +++ b/lib/kofficecore/KoGenStyles.cpp @@ -0,0 +1,405 @@ +/* This file is part of the KDE project + Copyright (C) 2004-2006 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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 "KoGenStyles.h" +#include <KoXmlWriter.h> +#include <float.h> +#include <kdebug.h> + +KoGenStyles::KoGenStyles() +{ +} + +KoGenStyles::~KoGenStyles() +{ +} + +QString KoGenStyles::lookup( const KoGenStyle& style, const QString& name, int flags ) +{ + StyleMap::iterator it = m_styleMap.find( style ); + if ( it == m_styleMap.end() ) { + // Not found, try if this style is in fact equal to its parent (the find above + // wouldn't have found it, due to m_parentName being set). + if ( !style.parentName().isEmpty() ) { + KoGenStyle testStyle( style ); + const KoGenStyle* parentStyle = this->style( style.parentName() ); // ## linear search + if( !parentStyle ) { + kdDebug(30003) << "KoGenStyles::lookup(" << name << "): parent style '" << style.parentName() << "' not found in collection" << endl; + } else { + if ( testStyle.m_familyName != parentStyle->m_familyName ) + { + kdWarning(30003) << "KoGenStyles::lookup(" << name << ", family=" << testStyle.m_familyName << ") parent style '" << style.parentName() << "' has a different family: " << parentStyle->m_familyName << endl; + } + + testStyle.m_parentName = parentStyle->m_parentName; + // Exclude the type from the comparison. It's ok for an auto style + // to have a user style as parent; they can still be identical + testStyle.m_type = parentStyle->m_type; + // Also it's ok to not have the display name of the parent style + // in the auto style + QMap<QString, QString>::const_iterator it = parentStyle->m_attributes.find( "style:display-name" ); + if ( it != parentStyle->m_attributes.end() ) + testStyle.addAttribute( "style:display-name", *it ); + + if ( *parentStyle == testStyle ) + return style.parentName(); + } + } + + QString styleName( name ); + if ( styleName.isEmpty() ) { + styleName = 'A'; // for "auto". + flags &= ~DontForceNumbering; // i.e. force numbering + } + styleName = makeUniqueName( styleName, flags ); + if ( style.autoStyleInStylesDotXml() ) + m_autoStylesInStylesDotXml.insert( styleName, true /*unused*/ ); + else + m_styleNames.insert( styleName, true /*unused*/ ); + it = m_styleMap.insert( style, styleName ); + NamedStyle s; + s.style = &it.key(); + s.name = styleName; + m_styleArray.append( s ); + } + return it.data(); +} + +QString KoGenStyles::makeUniqueName( const QString& base, int flags ) const +{ + // If this name is not used yet, and numbering isn't forced, then the given name is ok. + if ( ( flags & DontForceNumbering ) + && m_autoStylesInStylesDotXml.find( base ) == m_autoStylesInStylesDotXml.end() + && m_styleNames.find( base ) == m_styleNames.end() ) + return base; + int num = 1; + QString name; + do { + name = base; + name += QString::number( num++ ); + } while ( m_autoStylesInStylesDotXml.find( name ) != m_autoStylesInStylesDotXml.end() + || m_styleNames.find( name ) != m_styleNames.end() ); + return name; +} + +QValueList<KoGenStyles::NamedStyle> KoGenStyles::styles( int type, bool markedForStylesXml ) const +{ + QValueList<KoGenStyles::NamedStyle> lst; + const NameMap& nameMap = markedForStylesXml ? m_autoStylesInStylesDotXml : m_styleNames; + StyleArray::const_iterator it = m_styleArray.begin(); + const StyleArray::const_iterator end = m_styleArray.end(); + for ( ; it != end ; ++it ) { + // Look up if it's marked for styles.xml or not by looking up in the corresponding style map. + if ( (*it).style->type() == type && nameMap.find((*it).name) != nameMap.end() ) { + lst.append( *it ); + } + } + return lst; +} + +const KoGenStyle* KoGenStyles::style( const QString& name ) const +{ + StyleArray::const_iterator it = m_styleArray.begin(); + const StyleArray::const_iterator end = m_styleArray.end(); + for ( ; it != end ; ++it ) { + if ( (*it).name == name ) + return (*it).style; + } + return 0; +} + +KoGenStyle* KoGenStyles::styleForModification( const QString& name ) +{ + return const_cast<KoGenStyle *>( style( name ) ); +} + +void KoGenStyles::markStyleForStylesXml( const QString& name ) +{ + Q_ASSERT( m_styleNames.find( name ) != m_styleNames.end() ); + m_styleNames.remove( name ); + m_autoStylesInStylesDotXml.insert( name, true ); + styleForModification( name )->setAutoStyleInStylesDotXml( true ); +} + +void KoGenStyles::dump() +{ + kdDebug() << "Style array:" << endl; + StyleArray::const_iterator it = m_styleArray.begin(); + const StyleArray::const_iterator end = m_styleArray.end(); + for ( ; it != end ; ++it ) { + kdDebug() << (*it).name << endl; + } + for ( NameMap::const_iterator it = m_styleNames.begin(); it != m_styleNames.end(); ++it ) { + kdDebug() << "style: " << it.key() << endl; + } + for ( NameMap::const_iterator it = m_autoStylesInStylesDotXml.begin(); it != m_autoStylesInStylesDotXml.end(); ++it ) { + kdDebug() << "auto style for style.xml: " << it.key() << endl; + const KoGenStyle* s = style( it.key() ); + Q_ASSERT( s ); + Q_ASSERT( s->autoStyleInStylesDotXml() ); + } +} + +// Returns -1, 0 (equal) or 1 +static int compareMap( const QMap<QString, QString>& map1, const QMap<QString, QString>& map2 ) +{ + QMap<QString, QString>::const_iterator it = map1.begin(); + QMap<QString, QString>::const_iterator oit = map2.begin(); + for ( ; it != map1.end(); ++it, ++oit ) { // both maps have been checked for size already + if ( it.key() != oit.key() ) + return it.key() < oit.key() ? -1 : +1; + if ( it.data() != oit.data() ) + return it.data() < oit.data() ? -1 : +1; + } + return 0; // equal +} + +//// + + +KoGenStyle::KoGenStyle( int type, const char* familyName, + const QString& parentName ) + : m_type( type ), m_familyName( familyName ), m_parentName( parentName ), + m_autoStyleInStylesDotXml( false ), m_defaultStyle( false ) +{ +} + +KoGenStyle::~KoGenStyle() +{ +} + +void KoGenStyle::writeStyleProperties( KoXmlWriter* writer, PropertyType i, + const char* elementName, const KoGenStyle* parentStyle ) const +{ + if ( !m_properties[i].isEmpty() ) { + writer->startElement( elementName ); + QMap<QString, QString>::const_iterator it = m_properties[i].begin(); + const QMap<QString, QString>::const_iterator end = m_properties[i].end(); + for ( ; it != end; ++it ) { + if ( !parentStyle || parentStyle->property( it.key(), i ) != it.data() ) + writer->addAttribute( it.key().utf8(), it.data().utf8() ); + } + writer->endElement(); + } +} + +void KoGenStyle::writeStyle( KoXmlWriter* writer, KoGenStyles& styles, const char* elementName, const QString& name, const char* propertiesElementName, bool closeElement, bool drawElement ) const +{ + //kdDebug(30003) << "writing out style " << name << " display-name=" << m_attributes["style:display-name"] << " family=" << m_familyName << endl; + writer->startElement( elementName ); + const KoGenStyle* parentStyle = 0; + if ( !m_defaultStyle ) { + if ( !drawElement ) + writer->addAttribute( "style:name", name ); + else + writer->addAttribute( "draw:name", name ); + if ( !m_parentName.isEmpty() ) { + parentStyle = styles.style( m_parentName ); + if ( parentStyle && m_familyName.isEmpty() ) { + // get family from parent style, just in case + // Note: this is saving code, don't convert to attributeNS! + const_cast<KoGenStyle *>( this )-> + m_familyName = parentStyle->attribute( "style:family" ).latin1(); + //kdDebug(30003) << "Got familyname " << m_familyName << " from parent" << endl; + } + writer->addAttribute( "style:parent-style-name", m_parentName ); + } + } else { // default-style + Q_ASSERT( qstrcmp( elementName, "style:default-style" ) == 0 ); + Q_ASSERT( m_parentName.isEmpty() ); + } + if ( !m_familyName.isEmpty() ) + const_cast<KoGenStyle *>( this )-> + addAttribute( "style:family", QString::fromLatin1( m_familyName ) ); + else { + if ( qstrcmp( elementName, "style:style" ) == 0 ) + kdWarning(30003) << "User style " << name << " is without family - invalid. m_type=" << m_type << endl; + } + +#if 0 // #ifndef NDEBUG + kdDebug() << "style: " << name << endl; + printDebug(); + if ( parentStyle ) { + kdDebug() << " parent: " << m_parentName << endl; + parentStyle->printDebug(); + } +#endif + + // Write attributes [which differ from the parent style] + // We only look at the direct parent style because we assume + // that styles are fully specified, i.e. the inheritance is + // only in the final file, not in the caller's code. + QMap<QString, QString>::const_iterator it = m_attributes.begin(); + for ( ; it != m_attributes.end(); ++it ) { + bool writeit = true; + if ( parentStyle && it.key() != "style:family" // always write the family out + && parentStyle->attribute( it.key() ) == it.data() ) + writeit = false; + if ( writeit ) + writer->addAttribute( it.key().utf8(), it.data().utf8() ); + } + bool createPropertiesTag = propertiesElementName && propertiesElementName[0] != '\0'; + KoGenStyle::PropertyType i = KoGenStyle::DefaultType; + if ( !m_properties[i].isEmpty() || + !m_properties[KoGenStyle::ChildElement].isEmpty() ) { + if ( createPropertiesTag ) + writer->startElement( propertiesElementName ); // e.g. paragraph-properties + it = m_properties[i].begin(); + for ( ; it != m_properties[i].end(); ++it ) { + if ( !parentStyle || parentStyle->property( it.key(), i ) != it.data() ) + writer->addAttribute( it.key().utf8(), it.data().utf8() ); + } + i = KoGenStyle::ChildElement; + it = m_properties[i].begin(); + for ( ; it != m_properties[i].end(); ++it ) { + if ( !parentStyle || parentStyle->property( it.key(), i ) != it.data() ) { + writer->addCompleteElement( it.data().utf8() ); + } + } + if ( createPropertiesTag ) + writer->endElement(); + } + writeStyleProperties( writer, KoGenStyle::GraphicType, "style:graphic-properties", parentStyle ); + writeStyleProperties( writer, KoGenStyle::ParagraphType, "style:paragraph-properties", parentStyle ); + writeStyleProperties( writer, KoGenStyle::TextType, "style:text-properties", parentStyle ); + + // And now the style maps + for ( uint i = 0; i < m_maps.count(); ++i ) { + bool writeit = true; + if ( parentStyle && compareMap( m_maps[i], parentStyle->m_maps[i] ) == 0 ) + writeit = false; + if ( writeit ) { + writer->startElement( "style:map" ); + QMap<QString, QString>::const_iterator it = m_maps[i].begin(); + for ( ; it != m_maps[i].end(); ++it ) { + writer->addAttribute( it.key().utf8(), it.data().utf8() ); + } + writer->endElement(); // style:map + } + } + if ( closeElement ) + writer->endElement(); +} + +void KoGenStyle::addPropertyPt( const QString& propName, double propValue, PropertyType type ) +{ + QString str; + str.setNum( propValue, 'g', DBL_DIG ); + str += "pt"; + m_properties[type].insert( propName, str ); +} + +void KoGenStyle::addAttributePt( const QString& attrName, double attrValue ) +{ + QString str; + str.setNum( attrValue, 'g', DBL_DIG ); + str += "pt"; + m_attributes.insert( attrName, str ); +} + +#ifndef NDEBUG +void KoGenStyle::printDebug() const +{ + int i = DefaultType; + kdDebug() << m_properties[i].count() << " properties." << endl; + for( QMap<QString,QString>::ConstIterator it = m_properties[i].begin(); it != m_properties[i].end(); ++it ) { + kdDebug() << " " << it.key() << " = " << it.data() << endl; + } + i = TextType; + kdDebug() << m_properties[i].count() << " text properties." << endl; + for( QMap<QString,QString>::ConstIterator it = m_properties[i].begin(); it != m_properties[i].end(); ++it ) { + kdDebug() << " " << it.key() << " = " << it.data() << endl; + } + i = ParagraphType; + kdDebug() << m_properties[i].count() << " paragraph properties." << endl; + for( QMap<QString,QString>::ConstIterator it = m_properties[i].begin(); it != m_properties[i].end(); ++it ) { + kdDebug() << " " << it.key() << " = " << it.data() << endl; + } + i = ChildElement; + kdDebug() << m_properties[i].count() << " child elements." << endl; + for( QMap<QString,QString>::ConstIterator it = m_properties[i].begin(); it != m_properties[i].end(); ++it ) { + kdDebug() << " " << it.key() << " = " << it.data() << endl; + } + kdDebug() << m_attributes.count() << " attributes." << endl; + for( QMap<QString,QString>::ConstIterator it = m_attributes.begin(); it != m_attributes.end(); ++it ) { + kdDebug() << " " << it.key() << " = " << it.data() << endl; + } + kdDebug() << m_maps.count() << " maps." << endl; + for ( uint i = 0; i < m_maps.count(); ++i ) { + kdDebug() << "map " << i << ":" << endl; + for( QMap<QString,QString>::ConstIterator it = m_maps[i].begin(); it != m_maps[i].end(); ++it ) { + kdDebug() << " " << it.key() << " = " << it.data() << endl; + } + } + kdDebug() << endl; +} +#endif + +bool KoGenStyle::operator<( const KoGenStyle &other ) const +{ + if ( m_type != other.m_type ) return m_type < other.m_type; + if ( m_parentName != other.m_parentName ) return m_parentName < other.m_parentName; + if ( m_autoStyleInStylesDotXml != other.m_autoStyleInStylesDotXml ) return m_autoStyleInStylesDotXml; + for ( uint i = 0 ; i < N_NumTypes ; ++i ) + if ( m_properties[i].count() != other.m_properties[i].count() ) + return m_properties[i].count() < other.m_properties[i].count(); + if ( m_attributes.count() != other.m_attributes.count() ) return m_attributes.count() < other.m_attributes.count(); + if ( m_maps.count() != other.m_maps.count() ) return m_maps.count() < other.m_maps.count(); + // Same number of properties and attributes, no other choice than iterating + for ( uint i = 0 ; i < N_NumTypes ; ++i ) { + int comp = compareMap( m_properties[i], other.m_properties[i] ); + if ( comp != 0 ) + return comp < 0; + } + int comp = compareMap( m_attributes, other.m_attributes ); + if ( comp != 0 ) + return comp < 0; + for ( uint i = 0 ; i < m_maps.count() ; ++i ) { + int comp = compareMap( m_maps[i], other.m_maps[i] ); + if ( comp != 0 ) + return comp < 0; + } + return false; +} + +bool KoGenStyle::operator==( const KoGenStyle &other ) const +{ + if ( m_type != other.m_type ) return false; + if ( m_parentName != other.m_parentName ) return false; + if ( m_autoStyleInStylesDotXml != other.m_autoStyleInStylesDotXml ) return false; + for ( uint i = 0 ; i < N_NumTypes ; ++i ) + if ( m_properties[i].count() != other.m_properties[i].count() ) + return false; + if ( m_attributes.count() != other.m_attributes.count() ) return false; + if ( m_maps.count() != other.m_maps.count() ) return false; + // Same number of properties and attributes, no other choice than iterating + for ( uint i = 0 ; i < N_NumTypes ; ++i ) { + int comp = compareMap( m_properties[i], other.m_properties[i] ); + if ( comp != 0 ) + return false; + } + int comp = compareMap( m_attributes, other.m_attributes ); + if ( comp != 0 ) + return false; + for ( uint i = 0 ; i < m_maps.count() ; ++i ) { + int comp = compareMap( m_maps[i], other.m_maps[i] ); + if ( comp != 0 ) + return false; + } + return true; +} diff --git a/lib/kofficecore/KoGenStyles.h b/lib/kofficecore/KoGenStyles.h new file mode 100644 index 00000000..ad05d6e0 --- /dev/null +++ b/lib/kofficecore/KoGenStyles.h @@ -0,0 +1,463 @@ +/* This file is part of the KDE project + Copyright (C) 2004-2006 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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. +*/ + +#ifndef KOGENSTYLES_H +#define KOGENSTYLES_H + +#include <qdict.h> +#include <qmap.h> +#include <qvaluevector.h> + +#include <koffice_export.h> + +class KoXmlWriter; +class KoGenStyle; + +/** + * @brief Repository of styles used during saving of OASIS/OOo file. + * + * Each instance of KoGenStyles is a collection of styles whose names + * are in the same "namespace". + * This means there should be one instance for all styles in <office:styles>, + * and automatic-styles, another instance for number formats, another + * one for draw styles, and another one for list styles. + * + * "Style" in this context only means "a collection of properties". + * The "Gen" means both "Generic" and "Generated" :) + * + * The basic design principle is the flyweight pattern: if you need + * a style with the same properties from two different places, you + * get the same object. Here it means rather you get the same name for the style, + * and it will get saved only once to the file. + * + * KoGenStyles features sharing, creation on demand, and name generation. + * Since this is used for saving only, it doesn't feature refcounting, nor + * removal of individual styles. + * + * NOTE: the use of KoGenStyles isn't mandatory, of course. If the application + * is already designed with user and automatic styles in mind for a given + * set of properties, it can go ahead and save all styles directly (after + * ensuring they have unique names). + * + * @author David Faure <faure@kde.org> + */ +class KOFFICECORE_EXPORT KoGenStyles +{ +public: + KoGenStyles(); + ~KoGenStyles(); + + /** + * Those are flags for the lookup() call. + * + * By default, the generated style names will look like "name1", "name2". + * If DontForceNumbering is set, the first name that will be tried is "name", and only if + * that one exists, then "name1" is tried. Set DontForceNumbering if the name given as + * argument is supposed to be the full style name. + * + */ + enum Flags { // bitfield + NoFlag = 0, + ForceNumbering = 0, // it's the default anyway + DontForceNumbering = 1 + }; + // KDE4 TODO: use QFlags and change the arg type in lookup + + /** + * Look up a style in the collection, inserting it if necessary. + * This assigns a name to the style and returns it. + * + * @param style the style to look up. + * @param name proposed (base) name for the style. Note that with the OASIS format, + * the style name is never shown to the user (there's a separate display-name + * attribute for that). So there are little reasons to use named styles anyway. + * But this attribute can be used for clarity of the files. + * If this name is already in use (for another style), then a number is appended + * to it until unique. + * @param flags see Flags + * + * @return the name for this style + * @todo ### rename lookup to insert + */ + QString lookup( const KoGenStyle& style, const QString& name = QString::null, int flags = NoFlag ); + + typedef QMap<KoGenStyle, QString> StyleMap; + /** + * Return the entire collection of styles + * Use this for saving the styles + */ + const StyleMap& styles() const { return m_styleMap; } + + struct NamedStyle { + const KoGenStyle* style; ///< @note owned by the collection + QString name; + }; + /** + * Return all styles of a given type + * Use this for saving the styles + * + * @param type the style type, see the KoGenStyle constructor + * @param markedForStylesXml if true, return only style marked for styles.xml, + * otherwise only those NOT marked for styles.xml. + * @see lookup + */ + QValueList<NamedStyle> styles( int type, bool markedForStylesXml = false ) const; + + /** + * @return an existing style by name + */ + const KoGenStyle* style( const QString& name ) const; + + /** + * @return an existing style by name, which can be modified. + * @warning This is DANGEROUS. + * It basically defeats the purpose of lookup()! + * Only do this if you know for sure no other 'user' of that style will + * be affected. + */ + KoGenStyle* styleForModification( const QString& name ); + + /** + * Mark a given automatic style as being needed in styles.xml. + * For instance styles used by headers and footers need to go there, since + * they are saved in styles.xml, and styles.xml must be independent from content.xml. + * + * Equivalent to using KoGenStyle::setAutoStyleInStylesDotXml() but this can be done after lookup. + * + * This operation can't be undone; once styles are promoted they can't go back + * to being content.xml-only. + * + * @see styles, KoGenStyle::setAutoStyleInStylesDotXml + */ + void markStyleForStylesXml( const QString& name ); + + /** + * Outputs debug information + */ + void dump(); + +private: + QString makeUniqueName( const QString& base, int flags ) const; + + /// style definition -> name + StyleMap m_styleMap; + + /// Map with the style name as key. + /// This map is mainly used to check for name uniqueness + /// The value of the bool doesn't matter. + typedef QMap<QString, bool> NameMap; // KDE4: QSet + NameMap m_styleNames; + NameMap m_autoStylesInStylesDotXml; + + /// List of styles (used to preserve ordering) + typedef QValueVector<NamedStyle> StyleArray; + StyleArray m_styleArray; + + class Private; + Private *d; +}; + +/** + * A generic style, i.e. basically a collection of properties and a name. + * Instances of KoGenStyle can either be held in the KoGenStyles collection, + * or created (e.g. on the stack) and given to KoGenStyles::lookup. + * + * @author David Faure <faure@kde.org> + */ +class KOFFICECORE_EXPORT KoGenStyle +{ +public: + /** + * Possible values for the "type" argument of the KoGenStyle constructor. + * Those values can be extended by applications (starting at number 20), + * it's for their own consumption anyway. + * (The reason for having the very common ones here, is to make it possible to + * use them from libkotext). + */ + enum { STYLE_PAGELAYOUT = 0, + STYLE_USER = 1, + STYLE_AUTO = 2, + STYLE_MASTER = 3, + STYLE_LIST = 4, + STYLE_AUTO_LIST = 5, + STYLE_NUMERIC_NUMBER = 6, + STYLE_NUMERIC_DATE = 7, + STYLE_NUMERIC_TIME = 8, + STYLE_NUMERIC_FRACTION = 9, + STYLE_NUMERIC_PERCENTAGE = 10, + STYLE_NUMERIC_SCIENTIFIC = 11, + STYLE_NUMERIC_CURRENCY = 12, + STYLE_NUMERIC_TEXT = 13, + STYLE_HATCH = 14, + STYLE_GRAPHICAUTO = 15}; + + /** + * Start the definition of a new style. Its name will be set later by KoGenStyles::lookup(), + * but first you must define its properties and attributes. + * + * @param type this is a hook for the application to categorize styles + * See the STYLE_* enum. Ignored when writing out the style. + * + * @param familyName The value for style:family, e.g. text, paragraph, graphic etc. + * The family is for style:style elements only; number styles and list styles don't have one. + * + * @param parentName If set, name of the parent style from which this one inherits. + */ + explicit KoGenStyle( int type = 0, const char* familyName = 0, + const QString& parentName = QString::null ); + ~KoGenStyle(); + + /* + * setAutoStyleInStylesDotXml(true) marks a given automatic style as being needed in styles.xml. + * For instance styles used by headers and footers need to go there, since + * they are saved in styles.xml, and styles.xml must be independent from content.xml. + * + * The application should use KoGenStyles::styles( type, true ) in order to retrieve + * those styles and save them separately. + */ + void setAutoStyleInStylesDotXml( bool b ) { m_autoStyleInStylesDotXml = b; } + /// @return the value passed to setAutoStyleInStylesDotXml; false by default + bool autoStyleInStylesDotXml() const { return m_autoStyleInStylesDotXml; } + + /* + * setDefaultStyle(true) marks a given style as being the default style. + * This means we expect that you will call writeStyle( ...,"style:default-style"), + * and its name will be ommitted in the output. + */ + void setDefaultStyle( bool b ) { m_defaultStyle = b; } + /// @return the value passed to setDefaultStyle; false by default + bool isDefaultStyle() const { return m_defaultStyle; } + + /// Return the type of this style, as set in the constructor + int type() const { return m_type; } + + /// Return the family name + const char* familyName() const { return m_familyName.data(); } + + /// Return the name of style's parent, if set + QString parentName() const { return m_parentName; } + + /** + * @brief The types of properties + * + * Simple styles only write one foo-properties tag, in which case they can just use DefaultType. + * However a given style might want to write several kinds of properties, in which case it would + * need to use other property types than the default one. + * + * For instance this style: + * @code + * <style:style style:family="chart"> + * <style:chart-properties .../> + * <style:graphic-properties .../> + * <style:text-properties .../> + * </style:style> + * @endcode + * would use DefaultType for chart-properties (and would pass "style:chart-properties" to writeStyle(), + * and would use GraphicType and TextType. + */ + enum PropertyType + { + /** + * DefaultType depends on family: e.g. paragraph-properties if family=paragraph + * or on the type of style (e.g. page-layout -> page-layout-properties). + * (In fact that tag name is the one passed to writeStyle) + */ + DefaultType = 0, + /// TextType is always text-properties. + TextType, + /// ParagraphType is always paragraph-properties. + ParagraphType, + /// GraphicType is always graphic-properties. + GraphicType, + Reserved1, ///< @internal for binary compatible extensions + Reserved2, ///< @internal for binary compatible extensions + ChildElement, ///< @internal + N_NumTypes ///< @internal - warning, adding items here affects binary compatibility (size of KoGenStyle) + }; + + /// Add a property to the style + void addProperty( const QString& propName, const QString& propValue, PropertyType type = DefaultType ) { + m_properties[type].insert( propName, propValue ); + } + /// Overloaded version of addProperty that takes a char*, usually for "..." + void addProperty( const QString& propName, const char* propValue, PropertyType type = DefaultType ) { + m_properties[type].insert( propName, QString::fromUtf8( propValue ) ); + } + /// Overloaded version of addProperty that converts an int to a string + void addProperty( const QString& propName, int propValue, PropertyType type = DefaultType ) { + m_properties[type].insert( propName, QString::number( propValue ) ); + } + /// Overloaded version of addProperty that converts a bool to a string (false/true) + void addProperty( const QString& propName, bool propValue, PropertyType type = DefaultType ) { + m_properties[type].insert( propName, propValue ? "true" : "false" ); + } + + /** + * Add a property which represents a distance, measured in pt + * The number is written out with the highest possible precision + * (unlike QString::number and setNum, which default to 6 digits), + * and the unit name ("pt") is appended to it. + */ + void addPropertyPt( const QString& propName, double propValue, PropertyType type = DefaultType ); + + /** + * Add an attribute to the style + * The difference between property and attributes is a bit oasis-format-specific: + * attributes are for the style element itself, and properties are in the style:properties child element + */ + void addAttribute( const QString& attrName, const QString& attrValue ) { + m_attributes.insert( attrName, attrValue ); + } + /// Overloaded version of addAttribute that takes a char*, usually for "..." + void addAttribute( const QString& attrName, const char* attrValue ) { + m_attributes.insert( attrName, QString::fromUtf8( attrValue ) ); + } + /// Overloaded version of addAttribute that converts an int to a string + void addAttribute( const QString& attrName, int attrValue ) { + m_attributes.insert( attrName, QString::number( attrValue ) ); + } + + /// Overloaded version of addAttribute that converts a bool to a string + void addAttribute( const QString& attrName, bool attrValue ) { + m_attributes.insert( attrName, attrValue ? "true" : "false" ); + } + + /** + * Add an attribute which represents a distance, measured in pt + * The number is written out with the highest possible precision + * (unlike QString::number and setNum, which default to 6 digits), + * and the unit name ("pt") is appended to it. + */ + void addAttributePt( const QString& attrName, double attrValue ); + + /** + * @brief Add a child element to the style properties. + * + * What is meant here is that the contents of the QString + * will be written out literally. This means you should use + * KoXmlWriter to generate it: + * @code + * QBuffer buffer; + * buffer.open( IO_WriteOnly ); + * KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + * elementWriter.startElement( "..." ); + * ... + * elementWriter.endElement(); + * QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + * gs.addChildElement( "...", elementContents ); + * @endcode + * + * The value of @p elementName isn't used, except that it must be unique. + */ + void addChildElement( const QString& elementName, const QString& elementContents ) { + m_properties[ChildElement].insert( elementName, elementContents ); + } + + /** + * @brief Add a style:map to the style. + * @param styleMap the attributes for the map, associated as (name,value). + */ + void addStyleMap( const QMap<QString, QString>& styleMap ) { + m_maps.append( styleMap ); + } + + /** + * @return true if the style has no attributes, no properties, no style map etc. + * This can be used by applications which do not save all attributes unconditionally, + * but only those that differ from the parent. But note that lookup() can't find this out... + */ + bool isEmpty() const { + if ( !m_attributes.isEmpty() || ! m_maps.isEmpty() ) + return false; + for ( uint i = 0 ; i < N_NumTypes ; ++i ) + if ( ! m_properties[i].isEmpty() ) + return false; + return true; + } + + /** + * Write the definition of this style to @p writer, using the OASIS format. + * @param writer the KoXmlWriter in which @p elementName will be created and filled in + * @param styles the styles collection, used to look up the parent style + * @param elementName the name of the XML element, e.g. "style:style". Don't forget to + * pass style:default-style if isDefaultStyle(). + * @param name must come from the collection. It will be ignored if isDefaultStyle() is true. + * @param propertiesElementName the name of the XML element with the style properties, + * e.g. "style:text-properties". Can be 0 in special cases where there should be no such item, + * in which case the attributes and elements are added under the style itself. + * @param closeElement set it to false to be able to add more child elements to the style element + * @param drawElement set it to true to add "draw:name" (used for gradient/hatch style) otherwise add "style:name" + */ + void writeStyle( KoXmlWriter* writer, KoGenStyles& styles, const char* elementName, const QString& name, + const char* propertiesElementName, bool closeElement = true, bool drawElement = false ) const; + + /** + * QMap requires a complete sorting order. + * Another solution would have been a qdict and a key() here, a la KoTextFormat, + * but the key was difficult to generate. + * Solutions with only a hash value (not representative of the whole data) + * require us to write a hashtable by hand.... + */ + bool operator<( const KoGenStyle &other ) const; + + /// Not needed for QMap, but can still be useful + bool operator==( const KoGenStyle &other ) const; + +private: + QString property( const QString& propName, PropertyType type ) const { + QMap<QString, QString>::const_iterator it = m_properties[type].find( propName ); + if ( it != m_properties[type].end() ) + return it.data(); + return QString::null; + } + + QString attribute( const QString& propName ) const { + QMap<QString, QString>::const_iterator it = m_attributes.find( propName ); + if ( it != m_attributes.end() ) + return it.data(); + return QString::null; + } + + void writeStyleProperties( KoXmlWriter* writer, PropertyType i, + const char* elementName, const KoGenStyle* parentStyle ) const; + +#ifndef NDEBUG + void printDebug() const; +#endif + +private: + // Note that the copy constructor and assignment operator are allowed. + // Better not use pointers below! + int m_type; + QCString m_familyName; + QString m_parentName; + /// We use QMaps since they provide automatic sorting on the key (important for unicity!) + QMap<QString, QString> m_properties[N_NumTypes]; + QMap<QString, QString> m_attributes; + typedef QMap<QString, QString> StyleMap; + QValueVector<StyleMap> m_maps; // we can't really sort the maps between themselves... + + bool m_autoStyleInStylesDotXml; + bool m_defaultStyle; + short m_unused2; + + // For lookup + friend class KoGenStyles; +}; + +#endif /* KOGENSTYLES_H */ diff --git a/lib/kofficecore/KoGlobal.cpp b/lib/kofficecore/KoGlobal.cpp new file mode 100644 index 00000000..87717172 --- /dev/null +++ b/lib/kofficecore/KoGlobal.cpp @@ -0,0 +1,205 @@ +/* This file is part of the KDE project + Copyright (C) 2001 David Faure <faure@kde.org> + Copyright 2003 Nicolas GOUTTE <goutte@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 "config.h" +#include <KoGlobal.h> +#include <kdebug.h> +#include <qfont.h> +#include <qfontinfo.h> +#include <kglobalsettings.h> +#include <kglobal.h> +#include <klocale.h> +#include <ksimpleconfig.h> +#include <kstandarddirs.h> +#include <kstaticdeleter.h> +#include <kimageio.h> +#include <kiconloader.h> +#include <kstandarddirs.h> +#include <locale.h> + +KoGlobal* KoGlobal::s_global = 0L; +static KStaticDeleter<KoGlobal> sdg; + +KoGlobal* KoGlobal::self() +{ + if ( !s_global ) + sdg.setObject( s_global, new KoGlobal ); + return s_global; +} + +KoGlobal::KoGlobal() + : m_pointSize( -1 ), m_kofficeConfig( 0L ) +{ + // Make sure that QCString::setNum doesn't give us "," as decimal point, e.g. when saving to OpenDocument. + setlocale( LC_NUMERIC, "C" ); + + // Install the libkoffice* translations + KGlobal::locale()->insertCatalogue("koffice"); + + KImageIO::registerFormats(); + + // Tell KStandardDirs about the koffice prefix + KGlobal::dirs()->addPrefix(PREFIX); + + // Tell the iconloader about share/apps/koffice/icons + KGlobal::iconLoader()->addAppDir("koffice"); + + // Another way to get the DPI of the display would be QPaintDeviceMetrics, + // but we have no widget here (and moving this to KoView wouldn't allow + // using this from the document easily). +#ifdef Q_WS_X11 + m_dpiX = QPaintDevice::x11AppDpiX(); + m_dpiY = QPaintDevice::x11AppDpiY(); +#else + m_dpiX = 75; + m_dpiY = 75; +#endif +} + +KoGlobal::~KoGlobal() +{ + delete m_kofficeConfig; +} + +QFont KoGlobal::_defaultFont() +{ + QFont font = KGlobalSettings::generalFont(); + // we have to use QFontInfo, in case the font was specified with a pixel size + if ( font.pointSize() == -1 ) + { + // cache size into m_pointSize, since QFontInfo loads the font -> slow + if ( m_pointSize == -1 ) + m_pointSize = QFontInfo(font).pointSize(); + Q_ASSERT( m_pointSize != -1 ); + font.setPointSize( m_pointSize ); + } + //kdDebug()<<k_funcinfo<<"QFontInfo(font).pointSize() :"<<QFontInfo(font).pointSize()<<endl; + //kdDebug()<<k_funcinfo<<"font.name() :"<<font.family ()<<endl; + return font; +} + +QStringList KoGlobal::_listOfLanguageTags() +{ + if ( m_langMap.isEmpty() ) + createListOfLanguages(); + return m_langMap.values(); +} + +QStringList KoGlobal::_listOfLanguages() +{ + if ( m_langMap.empty() ) + createListOfLanguages(); + return m_langMap.keys(); +} + +void KoGlobal::createListOfLanguages() +{ + KConfig config( "all_languages", true, false, "locale" ); + // Note that we could also use KLocale::allLanguagesTwoAlpha + + QMap<QString, bool> seenLanguages; + const QStringList langlist = config.groupList(); + for ( QStringList::ConstIterator itall = langlist.begin(); + itall != langlist.end(); ++itall ) + { + const QString tag = *itall; + config.setGroup( tag ); + const QString name = config.readEntry("Name", tag); + // e.g. name is "French" and tag is "fr" + + // The QMap does the sorting on the display-name, so that + // comboboxes are sorted. + m_langMap.insert( name, tag ); + + seenLanguages.insert( tag, true ); + } + + // Also take a look at the installed translations. + // Many of them are already in all_languages but all_languages doesn't + // currently have en_GB or en_US etc. + + const QStringList translationList = KGlobal::dirs()->findAllResources("locale", + QString::fromLatin1("*/entry.desktop")); + for ( QStringList::ConstIterator it = translationList.begin(); + it != translationList.end(); ++it ) + { + // Extract the language tag from the directory name + QString tag = *it; + int index = tag.findRev('/'); + tag = tag.left(index); + index = tag.findRev('/'); + tag = tag.mid(index+1); + + if ( seenLanguages.find( tag ) == seenLanguages.end() ) { + KSimpleConfig entry(*it); + entry.setGroup("KCM Locale"); + + const QString name = entry.readEntry("Name", tag); + // e.g. name is "US English" and tag is "en_US" + m_langMap.insert( name, tag ); + + // enable this if writing a third way of finding languages below + //seenLanguages.insert( tag, true ); + } + + } + + // #### We also might not have an entry for a language where spellchecking is supported, + // but no KDE translation is available, like fr_CA. + // How to add them? +} + +QString KoGlobal::tagOfLanguage( const QString & _lang) +{ + const LanguageMap& map = self()->m_langMap; + QMap<QString,QString>::ConstIterator it = map.find( _lang ); + if ( it != map.end() ) + return *it; + return QString::null; +} + +QString KoGlobal::languageFromTag( const QString &langTag ) +{ + const LanguageMap& map = self()->m_langMap; + QMap<QString,QString>::ConstIterator it = map.begin(); + const QMap<QString,QString>::ConstIterator end = map.end(); + for ( ; it != end; ++it ) + if ( it.data() == langTag ) + return it.key(); + + // Language code not found. Better return the code (tag) than nothing. + return langTag; +} + +KConfig* KoGlobal::_kofficeConfig() +{ + if ( !m_kofficeConfig ) { + m_kofficeConfig = new KConfig( "kofficerc" ); + } + return m_kofficeConfig; +} + +void KoGlobal::setDPI( int x, int y ) +{ + //kdDebug() << k_funcinfo << x << "," << y << endl; + KoGlobal* s = self(); + s->m_dpiX = x; + s->m_dpiY = y; +} diff --git a/lib/kofficecore/KoGlobal.h b/lib/kofficecore/KoGlobal.h new file mode 100644 index 00000000..bab9c261 --- /dev/null +++ b/lib/kofficecore/KoGlobal.h @@ -0,0 +1,107 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright 2002, 2003 David Faure <faure@kde.org> + Copyright 2003 Nicolas GOUTTE <goutte@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. +*/ + +#ifndef koGlobal_h +#define koGlobal_h + +#include <qstringlist.h> +#include <qfont.h> +#include <qmap.h> +class KConfig; + +#include <koffice_export.h> +class KOFFICECORE_EXPORT KoGlobal +{ +public: + /// For KoApplication + static void initialize() { + (void)self(); // I don't want to make KGlobal instances public, so self() is private + } + /** + * Return the default font for KOffice programs. + * This is (currently) the same as the KDE-global default font, + * except that it is guaranteed to have a point size set, + * never a pixel size (see @ref QFont). + */ + static QFont defaultFont() { + return self()->_defaultFont(); + } + + /** + * @return the global KConfig object around kofficerc. + * kofficerc is used for KOffice-wide settings, from totally unrelated classes, + * so this is the centralization of the KConfig object so that the file is + * parsed only once + */ + static KConfig* kofficeConfig() { + return self()->_kofficeConfig(); + } + + static int dpiX() { + return self()->m_dpiX; + } + static int dpiY() { + return self()->m_dpiY; + } + /// @internal, for KoApplication + static void setDPI( int x, int y ); + + /// Return the list of available languages, in their displayable form + /// (translated names) + static QStringList listOfLanguages() { + return self()->_listOfLanguages(); + } + /// Return the list of available languages, in their internal form + /// e.g. "fr" or "en_US", here called "tag" + static QStringList listTagOfLanguages() { // TODO rename to listOfLanguageTags + return self()->_listOfLanguageTags(); + } + /// For a given language display name, return its tag + static QString tagOfLanguage( const QString & _lang ); + /// For a given language tag, return its display name + static QString languageFromTag( const QString &_lang ); + + ~KoGlobal(); + +private: + static KoGlobal* self(); + KoGlobal(); + QFont _defaultFont(); + QStringList _listOfLanguages(); + QStringList _listOfLanguageTags(); + KConfig* _kofficeConfig(); + void createListOfLanguages(); + + int m_pointSize; + typedef QMap<QString, QString> LanguageMap; + LanguageMap m_langMap; // display-name -> language tag + KConfig* m_kofficeConfig; + int m_dpiX; + int m_dpiY; + // No BC problem here, constructor is private, feel free to add members + + // Singleton pattern. Maybe this should even be refcounted, so + // that it gets cleaned up when closing all koffice parts in e.g. konqueror? + static KoGlobal* s_global; + friend class this_is_a_singleton; // work around gcc warning +}; + +#endif // koGlobal diff --git a/lib/kofficecore/KoMainWindow.cpp b/lib/kofficecore/KoMainWindow.cpp new file mode 100644 index 00000000..bdcfaa6a --- /dev/null +++ b/lib/kofficecore/KoMainWindow.cpp @@ -0,0 +1,1701 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright (C) 2000-2006 David Faure <faure@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 "KoMainWindow.h" +#include "KoDocument.h" +#include "KoView.h" +#include "KoFilterManager.h" +#include "KoDocumentInfo.h" +#include "KoDocumentInfoDlg.h" +#include "KoQueryTrader.h" +#include "KoMainWindowIface.h" +#include "KoFrame.h" +#include "KoFileDialog.h" +#include "Koversiondialog.h" +#include "kkbdaccessextensions.h" +#include "KoSpeaker.h" + +#include <kaboutdata.h> +#include <kprinter.h> +#include <kdeversion.h> +#include <kstdaction.h> +#include <kapplication.h> +#include <kmessagebox.h> +#include <kstandarddirs.h> +#include <kio/netaccess.h> +#include <kkeydialog.h> +#include <kedittoolbar.h> +#include <kprogress.h> +#include <kpushbutton.h> +#include <kdebug.h> +#include <ktempfile.h> +#include <krecentdocument.h> +#include <kparts/partmanager.h> +#include <kparts/plugin.h> +#include <kparts/event.h> +#include <klocale.h> +#include <kstatusbar.h> +#include <kglobalsettings.h> +#include <ksharedptr.h> + +#include <qobjectlist.h> + +#include <unistd.h> +#include <stdlib.h> + +class KoPartManager : public KParts::PartManager +{ +public: + KoPartManager( QWidget * parent, const char * name = 0L ) + : KParts::PartManager( parent, name ) + { + setSelectionPolicy( KParts::PartManager::TriState ); + setAllowNestedParts( true ); + setIgnoreScrollBars( true ); + // Allow right-click on embedded objects (without activating them) + // But beware: this means right-click on parent, from embedded object, + // doesn't make the parent active first... + setActivationButtonMask( Qt::LeftButton | Qt::MidButton ); + } + virtual bool eventFilter( QObject *obj, QEvent *ev ) + { + if ( !obj->isWidgetType() || ::qt_cast<KoFrame *>( obj ) ) + return false; + return KParts::PartManager::eventFilter( obj, ev ); + } +}; + +class KoMainWindowPrivate +{ +public: + KoMainWindowPrivate() + { + m_rootDoc = 0L; + m_docToOpen = 0L; + m_manager = 0L; + bMainWindowGUIBuilt = false; + m_forQuit=false; + m_splitted=false; + m_activePart = 0L; + m_activeView = 0L; + m_splitter=0L; + m_orientation=0L; + m_removeView=0L; + m_toolbarList.setAutoDelete( true ); + m_firstTime=true; + m_progress=0L; + m_paDocInfo = 0; + m_paSave = 0; + m_paSaveAs = 0; + m_paPrint = 0; + m_paPrintPreview = 0; + statusBarLabel = 0L; + m_dcopObject = 0; + m_sendfile = 0; + m_paCloseFile = 0L; + m_reloadfile = 0L; + m_versionsfile = 0L; + m_importFile = 0; + m_exportFile = 0; + m_isImporting = false; + m_isExporting = false; + m_windowSizeDirty = false; + m_lastExportSpecialOutputFlag = 0; + + // TTS accessibility enhancement (only if KDE 3.4 or later and KTTSD daemon is installed.) + if (KoSpeaker::isKttsdInstalled()) { + if (kospeaker) + m_koSpeaker = kospeaker; + else + m_koSpeaker = new KoSpeaker(); + } else + m_koSpeaker = 0; + } + ~KoMainWindowPrivate() + { + delete m_dcopObject; + } + + KoDocument *m_rootDoc; + KoDocument *m_docToOpen; + QPtrList<KoView> m_rootViews; + KParts::PartManager *m_manager; + + KParts::Part *m_activePart; + KoView *m_activeView; + + QLabel * statusBarLabel; + KProgress *m_progress; + + QPtrList<KAction> m_splitViewActionList; + // This additional list is needed, because we don't plug + // the first list, when an embedded view gets activated (Werner) + QPtrList<KAction> m_veryHackyActionList; + QSplitter *m_splitter; + KSelectAction *m_orientation; + KAction *m_removeView; + KoMainWindowIface *m_dcopObject; + + QPtrList <KAction> m_toolbarList; + + bool bMainWindowGUIBuilt; + bool m_splitted; + bool m_forQuit; + bool m_firstTime; + bool m_windowSizeDirty; + + KAction *m_paDocInfo; + KAction *m_paSave; + KAction *m_paSaveAs; + KAction *m_paPrint; + KAction *m_paPrintPreview; + KAction *m_sendfile; + KAction *m_paCloseFile; + KAction *m_reloadfile; + KAction *m_versionsfile; + KAction *m_importFile; + KAction *m_exportFile; + + bool m_isImporting; + bool m_isExporting; + + KURL m_lastExportURL; + QCString m_lastExportFormat; + int m_lastExportSpecialOutputFlag; + + KSharedPtr<KoSpeaker> m_koSpeaker; +}; + +KoMainWindow::KoMainWindow( KInstance *instance, const char* name ) + : KParts::MainWindow( name ) +{ + setStandardToolBarMenuEnabled(true); // should there be a check for >= 3.1 ? + Q_ASSERT(instance); + d = new KoMainWindowPrivate; + + d->m_manager = new KoPartManager( this ); + + connect( d->m_manager, SIGNAL( activePartChanged( KParts::Part * ) ), + this, SLOT( slotActivePartChanged( KParts::Part * ) ) ); + + if ( instance ) + setInstance( instance, false ); // don't load plugins! we don't want + // the part's plugins with this shell, even though we are using the + // part's instance! (Simon) + + QString doc; + QStringList allFiles = KGlobal::dirs()->findAllResources( "data", "koffice/koffice_shell.rc" ); + setXMLFile( findMostRecentXMLFile( allFiles, doc ) ); + setLocalXMLFile( locateLocal( "data", "koffice/koffice_shell.rc" ) ); + + KStdAction::openNew( this, SLOT( slotFileNew() ), actionCollection(), "file_new" ); + KStdAction::open( this, SLOT( slotFileOpen() ), actionCollection(), "file_open" ); + m_recent = KStdAction::openRecent( this, SLOT(slotFileOpenRecent(const KURL&)), actionCollection() ); + d->m_paSave = KStdAction::save( this, SLOT( slotFileSave() ), actionCollection(), "file_save" ); + d->m_paSaveAs = KStdAction::saveAs( this, SLOT( slotFileSaveAs() ), actionCollection(), "file_save_as" ); + d->m_paPrint = KStdAction::print( this, SLOT( slotFilePrint() ), actionCollection(), "file_print" ); + d->m_paPrintPreview = KStdAction::printPreview( this, SLOT( slotFilePrintPreview() ), actionCollection(), "file_print_preview" ); + d->m_sendfile = KStdAction::mail( this, SLOT( slotEmailFile() ), actionCollection(), "file_send_file"); + + d->m_paCloseFile = KStdAction::close( this, SLOT( slotFileClose() ), actionCollection(), "file_close" ); + KStdAction::quit( this, SLOT( slotFileQuit() ), actionCollection(), "file_quit" ); + + d->m_reloadfile = new KAction( i18n( "Reload"), 0, + this, SLOT( slotReloadFile() ), + actionCollection(), "file_reload_file"); + + d->m_versionsfile = new KAction( i18n( "Versions..."), 0, + this, SLOT( slotVersionsFile() ), + actionCollection(), "file_versions_file"); + + d->m_importFile = new KAction( i18n( "I&mport..." ), 0, // clashing accel key :( + this, SLOT( slotImportFile() ), + actionCollection(), "file_import_file"); + d->m_exportFile = new KAction( i18n( "E&xport..." ), 0, + this, SLOT( slotExportFile() ), + actionCollection(), "file_export_file"); + + /* The following entry opens the document information dialog. Since the action is named so it + intends to show data this entry should not have a trailing ellipses (...). */ + d->m_paDocInfo = new KAction( i18n( "&Document Information" ), "documentinfo", 0, + this, SLOT( slotDocumentInfo() ), + actionCollection(), "file_documentinfo" ); + + KStdAction::keyBindings( this, SLOT( slotConfigureKeys() ), actionCollection() ); + KStdAction::configureToolbars( this, SLOT( slotConfigureToolbars() ), actionCollection() ); + + d->m_paDocInfo->setEnabled( false ); + d->m_paSaveAs->setEnabled( false ); + d->m_reloadfile->setEnabled( false ); + d->m_versionsfile->setEnabled( false ); + d->m_importFile->setEnabled( true ); // always enabled like File --> Open + d->m_exportFile->setEnabled( false ); + d->m_paSave->setEnabled( false ); + d->m_paPrint->setEnabled( false ); + d->m_paPrintPreview->setEnabled( false ); + d->m_sendfile->setEnabled( false); + d->m_paCloseFile->setEnabled( false); + + d->m_splitter=new QSplitter(Qt::Vertical, this, "mw-splitter"); + setCentralWidget( d->m_splitter ); + // Keyboard accessibility enhancements. + new KKbdAccessExtensions(this, "mw-panelSizer"); + // set up the action "list" for "Close all Views" (hacky :) (Werner) + d->m_veryHackyActionList.append( + new KAction(i18n("&Close All Views"), "fileclose", + "ctrl+shift+w", this, SLOT(slotCloseAllViews()), + actionCollection(), "view_closeallviews") ); + + // set up the action list for the splitter stuff + d->m_splitViewActionList.append(new KAction(i18n("&Split View"), "view_split", 0, + this, SLOT(slotSplitView()), + actionCollection(), "view_split")); + d->m_removeView=new KAction(i18n("&Remove View"), "view_remove", 0, + this, SLOT(slotRemoveView()), + actionCollection(), "view_rm_splitter"); + d->m_splitViewActionList.append(d->m_removeView); + d->m_removeView->setEnabled(false); + d->m_orientation=new KSelectAction(i18n("Splitter &Orientation"), "view_orientation", 0, + this, SLOT(slotSetOrientation()), + actionCollection(), "view_splitter_orientation"); + QStringList items; + items << i18n("&Vertical") + << i18n("&Horizontal"); + d->m_orientation->setItems(items); + d->m_orientation->setCurrentItem(static_cast<int>(d->m_splitter->orientation())); + d->m_splitViewActionList.append(d->m_orientation); + d->m_splitViewActionList.append(new KActionSeparator(this)); + + // Load list of recent files + KConfig * config = instance ? instance->config() : KGlobal::config(); + m_recent->loadEntries( config ); + + createShellGUI(); + d->bMainWindowGUIBuilt = true; + + if ( !initialGeometrySet() ) + { + // Default size + const int deskWidth = KGlobalSettings::desktopGeometry(this).width(); + if (deskWidth > 1100) // very big desktop ? + resize( 1000, 800 ); + if (deskWidth > 850) // big desktop ? + resize( 800, 600 ); + else // small (800x600, 640x480) desktop + resize( 600, 400 ); + } + + // Saved size + config->setGroup( "MainWindow" ); + //kdDebug(30003) << "KoMainWindow::restoreWindowSize" << endl; + restoreWindowSize( config ); +} + +KoMainWindow::~KoMainWindow() +{ + // The doc and view might still exist (this is the case when closing the window) + if (d->m_rootDoc) + d->m_rootDoc->removeShell(this); + + if (d->m_docToOpen) { + d->m_docToOpen->removeShell(this); + delete d->m_docToOpen; + } + + // safety first ;) + d->m_manager->setActivePart(0); + + if(d->m_rootViews.findRef(d->m_activeView)==-1) { + delete d->m_activeView; + d->m_activeView=0L; + } + d->m_rootViews.setAutoDelete( true ); + d->m_rootViews.clear(); + + // We have to check if this was a root document. + // -> We aren't allowed to delete the (embedded) document! + // This has to be checked from queryClose, too :) + if ( d->m_rootDoc && d->m_rootDoc->viewCount() == 0 && + !d->m_rootDoc->isEmbedded()) + { + //kdDebug(30003) << "Destructor. No more views, deleting old doc " << d->m_rootDoc << endl; + delete d->m_rootDoc; + } + + delete d->m_manager; + delete d; +} + +void KoMainWindow::setRootDocument( KoDocument *doc ) +{ + if ( d->m_rootDoc == doc ) + return; + + if (d->m_docToOpen && d->m_docToOpen != doc) { + d->m_docToOpen->removeShell(this); + delete d->m_docToOpen; + d->m_docToOpen = 0; + } else { + d->m_docToOpen = 0; + } + + //kdDebug(30003) << "KoMainWindow::setRootDocument this = " << this << " doc = " << doc << endl; + QPtrList<KoView> oldRootViews = d->m_rootViews; + d->m_rootViews.clear(); + KoDocument *oldRootDoc = d->m_rootDoc; + + if ( oldRootDoc ) + oldRootDoc->removeShell( this ); + + d->m_rootDoc = doc; + + if ( doc ) + { + doc->setSelectable( false ); + //d->m_manager->addPart( doc, false ); // done by KoView::setPartManager + d->m_rootViews.append( doc->createView( d->m_splitter, "view" /*not unique, but better than unnamed*/ ) ); + d->m_rootViews.current()->setPartManager( d->m_manager ); + + d->m_rootViews.current()->show(); + // The addShell has been done already if using openURL + if ( !d->m_rootDoc->shells().contains( this ) ) + d->m_rootDoc->addShell( this ); + d->m_removeView->setEnabled(false); + d->m_orientation->setEnabled(false); + } + + bool enable = d->m_rootDoc != 0 ? true : false; + d->m_paDocInfo->setEnabled( enable ); + d->m_paSave->setEnabled( enable ); + d->m_paSaveAs->setEnabled( enable ); + d->m_importFile->setEnabled( enable ); + d->m_exportFile->setEnabled( enable ); + d->m_paPrint->setEnabled( enable ); + d->m_paPrintPreview->setEnabled( enable ); + d->m_sendfile->setEnabled( enable); + d->m_paCloseFile->setEnabled( enable); + updateCaption(); + + d->m_manager->setActivePart( d->m_rootDoc, d->m_rootViews.current() ); + + oldRootViews.setAutoDelete( true ); + oldRootViews.clear(); + + if ( oldRootDoc && oldRootDoc->viewCount() == 0 ) + { + //kdDebug(30003) << "No more views, deleting old doc " << oldRootDoc << endl; + delete oldRootDoc; + } +} + +void KoMainWindow::updateReloadFileAction(KoDocument *doc) +{ + d->m_reloadfile->setEnabled( doc && !doc->url().isEmpty() ); +} + +void KoMainWindow::updateVersionsFileAction(KoDocument *doc) +{ + //TODO activate it just when we save it in oasis file format + d->m_versionsfile->setEnabled( doc && !doc->url().isEmpty()&&doc->isModified()); +} + + +void KoMainWindow::setRootDocumentDirect( KoDocument *doc, const QPtrList<KoView> & views ) +{ + d->m_rootDoc = doc; + d->m_rootViews = views; + bool enable = d->m_rootDoc != 0 ? true : false; + d->m_paDocInfo->setEnabled( enable ); + d->m_paSave->setEnabled( enable ); + d->m_paSaveAs->setEnabled( enable ); + d->m_exportFile->setEnabled( enable ); + d->m_paPrint->setEnabled( enable ); + d->m_paPrintPreview->setEnabled( enable ); + d->m_sendfile->setEnabled( enable); + d->m_paCloseFile->setEnabled( enable ); +} + +void KoMainWindow::addRecentURL( const KURL& url ) +{ + kdDebug(30003) << "KoMainWindow::addRecentURL url=" << url.prettyURL() << endl; + // Add entry to recent documents list + // (call coming from KoDocument because it must work with cmd line, template dlg, file/open, etc.) + if ( !url.isEmpty() ) + { + bool ok = true; + if ( url.isLocalFile() ) + { + QString path = url.path( -1 ); + QStringList tmpDirs = KGlobal::dirs()->resourceDirs( "tmp" ); + for ( QStringList::Iterator it = tmpDirs.begin() ; ok && it != tmpDirs.end() ; ++it ) + if ( path.contains( *it ) ) + ok = false; // it's in the tmp resource + if ( ok ) + KRecentDocument::add(path); + } + else + KRecentDocument::add(url.url(-1), true); + + if ( ok ) + m_recent->addURL( url ); + saveRecentFiles(); + } +} + +void KoMainWindow::saveRecentFiles() +{ + // Save list of recent files + KConfig * config = instance() ? instance()->config() : KGlobal::config(); + kdDebug(30003) << this << " Saving recent files list into config. instance()=" << instance() << endl; + m_recent->saveEntries( config ); + config->sync(); + if (KMainWindow::memberList) + { + // Tell all windows to reload their list, after saving + // Doesn't work multi-process, but it's a start + KMainWindow *window = KMainWindow::memberList->first(); + for (; window; window = KMainWindow::memberList->next()) + static_cast<KoMainWindow *>(window)->reloadRecentFileList(); + } +} + +void KoMainWindow::reloadRecentFileList() +{ + KConfig * config = instance() ? instance()->config() : KGlobal::config(); + m_recent->loadEntries( config ); +} + +KoDocument* KoMainWindow::createDoc() const +{ + KoDocumentEntry entry = KoDocumentEntry( KoDocument::readNativeService() ); + return entry.createDoc(); +} + +void KoMainWindow::updateCaption() +{ + //kdDebug(30003) << "KoMainWindow::updateCaption()" << endl; + if ( !d->m_rootDoc ) + setCaption(QString::null); + else if ( rootDocument()->isCurrent() ) + { + QString caption; + // Get caption from document info (title(), in about page) + if ( rootDocument()->documentInfo() ) + { + KoDocumentInfoPage * page = rootDocument()->documentInfo()->page( QString::fromLatin1("about") ); + if (page) + caption = static_cast<KoDocumentInfoAbout *>(page)->title(); + } + const QString url = rootDocument()->url().prettyURL( 0, KURL::StripFileProtocol ); + if ( !caption.isEmpty() && !url.isEmpty() ) + caption = QString( "%1 - %2" ).arg( caption ).arg( url ); + else if ( caption.isEmpty() ) + caption = url; + + setCaption( caption, rootDocument()->isModified() ); + if ( !rootDocument()->url().fileName(false).isEmpty() ) + d->m_paSave->setToolTip( i18n("Save as %1").arg(rootDocument()->url().fileName(false)) ); + else + d->m_paSave->setToolTip( i18n("Save") ); + } +} + +void KoMainWindow::updateCaption( QString caption, bool mod ) +{ + //kdDebug(30003)<<"KoMainWindow::updateCaption("<<caption<<","<<mod<<")"<<endl; + setCaption( caption, mod ); +} + +KoDocument *KoMainWindow::rootDocument() const +{ + return d->m_rootDoc; +} + +KoView *KoMainWindow::rootView() const +{ + if(d->m_rootViews.find(d->m_activeView)!=-1) + return d->m_activeView; + return d->m_rootViews.first(); +} + +KParts::PartManager *KoMainWindow::partManager() +{ + return d->m_manager; +} + +bool KoMainWindow::openDocument( const KURL & url ) +{ + if ( !KIO::NetAccess::exists(url,true,0) ) + { + KMessageBox::error(0L, i18n("The file %1 does not exist.").arg(url.url()) ); + m_recent->removeURL(url); //remove the file from the recent-opened-file-list + saveRecentFiles(); + return false; + } + return openDocumentInternal( url ); +} + +// (not virtual) +bool KoMainWindow::openDocument( KoDocument *newdoc, const KURL & url ) +{ + if (!KIO::NetAccess::exists(url,true,0) ) + { + if (!newdoc->checkAutoSaveFile()) + { + newdoc->initEmpty(); //create an emtpy document + } + + setRootDocument( newdoc ); + newdoc->setURL(url); + QString mime = KMimeType::findByURL(url)->name(); + if ( mime.isEmpty() || mime == KMimeType::defaultMimeType() ) + mime = newdoc->nativeFormatMimeType(); + if ( url.isLocalFile() ) // workaround for kde<=3.3 kparts bug, fixed for 3.4 + newdoc->setFile(url.path()); + newdoc->setMimeTypeAfterLoading( mime ); + updateCaption(); + return true; + } + return openDocumentInternal( url, newdoc ); +} + +// ## If you modify anything here, please check KoShellWindow::openDocumentInternal +bool KoMainWindow::openDocumentInternal( const KURL & url, KoDocument *newdoc ) +{ + //kdDebug(30003) << "KoMainWindow::openDocument " << url.url() << endl; + + if ( !newdoc ) + newdoc = createDoc(); + if ( !newdoc ) + return false; + + d->m_firstTime=true; + connect(newdoc, SIGNAL(sigProgress(int)), this, SLOT(slotProgress(int))); + connect(newdoc, SIGNAL(completed()), this, SLOT(slotLoadCompleted())); + connect(newdoc, SIGNAL(canceled( const QString & )), this, SLOT(slotLoadCanceled( const QString & ))); + newdoc->addShell( this ); // used by openURL + bool openRet = (!isImporting ()) ? newdoc->openURL(url) : newdoc->import(url); + if(!openRet) + { + newdoc->removeShell(this); + delete newdoc; + return false; + } + updateReloadFileAction(newdoc); + updateVersionsFileAction( newdoc ); + return true; +} + +// Separate from openDocument to handle async loading (remote URLs) +void KoMainWindow::slotLoadCompleted() +{ + kdDebug(30003) << "KoMainWindow::slotLoadCompleted" << endl; + KoDocument* doc = rootDocument(); + KoDocument* newdoc = (KoDocument *)(sender()); + + if ( doc && doc->isEmpty() && !doc->isEmbedded() ) + { + // Replace current empty document + setRootDocument( newdoc ); + } + else if ( doc && !doc->isEmpty() ) + { + // Open in a new shell + // (Note : could create the shell first and the doc next for this + // particular case, that would give a better user feedback...) + KoMainWindow *s = new KoMainWindow( newdoc->instance() ); + s->show(); + newdoc->removeShell( this ); + s->setRootDocument( newdoc ); + } + else + { + // We had no document, set the new one + setRootDocument( newdoc ); + } + disconnect(newdoc, SIGNAL(sigProgress(int)), this, SLOT(slotProgress(int))); + disconnect(newdoc, SIGNAL(completed()), this, SLOT(slotLoadCompleted())); + disconnect(newdoc, SIGNAL(canceled( const QString & )), this, SLOT(slotLoadCanceled( const QString & ))); +} + +void KoMainWindow::slotLoadCanceled( const QString & errMsg ) +{ + kdDebug(30003) << "KoMainWindow::slotLoadCanceled" << endl; + if ( !errMsg.isEmpty() ) // empty when canceled by user + KMessageBox::error( this, errMsg ); + // ... can't delete the document, it's the one who emitted the signal... + + KoDocument* newdoc = (KoDocument *)(sender()); + disconnect(newdoc, SIGNAL(sigProgress(int)), this, SLOT(slotProgress(int))); + disconnect(newdoc, SIGNAL(completed()), this, SLOT(slotLoadCompleted())); + disconnect(newdoc, SIGNAL(canceled( const QString & )), this, SLOT(slotLoadCanceled( const QString & ))); + + newdoc->removeShell(this); + delete newdoc; +} + +void KoMainWindow::slotSaveCanceled( const QString &errMsg ) +{ + kdDebug(30003) << "KoMainWindow::slotSaveCanceled" << endl; + if ( !errMsg.isEmpty() ) // empty when canceled by user + KMessageBox::error( this, errMsg ); + slotSaveCompleted(); +} + +void KoMainWindow::slotSaveCompleted() +{ + kdDebug(30003) << "KoMainWindow::slotSaveCompleted" << endl; + KoDocument* pDoc = (KoDocument *)(sender()); + disconnect(pDoc, SIGNAL(sigProgress(int)), this, SLOT(slotProgress(int))); + disconnect(pDoc, SIGNAL(completed()), this, SLOT(slotSaveCompleted())); + disconnect(pDoc, SIGNAL(canceled( const QString & )), + this, SLOT(slotSaveCanceled( const QString & ))); +} + +// returns true if we should save, false otherwise. +bool KoMainWindow::exportConfirmation( const QCString &outputFormat ) +{ + if (!rootDocument()->wantExportConfirmation()) return true; + KMimeType::Ptr mime = KMimeType::mimeType( outputFormat ); + + const bool neverHeardOfIt = ( mime->name() == KMimeType::defaultMimeType() ); + QString comment = neverHeardOfIt ? + i18n( "%1 (unknown file type)" ).arg( outputFormat ) + : mime->comment(); + + // Warn the user + int ret; + if (!isExporting ()) // File --> Save + { + ret = KMessageBox::warningContinueCancel + ( + this, + i18n( "<qt>Saving as a %1 may result in some loss of formatting." + "<p>Do you still want to save in this format?</qt>" ) + .arg( QString( "<b>%1</b>" ).arg( comment ) ), // in case we want to remove the bold later + i18n( "Confirm Save" ), + KStdGuiItem::save (), + "NonNativeSaveConfirmation", + true + ); + } + else // File --> Export + { + ret = KMessageBox::warningContinueCancel + ( + this, + i18n( "<qt>Exporting as a %1 may result in some loss of formatting." + "<p>Do you still want to export to this format?</qt>" ) + .arg( QString( "<b>%1</b>" ).arg( comment ) ), // in case we want to remove the bold later + i18n( "Confirm Export" ), + i18n ("Export"), + "NonNativeExportConfirmation", // different to the one used for Save (above) + true + ); + } + + return (ret == KMessageBox::Continue); +} + +bool KoMainWindow::saveDocument( bool saveas, bool silent ) +{ + KoDocument* pDoc = rootDocument(); + if(!pDoc) + return true; + + bool reset_url; + if ( pDoc->url().isEmpty() ) + { + emit saveDialogShown(); + reset_url = true; + saveas = true; + } + else + reset_url = false; + + connect(pDoc, SIGNAL(sigProgress(int)), this, SLOT(slotProgress(int))); + connect(pDoc, SIGNAL(completed()), this, SLOT(slotSaveCompleted())); + connect(pDoc, SIGNAL(canceled( const QString & )), + this, SLOT(slotSaveCanceled( const QString & ))); + + KURL oldURL = pDoc->url(); + QString oldFile = pDoc->file(); + QCString _native_format = pDoc->nativeFormatMimeType(); + QCString oldOutputFormat = pDoc->outputMimeType(); + int oldSpecialOutputFlag = pDoc->specialOutputFlag(); + KURL suggestedURL = pDoc->url(); + + QStringList mimeFilter = KoFilterManager::mimeFilter( _native_format, KoFilterManager::Export, pDoc->extraNativeMimeTypes() ); + if (mimeFilter.findIndex (oldOutputFormat) < 0 && !isExporting()) + { + kdDebug(30003) << "KoMainWindow::saveDocument no export filter for '" << oldOutputFormat << "'" << endl; + + // --- don't setOutputMimeType in case the user cancels the Save As + // dialog and then tries to just plain Save --- + + // suggest a different filename extension (yes, we fortunately don't all live in a world of magic :)) + QString suggestedFilename = suggestedURL.fileName (); + if ( !suggestedFilename.isEmpty () ) // ".kwd" looks strange for a name + { + int c = suggestedFilename.findRev ('.'); + + KMimeType::Ptr mime = KMimeType::mimeType( _native_format ); + QString ext = mime->property( "X-KDE-NativeExtension" ).toString(); + if (!ext.isEmpty ()) + { + if (c < 0) + suggestedFilename += ext; + else + suggestedFilename = suggestedFilename.left (c) + ext; + } + else // current filename extension wrong anyway + { + if (c > 0) + { + // this assumes that a . signifies an extension, not just a . + suggestedFilename = suggestedFilename.left (c); + } + } + + suggestedURL.setFileName (suggestedFilename); + } + + // force the user to choose outputMimeType + saveas = true; + } + + bool ret = false; + + if ( pDoc->url().isEmpty() || saveas ) + { + // if you're just File/Save As'ing to change filter options you + // don't want to be reminded about overwriting files etc. + bool justChangingFilterOptions = false; + + KoFileDialog *dialog = new KoFileDialog( (isExporting() && !d->m_lastExportURL.isEmpty() )? d->m_lastExportURL.url () : suggestedURL.url (), + QString::null, this, "file dialog", true); + + if (!isExporting()) + dialog->setCaption( i18n("Save Document As") ); + else + dialog->setCaption( i18n("Export Document As") ); + + dialog->setOperationMode( KFileDialog::Saving ); + dialog->setSpecialMimeFilter( mimeFilter, + isExporting() ? d->m_lastExportFormat : pDoc->mimeType(), + isExporting() ? d->m_lastExportSpecialOutputFlag : oldSpecialOutputFlag, + _native_format, + pDoc->supportedSpecialFormats() ); + + KURL newURL; + QCString outputFormat = _native_format; + int specialOutputFlag = 0; + bool bOk; + do { + bOk=true; + if(dialog->exec()==QDialog::Accepted) { + newURL=dialog->selectedURL(); + outputFormat=dialog->currentMimeFilter().latin1(); + specialOutputFlag = dialog->specialEntrySelected(); + kdDebug(30003) << "KoMainWindow::saveDocument outputFormat = " << outputFormat << endl; + + if (!isExporting()) + justChangingFilterOptions = (newURL == pDoc->url()) && + (outputFormat == pDoc->mimeType()) && + (specialOutputFlag == oldSpecialOutputFlag); + else + justChangingFilterOptions = (newURL == d->m_lastExportURL) && + (outputFormat == d->m_lastExportFormat) && + (specialOutputFlag == d->m_lastExportSpecialOutputFlag); + } + else + { + bOk = false; + break; + } + + if ( newURL.isEmpty() ) + { + bOk = false; + break; + } + + // adjust URL before doing checks on whether the file exists. + if ( specialOutputFlag == KoDocument::SaveAsDirectoryStore ) { + QString fileName = newURL.fileName(); + if ( fileName != "content.xml" ) { + newURL.addPath( "content.xml" ); + } + } + + // this file exists and we are not just clicking "Save As" to change filter options + // => ask for confirmation + if ( KIO::NetAccess::exists( newURL, false /*will write*/, this ) && !justChangingFilterOptions ) + { + bOk = KMessageBox::questionYesNo( this, + i18n("A document with this name already exists.\n"\ + "Do you want to overwrite it?"), + i18n("Warning") ) == KMessageBox::Yes; + } + } while ( !bOk ); + + delete dialog; + + if (bOk) + { + bool wantToSave = true; + + // don't change this line unless you know what you're doing :) + if (!justChangingFilterOptions || pDoc->confirmNonNativeSave (isExporting ())) { + if ( !pDoc->isNativeFormat( outputFormat ) ) + wantToSave = exportConfirmation( outputFormat ); + } + + if (wantToSave) + { + // + // Note: + // If the user is stupid enough to Export to the current URL, + // we do _not_ change this operation into a Save As. Reasons + // follow: + // + // 1. A check like "isExporting() && oldURL == newURL" + // doesn't _always_ work on case-insensitive filesystems + // and inconsistent behaviour is bad. + // 2. It is probably not a good idea to change pDoc->mimeType + // and friends because the next time the user File/Save's, + // (not Save As) they won't be expecting that they are + // using their File/Export settings + // + // As a bad side-effect of this, the modified flag will not + // be updated and it is possible that what is currently on + // their screen is not what is stored on disk (through loss + // of formatting). But if you are dumb enough to change + // mimetype but not the filename, then arguably, _you_ are + // the "bug" :) + // + // - Clarence + // + + + pDoc->setOutputMimeType( outputFormat, specialOutputFlag ); + if (!isExporting ()) // Save As + { + ret = pDoc->saveAs( newURL ); + + if (ret) + { + kdDebug(30003) << "Successful Save As!" << endl; + addRecentURL( newURL ); + } + else + { + kdDebug(30003) << "Failed Save As!" << endl; + pDoc->setURL( oldURL ), pDoc->setFile( oldFile ); + pDoc->setOutputMimeType( oldOutputFormat, oldSpecialOutputFlag ); + } + } + else // Export + { + ret = pDoc->exp0rt( newURL ); + + if (ret) + { + // a few file dialog convenience things + d->m_lastExportURL = newURL; + d->m_lastExportFormat = outputFormat; + d->m_lastExportSpecialOutputFlag = specialOutputFlag; + } + + // always restore output format + pDoc->setOutputMimeType( oldOutputFormat, oldSpecialOutputFlag ); + } + } // if (wantToSave) + else + ret = false; + } // if (bOk) + else + ret = false; + } + else { // saving + bool needConfirm = pDoc->confirmNonNativeSave( false ) && + !pDoc->isNativeFormat( oldOutputFormat ); + if (!needConfirm || + (needConfirm && exportConfirmation( oldOutputFormat /* not so old :) */ )) + ) + { + // be sure pDoc has the correct outputMimeType! + ret = pDoc->save(); + + if (!ret) + { + kdDebug(30003) << "Failed Save!" << endl; + pDoc->setURL( oldURL ), pDoc->setFile( oldFile ); + } + } + else + ret = false; + } + +// Now that there's a File/Export option, this is no longer necessary. +// If you continue to use File/Save to export to a foreign format, +// this signals your intention to continue working in a foreign format. +// You have already been warned by the DoNotAskAgain exportConfirmation +// about losing formatting when you first saved so don't set modified +// here or else it will be reported as a bug by some MSOffice user. +// You have been warned! Do not click DoNotAskAgain!!! +#if 0 + if (ret && !isExporting()) + { + // When exporting to a non-native format, we don't reset modified. + // This way the user will be reminded to save it again in the native format, + // if he/she doesn't want to lose formatting. + if ( wasModified && pDoc->outputMimeType() != _native_format ) + pDoc->setModified( true ); + } +#endif + + if (!ret && reset_url) + pDoc->resetURL(); //clean the suggested filename as the save dialog was rejected + else if (! silent) // don't let the document change the window caption + pDoc->setTitleModified(); + return ret; +} + +void KoMainWindow::closeEvent(QCloseEvent *e) { + if(queryClose()) { + saveWindowSettings(); + setRootDocument(0L); + KParts::MainWindow::closeEvent(e); + } +} + +void KoMainWindow::saveWindowSettings() +{ + if (d->m_windowSizeDirty && rootDocument()) + { + // Save window size into the config file of our instance + instance()->config()->setGroup( "MainWindow" ); + //kdDebug(30003) << "KoMainWindow::saveWindowSettings" << endl; + saveWindowSize( instance()->config() ); + d->m_windowSizeDirty = false; + // Save toolbar position into the config file of the app, under the doc's instance name + //kdDebug(30003) << "KoMainWindow::closeEvent -> saveMainWindowSettings rootdoc's instance=" << rootDocument()->instance()->instanceName() << endl; + saveMainWindowSettings( KGlobal::config(), rootDocument()->instance()->instanceName() ); + KGlobal::config()->sync(); + resetAutoSaveSettings(); // Don't let KMainWindow override the good stuff we wrote down + } +} + +void KoMainWindow::resizeEvent( QResizeEvent * e ) +{ + d->m_windowSizeDirty = true; + KParts::MainWindow::resizeEvent( e ); +} + +bool KoMainWindow::queryClose() +{ + if ( rootDocument() == 0 ) + return true; + //kdDebug(30003) << "KoMainWindow::queryClose() viewcount=" << rootDocument()->viewCount() + // << " shellcount=" << rootDocument()->shellCount() << endl; + if ( !d->m_forQuit && rootDocument()->shellCount() > 1 ) + // there are more open, and we are closing just one, so no problem for closing + return true; + + // see DTOR for a descr. of the test + if ( d->m_rootDoc->isEmbedded() ) + return true; + + // main doc + internally stored child documents + if ( d->m_rootDoc->isModified() ) + { + QString name; + if ( rootDocument()->documentInfo() ) + { + name = rootDocument()->documentInfo()->title(); + } + if ( name.isEmpty() ) + name = rootDocument()->url().fileName(); + + if ( name.isEmpty() ) + name = i18n( "Untitled" ); + + int res = KMessageBox::warningYesNoCancel( this, + i18n( "<p>The document <b>'%1'</b> has been modified.</p><p>Do you want to save it?</p>" ).arg(name), + QString::null, + KStdGuiItem::save(), + KStdGuiItem::discard()); + + switch(res) { + case KMessageBox::Yes : { + d->m_rootDoc->setDoNotSaveExtDoc(); // external docs are saved later + bool isNative = ( d->m_rootDoc->outputMimeType() == d->m_rootDoc->nativeFormatMimeType() ); + if (! saveDocument( !isNative ) ) + return false; + break; + } + case KMessageBox::No : + rootDocument()->removeAutoSaveFiles(); + rootDocument()->setModified( false ); // Now when queryClose() is called by closeEvent it won't do anything. + break; + default : // case KMessageBox::Cancel : + return false; + } + } + + if ( d->m_rootDoc->queryCloseExternalChildren() == KMessageBox::Cancel ) + { + return false; + } + + return true; +} + +// Helper method for slotFileNew and slotFileClose +void KoMainWindow::chooseNewDocument( int /*KoDocument::InitDocFlags*/ initDocFlags ) +{ + KoDocument* doc = rootDocument(); + KoDocument *newdoc = createDoc(); + + if ( !newdoc ) + return; + + //FIXME: This needs to be handled differently + connect(newdoc, SIGNAL(sigProgress(int)), this, SLOT(slotProgress(int))); + disconnect(newdoc, SIGNAL(sigProgress(int)), this, SLOT(slotProgress(int))); + + if ( ( !doc && ( initDocFlags == KoDocument::InitDocFileNew ) ) || ( doc && !doc->isEmpty() ) ) + { + KoMainWindow *s = new KoMainWindow( newdoc->instance() ); + s->show(); + newdoc->addShell( s ); + newdoc->showStartUpWidget( s, true /*Always show widget*/ ); + return; + } + + if( doc ) { + setRootDocument( 0 ); + delete d->m_rootDoc; + d->m_rootDoc = 0; + } + + newdoc->addShell( this ); + newdoc->showStartUpWidget( this, true /*Always show widget*/ ); +} + +void KoMainWindow::slotFileNew() +{ + chooseNewDocument( KoDocument::InitDocFileNew ); +} + +void KoMainWindow::slotFileOpen() +{ + KFileDialog *dialog = new KFileDialog(":OpenDialog", QString::null, this, "file dialog", true); + if (!isImporting()) + dialog->setCaption( i18n("Open Document") ); + else + dialog->setCaption( i18n("Import Document") ); + + // The few lines below need to be kept in sync with KoTemplateChooseDia::setupFileDialog + const QStringList mimeFilter = KoFilterManager::mimeFilter( KoDocument::readNativeFormatMimeType(), + KoFilterManager::Import, + KoDocument::readExtraNativeMimeTypes() ); + dialog->setMimeFilter( mimeFilter ); + if(dialog->exec()!=QDialog::Accepted) { + delete dialog; + return; + } + KURL url( dialog->selectedURL() ); + delete dialog; + + if ( url.isEmpty() ) + return; + + (void) openDocument( url ); +} + +void KoMainWindow::slotFileOpenRecent( const KURL & url ) +{ + (void) openDocument( url ); +} + +void KoMainWindow::slotFileSave() +{ + if ( saveDocument() ) + emit documentSaved(); +} + +void KoMainWindow::slotFileSaveAs() +{ + if ( saveDocument( true ) ) + emit documentSaved(); +} + +void KoMainWindow::slotDocumentInfo() +{ + if ( !rootDocument() ) + return; + + KoDocumentInfo *docInfo = rootDocument()->documentInfo(); + + if ( !docInfo ) + return; + + KoDocumentInfoDlg *dlg = new KoDocumentInfoDlg( docInfo, this, "documentInfoDlg" ); + if ( dlg->exec() ) + { + dlg->save(); + rootDocument()->setModified( true ); + rootDocument()->setTitleModified(); + } + + delete dlg; +} + +void KoMainWindow::slotFileClose() +{ + if (queryClose()) + { + saveWindowSettings(); + setRootDocument( 0 ); // don't delete this shell when deleting the document + delete d->m_rootDoc; + d->m_rootDoc = 0; + chooseNewDocument( KoDocument::InitDocFileClose ); + } +} + +void KoMainWindow::slotFileQuit() +{ + close(); +} + +void KoMainWindow::print(bool quick) { + if ( !rootView() ) + { + kdDebug(30003) << "KoMainWindow::slotFilePrint : No root view!" << endl; + return; + } + + KPrinter printer( true /*, QPrinter::HighResolution*/ ); + QString title = rootView()->koDocument()->documentInfo()->title(); + QString fileName = rootView()->koDocument()->url().fileName(); + + // strip off the native extension (I don't want foobar.kwd.ps when printing into a file) + KMimeType::Ptr mime = KMimeType::mimeType( rootView()->koDocument()->outputMimeType() ); + if ( mime ) { + QString extension = mime->property( "X-KDE-NativeExtension" ).toString(); + + if ( fileName.endsWith( extension ) ) + fileName.truncate( fileName.length() - extension.length() ); + } + + if ( title.isEmpty() ) + title = fileName; + if ( title.isEmpty() ) { + // #139905 - breaks message freeze though + //const QString programName = instance()->aboutData() ? instance()->aboutData()->programName() : instance()->instanceName(); + //title = i18n("%1 unsaved document (%2)").arg(programName).arg(KGlobal::locale()->formatDate(QDate::currentDate(), true/*short*/)); + } + printer.setDocName( title ); + printer.setDocFileName( fileName ); + printer.setDocDirectory( rootView()->koDocument()->url().directory() ); + + // ### TODO: apply global koffice settings here + + rootView()->setupPrinter( printer ); + + if ( quick || printer.setup( this ) ) + rootView()->print( printer ); +} + + +void KoMainWindow::slotFilePrint() +{ + print(false); +} + +void KoMainWindow::slotFilePrintPreview() +{ + if ( !rootView() ) + { + kdWarning() << "KoMainWindow::slotFilePrint : No root view!" << endl; + return; + } + KPrinter printer( false ); + KTempFile tmpFile; + // The temp file is deleted by KoPrintPreview + + // This line has to be before setupPrinter to let the apps decide what to + // print and what not (if they want to :) + printer.setFromTo( printer.minPage(), printer.maxPage() ); + printer.setPreviewOnly( true ); + rootView()->setupPrinter( printer ); + + QString oldFileName = printer.outputFileName(); + printer.setOutputFileName( tmpFile.name() ); + int oldNumCopies = printer.numCopies(); + printer.setNumCopies( 1 ); + // Disable kdeprint's own preview, we'd get two. This shows that KPrinter needs + // a "don't use the previous settings" mode. The current way is really too much of a hack. + QString oldKDEPreview = printer.option( "kde-preview" ); + printer.setOption( "kde-preview", "0" ); + + rootView()->print(printer); + //KoPrintPreview::preview(this, "KoPrintPreviewDialog", tmpFile.name()); + + // Restore previous values + printer.setOutputFileName( oldFileName ); + printer.setNumCopies( oldNumCopies ); + printer.setOption( "kde-preview", oldKDEPreview ); +} + +void KoMainWindow::slotConfigureKeys() +{ + guiFactory()->configureShortcuts(); +} + +void KoMainWindow::slotConfigureToolbars() +{ + if (rootDocument()) + saveMainWindowSettings( KGlobal::config(), rootDocument()->instance()->instanceName() ); + KEditToolbar edit(factory(), this); + connect(&edit,SIGNAL(newToolbarConfig()),this,SLOT(slotNewToolbarConfig())); + (void) edit.exec(); +} + +void KoMainWindow::slotNewToolbarConfig() +{ + if (rootDocument()) + applyMainWindowSettings( KGlobal::config(), rootDocument()->instance()->instanceName() ); + KXMLGUIFactory *factory = guiFactory(); + + // Check if there's an active view + if( !d->m_activeView ) + return; + + // This gets plugged in even for embedded views + factory->plugActionList(d->m_activeView, "view_closeallviews", + d->m_veryHackyActionList); + + // This one only for root views + if(d->m_rootViews.findRef(d->m_activeView)!=-1) + factory->plugActionList(d->m_activeView, "view_split", + d->m_splitViewActionList ); + plugActionList( "toolbarlist", d->m_toolbarList ); +} + +void KoMainWindow::slotToolbarToggled( bool toggle ) +{ + //kdDebug(30003) << "KoMainWindow::slotToolbarToggled " << sender()->name() << " toggle=" << true << endl; + // The action (sender) and the toolbar have the same name + KToolBar * bar = toolBar( sender()->name() ); + if (bar) + { + if (toggle) + bar->show(); + else + bar->hide(); + + if (rootDocument()) + saveMainWindowSettings( KGlobal::config(), rootDocument()->instance()->instanceName() ); + } + else + kdWarning(30003) << "slotToolbarToggled : Toolbar " << sender()->name() << " not found!" << endl; +} + +bool KoMainWindow::toolbarIsVisible(const char *tbName) +{ + QWidget *tb = toolBar( tbName); + return !tb->isHidden(); +} + +void KoMainWindow::showToolbar( const char * tbName, bool shown ) +{ + QWidget * tb = toolBar( tbName ); + if ( !tb ) + { + kdWarning(30003) << "KoMainWindow: toolbar " << tbName << " not found." << endl; + return; + } + if ( shown ) + tb->show(); + else + tb->hide(); + + // Update the action appropriately + QPtrListIterator<KAction> it( d->m_toolbarList ); + for ( ; it.current() ; ++it ) + if ( !strcmp( it.current()->name(), tbName ) ) + { + //kdDebug(30003) << "KoMainWindow::showToolbar setChecked " << shown << endl; + static_cast<KToggleAction *>(it.current())->setChecked( shown ); + break; + } +} + +void KoMainWindow::slotSplitView() { + d->m_splitted=true; + d->m_rootViews.append(d->m_rootDoc->createView(d->m_splitter, "splitted-view")); + d->m_rootViews.current()->show(); + d->m_rootViews.current()->setPartManager( d->m_manager ); + d->m_manager->setActivePart( d->m_rootDoc, d->m_rootViews.current() ); + d->m_removeView->setEnabled(true); + d->m_orientation->setEnabled(true); +} + +void KoMainWindow::slotCloseAllViews() { + + // Attention: Very touchy code... you know what you're doing? Goooood :) + d->m_forQuit=true; + if(queryClose()) { + // In case the document is embedded we close all open "extra-shells" + if(d->m_rootDoc && d->m_rootDoc->isEmbedded()) { + hide(); + d->m_rootDoc->removeShell(this); + QPtrListIterator<KoMainWindow> it(d->m_rootDoc->shells()); + while (it.current()) { + it.current()->hide(); + delete it.current(); // this updates the lists' current pointer and thus + // the iterator (the shell dtor calls removeShell) + d->m_rootDoc=0; + } + } + // not embedded -> destroy the document and all shells/views ;) + else + setRootDocument( 0L ); + close(); // close this window (and quit the app if necessary) + } + d->m_forQuit=false; +} + +void KoMainWindow::slotRemoveView() { + KoView *view; + if(d->m_rootViews.findRef(d->m_activeView)!=-1) + view=d->m_rootViews.current(); + else + view=d->m_rootViews.first(); + view->hide(); + if ( !d->m_rootViews.removeRef(view) ) + kdWarning() << "view not found in d->m_rootViews!" << endl; + + if(d->m_rootViews.count()==1) + { + d->m_removeView->setEnabled(false); + d->m_orientation->setEnabled(false); + } + // Prevent the view's destroyed() signal from triggering GUI rebuilding (too early) + d->m_manager->setActivePart( 0, 0 ); + + delete view; + view=0L; + + d->m_rootViews.first()->setPartManager( d->m_manager ); + d->m_manager->setActivePart( d->m_rootDoc, d->m_rootViews.first() ); + + if(d->m_rootViews.count()==1) + d->m_splitted=false; +} + +void KoMainWindow::slotSetOrientation() { + d->m_splitter->setOrientation(static_cast<Qt::Orientation> + (d->m_orientation->currentItem())); +} + +void KoMainWindow::slotProgress(int value) { + //kdDebug(30003) << "KoMainWindow::slotProgress " << value << endl; + if(value==-1) { + if ( d->m_progress ) + { + statusBar()->removeWidget(d->m_progress); + delete d->m_progress; + d->m_progress=0L; + } + d->m_firstTime=true; + return; + } + if(d->m_firstTime) + { + // The statusbar might not even be created yet. + // So check for that first, and create it if necessary + QObjectList *l = queryList( "QStatusBar" ); + if ( !l || !l->first() ) { + statusBar()->show(); + QApplication::sendPostedEvents( this, QEvent::ChildInserted ); + setUpLayout(); + } + delete l; + + if ( d->m_progress ) + { + statusBar()->removeWidget(d->m_progress); + delete d->m_progress; + d->m_progress=0L; + } + statusBar()->setMaximumHeight(statusBar()->height()); + d->m_progress=new KProgress(statusBar()); + //d->m_progress->setMaximumHeight(statusBar()->height()); + statusBar()->addWidget( d->m_progress, 0, true ); + d->m_progress->show(); + d->m_firstTime=false; + } + d->m_progress->setProgress(value); + kapp->processEvents(); +} + + +void KoMainWindow::slotActivePartChanged( KParts::Part *newPart ) +{ + + // This looks very much like KParts::MainWindow::createGUI, but we have + // to reimplement it because it works with an active part, whereas we work + // with an active view _and_ an active part, depending for what. + // Both are KXMLGUIClients, but e.g. the plugin query needs a QObject. + //kdDebug(30003) << "KoMainWindow::slotActivePartChanged( Part * newPart) newPart = " << newPart << endl; + //kdDebug(30003) << "current active part is " << d->m_activePart << endl; + + if ( d->m_activePart && d->m_activePart == newPart && !d->m_splitted ) + { + //kdDebug(30003) << "no need to change the GUI" << endl; + return; + } + + KXMLGUIFactory *factory = guiFactory(); + + setUpdatesEnabled( false ); + + if ( d->m_activeView ) + { + KParts::GUIActivateEvent ev( false ); + QApplication::sendEvent( d->m_activePart, &ev ); + QApplication::sendEvent( d->m_activeView, &ev ); + + + factory->removeClient( d->m_activeView ); + + unplugActionList( "toolbarlist" ); + d->m_toolbarList.clear(); // deletes the actions + } + + if ( !d->bMainWindowGUIBuilt ) + { + // Load mainwindow plugins + KParts::Plugin::loadPlugins( this, this, instance(), true ); + createShellGUI(); + } + + if ( newPart && d->m_manager->activeWidget() && d->m_manager->activeWidget()->inherits( "KoView" ) ) + { + d->m_activeView = (KoView *)d->m_manager->activeWidget(); + d->m_activePart = newPart; + //kdDebug(30003) << "new active part is " << d->m_activePart << endl; + + factory->addClient( d->m_activeView ); + + + // This gets plugged in even for embedded views + factory->plugActionList(d->m_activeView, "view_closeallviews", + d->m_veryHackyActionList); + // This one only for root views + if(d->m_rootViews.findRef(d->m_activeView)!=-1) + factory->plugActionList(d->m_activeView, "view_split", d->m_splitViewActionList ); + + // Position and show toolbars according to user's preference + setAutoSaveSettings( newPart->instance()->instanceName(), false ); + + // Create and plug toolbar list for Settings menu + //QPtrListIterator<KToolBar> it = toolBarIterator(); + QPtrList<QWidget> toolBarList = factory->containers( "ToolBar" ); + QPtrListIterator<QWidget> it( toolBarList ); + for ( ; it.current() ; ++it ) + { + if ( it.current()->inherits("KToolBar") ) + { + KToolBar * tb = static_cast<KToolBar *>(it.current()); + KToggleAction * act = new KToggleAction( i18n("Show %1 Toolbar").arg( tb->text() ), 0, + actionCollection(), tb->name() ); + act->setCheckedState(i18n("Hide %1 Toolbar").arg( tb->text() )); + connect( act, SIGNAL( toggled( bool ) ), this, SLOT( slotToolbarToggled( bool ) ) ); + act->setChecked ( !tb->isHidden() ); + d->m_toolbarList.append( act ); + } + else + kdWarning(30003) << "Toolbar list contains a " << it.current()->className() << " which is not a toolbar!" << endl; + } + plugActionList( "toolbarlist", d->m_toolbarList ); + + // Send the GUIActivateEvent only now, since it might show/hide toolbars too + // (and this has priority over applyMainWindowSettings) + KParts::GUIActivateEvent ev( true ); + QApplication::sendEvent( d->m_activePart, &ev ); + QApplication::sendEvent( d->m_activeView, &ev ); + } + else + { + d->m_activeView = 0L; + d->m_activePart = 0L; + } + setUpdatesEnabled( true ); +} + +QLabel * KoMainWindow::statusBarLabel() +{ + if ( !d->statusBarLabel ) + { + d->statusBarLabel = new QLabel( statusBar() ); + statusBar()->addWidget( d->statusBarLabel, 1, true ); + } + return d->statusBarLabel; +} + +void KoMainWindow::setMaxRecentItems(uint _number) +{ + m_recent->setMaxItems( _number ); +} + +DCOPObject * KoMainWindow::dcopObject() +{ + if ( !d->m_dcopObject ) + { + d->m_dcopObject = new KoMainWindowIface( this ); + } + + return d->m_dcopObject; +} + +void KoMainWindow::slotEmailFile() +{ + if (!rootDocument()) + return; + + // Subject = Document file name + // Attachment = The current file + // Message Body = The current document in HTML export? <-- This may be an option. + QString theSubject; + QStringList urls; + QString fileURL; + if (rootDocument()->url ().isEmpty () || + rootDocument()->isModified()) + { + //Save the file as a temporary file + bool const tmp_modified = rootDocument()->isModified(); + KURL const tmp_url = rootDocument()->url(); + QCString const tmp_mimetype = rootDocument()->outputMimeType(); + KTempFile tmpfile; //TODO: The temorary file should be deleted when the mail program is closed + KURL u; + u.setPath(tmpfile.name()); + rootDocument()->setURL(u); + rootDocument()->setModified(true); + rootDocument()->setOutputMimeType(rootDocument()->nativeFormatMimeType()); + + saveDocument(false, true); + + fileURL = tmpfile.name(); + theSubject = i18n("Document"); + urls.append( fileURL ); + + rootDocument()->setURL(tmp_url); + rootDocument()->setModified(tmp_modified); + rootDocument()->setOutputMimeType(tmp_mimetype); + } + else + { + fileURL = rootDocument()->url().url(); + theSubject = i18n("Document - %1").arg(rootDocument()->url().fileName(false)); + urls.append( fileURL ); + } + + kdDebug(30003) << "(" << fileURL <<")" << endl; + + if (!fileURL.isEmpty()) + { + kapp->invokeMailer(QString::null, QString::null, QString::null, theSubject, + QString::null, //body + QString::null, + urls); // attachments + } +} + +void KoMainWindow::slotVersionsFile() +{ + KoVersionDialog *dlg = new KoVersionDialog( this ); + dlg->exec(); + delete dlg; +} + +void KoMainWindow::slotReloadFile() +{ + KoDocument* pDoc = rootDocument(); + if(!pDoc || pDoc->url().isEmpty() || !pDoc->isModified()) + return; + + bool bOk = KMessageBox::questionYesNo( this, + i18n("You will lose all your changes!\n" + "Do you want to continue?"), + i18n("Warning") ) == KMessageBox::Yes; + if ( !bOk ) + return; + + KURL url = pDoc->url(); + if ( pDoc && !pDoc->isEmpty() ) + { + setRootDocument( 0L ); // don't delete this shell when deleting the document + delete d->m_rootDoc; + d->m_rootDoc = 0L; + } + openDocument( url ); + return; + +} + +void KoMainWindow::slotImportFile() +{ + kdDebug(30003) << "slotImportFile()" << endl; + + d->m_isImporting = true; + slotFileOpen(); + d->m_isImporting = false; +} + +void KoMainWindow::slotExportFile() +{ + kdDebug(30003) << "slotExportFile()" << endl; + + d->m_isExporting = true; + slotFileSaveAs(); + d->m_isExporting = false; +} + +bool KoMainWindow::isImporting() const +{ + return d->m_isImporting; +} + +bool KoMainWindow::isExporting() const +{ + return d->m_isExporting; +} + +void KoMainWindow::setDocToOpen( KoDocument *doc ) +{ + d->m_docToOpen = doc; +} + +#include <KoMainWindow.moc> diff --git a/lib/kofficecore/KoMainWindow.h b/lib/kofficecore/KoMainWindow.h new file mode 100644 index 00000000..2b738db9 --- /dev/null +++ b/lib/kofficecore/KoMainWindow.h @@ -0,0 +1,387 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright (C) 2000-2004 David Faure <faure@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. +*/ + +#ifndef __ko_main_window_h__ +#define __ko_main_window_h__ + +#include <kparts/mainwindow.h> +#include <kfiledialog.h> +#include <koffice_export.h> +#include "KoDocument.h" + +class QLabel; +class KoView; +class KoMainWindowPrivate; +class KURL; +class KRecentFilesAction; +class KoFilterManager; +class DCOPObject; + +namespace KParts +{ + class PartManager; +} + +/** + * @brief Main window for a KOffice application + * + * This class is used to represent a main window + * of a KOffice component. Each main window contains + * a menubar and some toolbars. + * + * @note This class does NOT need to be subclassed in your application. + */ +class KOFFICECORE_EXPORT KoMainWindow : public KParts::MainWindow +{ + Q_OBJECT +public: + + /** + * Constructor. + * + * Initializes a KOffice main window (with its basic GUI etc.). + */ + KoMainWindow( KInstance *instance, const char *_name = 0 ); + + /** + * Destructor. + */ + ~KoMainWindow(); + + /** + * Called when a document is assigned to this mainwindow. + * This creates a view for this document, makes it the active part, etc. + */ + virtual void setRootDocument( KoDocument *doc ); + + /** + * This is used to handle the document used at start up before it actually + * added as root document. + */ + void setDocToOpen( KoDocument *doc ); + + /** + * Update caption from document info - call when document info + * (title in the about page) changes. + */ + virtual void updateCaption(); + + /** + * Retrieves the document that is displayed in the mainwindow. + */ + virtual KoDocument* rootDocument() const; + + virtual KoView *rootView() const; + + virtual KParts::PartManager *partManager(); + + /** + * Prints the document + * @param quick whether the print setup dialog is to be displayed + **/ + void print(bool quick); + + /** + * The application should call this to show or hide a toolbar. + * It also takes care of the corresponding action in the settings menu. + */ + void showToolbar( const char * tbName, bool shown ); + + /** + * @return TRUE if the toolbar @p tbName is visible + */ + bool toolbarIsVisible(const char *tbName); + + /** + * Get hold of the label in the statusbar, to write messages to it. + * You can also insert other items in the status bar by using QStatusBar::addWidget. + */ + QLabel * statusBarLabel(); + + /** + * Sets the maximum number of recent documents entries. + */ + void setMaxRecentItems(uint _number); + + /** + * The document opened a URL -> store into recent documents list. + */ + void addRecentURL( const KURL& url ); + + /** + * Load the desired document and show it. + * @param url the URL to open + * + * @return TRUE on success. + */ + virtual bool openDocument( const KURL & url ); + + /** + * Load the URL into this document (and make it root doc after loading) + * + * Special method for KoApplication::start, don't use. + */ + bool openDocument( KoDocument *newdoc, const KURL & url ); + + virtual DCOPObject * dcopObject(); + + /** + * Reloads the recent documents list. + */ + void reloadRecentFileList(); + + /** + * Updates the window caption based on the document info and path. + */ + virtual void updateCaption( const QString caption, bool mod ); + void updateReloadFileAction(KoDocument *doc); + void updateVersionsFileAction(KoDocument *doc); + +signals: + /** + * This signal is emitted if the document has been saved successfully. + */ + void documentSaved(); + /// This signals is emmitted before the save dialog is shown + void saveDialogShown(); + +public slots: + + /** + * Slot for eMailing the document using KMail + * + * This is a very simple extension that will allow any document + * that is currently being edited to be emailed using KMail. + */ + void slotEmailFile(); + + /** + * Slot for opening a new document. + * + * If the current document is empty, the new document replaces it. + * If not, a new shell will be opened for showing the document. + */ + virtual void slotFileNew(); + + /** + * Slot for opening a saved file. + * + * If the current document is empty, the opened document replaces it. + * If not a new shell will be opened for showing the opened file. + */ + virtual void slotFileOpen(); + + /** + * Slot for opening a file among the recently opened files. + * + * If the current document is empty, the opened document replaces it. + * If not a new shell will be opened for showing the opened file. + */ + virtual void slotFileOpenRecent( const KURL & ); + + /** + * Saves the current document with the current name. + */ + virtual void slotFileSave(); + + /** + * Saves the current document with a new name. + */ + virtual void slotFileSaveAs(); + + /** + * Prints the actual document. + */ + virtual void slotFilePrint(); + + /** + * Show a print preview + */ + void slotFilePrintPreview(); // make virtual later + + /** + * Show a dialog with author and document information. + */ + virtual void slotDocumentInfo(); + + /** + * Closes the document. + */ + virtual void slotFileClose(); + + /** + * Closes the shell. + */ + virtual void slotFileQuit(); + + /** + * Configure key bindings. + */ + virtual void slotConfigureKeys(); + + /** + * Configure toolbars. + */ + virtual void slotConfigureToolbars(); + + /** + * Post toolbar config. + * (Plug action lists back in, etc.) + */ + virtual void slotNewToolbarConfig(); + + /** + * Shows or hides a toolbar + */ + virtual void slotToolbarToggled( bool toggle ); + + /** + * View splitting stuff + */ + virtual void slotSplitView(); + virtual void slotRemoveView(); + virtual void slotSetOrientation(); + + /** + * Close all views + */ + virtual void slotCloseAllViews(); + + /** + * Reload file + */ + void slotReloadFile(); + + /** + * This will call a dialogbox to add version to list of files + */ + void slotVersionsFile(); + + /** + * File --> Import + * + * This will call slotFileOpen(). To differentiate this from an ordinary + * call to slotFileOpen() call @ref isImporting(). + */ + void slotImportFile(); + + /** + * File --> Export + * + * This will call slotFileSaveAs(). To differentiate this from an ordinary + * call to slotFileSaveAs() call @ref isExporting(). + */ + void slotExportFile(); + +protected: // protected methods are mostly for koshell, it's the only one deriving from KoMainWindow + + /// Helper method for slotFileNew and slotFileClose + void chooseNewDocument( int /*KoDocument::InitDocFlags*/ initDocFlags ); + /** + * Special method for KOShell, to allow switching the root + * document (and its views) among a set of them. + */ + void setRootDocumentDirect( KoDocument *doc, const QPtrList<KoView> & views ); + + /** + * Create a new empty document. + */ + virtual KoDocument* createDoc() const; + + /** + * Saves the document, asking for a filename if necessary. + * + * @param saveas if set to TRUE the user is always prompted for a filename + * + * @param silent if set to TRUE rootDocument()->setTitleModified will not be called. + * + * @return TRUE on success, false on error or cancel + * (don't display anything in this case, the error dialog box is also implemented here + * but restore the original URL in slotFileSaveAs) + */ + virtual bool saveDocument( bool saveas = false, bool silent = false ); + + virtual void closeEvent( QCloseEvent * e ); + virtual void resizeEvent( QResizeEvent * e ); + + /** + * Ask user about saving changes to the document upon exit. + */ + virtual bool queryClose(); + + virtual bool openDocumentInternal( const KURL & url, KoDocument * newdoc = 0L ); + + /** + * Returns whether or not the current slotFileSave[As]() or saveDocument() + * call is actually an export operation (like File --> Export). + * + * If this is true, you must call KoDocument::export() instead of + * KoDocument::save() or KoDocument::saveAs(), in any reimplementation of + * saveDocument(). + */ + bool isExporting() const; + + /** + * Returns whether or not the current slotFileOpen() or openDocument() + * call is actually an import operation (like File --> Import). + * + * If this is true, you must call KoDocument::import() instead of + * KoDocument::openURL(), in any reimplementation of openDocument() or + * openDocumentInternal(). + */ + bool isImporting() const; + + /** + * Save the list of recent files. + */ + void saveRecentFiles(); + + KRecentFilesAction *recentAction() const { return m_recent; } + +private: + + /** + * Asks the user if they really want to save the document. + * Called only if outputFormat != nativeFormat. + * + * @return true if the document should be saved + */ + bool exportConfirmation( const QCString &outputFormat ); + + void saveWindowSettings(); + + KRecentFilesAction *m_recent; + +protected slots: + virtual void slotActivePartChanged( KParts::Part *newPart ); + +private slots: + void slotProgress(int value); + void slotLoadCompleted(); + void slotLoadCanceled (const QString &); + void slotSaveCompleted(); + void slotSaveCanceled(const QString &); + +private: + KoMainWindowPrivate *d; + +}; + +#endif diff --git a/lib/kofficecore/KoMainWindowIface.cc b/lib/kofficecore/KoMainWindowIface.cc new file mode 100644 index 00000000..356bd80b --- /dev/null +++ b/lib/kofficecore/KoMainWindowIface.cc @@ -0,0 +1,65 @@ +/* This file is part of the KDE project + Copyright (c) 2001 David Faure <faure@kde.org> + + $Id: KoMainWindowIface.cc 508787 2006-02-12 18:28:12Z ingwa $ + + 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 "KoMainWindowIface.h" + +#include "KoMainWindow.h" + +#include <kapplication.h> +#include <dcopclient.h> +#include <kdcopactionproxy.h> + +KoMainWindowIface::KoMainWindowIface( KoMainWindow *mainwindow ) + : DCOPObject( mainwindow ) +{ + m_pMainWindow = mainwindow; + m_actionProxy = new KDCOPActionProxy( mainwindow->actionCollection(), this ); +} + +KoMainWindowIface::~KoMainWindowIface() +{ + delete m_actionProxy; +} + +DCOPRef KoMainWindowIface::action( const QCString &name ) +{ + return DCOPRef( kapp->dcopClient()->appId(), m_actionProxy->actionObjectId( name ) ); +} + +QCStringList KoMainWindowIface::actions() +{ + QCStringList res; + QValueList<KAction *> lst = m_actionProxy->actions(); + QValueList<KAction *>::ConstIterator it = lst.begin(); + QValueList<KAction *>::ConstIterator end = lst.end(); + for (; it != end; ++it ) + res.append( (*it)->name() ); + + return res; +} + +QMap<QCString,DCOPRef> KoMainWindowIface::actionMap() +{ + return m_actionProxy->actionMap(); +} + +ASYNC KoMainWindowIface::print(bool quick) { + m_pMainWindow->print(quick); +} diff --git a/lib/kofficecore/KoMainWindowIface.h b/lib/kofficecore/KoMainWindowIface.h new file mode 100644 index 00000000..6b7c3cb6 --- /dev/null +++ b/lib/kofficecore/KoMainWindowIface.h @@ -0,0 +1,48 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Davud Faure <faure@kde.org> + + $Id: KoMainWindowIface.h 466447 2005-10-02 17:54:10Z zander $ + + 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. +*/ +#ifndef __KoMainWindowIface_h__ +#define __KoMainWindowIface_h__ + +#include <dcopobject.h> +#include <dcopref.h> + +class KDCOPActionProxy; +class KoMainWindow; + +class KoMainWindowIface : public DCOPObject +{ + K_DCOP +public: + KoMainWindowIface( KoMainWindow *mainwindow ); + virtual ~KoMainWindowIface(); + +k_dcop: + DCOPRef action( const QCString &name ); + QCStringList actions(); + QMap<QCString,DCOPRef> actionMap(); + ASYNC print(bool quick); + +protected: + KoMainWindow *m_pMainWindow; + KDCOPActionProxy *m_actionProxy; +}; + +#endif diff --git a/lib/kofficecore/KoOasisLoadingContext.cpp b/lib/kofficecore/KoOasisLoadingContext.cpp new file mode 100644 index 00000000..fdad0ecc --- /dev/null +++ b/lib/kofficecore/KoOasisLoadingContext.cpp @@ -0,0 +1,124 @@ +/* This file is part of the KDE project + Copyright (C) 2005 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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 "KoOasisLoadingContext.h" +#include <KoOasisStore.h> +#include <KoOasisStyles.h> +#include <KoStore.h> +#include <KoXmlNS.h> +#include <kdebug.h> +#include <KoDom.h> + +KoOasisLoadingContext::KoOasisLoadingContext( KoDocument* doc, + KoOasisStyles& styles, KoStore* store ) + : m_doc( doc ), m_store( store ), m_styles( styles ), + m_metaXmlParsed( false ), m_useStylesAutoStyles( false ) +{ + // Ideally this should be done by KoDocument and passed as argument here... + KoOasisStore oasisStore( store ); + QString dummy; + (void)oasisStore.loadAndParse( "tar:/META-INF/manifest.xml", m_manifestDoc, dummy ); +} + + +KoOasisLoadingContext::~KoOasisLoadingContext() +{ + +} + +void KoOasisLoadingContext::fillStyleStack( const QDomElement& object, const char* nsURI, const char* attrName, const char* family ) +{ + // find all styles associated with an object and push them on the stack + if ( object.hasAttributeNS( nsURI, attrName ) ) { + const QString styleName = object.attributeNS( nsURI, attrName, QString::null ); + const QDomElement* style = 0; + bool isStyleAutoStyle = false; + if ( m_useStylesAutoStyles ) { + // When loading something from styles.xml, look into the styles.xml auto styles first + style = m_styles.findStyleAutoStyle( styleName, family ); + // and fallback to looking at styles(), which includes the user styles from styles.xml + if ( style ) + isStyleAutoStyle = true; + } + if ( !style ) + style = m_styles.findStyle( styleName, family ); + if ( style ) + addStyles( style, family, isStyleAutoStyle ); + else + kdWarning(32500) << "fillStyleStack: no style named " << styleName << " found." << endl; + } +} + +void KoOasisLoadingContext::addStyles( const QDomElement* style, const char* family, bool usingStylesAutoStyles ) +{ + Q_ASSERT( style ); + if ( !style ) return; + // this recursive function is necessary as parent styles can have parents themselves + if ( style->hasAttributeNS( KoXmlNS::style, "parent-style-name" ) ) { + const QString parentStyleName = style->attributeNS( KoXmlNS::style, "parent-style-name", QString::null ); + const QDomElement* parentStyle = 0; + if ( usingStylesAutoStyles ) { + // When loading something from styles.xml, look into the styles.xml auto styles first + parentStyle = m_styles.findStyleAutoStyle( parentStyleName, family ); + // and fallback to looking at styles(), which includes the user styles from styles.xml + } + if ( !parentStyle ) + parentStyle = m_styles.findStyle( parentStyleName, family ); + if ( parentStyle ) + addStyles( parentStyle, family, usingStylesAutoStyles ); + else + kdWarning(32500) << "Parent style not found: " << parentStyleName << endl; + } + else if ( family ) { + const QDomElement* def = m_styles.defaultStyle( family ); + if ( def ) { // on top of all, the default style for this family + //kdDebug(32500) << "pushing default style " << style->attributeNS( KoXmlNS::style, "name", QString::null ) << endl; + m_styleStack.push( *def ); + } + } + + //kdDebug(32500) << "pushing style " << style->attributeNS( KoXmlNS::style, "name", QString::null ) << endl; + m_styleStack.push( *style ); +} + +QString KoOasisLoadingContext::generator() const +{ + parseMeta(); + return m_generator; +} + +void KoOasisLoadingContext::parseMeta() const +{ + if ( !m_metaXmlParsed && m_store ) + { + if ( m_store->hasFile( "meta.xml" ) ) + { + QDomDocument metaDoc; + KoOasisStore oasisStore( m_store ); + QString errorMsg; + if ( oasisStore.loadAndParse( "meta.xml", metaDoc, errorMsg ) ) { + QDomNode meta = KoDom::namedItemNS( metaDoc, KoXmlNS::office, "document-meta" ); + QDomNode office = KoDom::namedItemNS( meta, KoXmlNS::office, "meta" ); + QDomElement generator = KoDom::namedItemNS( office, KoXmlNS::meta, "generator" ); + if ( !generator.isNull() ) + m_generator = generator.text(); + } + } + m_metaXmlParsed = true; + } +} diff --git a/lib/kofficecore/KoOasisLoadingContext.h b/lib/kofficecore/KoOasisLoadingContext.h new file mode 100644 index 00000000..cac1786b --- /dev/null +++ b/lib/kofficecore/KoOasisLoadingContext.h @@ -0,0 +1,122 @@ +/* This file is part of the KDE project + Copyright (C) 2005 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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. +*/ + +#ifndef KOOASISLOADINGCONTEXT_H +#define KOOASISLOADINGCONTEXT_H + +class KoXmlWriter; +class QDomElement; +class KoDocument; +class KoOasisStyles; +class KoPictureCollection; +class KoStore; + +#include <qmap.h> +#include <koffice_export.h> +#include <qstringlist.h> +#include <KoStyleStack.h> + +/** + * Used during loading of Oasis format (and discarded at the end of the loading). + * + * @author David Faure <faure@kde.org> + */ +class KOFFICECORE_EXPORT KoOasisLoadingContext +{ +public: + /** + * Stores reference to the KoOasisStyles and stored passed by KoDocument. + * Make sure that the KoOasisStyles instance outlives this KoOasisLoadingContext instance. + * (This is the case during loading, when using the KoOasisStyles given by KoDocument) + * + * @param doc the KoDocument being loaded + * @param styles reference to the KoOasisStyles parsed by KoDocument + * @param store pointer to store, if available, for e.g. loading images. + */ + KoOasisLoadingContext( KoDocument* doc, KoOasisStyles& styles, KoStore* store ); + ~KoOasisLoadingContext(); + + KoDocument* koDocument() { return m_doc; } + KoStore* store() { return m_store; } + + KoOasisStyles& oasisStyles() { return m_styles; } + KoStyleStack& styleStack() { return m_styleStack; } + + const QDomDocument& manifestDocument() const { return m_manifestDoc; } + + /// Return the <meta:generator> of the document, e.g. "KOffice/1.4.0a" + QString generator() const; + + /** + * Convenience method for loading the style of an object + * before loading that object. + * + * Read attribute (nsURI,attrName) from the given dom element, + * treat that attribute as a style name, and load that style + * including all its parent styles. + * @param element the dom element to read the attribute from + * @param nsURI the namespace URI of the attribute to read + * @param attrName the name of the attribute to read + * @param family the style family used for this object + */ + void fillStyleStack( const QDomElement& element, const char* nsURI, const char* attrName, const char* family ); + + /** + * Add @p style to the stack, as well as all its parent styles + * and the default style for this style family. + * + * @param style the dom element containing the style to add to the stack + * @param family the family to use when looking up parent styles + * @param usingStylesAutoStyles if true, the parent styles are looked up + * in the automatic styles from styles.xml, instead of looking up + * in the automatic styles from content.xml as we usually do. + * This is useful for loading headers and footers for instance. + * See setUseStylesAutoStyles(), which makes fillStyleStack() set this bool. + * + * Usually you would call fillStyleStack() instead. + */ + void addStyles( const QDomElement* style, const char* family, bool usingStylesAutoStyles = false ); + + /// Set to true while loading headers and footers, to remember to use auto styles + /// from styles.xml + void setUseStylesAutoStyles( bool useStylesAutoStyles ) { m_useStylesAutoStyles = useStylesAutoStyles; } + //bool useStylesAutoStyles() const { return m_useStylesAutoStyles; } + +private: + void parseMeta() const; + +private: + KoDocument* m_doc; + KoStore* m_store; + KoOasisStyles& m_styles; + KoStyleStack m_styleStack; + + mutable QString m_generator; + mutable bool m_metaXmlParsed; + bool m_useStylesAutoStyles; + bool m_unused1; // padding, can be used later + bool m_unused2; // padding, can be used later + + QDomDocument m_manifestDoc; + + class Private; + Private *d; +}; + +#endif /* KOOASISLOADINGCONTEXT_H */ + diff --git a/lib/kofficecore/KoOasisSettings.cpp b/lib/kofficecore/KoOasisSettings.cpp new file mode 100644 index 00000000..99f63524 --- /dev/null +++ b/lib/kofficecore/KoOasisSettings.cpp @@ -0,0 +1,224 @@ +/* This file is part of the KDE project + Copyright (C) 2004 Laurent Montel <montel@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 "KoOasisSettings.h" +#include "KoXmlNS.h" +#include "KoDom.h" +#include <kdebug.h> + +KoOasisSettings::KoOasisSettings( const QDomDocument& doc ) + : m_settingsElement( KoDom::namedItemNS( doc.documentElement(), KoXmlNS::office, "settings" ) ), + m_configNSURI( KoXmlNS::config ) +{ + const QDomElement contents = doc.documentElement(); + if ( m_settingsElement.isNull() ) + kdDebug() << " document doesn't have tag 'office:settings'\n"; +} + +KoOasisSettings::KoOasisSettings( const QDomDocument& doc, const char* officeNSURI, const char* configNSURI ) + : m_settingsElement( KoDom::namedItemNS( doc.documentElement(), officeNSURI, "settings" ) ), + m_configNSURI( configNSURI ) +{ + const QDomElement contents = doc.documentElement(); + if ( m_settingsElement.isNull() ) + kdDebug() << " document doesn't have tag 'office:settings'\n"; +} + +KoOasisSettings::Items KoOasisSettings::itemSet( const QString& itemSetName ) const +{ + QDomElement e; + forEachElement( e, m_settingsElement ) + { + if ( e.localName() == "config-item-set" && + e.namespaceURI() == m_configNSURI && + e.attributeNS( m_configNSURI, "name", QString::null ) == itemSetName ) + { + return Items( e, this ); + } + } + + return Items( QDomElement(), this ); +} + +KoOasisSettings::IndexedMap KoOasisSettings::Items::indexedMap( const QString& itemMapName ) const +{ + QDomElement configItem; + forEachElement( configItem, m_element ) + { + if ( configItem.localName() == "config-item-map-indexed" && + configItem.namespaceURI() == m_settings->m_configNSURI && + configItem.attributeNS( m_settings->m_configNSURI, "name", QString::null ) == itemMapName ) + { + return IndexedMap( configItem, m_settings ); + } + } + return IndexedMap( QDomElement(), m_settings ); +} + +KoOasisSettings::NamedMap KoOasisSettings::Items::namedMap( const QString& itemMapName ) const +{ + QDomElement configItem; + forEachElement( configItem, m_element ) + { + if ( configItem.localName() == "config-item-map-named" && + configItem.namespaceURI() == m_settings->m_configNSURI && + configItem.attributeNS( m_settings->m_configNSURI, "name", QString::null ) == itemMapName ) + { + return NamedMap( configItem, m_settings ); + } + } + return NamedMap( QDomElement(), m_settings ); +} + +KoOasisSettings::Items KoOasisSettings::IndexedMap::entry( int entryIndex ) const +{ + int i = 0; + QDomElement entry; + forEachElement( entry, m_element ) + { + if ( entry.localName() == "config-item-map-entry" && + entry.namespaceURI() == m_settings->m_configNSURI ) + { + if ( i == entryIndex ) + return Items( entry, m_settings ); + else + ++i; + } + } + return Items( QDomElement(), m_settings ); +} + +KoOasisSettings::Items KoOasisSettings::NamedMap::entry( const QString& entryName ) const +{ + QDomElement entry; + forEachElement( entry, m_element ) + { + if ( entry.localName() == "config-item-map-entry" && + entry.namespaceURI() == m_settings->m_configNSURI && + entry.attributeNS( m_settings->m_configNSURI, "name", QString::null ) == entryName ) + { + return Items( entry, m_settings ); + } + } + return Items( QDomElement(), m_settings ); +} + +// helper method +QString KoOasisSettings::Items::findConfigItem( const QDomElement& element, + const QString& item, bool* ok ) const +{ + QDomElement it; + forEachElement( it, element ) + { + if ( it.localName() == "config-item" && + it.namespaceURI() == m_settings->m_configNSURI && + it.attributeNS( m_settings->m_configNSURI, "name", QString::null ) == item ) + { + *ok = true; + return it.text(); + } + } + *ok = false; + return QString::null; +} + + +QString KoOasisSettings::Items::findConfigItem( const QString& item, bool* ok ) const +{ + return findConfigItem( m_element, item, ok ); +} + +#if 0 // does anyone need this one? passing a default value does the job, too +bool KoOasisSettings::Items::hasConfigItem( const QString& configName ) const +{ + bool ok; + (void)findConfigItem( configName, &ok ); + return ok; +} +#endif + +QString KoOasisSettings::Items::parseConfigItemString( const QString& configName, const QString& defValue ) const +{ + bool ok; + const QString str = findConfigItem( configName, &ok ); + return ok ? str : defValue; +} + +int KoOasisSettings::Items::parseConfigItemInt( const QString& configName, int defValue ) const +{ + bool ok; + const QString str = findConfigItem( configName, &ok ); + int value; + if ( ok ) { + value = str.toInt( &ok ); + if ( ok ) + return value; + } + return defValue; +} + +double KoOasisSettings::Items::parseConfigItemDouble( const QString& configName, double defValue ) const +{ + bool ok; + const QString str = findConfigItem( configName, &ok ); + double value; + if ( ok ) { + value = str.toDouble( &ok ); + if ( ok ) + return value; + } + return defValue; +} + +bool KoOasisSettings::Items::parseConfigItemBool( const QString& configName, bool defValue ) const +{ + bool ok; + const QString str = findConfigItem( configName, &ok ); + if ( str == "true" ) + return true; + else if ( str == "false" ) + return false; + return defValue; +} + +short KoOasisSettings::Items::parseConfigItemShort( const QString& configName, short defValue ) const +{ + bool ok; + const QString str = findConfigItem( configName, &ok ); + short value; + if ( ok ) { + value = str.toShort( &ok ); + if ( ok ) + return value; + } + return defValue; +} + +long KoOasisSettings::Items::parseConfigItemLong( const QString& configName, long defValue ) const +{ + bool ok; + const QString str = findConfigItem( configName, &ok ); + long value; + if ( ok ) { + value = str.toLong( &ok ); + if ( ok ) + return value; + } + return defValue; +} diff --git a/lib/kofficecore/KoOasisSettings.h b/lib/kofficecore/KoOasisSettings.h new file mode 100644 index 00000000..cd1ae112 --- /dev/null +++ b/lib/kofficecore/KoOasisSettings.h @@ -0,0 +1,179 @@ +/* This file is part of the KDE project + Copyright (C) 2004 Laurent Montel <montel@kde.org> + David Faure <faure@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. +*/ + + +#ifndef KOOASISSETTINGS_H +#define KOOASISSETTINGS_H + +#include <qdom.h> +#include <koffice_export.h> + +/** + * @brief Parse settings.xml file. + * + * This class helps parsing the settings.xml file of an OASIS document. + * + * For reference, the structure of settings.xml looks like: + * <pre> + * \<office:settings\> + * \<config:config-item-set config:name="configure-settings"\> + * .... + * \</config:config-item-set\> + * \<config:config-item-set config:name="view-settings"\> + * \<config:config-item-map-indexed config:name="Views"\> + * \<config:config-item-map-entry\> + * \<config:config-item config:name="SnapLinesDrawing" config:type="string"\>value\</config:config-item\> + * .... + * \<config:config-item-map-named config:name="Tables"\> + * \<config:config-item-map-entry config:name="Sheet1"\> + * \<config:config-item config:name="CursorPositionX"\> + * ...... + * \</config:config-item-map-entry\> + * \<config:config-item-map-entry config:name="Sheet2"\> + * .... + * \</config:config-item-map-entry\> + * \</config:config-item-map-named\> + * ..... + * \</config:config-item-map-entry\> + * \</config:config-item-map-indexed\> + * \<config:config-item-map-indexed config:name="Interface"\> + * ....... + * \</config:config-item-map-indexed\> + * \</config:config-item-set\> + * \</office:settings\> + * </pre> + * Basically, an item-set is a set of named \<config-item\>s and/or maps. + * There are two kinds of maps (by-index or by-name), and entries in the + * maps contain \<config-item\>s too, or nested maps. + * + * The API of KoOasisSettings allows the caller to look for a given item-set + * or item-map once, and then lookup multiple items inside it. + * It also allows "drilling down" inside the tree in case of nesting. + */ +class KOFFICECORE_EXPORT KoOasisSettings +{ +public: + /** + * Normal KoOasisSettings constructor, for an OASIS settings.xml + */ + explicit KoOasisSettings( const QDomDocument& doc ); + + /** + * KoOasisSettings constructor for an OpenOffice-1.1 file + */ + KoOasisSettings( const QDomDocument& doc, const char* officeNSURI, const char* configNSURI ); + + class Items; + + /** + * Returns the toplevel item-set named @p itemSetName. + * If not found, the returned items instance is null. + */ + Items itemSet( const QString& itemSetName ) const; + + class IndexedMap; + class NamedMap; + /// Represents a collection of items (config-item or maps). + class KOFFICECORE_EXPORT Items + { + friend class KoOasisSettings; + friend class IndexedMap; + friend class NamedMap; + Items( const QDomElement& elem, const KoOasisSettings* settings ) + : m_element( elem ), m_settings( settings ) {} + public: + bool isNull() const { return m_element.isNull(); } + + /** + * Look for the config-item-map-indexed named @p itemMapName and return it. + * + * An indexed map is an array (or sequence), i.e. items are supposed to + * be retrieved by index. This is useful for e.g. "view 0", "view 1" etc. + */ + IndexedMap indexedMap( const QString& itemMapName ) const; + + /** + * Look for the config-item-map-named named @p mapItemName and return it. + * + * A named map is a map where items are retrieved by entry name, @see selectItemMapEntry + * @return false if no such map was found + */ + NamedMap namedMap( const QString& itemMapName ) const; + + int parseConfigItemInt( const QString& configName, int defValue = 0 ) const; + double parseConfigItemDouble( const QString& configName, double defValue = 0 ) const; + QString parseConfigItemString( const QString& configName, const QString& defValue = QString::null ) const; + bool parseConfigItemBool( const QString& configName, bool defValue = false ) const; + short parseConfigItemShort( const QString& configName, short defValue = 0 ) const; + long parseConfigItemLong( const QString& configName, long defValue = 0 ) const; + private: + /// @internal + QString findConfigItem( const QString& item, bool* ok ) const; + /// @internal + QString findConfigItem( const QDomElement& element, const QString& item, bool* ok ) const; + + QDomElement m_element; + const KoOasisSettings* m_settings; + }; + + /// Internal base class for IndexedMap and NamedMap + class Map + { + public: + bool isNull() const { return m_element.isNull(); } + protected: + Map( const QDomElement& elem, const KoOasisSettings* settings ) + : m_element( elem ), m_settings( settings ) {} + const QDomElement m_element; + const KoOasisSettings* m_settings; + }; + + class KOFFICECORE_EXPORT IndexedMap : public Map + { + friend class Items; + IndexedMap( const QDomElement& elem, const KoOasisSettings* settings ) + : Map( elem, settings ) {} + public: + /// Returns an entry in an indexed map + Items entry( int entryIndex ) const; + }; + + class KOFFICECORE_EXPORT NamedMap : public Map + { + friend class Items; + NamedMap( const QDomElement& elem, const KoOasisSettings* settings ) + : Map( elem, settings ) {} + public: + /// Returns an entry in a named map + Items entry( const QString& entryName ) const; + }; + +private: + friend class Items; + friend class IndexedMap; + friend class NamedMap; + const QDomElement m_settingsElement; + const char* m_configNSURI; + + class Private; + Private* d; +}; + +#endif diff --git a/lib/kofficecore/KoOasisStore.cpp b/lib/kofficecore/KoOasisStore.cpp new file mode 100644 index 00000000..1b7537ae --- /dev/null +++ b/lib/kofficecore/KoOasisStore.cpp @@ -0,0 +1,194 @@ +/* This file is part of the KDE project + Copyright (C) 2005 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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 "KoOasisStore.h" + +#include "KoDocument.h" +#include "KoXmlNS.h" +#include "KoDom.h" +#include <KoStore.h> +#include <KoStoreDevice.h> +#include <KoXmlWriter.h> + +#include <ktempfile.h> +#include <kdebug.h> +#include <klocale.h> + +#include <qfile.h> +#include <qxml.h> +#include <qbuffer.h> + +KoOasisStore::KoOasisStore( KoStore* store ) + : m_store( store ), + m_storeDevice( 0 ), + m_contentWriter( 0 ), + m_bodyWriter( 0 ), + m_manifestWriter( 0 ), + m_contentTmpFile( 0 ) +{ +} + +KoOasisStore::~KoOasisStore() +{ + // If all the right close methods were called, nothing should remain, + // so those deletes are really just in case. + Q_ASSERT( !m_contentWriter ); + delete m_contentWriter; + Q_ASSERT( !m_bodyWriter ); + delete m_bodyWriter; + Q_ASSERT( !m_storeDevice ); + delete m_storeDevice; + Q_ASSERT( !m_contentTmpFile ); + delete m_contentTmpFile; + Q_ASSERT( !m_manifestWriter ); + delete m_manifestWriter; +} + +KoXmlWriter* KoOasisStore::contentWriter() +{ + if ( !m_contentWriter ) + { + if ( !m_store->open( "content.xml" ) ) + return 0; + m_storeDevice = new KoStoreDevice( m_store ); + m_contentWriter = KoDocument::createOasisXmlWriter( m_storeDevice, "office:document-content" ); + } + return m_contentWriter; +} + +KoXmlWriter* KoOasisStore::bodyWriter() +{ + if ( !m_bodyWriter ) + { + Q_ASSERT( !m_contentTmpFile ); + m_contentTmpFile = new KTempFile; + m_contentTmpFile->setAutoDelete( true ); + m_bodyWriter = new KoXmlWriter( m_contentTmpFile->file(), 1 ); + } + return m_bodyWriter; +} + +bool KoOasisStore::closeContentWriter() +{ + Q_ASSERT( m_bodyWriter ); + Q_ASSERT( m_contentTmpFile ); + + delete m_bodyWriter; m_bodyWriter = 0; + // copy over the contents from the tempfile to the real one + QFile* tmpFile = m_contentTmpFile->file(); + tmpFile->close(); + m_contentWriter->addCompleteElement( tmpFile ); + m_contentTmpFile->close(); + delete m_contentTmpFile; m_contentTmpFile = 0; + + Q_ASSERT( m_contentWriter ); + m_contentWriter->endElement(); // document-content + m_contentWriter->endDocument(); + delete m_contentWriter; m_contentWriter = 0; + delete m_storeDevice; m_storeDevice = 0; + if ( !m_store->close() ) // done with content.xml + return false; + return true; +} + +KoXmlWriter* KoOasisStore::manifestWriter( const char* mimeType ) +{ + if ( !m_manifestWriter ) + { + // the pointer to the buffer is already stored in the KoXmlWriter, no need to store it here as well + QBuffer *manifestBuffer = new QBuffer; + manifestBuffer->open( IO_WriteOnly ); + m_manifestWriter = new KoXmlWriter( manifestBuffer ); + m_manifestWriter->startDocument( "manifest:manifest" ); + m_manifestWriter->startElement( "manifest:manifest" ); + m_manifestWriter->addAttribute( "xmlns:manifest", KoXmlNS::manifest ); + m_manifestWriter->addManifestEntry( "/", mimeType ); + } + return m_manifestWriter; +} + +bool KoOasisStore::closeManifestWriter() +{ + m_manifestWriter->endElement(); + m_manifestWriter->endDocument(); + QBuffer* buffer = static_cast<QBuffer *>( m_manifestWriter->device() ); + delete m_manifestWriter; m_manifestWriter = 0; + bool ok = false; + if ( m_store->open( "META-INF/manifest.xml" ) ) + { + Q_LONG written = m_store->write( buffer->buffer() ); + ok = ( written == (Q_LONG)buffer->buffer().size() && m_store->close() ); + } + delete buffer; + return ok; +} + +bool KoOasisStore::loadAndParse( const QString& fileName, QDomDocument& doc, QString& errorMessage ) +{ + //kdDebug(30003) << "loadAndParse: Trying to open " << fileName << endl; + + if (!m_store->open(fileName)) + { + kdWarning(30003) << "Entry " << fileName << " not found!" << endl; + errorMessage = i18n( "Could not find %1" ).arg( fileName ); + return false; + } + // Error variables for QDomDocument::setContent + QString errorMsg; + int errorLine, errorColumn; + + // We need to be able to see the space in <text:span> </text:span>, this is why + // we activate the "report-whitespace-only-CharData" feature. + // Unfortunately this leads to lots of whitespace text nodes in between real + // elements in the rest of the document, watch out for that. + QXmlInputSource source( m_store->device() ); + // Copied from QDomDocumentPrivate::setContent, to change the whitespace thing + QXmlSimpleReader reader; + KoDocument::setupXmlReader( reader, true /*namespaceProcessing*/ ); + + bool ok = doc.setContent( &source, &reader, &errorMsg, &errorLine, &errorColumn ); + if ( !ok ) + { + kdError(30003) << "Parsing error in " << fileName << "! Aborting!" << endl + << " In line: " << errorLine << ", column: " << errorColumn << endl + << " Error message: " << errorMsg << endl; + errorMessage = i18n( "Parsing error in the main document at line %1, column %2\nError message: %3" ) + .arg( errorLine ).arg( errorColumn ).arg( i18n ( "QXml", errorMsg.utf8() ) ); + } + else + { + kdDebug(30003) << "File " << fileName << " loaded and parsed" << endl; + } + m_store->close(); + return ok; +} + +QString KoOasisStore::mimeForPath( const QDomDocument& doc, const QString& fullPath ) +{ + QDomElement docElem = doc.documentElement(); + QDomElement elem; + forEachElement( elem, docElem ) + { + if ( elem.localName() == "file-entry" && elem.namespaceURI() == KoXmlNS::manifest ) + { + if ( elem.attributeNS( KoXmlNS::manifest, "full-path", QString::null ) == fullPath ) + return elem.attributeNS( KoXmlNS::manifest, "media-type", QString::null ); + } + } + return QString::null; +} diff --git a/lib/kofficecore/KoOasisStore.h b/lib/kofficecore/KoOasisStore.h new file mode 100644 index 00000000..506700b0 --- /dev/null +++ b/lib/kofficecore/KoOasisStore.h @@ -0,0 +1,96 @@ +/* This file is part of the KDE project + Copyright (C) 2005 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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. +*/ +#ifndef KOOASISSTORE_H +#define KOOASISSTORE_H + +class QString; +class QDomDocument; +class KTempFile; +class KoXmlWriter; +class KoStore; +class KoStoreDevice; +class QIODevice; + +/** + * Helper class around KoStore for writing out OASIS files. + * This class helps solving the problem that automatic styles must be before + * the body, but it's easier to iterate over the application's objects only + * once. So we open a KoXmlWriter into a memory buffer, write the body into it, + * collect automatic styles while doing that, write out automatic styles, + * and then copy the body XML from the buffer into the real KoXmlWriter. + * + * The typical use of this class is therefore: + * - write body into bodyWriter() and collect auto styles + * - write auto styles into contentWriter() + * - call closeContentWriter() + * - write other files into the store (styles.xml, settings.xml etc.) + * + * + * TODO: maybe we could encapsulate a bit more things, to e.g. handle + * adding manifest entries automatically. + * + * @author: David Faure <faure@kde.org> + */ +#include <koffice_export.h> + +class KOFFICECORE_EXPORT KoOasisStore +{ +public: + /// @param store recontents the property of the caller + KoOasisStore( KoStore* store ); + + ~KoOasisStore(); + + KoStore* store() const { return m_store; } + + /// Open contents.xml for writing and return the KoXmlWriter + KoXmlWriter* contentWriter(); + + /// Open another KoXmlWriter for writing out the contents + /// into a temporary file, to collect automatic styles while doing that. + KoXmlWriter* bodyWriter(); + + /// This will copy the body into the content writer, + /// delete the bodyWriter and the contentWriter, and then + /// close contents.xml. + bool closeContentWriter(); + + // For other files in the store, use open/addManifestEntry/KoStoreDevice/createOasisXmlWriter/close + + /// Create and return a manifest writer. It will write to a memory buffer. + KoXmlWriter* manifestWriter( const char* mimeType ); + + /// Close the manifest writer, writing its contents to manifest.xml + bool closeManifestWriter(); + + /// A completely unrelated method, for loading a file from an oasis store + bool loadAndParse( const QString& fileName, QDomDocument& doc, QString& errorMessage ); + + /// Another method for loading: get mimetype from full path, using the manifest + static QString mimeForPath( const QDomDocument& doc, const QString& fullPath ); + +private: + KoStore* m_store; + KoStoreDevice* m_storeDevice; + KoXmlWriter* m_contentWriter; + KoXmlWriter* m_bodyWriter; + KoXmlWriter* m_manifestWriter; + KTempFile* m_contentTmpFile; +}; + +#endif /* KOOASISSTORE_H */ diff --git a/lib/kofficecore/KoOasisStyles.cpp b/lib/kofficecore/KoOasisStyles.cpp new file mode 100644 index 00000000..3a018767 --- /dev/null +++ b/lib/kofficecore/KoOasisStyles.cpp @@ -0,0 +1,1604 @@ +/* This file is part of the KDE project + Copyright (C) 2004-2006 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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 "KoOasisStyles.h" +#include <KoXmlWriter.h> +#include <kdebug.h> +#include <qdom.h> +#include "KoDom.h" +#include "KoXmlNS.h" +#include "KoGenStyles.h" +#include <qbuffer.h> +#include <kglobal.h> +#include <klocale.h> +#include <qbrush.h> +#include <KoStyleStack.h> + +class KoOasisStyles::Private +{ +public: + // The key of the map is the family + QMap<QString, QDict<QDomElement> > m_styles; + QMap<QString, QDict<QDomElement> > m_stylesAutoStyles; +}; + +KoOasisStyles::KoOasisStyles() + : d( new Private ) +{ + m_defaultStyle.setAutoDelete( true ); + m_masterPages.setAutoDelete( true ); + m_listStyles.setAutoDelete( true ); + m_drawStyles.setAutoDelete( true ); +} + +KoOasisStyles::~KoOasisStyles() +{ + delete d; +} + +void KoOasisStyles::createStyleMap( const QDomDocument& doc, bool stylesDotXml ) +{ + const QDomElement docElement = doc.documentElement(); + // We used to have the office:version check here, but better let the apps do that + QDomElement fontStyles = KoDom::namedItemNS( docElement, KoXmlNS::office, "font-decls" ); + + if ( !fontStyles.isNull() ) { + //kdDebug(30003) << "Starting reading in font-decl..." << endl; + insertStyles( fontStyles, false ); + }// else + // kdDebug(30003) << "No items found" << endl; + + //kdDebug(30003) << "Starting reading in office:automatic-styles. stylesDotXml=" << stylesDotXml << endl; + + QDomElement autoStyles = KoDom::namedItemNS( docElement, KoXmlNS::office, "automatic-styles" ); + if ( !autoStyles.isNull() ) { + insertStyles( autoStyles, stylesDotXml ); + }// else + // kdDebug(30003) << "No items found" << endl; + + + //kdDebug(30003) << "Reading in master styles" << endl; + + QDomNode masterStyles = KoDom::namedItemNS( docElement, KoXmlNS::office, "master-styles" ); + + if ( !masterStyles.isNull() ) { + QDomElement master; + forEachElement( master, masterStyles ) + { + if ( master.localName() == "master-page" && + master.namespaceURI() == KoXmlNS::style ) { + const QString name = master.attributeNS( KoXmlNS::style, "name", QString::null ); + kdDebug(30003) << "Master style: '" << name << "' loaded " << endl; + m_masterPages.insert( name, new QDomElement( master ) ); + } else + // OASIS docu mentions style:handout-master and draw:layer-set here + kdWarning(30003) << "Unknown tag " << master.tagName() << " in office:master-styles" << endl; + } + } + + + kdDebug(30003) << "Starting reading in office:styles" << endl; + + const QDomElement officeStyle = KoDom::namedItemNS( docElement, KoXmlNS::office, "styles" ); + + if ( !officeStyle.isNull() ) { + m_officeStyle = officeStyle; + insertOfficeStyles( m_officeStyle ); + + } + + //kdDebug(30003) << "Styles read in." << endl; +} + +QValueVector<QDomElement> KoOasisStyles::userStyles() const +{ + QValueVector<QDomElement> vec; + // Collect user styles + unsigned int i = 0; + QDomElement e; + forEachElement( e, m_officeStyle ) + { + if ( e.localName() == "style" && + e.namespaceURI() == KoXmlNS::style ) + { + vec.resize( i+1 ); + vec[i++] = e; + } + } + return vec; +} + +const QDict<QDomElement>& KoOasisStyles::styles(const QString& family) const +{ + // hmm this can create an empty item in the map, but otherwise we couldn't + // return a const reference. + return d->m_styles[family]; +} + +void KoOasisStyles::insertOfficeStyles( const QDomElement& styles ) +{ + QDomElement e; + forEachElement( e, styles ) + { + const QString localName = e.localName(); + const QString ns = e.namespaceURI(); + if ( ( ns == KoXmlNS::svg && ( + localName == "linearGradient" + || localName == "radialGradient" ) ) + || ( ns == KoXmlNS::draw && ( + localName == "gradient" + || localName == "hatch" + || localName == "fill-image" + || localName == "marker" + || localName == "stroke-dash" + || localName == "opacity" ) ) + ) + { + const QString name = e.attributeNS( KoXmlNS::draw, "name", QString::null ); + Q_ASSERT( !name.isEmpty() ); + QDomElement* ep = new QDomElement( e ); + m_drawStyles.insert( name, ep ); + } + else + insertStyle( e, false ); + } +} + + +void KoOasisStyles::insertStyles( const QDomElement& styles, bool styleAutoStyles ) +{ + //kdDebug(30003) << "Inserting styles from " << styles.tagName() << endl; + QDomElement e; + forEachElement( e, styles ) + insertStyle( e, styleAutoStyles ); +} + +void KoOasisStyles::insertStyle( const QDomElement& e, bool styleAutoStyles ) +{ + const QString localName = e.localName(); + const QString ns = e.namespaceURI(); + + const QString name = e.attributeNS( KoXmlNS::style, "name", QString::null ); + if ( ns == KoXmlNS::style && localName == "style" ) { + const QString family = e.attributeNS( KoXmlNS::style, "family", QString::null ); + + if ( styleAutoStyles ) { + QDict<QDomElement>& dict = d->m_stylesAutoStyles[ family ]; + dict.setAutoDelete( true ); + if ( dict.find( name ) != 0 ) + kdDebug(30003) << "Auto-style: '" << name << "' already exists" << endl; + dict.insert( name, new QDomElement( e ) ); + //kdDebug(30003) << "Style: '" << name << "' loaded as a style auto style" << endl; + } else { + QDict<QDomElement>& dict = d->m_styles[ family ]; + dict.setAutoDelete( true ); + + if ( dict.find( name ) != 0 ) + kdDebug(30003) << "Style: '" << name << "' already exists" << endl; + dict.insert( name, new QDomElement( e ) ); + //kdDebug(30003) << "Style: '" << name << "' loaded " << endl; + } + } else if ( ns == KoXmlNS::style && ( + localName == "page-layout" + || localName == "font-decl" + || localName == "presentation-page-layout" ) ) + { + if ( m_styles.find( name ) != 0 ) + kdDebug(30003) << "Style: '" << name << "' already exists" << endl; + m_styles.insert( name, new QDomElement( e ) ); + } else if ( localName == "default-style" && ns == KoXmlNS::style ) { + const QString family = e.attributeNS( KoXmlNS::style, "family", QString::null ); + if ( !family.isEmpty() ) + m_defaultStyle.insert( family, new QDomElement( e ) ); + } else if ( localName == "list-style" && ns == KoXmlNS::text ) { + m_listStyles.insert( name, new QDomElement( e ) ); + //kdDebug(30003) << "List style: '" << name << "' loaded " << endl; + } else if ( ns == KoXmlNS::number && ( + localName == "number-style" + || localName == "currency-style" + || localName == "percentage-style" + || localName == "boolean-style" + || localName == "text-style" + || localName == "date-style" + || localName == "time-style" ) ) { + importDataStyle( e ); + } + // The rest (text:*-configuration and text:outline-style) is to be done by the apps. +} + +// OO spec 2.5.4. p68. Conversion to Qt format: see qdate.html +// OpenCalcImport::loadFormat has similar code, but slower, intermixed with other stuff, +// lacking long-textual forms. +void KoOasisStyles::importDataStyle( const QDomElement& parent ) +{ + NumericStyleFormat dataStyle; + + const QString localName = parent.localName(); + if (localName == "number-style") + dataStyle.type = NumericStyleFormat::Number; + else if (localName == "currency-style") + dataStyle.type = NumericStyleFormat::Currency; + else if (localName == "percentage-style") + dataStyle.type = NumericStyleFormat::Percentage; + else if (localName == "boolean-style") + dataStyle.type = NumericStyleFormat::Boolean; + else if (localName == "text-style") + dataStyle.type = NumericStyleFormat::Text; + else if (localName == "date-style") + dataStyle.type = NumericStyleFormat::Date; + else if (localName == "time-style") + dataStyle.type = NumericStyleFormat::Time; + + QString format; + int precision = -1; + int leadingZ = 1; + bool thousandsSep = false; + //todo negred + //bool negRed = false; + bool ok = false; + int i = 0; + QDomElement e; + QString prefix; + QString suffix; + forEachElement( e, parent ) + { + if ( e.namespaceURI() != KoXmlNS::number ) + continue; + QString localName = e.localName(); + const QString numberStyle = e.attributeNS( KoXmlNS::number, "style", QString::null ); + const bool shortForm = numberStyle == "short" || numberStyle.isEmpty(); + if ( localName == "day" ) { + format += shortForm ? "d" : "dd"; + } else if ( localName == "day-of-week" ) { + format += shortForm ? "ddd" : "dddd"; + } else if ( localName == "month" ) { + if ( e.attributeNS( KoXmlNS::number, "possessive-form", QString::null ) == "true" ) { + format += shortForm ? "PPP" : "PPPP"; + } + // TODO the spec has a strange mention of number:format-source + else if ( e.attributeNS( KoXmlNS::number, "textual", QString::null ) == "true" ) { + format += shortForm ? "MMM" : "MMMM"; + } else { // month number + format += shortForm ? "M" : "MM"; + } + } else if ( localName == "year" ) { + format += shortForm ? "yy" : "yyyy"; + } else if ( localName == "era" ) { + //TODO I don't know what is it... (define into oo spec) + } else if ( localName == "week-of-year" || localName == "quarter") { + // ### not supported in Qt + } else if ( localName == "hours" ) { + format += shortForm ? "h" : "hh"; + } else if ( localName == "minutes" ) { + format += shortForm ? "m" : "mm"; + } else if ( localName == "seconds" ) { + format += shortForm ? "s" : "ss"; + } else if ( localName == "am-pm" ) { + format += "ap"; + } else if ( localName == "text" ) { // litteral + format += e.text(); + } else if ( localName == "suffix" ) { + suffix = e.text(); + kdDebug()<<" suffix :"<<suffix<<endl; + } else if ( localName == "prefix" ) { + prefix = e.text(); + kdDebug()<<" prefix :"<<prefix<<endl; + } else if ( localName == "currency-symbol" ) { + dataStyle.currencySymbol = e.text(); + kdDebug()<<" currency-symbol: "<<dataStyle.currencySymbol<<endl; + format += e.text(); + //TODO + // number:language="de" number:country="DE">€</number:currency-symbol> + // Stefan: localization of the symbol? + } else if ( localName == "number" ) { + // TODO: number:grouping="true" + if ( e.hasAttributeNS( KoXmlNS::number, "decimal-places" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "decimal-places", QString::null ).toInt( &ok ); + if ( ok ) + precision = d; + } + if ( e.hasAttributeNS( KoXmlNS::number, "min-integer-digits" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "min-integer-digits", QString::null ).toInt( &ok ); + if ( ok ) + leadingZ = d; + } + if ( thousandsSep && leadingZ <= 3 ) + { + format += "#,"; + for ( i = leadingZ; i <= 3; ++i ) + format += '#'; + } + for ( i = 1; i <= leadingZ; ++i ) + { + format += '0'; + if ( ( i % 3 == 0 ) && thousandsSep ) + format =+ ',' ; + } + if (precision > -1) + { + format += '.'; + for ( i = 0; i < precision; ++i ) + format += '0'; + } + } + else if ( localName == "scientific-number" ) { + if (dataStyle.type == NumericStyleFormat::Number) + dataStyle.type = NumericStyleFormat::Scientific; + int exp = 2; + + if ( e.hasAttributeNS( KoXmlNS::number, "decimal-places" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "decimal-places", QString::null ).toInt( &ok ); + if ( ok ) + precision = d; + } + + if ( e.hasAttributeNS( KoXmlNS::number, "min-integer-digits" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "min-integer-digits", QString::null ).toInt( &ok ); + if ( ok ) + leadingZ = d; + } + + if ( e.hasAttributeNS( KoXmlNS::number, "min-exponent-digits" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "min-exponent-digits", QString::null ).toInt( &ok ); + if ( ok ) + exp = d; + if ( exp <= 0 ) + exp = 1; + } + + if ( thousandsSep && leadingZ <= 3 ) + { + format += "#,"; + for ( i = leadingZ; i <= 3; ++i ) + format += '#'; + } + + for ( i = 1; i <= leadingZ; ++i ) + { + format+='0'; + if ( ( i % 3 == 0 ) && thousandsSep ) + format+=','; + } + + if (precision > -1) + { + format += '.'; + for ( i = 0; i < precision; ++i ) + format += '0'; + } + + format+="E+"; + for ( i = 0; i < exp; ++i ) + format+='0'; + } else if ( localName == "fraction" ) { + if (dataStyle.type == NumericStyleFormat::Number) + dataStyle.type = NumericStyleFormat::Fraction; + int integer = 0; + int numerator = 1; + int denominator = 1; + int denominatorValue=0; + if ( e.hasAttributeNS( KoXmlNS::number, "min-integer-digits" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "min-integer-digits", QString::null ).toInt( &ok ); + if ( ok ) + integer = d; + } + if ( e.hasAttributeNS( KoXmlNS::number, "min-numerator-digits" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "min-numerator-digits", QString::null ).toInt( &ok ); + if ( ok ) + numerator = d; + } + if ( e.hasAttributeNS( KoXmlNS::number, "min-denominator-digits" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "min-denominator-digits", QString::null ).toInt( &ok ); + if ( ok ) + denominator = d; + } + if ( e.hasAttributeNS( KoXmlNS::number, "denominator-value" ) ) + { + int d = e.attributeNS( KoXmlNS::number, "denominator-value", QString::null ).toInt( &ok ); + if ( ok ) + denominatorValue = d; + } + + for ( i = 0; i < integer; ++i ) + format+='#'; + + format+=' '; + + for ( i = 0; i < numerator; ++i ) + format+='?'; + + format+='/'; + + if ( denominatorValue != 0 ) + format+=QString::number( denominatorValue ); + else + { + for ( i = 0; i < denominator; ++i ) + format+='?'; + } + } + // Not needed: + // <style:map style:condition="value()>=0" style:apply-style-name="N106P0"/> + // we handle painting negative numbers in red differently + + } + + const QString styleName = parent.attributeNS( KoXmlNS::style, "name", QString::null ); + kdDebug(30003) << "data style: " << styleName << " qt format=" << format << endl; + if ( !prefix.isEmpty() ) + { + kdDebug(30003)<<" format.left( prefix.length() ) :"<<format.left( prefix.length() )<<" prefix :"<<prefix<<endl; + if ( format.left( prefix.length() )==prefix ) + { + format = format.right( format.length()-prefix.length() ); + } + else + prefix = QString::null; + } + if ( !suffix.isEmpty() ) + { + kdDebug(30003)<<"format.right( suffix.length() ) :"<<format.right( suffix.length() )<<" suffix :"<<suffix<<endl; + if ( format.right( suffix.length() )==suffix ) + { + format = format.left( format.length()-suffix.length() ); + } + else + suffix = QString::null; + } + + dataStyle.formatStr = format; + dataStyle.prefix = prefix; + dataStyle.suffix = suffix; + dataStyle.precision = precision; + kdDebug()<<" finish insert format :"<<format<<" prefix :"<<prefix<<" suffix :"<<suffix<<endl; + m_dataFormats.insert( styleName, dataStyle ); +} + +#define addTextNumber( text, elementWriter ) { \ + if ( !text.isEmpty() ) \ + { \ + elementWriter.startElement( "number:text" ); \ + elementWriter.addTextNode( text ); \ + elementWriter.endElement(); \ + text=""; \ + } \ +} + +void KoOasisStyles::parseOasisTimeKlocale(KoXmlWriter &elementWriter, QString & format, QString & text ) +{ + kdDebug(30003)<<"parseOasisTimeKlocale(KoXmlWriter &elementWriter, QString & format, QString & text ) :"<<format<<endl; + do + { + if ( !saveOasisKlocaleTimeFormat( elementWriter, format, text ) ) + { + text += format[0]; + format = format.remove( 0, 1 ); + } + } + while ( format.length() > 0 ); + addTextNumber( text, elementWriter ); +} + +bool KoOasisStyles::saveOasisKlocaleTimeFormat( KoXmlWriter &elementWriter, QString & format, QString & text ) +{ + bool changed = false; + if ( format.startsWith( "%H" ) ) //hh + { + //hour in 24h + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:hours" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + else if ( format.startsWith( "%k" ) )//h + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:hours" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + else if ( format.startsWith( "%I" ) )// ????? + { + //TODO hour in 12h + changed = true; + } + else if ( format.startsWith( "%l" ) ) + { + //TODO hour in 12h with 1 digit + changed = true; + } + else if ( format.startsWith( "%M" ) )// mm + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:minutes" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + + } + else if ( format.startsWith( "%S" ) ) //ss + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:seconds" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + else if ( format.startsWith( "%p" ) ) + { + //TODO am or pm + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:am-pm" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + return changed; +} + + +bool KoOasisStyles::saveOasisTimeFormat( KoXmlWriter &elementWriter, QString & format, QString & text, bool &antislash ) +{ + bool changed = false; + //we can also add time to date. + if ( antislash ) + { + text+=format[0]; + format = format.remove( 0, 1 ); + antislash = false; + changed = true; + } + else if ( format.startsWith( "hh" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:hours" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + else if ( format.startsWith( "h" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:hours" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 1 ); + changed = true; + } + else if ( format.startsWith( "mm" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:minutes" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + else if ( format.startsWith( "m" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:minutes" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 1 ); + changed = true; + } + else if ( format.startsWith( "ss" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:seconds" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + else if ( format.startsWith( "s" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:seconds" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 1 ); + changed = true; + } + else if ( format.startsWith( "ap" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:am-pm" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + changed = true; + } + return changed; +} + +QString KoOasisStyles::saveOasisTimeStyle( KoGenStyles &mainStyles, const QString & _format, bool klocaleFormat, + const QString & _prefix, const QString & _suffix ) +{ + Q_UNUSED(_prefix); + Q_UNUSED(_suffix); + kdDebug(30003)<<"QString KoOasisStyles::saveOasisTimeStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_TIME ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + if ( klocaleFormat ) + { + parseOasisTimeKlocale( elementWriter, format, text ); + } + else + { + bool antislash = false; + do + { + if ( !saveOasisTimeFormat( elementWriter, format, text, antislash ) ) + { + QString elem( format[0] ); + format = format.remove( 0, 1 ); + if ( elem == "\\" ) + { + antislash = true; + } + else + { + text += elem; + antislash = false; + } + } + } + while ( format.length() > 0 ); + addTextNumber( text, elementWriter ); + } + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); +} + +//convert klocale string to good format +void KoOasisStyles::parseOasisDateKlocale(KoXmlWriter &elementWriter, QString & format, QString & text ) +{ + kdDebug(30003)<<"KoOasisStyles::parseOasisDateKlocale(KoXmlWriter &elementWriter, QString & format, QString & text ) :"<<format<<endl; + do + { + if ( format.startsWith( "%Y" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:year" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%y" ) ) + { + + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:year" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%n" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.addAttribute( "number:textual", "false"); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%m" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.addAttribute( "number:textual", "false"); //not necessary remove it + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%e" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:day" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%d" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:day" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%b" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.addAttribute( "number:textual", "true"); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%B" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.addAttribute( "number:textual", "true"); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%a" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:day-of-week" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "%A" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:day-of-week" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else + { + if ( !saveOasisKlocaleTimeFormat( elementWriter, format, text ) ) + { + text += format[0]; + format = format.remove( 0, 1 ); + } + } + } + while ( format.length() > 0 ); + addTextNumber( text, elementWriter ); +} + +QString KoOasisStyles::saveOasisDateStyle( KoGenStyles &mainStyles, const QString & _format, bool klocaleFormat, + const QString & _prefix, const QString & _suffix ) +{ + Q_UNUSED(_prefix); + Q_UNUSED(_suffix); + kdDebug(30003)<<"QString KoOasisStyles::saveOasisDateStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + + // Not supported into Qt: "era" "week-of-year" "quarter" + + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_DATE ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + if ( klocaleFormat ) + { + parseOasisDateKlocale( elementWriter, format, text ); + } + else + { + bool antislash = false; + do + { + if ( antislash ) + { + text+=format[0]; + format = format.remove( 0, 1 ); + } + //TODO implement loading ! What is it ? + else if ( format.startsWith( "MMMMM" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:textual", "true"); + elementWriter.endElement(); + format = format.remove( 0, 5 ); + } + else if ( format.startsWith( "MMMM" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.addAttribute( "number:textual", "true"); + elementWriter.endElement(); + format = format.remove( 0, 4 ); + } + else if ( format.startsWith( "MMM" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.addAttribute( "number:textual", "true"); + elementWriter.endElement(); + format = format.remove( 0, 3 ); + } + else if ( format.startsWith( "MM" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.addAttribute( "number:textual", "false"); //not necessary remove it + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "M" ) ) + { + addTextNumber( text, elementWriter ); + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.addAttribute( "number:textual", "false"); + elementWriter.endElement(); + format = format.remove( 0, 1 ); + } + else if ( format.startsWith( "PPPP" ) ) + { + addTextNumber( text, elementWriter ); + //<number:month number:possessive-form="true" number:textual="true" number:style="long"/> + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.addAttribute( "number:textual", "false"); + elementWriter.addAttribute( "number:possessive-form", "true" ); + elementWriter.endElement(); + format = format.remove( 0, 4 ); + } + else if ( format.startsWith( "PPP" ) ) + { + addTextNumber( text, elementWriter ); + //<number:month number:possessive-form="true" number:textual="true" number:style="short"/> + elementWriter.startElement( "number:month" ); + elementWriter.addAttribute( "number:possessive-form", "true" ); + + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.addAttribute( "number:textual", "false"); + elementWriter.endElement(); + format = format.remove( 0, 3 ); + } + else if ( format.startsWith( "dddd" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:day-of-week" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 4 ); + } + else if ( format.startsWith( "ddd" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:day-of-week" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 3 ); + } + else if ( format.startsWith( "dd" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:day" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else if ( format.startsWith( "d" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:day" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 1 ); + } + else if ( format.startsWith( "yyyy" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:year" ); + elementWriter.addAttribute( "number:style", "long" ); + elementWriter.endElement(); + format = format.remove( 0, 4 ); + } + else if ( format.startsWith( "yy" ) ) + { + addTextNumber( text, elementWriter ); + + elementWriter.startElement( "number:year" ); + elementWriter.addAttribute( "number:style", "short" ); + elementWriter.endElement(); + format = format.remove( 0, 2 ); + } + else + { + if ( !saveOasisTimeFormat( elementWriter, format, text,antislash ) ) + { + QString elem( format[0] ); + format = format.remove( 0, 1 ); + if ( elem == "\\" ) + { + antislash = true; + } + else + { + text += elem; + antislash = false; + } + } + } + } + while ( format.length() > 0 ); + addTextNumber( text, elementWriter ); + } + + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); +} + + +QString KoOasisStyles::saveOasisFractionStyle( KoGenStyles &mainStyles, const QString & _format, const QString &_prefix, const QString &_suffix ) +{ + kdDebug(30003)<<"QString saveOasisFractionStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_FRACTION ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + int integer = 0; + int numerator = 0; + int denominator = 0; + int denominatorValue = 0; + bool beforeSlash = true; + do + { + if ( format[0]=='#' ) + integer++; + else if ( format[0]=='/' ) + beforeSlash = false; + else if ( format[0]=='?' ) + { + if ( beforeSlash ) + numerator++; + else + denominator++; + } + else + { + bool ok; + int value = format.toInt( &ok ); + if ( ok ) + { + denominatorValue=value; + break; + } + } + format.remove( 0,1 ); + } + while ( format.length() > 0 ); + + text= _prefix; + addTextNumber(text, elementWriter ); + + elementWriter.startElement( "number:fraction" ); + elementWriter.addAttribute( "number:min-integer-digits", integer ); + elementWriter.addAttribute( "number:min-numerator-digits",numerator ); + elementWriter.addAttribute( "number:min-denominator-digits",denominator ); + if ( denominatorValue != 0 ) + elementWriter.addAttribute( "number:denominator-value",denominatorValue ); + elementWriter.endElement(); + + addKofficeNumericStyleExtension( elementWriter, _suffix, _prefix ); + + text=_suffix; + addTextNumber(text, elementWriter ); + + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); +} + + +QString KoOasisStyles::saveOasisNumberStyle( KoGenStyles &mainStyles, const QString & _format, const QString &_prefix, const QString &_suffix ) +{ + kdDebug(30003)<<"QString saveOasisNumberStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_NUMBER ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + int decimalplaces = 0; + int integerdigits = 0; + bool beforeSeparator = true; + do + { + if ( format[0]=='.' || format[0]==',' ) + beforeSeparator = false; + else if ( format[0]=='0' && beforeSeparator ) + integerdigits++; + else if ( format[0]=='0' && !beforeSeparator ) + decimalplaces++; + else + kdDebug(30003)<<" error format 0 \n"; + format.remove( 0,1 ); + } + while ( format.length() > 0 ); + text= _prefix ; + addTextNumber(text, elementWriter ); + elementWriter.startElement( "number:number" ); + kdDebug(30003)<<" decimalplaces :"<<decimalplaces<<" integerdigits :"<<integerdigits<<endl; + if (!beforeSeparator) + elementWriter.addAttribute( "number:decimal-places", decimalplaces ); + elementWriter.addAttribute( "number:min-integer-digits", integerdigits ); + elementWriter.endElement(); + + text =_suffix ; + addTextNumber(text, elementWriter ); + addKofficeNumericStyleExtension( elementWriter, _suffix,_prefix ); + + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); +} + +QString KoOasisStyles::saveOasisPercentageStyle( KoGenStyles &mainStyles, const QString & _format, const QString &_prefix, const QString &_suffix ) +{ + //<number:percentage-style style:name="N11"> + //<number:number number:decimal-places="2" number:min-integer-digits="1"/> + //<number:text>%</number:text> + //</number:percentage-style> + + kdDebug(30003)<<"QString saveOasisPercentageStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_PERCENTAGE ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + int decimalplaces = 0; + int integerdigits = 0; + bool beforeSeparator = true; + do + { + if ( format[0]=='.' || format[0]==',' ) + beforeSeparator = false; + else if ( format[0]=='0' && beforeSeparator ) + integerdigits++; + else if ( format[0]=='0' && !beforeSeparator ) + decimalplaces++; + else + kdDebug(30003)<<" error format 0 \n"; + format.remove( 0,1 ); + } + while ( format.length() > 0 ); + text= _prefix ; + addTextNumber(text, elementWriter ); + elementWriter.startElement( "number:number" ); + if (!beforeSeparator) + elementWriter.addAttribute( "number:decimal-places", decimalplaces ); + elementWriter.addAttribute( "number:min-integer-digits", integerdigits ); + elementWriter.endElement(); + + addTextNumber(QString( "%" ), elementWriter ); + + text =_suffix ; + addTextNumber(text, elementWriter ); + addKofficeNumericStyleExtension( elementWriter, _suffix,_prefix ); + + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); + +} + +QString KoOasisStyles::saveOasisScientificStyle( KoGenStyles &mainStyles, const QString & _format, const QString &_prefix, const QString &_suffix ) +{ + //<number:number-style style:name="N60"> + //<number:scientific-number number:decimal-places="2" number:min-integer-digits="1" number:min-exponent-digits="3"/> + //</number:number-style> + + //example 000,000e+0000 + kdDebug(30003)<<"QString saveOasisScientificStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_SCIENTIFIC ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + int decimalplace = 0; + int integerdigits = 0; + int exponentdigits = 0; + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + bool beforeSeparator = true; + bool exponential = false; + bool positive = true; + do + { + if ( !exponential ) + { + if ( format[0]=='0' && beforeSeparator ) + integerdigits++; + else if ( format[0]==',' || format[0]=='.' ) + beforeSeparator = false; + else if ( format[0]=='0' && !beforeSeparator ) + decimalplace++; + else if ( format[0].lower()=='e' ) + { + format.remove( 0, 1 ); + if ( format[0]=='+' ) + positive = true; + else if ( format[0]=='-' ) + positive = false; + else + kdDebug(30003)<<"Error into scientific number\n"; + exponential = true; + } + } + else + { + if ( format[0]=='0' && positive ) + exponentdigits++; + else if ( format[0]=='0' && !positive ) + exponentdigits--; + else + kdDebug(30003)<<" error into scientific number exponential value\n"; + } + format.remove( 0,1 ); + } + while ( format.length() > 0 ); + text = _prefix ; + addTextNumber(text, elementWriter ); + + elementWriter.startElement( "number:scientific-number" ); + kdDebug(30003)<<" decimalplace :"<<decimalplace<<" integerdigits :"<<integerdigits<<" exponentdigits :"<<exponentdigits<<endl; + if (!beforeSeparator) + elementWriter.addAttribute( "number:decimal-places", decimalplace ); + elementWriter.addAttribute( "number:min-integer-digits",integerdigits ); + elementWriter.addAttribute( "number:min-exponent-digits",exponentdigits ); + elementWriter.endElement(); + + text = _suffix; + addTextNumber(text, elementWriter ); + addKofficeNumericStyleExtension( elementWriter, _suffix,_prefix ); + + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); +} + +QString KoOasisStyles::saveOasisCurrencyStyle( KoGenStyles &mainStyles, + const QString & _format, const QString &symbol, + const QString &_prefix, const QString &_suffix ) +{ + + //<number:currency-style style:name="N107P0" style:volatile="true"> + //<number:number number:decimal-places="2" number:min-integer-digits="1" number:grouping="true"/> + //<number:text> </number:text> + //<number:currency-symbol>VEB</number:currency-symbol> + //</number:currency-style> + + kdDebug(30003)<<"QString saveOasisCurrencyStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_CURRENCY ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + int decimalplaces = 0; + int integerdigits = 0; + bool beforeSeparator = true; + do + { + if ( format[0]=='.' || format[0]==',' ) + beforeSeparator = false; + else if ( format[0]=='0' && beforeSeparator ) + integerdigits++; + else if ( format[0]=='0' && !beforeSeparator ) + decimalplaces++; + else + kdDebug(30003)<<" error format 0 \n"; + format.remove( 0,1 ); + } + while ( format.length() > 0 ); + + text = _prefix ; + addTextNumber(text, elementWriter ); + + elementWriter.startElement( "number:number" ); + kdDebug(30003)<<" decimalplaces :"<<decimalplaces<<" integerdigits :"<<integerdigits<<endl; + if (!beforeSeparator) + elementWriter.addAttribute( "number:decimal-places", decimalplaces ); + elementWriter.addAttribute( "number:min-integer-digits", integerdigits ); + elementWriter.endElement(); + + text = _suffix ; + addTextNumber(text, elementWriter ); + addKofficeNumericStyleExtension( elementWriter, _suffix,_prefix ); + + elementWriter.startElement( "number:currency-symbol" ); + kdDebug(30003) << " currency-symbol: " << symbol << endl; + elementWriter.addTextNode( symbol.utf8() ); + elementWriter.endElement(); + + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); +} + +QString KoOasisStyles::saveOasisTextStyle( KoGenStyles &mainStyles, const QString & _format, const QString &_prefix, const QString &_suffix ) +{ + + //<number:text-style style:name="N100"> + //<number:text-content/> + ///</number:text-style> + + kdDebug(30003)<<"QString saveOasisTextStyle( KoGenStyles &mainStyles, const QString & _format ) :"<<_format<<endl; + QString format( _format ); + + KoGenStyle currentStyle( KoGenStyle::STYLE_NUMERIC_TEXT ); + QBuffer buffer; + buffer.open( IO_WriteOnly ); + KoXmlWriter elementWriter( &buffer ); // TODO pass indentation level + QString text; + do + { + format.remove( 0,1 ); + } + while ( format.length() > 0 ); + text = _prefix ; + addTextNumber(text, elementWriter ); + + elementWriter.startElement( "number:text-style" ); + + text = _suffix ; + addTextNumber(text, elementWriter ); + addKofficeNumericStyleExtension( elementWriter, _suffix,_prefix ); + elementWriter.endElement(); + + QString elementContents = QString::fromUtf8( buffer.buffer(), buffer.buffer().size() ); + currentStyle.addChildElement( "number", elementContents ); + return mainStyles.lookup( currentStyle, "N" ); +} + +//This is an extension of numeric style. For the moment we used namespace of oasis format for specific koffice extention. change it for the futur. +void KoOasisStyles::addKofficeNumericStyleExtension( KoXmlWriter & elementWriter, const QString &_suffix, const QString &_prefix ) + { + if ( !_suffix.isEmpty() ) + { + elementWriter.startElement( "number:suffix" ); + elementWriter.addTextNode( _suffix ); + elementWriter.endElement(); + } + if ( !_prefix.isEmpty() ) + { + elementWriter.startElement( "number:prefix" ); + elementWriter.addTextNode( _prefix ); + elementWriter.endElement(); + } +} + +void KoOasisStyles::saveOasisFillStyle( KoGenStyle &styleFill, KoGenStyles& mainStyles, const QBrush & brush ) +{ + if ( brush.style() == Qt::SolidPattern ) + { + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else if ( brush.style() == Qt::Dense1Pattern ) + { + styleFill.addProperty( "draw:transparency", "94%" ); + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else if ( brush.style() == Qt::Dense2Pattern ) + { + styleFill.addProperty( "draw:transparency", "88%" ); + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else if ( brush.style() == Qt::Dense3Pattern ) + { + styleFill.addProperty( "draw:transparency", "63%" ); + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else if ( brush.style() == Qt::Dense4Pattern ) + { + styleFill.addProperty( "draw:transparency", "50%" ); + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else if ( brush.style() == Qt::Dense5Pattern ) + { + styleFill.addProperty( "draw:transparency", "37%" ); + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else if ( brush.style() == Qt::Dense6Pattern ) + { + styleFill.addProperty( "draw:transparency", "12%" ); + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else if ( brush.style() == Qt::Dense7Pattern ) + { + styleFill.addProperty( "draw:transparency", "6%" ); + styleFill.addProperty( "draw:fill","solid" ); + styleFill.addProperty( "draw:fill-color", brush.color().name() ); + } + else //otherstyle + { + styleFill.addProperty( "draw:fill","hatch" ); + styleFill.addProperty( "draw:fill-hatch-name", saveOasisHatchStyle( mainStyles,brush ) ); + } + +} + +QString KoOasisStyles::saveOasisHatchStyle( KoGenStyles& mainStyles, const QBrush &brush ) +{ + KoGenStyle hatchStyle( KoGenStyle::STYLE_HATCH /*no family name*/); + hatchStyle.addAttribute( "draw:color", brush.color().name() ); + //hatchStyle.addAttribute( "draw:distance", m_distance ); not implemented into kpresenter + switch( brush.style() ) + { + case Qt::HorPattern: + hatchStyle.addAttribute( "draw:style", "single" ); + hatchStyle.addAttribute( "draw:rotation", 0); + break; + case Qt::BDiagPattern: + hatchStyle.addAttribute( "draw:style", "single" ); + hatchStyle.addAttribute( "draw:rotation", 450); + break; + case Qt::VerPattern: + hatchStyle.addAttribute( "draw:style", "single" ); + hatchStyle.addAttribute( "draw:rotation", 900); + break; + case Qt::FDiagPattern: + hatchStyle.addAttribute( "draw:style", "single" ); + hatchStyle.addAttribute( "draw:rotation", 1350); + break; + case Qt::CrossPattern: + hatchStyle.addAttribute( "draw:style", "double" ); + hatchStyle.addAttribute( "draw:rotation", 0); + break; + case Qt::DiagCrossPattern: + hatchStyle.addAttribute( "draw:style", "double" ); + hatchStyle.addAttribute( "draw:rotation", 450); + break; + default: + break; + } + + return mainStyles.lookup( hatchStyle, "hatch" ); +} + +QBrush KoOasisStyles::loadOasisFillStyle( const KoStyleStack &styleStack, const QString & fill, const KoOasisStyles & oasisStyles ) +{ + QBrush tmpBrush; + if ( fill == "solid" ) + { + tmpBrush.setStyle(static_cast<Qt::BrushStyle>( 1 ) ); + if ( styleStack.hasAttributeNS( KoXmlNS::draw, "fill-color" ) ) + tmpBrush.setColor(styleStack.attributeNS( KoXmlNS::draw, "fill-color" ) ); + if ( styleStack.hasAttributeNS( KoXmlNS::draw, "transparency" ) ) + { + QString transparency = styleStack.attributeNS( KoXmlNS::draw, "transparency" ); + if ( transparency == "94%" ) + { + tmpBrush.setStyle(Qt::Dense1Pattern); + } + else if ( transparency == "88%" ) + { + tmpBrush.setStyle(Qt::Dense2Pattern); + } + else if ( transparency == "63%" ) + { + tmpBrush.setStyle(Qt::Dense3Pattern); + + } + else if ( transparency == "50%" ) + { + tmpBrush.setStyle(Qt::Dense4Pattern); + + } + else if ( transparency == "37%" ) + { + tmpBrush.setStyle(Qt::Dense5Pattern); + + } + else if ( transparency == "12%" ) + { + tmpBrush.setStyle(Qt::Dense6Pattern); + + } + else if ( transparency == "6%" ) + { + tmpBrush.setStyle(Qt::Dense7Pattern); + + } + else + kdDebug()<<" transparency is not defined into kpresenter :"<<transparency<<endl; + } + } + else if ( fill == "hatch" ) + { + QString style = styleStack.attributeNS( KoXmlNS::draw, "fill-hatch-name" ); + kdDebug()<<" hatch style is : "<<style<<endl; + + //type not defined by default + //try to use style. + QDomElement* draw = oasisStyles.drawStyles()[style]; + if ( draw) + { + kdDebug()<<"We have a style\n"; + int angle = 0; + if( draw->hasAttributeNS( KoXmlNS::draw, "rotation" )) + { + angle = (draw->attributeNS( KoXmlNS::draw, "rotation", QString::null ).toInt())/10; + kdDebug()<<"angle :"<<angle<<endl; + } + if(draw->hasAttributeNS( KoXmlNS::draw, "color" ) ) + { + //kdDebug()<<" draw:color :"<<draw->attributeNS( KoXmlNS::draw, "color", QString::null )<<endl; + tmpBrush.setColor(draw->attributeNS( KoXmlNS::draw, "color", QString::null ) ); + } + if( draw->hasAttributeNS( KoXmlNS::draw, "distance" )) + { + //todo implemente it into kpresenter + } + if( draw->hasAttributeNS( KoXmlNS::draw, "display-name")) + { + //todo implement it into kpresenter + } + if( draw->hasAttributeNS( KoXmlNS::draw, "style" )) + { + //todo implemente it into kpresenter + QString styleHash = draw->attributeNS( KoXmlNS::draw, "style", QString::null ); + if( styleHash == "single") + { + switch( angle ) + { + case 0: + case 180: + tmpBrush.setStyle(Qt::HorPattern ); + break; + case 45: + case 225: + tmpBrush.setStyle(Qt::BDiagPattern ); + break; + case 90: + case 270: + tmpBrush.setStyle(Qt::VerPattern ); + break; + case 135: + case 315: + tmpBrush.setStyle(Qt::FDiagPattern ); + break; + default: + //todo fixme when we will have a kopaint + kdDebug()<<" draw:rotation 'angle' : "<<angle<<endl; + break; + } + } + else if( styleHash == "double") + { + switch( angle ) + { + case 0: + case 180: + case 90: + case 270: + tmpBrush.setStyle(Qt::CrossPattern ); + break; + case 45: + case 135: + case 225: + case 315: + tmpBrush.setStyle(Qt::DiagCrossPattern ); + break; + default: + //todo fixme when we will have a kopaint + kdDebug()<<" draw:rotation 'angle' : "<<angle<<endl; + break; + } + + } + else if( styleHash == "triple") + { + kdDebug()<<" it is not implemented :( \n"; + } + } + } + } + return tmpBrush; +} + +const QDomElement* KoOasisStyles::defaultStyle( const QString& family ) const +{ + return m_defaultStyle[family]; +} + +const QDomElement* KoOasisStyles::findStyle( const QString& name ) const +{ + return m_styles[ name ]; +} + +const QDomElement* KoOasisStyles::findStyle( const QString& styleName, const QString& family ) const +{ + const QDomElement* style = d->m_styles[ family ][ styleName ]; + if ( style && !family.isEmpty() ) { + const QString styleFamily = style->attributeNS( KoXmlNS::style, "family", QString::null ); + if ( styleFamily != family ) { + kdWarning() << "KoOasisStyles: was looking for style " << styleName + << " in family " << family << " but got " << styleFamily << endl; + } + } + return style; +} + +const QDomElement* KoOasisStyles::findStyleAutoStyle( const QString& styleName, const QString& family ) const +{ + const QDomElement* style = d->m_stylesAutoStyles[ family ][ styleName ]; + if ( style ) { + const QString styleFamily = style->attributeNS( KoXmlNS::style, "family", QString::null ); + if ( styleFamily != family ) { + kdWarning() << "KoOasisStyles: was looking for style " << styleName + << " in family " << family << " but got " << styleFamily << endl; + } + } + return style; +} diff --git a/lib/kofficecore/KoOasisStyles.h b/lib/kofficecore/KoOasisStyles.h new file mode 100644 index 00000000..caa8d513 --- /dev/null +++ b/lib/kofficecore/KoOasisStyles.h @@ -0,0 +1,165 @@ +/* This file is part of the KDE project + Copyright (C) 2004-2006 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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. +*/ + +#ifndef KOOASISSTYLES_H +#define KOOASISSTYLES_H + +#include <qdom.h> +#include <qdict.h> +#include <qvaluevector.h> +#include <qmap.h> +#include <koffice_export.h> + +class KoGenStyles; +class KoXmlWriter; +class QBrush; +class KoGenStyle; +class KoStyleStack; + +/** + * Repository of styles used during loading of OASIS/OOo file + */ +class KOFFICECORE_EXPORT KoOasisStyles +{ +public: + KoOasisStyles(); + ~KoOasisStyles(); + + /// Look into @p doc for styles and remember them + /// @param doc document to look into + /// @param stylesDotXml true when loading styles.xml, false otherwise + void createStyleMap( const QDomDocument& doc, bool stylesDotXml ); + + /** + * Look up a style by name. + * This method can find styles defined by the tags "style:page-layout", + * "style:presentation-page-layout", or "style:font-decl". + * Do NOT use this method for style:style styles. + * + * @param name the style name + * @return the dom element representing the style, or QString::null if it wasn't found. + */ + const QDomElement* findStyle( const QString& name ) const; + + /** + * Look up a style:style by name. + * @param name the style name + * @param family the style family (for a style:style, use 0 otherwise) + * @return the dom element representing the style, or QString::null if it wasn't found. + */ + const QDomElement* findStyle( const QString& name, const QString& family ) const; + + /// Similar to findStyle but for auto-styles in styles.xml only. + const QDomElement* findStyleAutoStyle( const QString& name, const QString& family ) const; + + /// @return the style:styles that are "user styles", i.e. those from office:styles + /// findStyle() is used for lookup. userStyles() is used to load all user styles upfront. + QValueVector<QDomElement> userStyles() const; + + /// @return the default style for a given family ("graphic","paragraph","table" etc.) + /// Returns 0 if no default style for this family is available + const QDomElement* defaultStyle( const QString& family ) const; + + /// @return the office:style element + const QDomElement& officeStyle() const { return m_officeStyle; } + + /// @return all list styles ("text:list-style" elements), hashed by name + const QDict<QDomElement>& listStyles() const { return m_listStyles; } + + /// @return master pages ("style:master-page" elements), hashed by name + const QDict<QDomElement>& masterPages() const { return m_masterPages; } + + /// @return draw styles, hashed by name + const QDict<QDomElement>& drawStyles() const { return m_drawStyles; } + + /// @return all styles ("style:style" elements) for a given family, hashed by name + const QDict<QDomElement>& styles(const QString& family) const; + + /// Prefix and suffix are always included into formatStr. Having them as separate fields simply + /// allows to extract them from formatStr, to display them in separate widgets. + struct NumericStyleFormat + { + QString formatStr; + QString prefix; + QString suffix; + enum { Number, Scientific, Fraction, Currency, Percentage, + Date, Time, Boolean, Text } type; + int precision; + QString currencySymbol; + }; + + typedef QMap<QString, NumericStyleFormat> DataFormatsMap; + /// Value (date/time/number...) formats found while parsing styles. Used e.g. for fields. + /// Key: format name. Value: + const DataFormatsMap& dataFormats() const { return m_dataFormats; } + + static QString saveOasisDateStyle( KoGenStyles &mainStyles, const QString & _format, bool klocaleFormat, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + static QString saveOasisTimeStyle( KoGenStyles &mainStyles, const QString & _format, bool klocaleFormat, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + static QString saveOasisFractionStyle( KoGenStyles &mainStyles, const QString & _format, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + static QString saveOasisScientificStyle( KoGenStyles &mainStyles, const QString & _format, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + static QString saveOasisNumberStyle( KoGenStyles &mainStyles, const QString & _format, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + static QString saveOasisPercentageStyle( KoGenStyles &mainStyles, const QString & _format, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + static QString saveOasisCurrencyStyle( KoGenStyles &mainStyles, const QString & _format, const QString &symbol, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + static QString saveOasisTextStyle( KoGenStyles &mainStyles, const QString & _format, + const QString &_prefix = QString::null , const QString &_suffix= QString::null ); + + static void saveOasisFillStyle( KoGenStyle &styleFill, KoGenStyles& mainStyles, const QBrush & brush ); + static QString saveOasisHatchStyle( KoGenStyles& mainStyles, const QBrush &brush ); + + static QBrush loadOasisFillStyle( const KoStyleStack &styleStack, const QString & fill, const KoOasisStyles & oasisStyles ); + +private: + /// Add styles to styles map + void insertStyles( const QDomElement& styles, bool styleAutoStyles = false ); + +private: + void insertOfficeStyles( const QDomElement& styles ); + void insertStyle( const QDomElement& style, bool styleAutoStyles ); + void importDataStyle( const QDomElement& parent ); + static bool saveOasisTimeFormat( KoXmlWriter &elementWriter, QString & format, QString & text, bool &antislash ); + static void parseOasisDateKlocale(KoXmlWriter &elementWriter, QString & format, QString & text ); + static bool saveOasisKlocaleTimeFormat( KoXmlWriter &elementWriter, QString & format, QString & text ); + static void parseOasisTimeKlocale(KoXmlWriter &elementWriter, QString & format, QString & text ); + static void addKofficeNumericStyleExtension( KoXmlWriter & elementWriter, const QString &_suffix, const QString &_prefix ); + + KoOasisStyles( const KoOasisStyles & ); // forbidden + KoOasisStyles& operator=( const KoOasisStyles & ); // forbidden + +private: + QDict<QDomElement> m_styles; // page-layout, font-decl etc. + QDict<QDomElement> m_defaultStyle; + QDomElement m_officeStyle; + + QDict<QDomElement> m_masterPages; + QDict<QDomElement> m_listStyles; + + QDict<QDomElement> m_drawStyles; + DataFormatsMap m_dataFormats; + + class Private; + Private * const d; +}; + +#endif /* KOOASISSTYLES_H */ diff --git a/lib/kofficecore/KoOpenPane.cpp b/lib/kofficecore/KoOpenPane.cpp new file mode 100644 index 00000000..07c9580d --- /dev/null +++ b/lib/kofficecore/KoOpenPane.cpp @@ -0,0 +1,315 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Peter Simonsson <psn@linux.se> + + 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 "KoOpenPane.h" + +#include <qvbox.h> +#include <qlayout.h> +#include <qheader.h> +#include <qwidgetstack.h> +#include <qlabel.h> +#include <qvaluelist.h> +#include <qimage.h> +#include <qpainter.h> +#include <qpen.h> + +#include <klocale.h> +#include <kfiledialog.h> +#include <kinstance.h> +#include <kpushbutton.h> +#include <kiconloader.h> +#include <kdebug.h> +#include <klistview.h> + +#include "KoFilterManager.h" +#include "KoTemplates.h" +#include "KoDocument.h" +#include "KoDetailsPane.h" +#include "koDetailsPaneBase.h" + +#include <limits.h> + +class KoSectionListItem : public QListViewItem +{ + public: + KoSectionListItem(KListView* listView, const QString& name, int sortWeight, int widgetIndex = -1) + : QListViewItem(listView, name), m_sortWeight(sortWeight), m_widgetIndex(widgetIndex) + { + } + + virtual int compare(QListViewItem* i, int, bool) const + { + KoSectionListItem* item = dynamic_cast<KoSectionListItem*>(i); + + if(!item) + return 0; + + return sortWeight() - item->sortWeight(); + } + + virtual void paintCell(QPainter* p, const QColorGroup& cg, int column, int width, int align) + { + if(widgetIndex() >= 0) { + QListViewItem::paintCell(p, cg, column, width, align); + } else { + int ypos = (height() - 2) / 2; + QPen pen(cg.foreground(), 2); + p->setPen(pen); + p->drawLine(0, ypos, width, ypos); + } + } + + int sortWeight() const { return m_sortWeight; } + int widgetIndex() const { return m_widgetIndex; } + + private: + int m_sortWeight; + int m_widgetIndex; +}; + +class KoOpenPanePrivate +{ + public: + KoOpenPanePrivate() : + m_instance(0) + { + } + + KInstance* m_instance; +}; + +KoOpenPane::KoOpenPane(QWidget *parent, KInstance* instance, const QString& templateType) + : KoOpenPaneBase(parent, "OpenPane") +{ + d = new KoOpenPanePrivate; + d->m_instance = instance; + + m_sectionList->header()->hide(); + m_sectionList->setSorting(0); +#if KDE_IS_VERSION(3,4,0) + m_sectionList->setShadeSortColumn(false); +#endif + connect(m_sectionList, SIGNAL(selectionChanged(QListViewItem*)), + this, SLOT(selectionChanged(QListViewItem*))); + connect(m_sectionList, SIGNAL(pressed(QListViewItem*)), + this, SLOT(itemClicked(QListViewItem*))); + connect(m_sectionList, SIGNAL(spacePressed(QListViewItem*)), + this, SLOT(itemClicked(QListViewItem*))); + connect(m_sectionList, SIGNAL(returnPressed(QListViewItem*)), + this, SLOT(itemClicked(QListViewItem*))); + + KGuiItem openExistingGItem(i18n("Open Existing Document..."), "fileopen"); + m_openExistingButton->setGuiItem(openExistingGItem); + connect(m_openExistingButton, SIGNAL(clicked()), this, SLOT(showOpenFileDialog())); + + initRecentDocs(); + initTemplates(templateType); + + KoSectionListItem* selectedItem = static_cast<KoSectionListItem*>(m_sectionList->selectedItem()); + + if(selectedItem) { + m_widgetStack->widget(selectedItem->widgetIndex())->setFocus(); + } + + QValueList<int> sizes; + sizes << 20 << width() - 20; + m_splitter->setSizes(sizes); + + // Set the sizes of the details pane splitters + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + sizes = cfgGrp.readIntListEntry("DetailsPaneSplitterSizes"); + emit splitterResized(0, sizes); + + connect(this, SIGNAL(splitterResized(KoDetailsPaneBase*, const QValueList<int>&)), + this, SLOT(saveSplitterSizes(KoDetailsPaneBase*, const QValueList<int>&))); +} + +KoOpenPane::~KoOpenPane() +{ + KoSectionListItem* item = dynamic_cast<KoSectionListItem*>(m_sectionList->selectedItem()); + + if(item) { + if(!dynamic_cast<KoDetailsPaneBase*>(m_widgetStack->widget(item->widgetIndex()))) { + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + cfgGrp.writeEntry("LastReturnType", "Custom"); + } + } + + delete d; +} + +void KoOpenPane::showOpenFileDialog() +{ + const QStringList mimeFilter = KoFilterManager::mimeFilter(KoDocument::readNativeFormatMimeType(), + KoFilterManager::Import, KoDocument::readExtraNativeMimeTypes()); + + KURL url = KFileDialog::getOpenURL(":OpenDialog", mimeFilter.join(" "), this); + + if(!url.isEmpty()) { + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + cfgGrp.writeEntry("LastReturnType", "File"); + emit openExistingFile(url.url()); + } +} + +void KoOpenPane::initRecentDocs() +{ + KoRecentDocumentsPane* recentDocPane = new KoRecentDocumentsPane(this, d->m_instance); + connect(recentDocPane, SIGNAL(openFile(const QString&)), this, SIGNAL(openExistingFile(const QString&))); + QListViewItem* item = addPane(i18n("Recent Documents"), "fileopen", recentDocPane, 0); + connect(recentDocPane, SIGNAL(splitterResized(KoDetailsPaneBase*, const QValueList<int>&)), + this, SIGNAL(splitterResized(KoDetailsPaneBase*, const QValueList<int>&))); + connect(this, SIGNAL(splitterResized(KoDetailsPaneBase*, const QValueList<int>&)), + recentDocPane, SLOT(resizeSplitter(KoDetailsPaneBase*, const QValueList<int>&))); + + KoSectionListItem* separator = new KoSectionListItem(m_sectionList, "", 1); + separator->setEnabled(false); + + if(d->m_instance->config()->hasGroup("RecentFiles")) { + m_sectionList->setSelected(item, true); + } +} + +void KoOpenPane::initTemplates(const QString& templateType) +{ + QListViewItem* selectItem = 0; + QListViewItem* firstItem = 0; + const int templateOffset = 1000; + + if(!templateType.isEmpty()) + { + KoTemplateTree templateTree(templateType.local8Bit(), d->m_instance, true); + + for (KoTemplateGroup *group = templateTree.first(); group != 0L; group = templateTree.next()) { + if (group->isHidden()) { + continue; + } + + KoTemplatesPane* pane = new KoTemplatesPane(this, d->m_instance, + group, templateTree.defaultTemplate()); + connect(pane, SIGNAL(openTemplate(const QString&)), this, SIGNAL(openTemplate(const QString&))); + connect(pane, SIGNAL(alwaysUseChanged(KoTemplatesPane*, const QString&)), + this, SIGNAL(alwaysUseChanged(KoTemplatesPane*, const QString&))); + connect(this, SIGNAL(alwaysUseChanged(KoTemplatesPane*, const QString&)), + pane, SLOT(changeAlwaysUseTemplate(KoTemplatesPane*, const QString&))); + connect(pane, SIGNAL(splitterResized(KoDetailsPaneBase*, const QValueList<int>&)), + this, SIGNAL(splitterResized(KoDetailsPaneBase*, const QValueList<int>&))); + connect(this, SIGNAL(splitterResized(KoDetailsPaneBase*, const QValueList<int>&)), + pane, SLOT(resizeSplitter(KoDetailsPaneBase*, const QValueList<int>&))); + QListViewItem* item = addPane(group->name(), group->first()->loadPicture(d->m_instance), + pane, group->sortingWeight() + templateOffset); + + if(!firstItem) { + firstItem = item; + } + + if(group == templateTree.defaultGroup()) { + firstItem = item; + } + + if(pane->isSelected()) { + selectItem = item; + } + } + } else { + firstItem = m_sectionList->firstChild(); + } + + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + + if(selectItem && (cfgGrp.readEntry("LastReturnType") == "Template")) { + m_sectionList->setSelected(selectItem, true); + } else if(!m_sectionList->selectedItem() && firstItem) { + m_sectionList->setSelected(firstItem, true); + } +} + +void KoOpenPane::setCustomDocumentWidget(QWidget *widget) { + Q_ASSERT(widget); + KoSectionListItem* separator = new KoSectionListItem(m_sectionList, "", INT_MAX-1); + separator->setEnabled(false); + + QListViewItem* item = addPane(i18n("Custom Document"), QString::null, widget, INT_MAX); + + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + + if(cfgGrp.readEntry("LastReturnType") == "Custom") { + m_sectionList->setSelected(item, true); + KoSectionListItem* selectedItem = static_cast<KoSectionListItem*>(item); + m_widgetStack->widget(selectedItem->widgetIndex())->setFocus(); + } +} + +QListViewItem* KoOpenPane::addPane(const QString& title, const QString& icon, QWidget* widget, int sortWeight) +{ + return addPane(title, SmallIcon(icon, KIcon::SizeLarge, KIcon::DefaultState, d->m_instance), + widget, sortWeight); +} + +QListViewItem* KoOpenPane::addPane(const QString& title, const QPixmap& icon, QWidget* widget, int sortWeight) +{ + if(!widget) { + return 0; + } + + int id = m_widgetStack->addWidget(widget); + KoSectionListItem* listItem = new KoSectionListItem(m_sectionList, title, sortWeight, id); + + if(!icon.isNull()) { + QImage image = icon.convertToImage(); + + if((image.width() > 48) || (image.height() > 48)) { + image = image.smoothScale(48, 48, QImage::ScaleMin); + } + + image.setAlphaBuffer(true); + image = image.copy((image.width() - 48) / 2, (image.height() - 48) / 2, 48, 48); + listItem->setPixmap(0, QPixmap(image)); + } + + return listItem; +} + +void KoOpenPane::selectionChanged(QListViewItem* item) +{ + KoSectionListItem* section = dynamic_cast<KoSectionListItem*>(item); + + if(!item) + return; + + m_headerLabel->setText(section->text(0)); + m_widgetStack->raiseWidget(section->widgetIndex()); +} + +void KoOpenPane::saveSplitterSizes(KoDetailsPaneBase* /*sender*/, const QValueList<int>& sizes) +{ + KConfigGroup cfgGrp(d->m_instance->config(), "TemplateChooserDialog"); + cfgGrp.writeEntry("DetailsPaneSplitterSizes", sizes); +} + +void KoOpenPane::itemClicked(QListViewItem* item) +{ + KoSectionListItem* selectedItem = static_cast<KoSectionListItem*>(item); + + if(selectedItem) { + m_widgetStack->widget(selectedItem->widgetIndex())->setFocus(); + } +} + +#include "KoOpenPane.moc" diff --git a/lib/kofficecore/KoOpenPane.h b/lib/kofficecore/KoOpenPane.h new file mode 100644 index 00000000..20bd425d --- /dev/null +++ b/lib/kofficecore/KoOpenPane.h @@ -0,0 +1,90 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Peter Simonsson <psn@linux.se> + + 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. +*/ +#ifndef KOOPENPANE_H +#define KOOPENPANE_H + +#include <koOpenPaneBase.h> + +class KoCustomDocumentCreator; +class KConfig; +class KoTemplateGroup; +class KoOpenPanePrivate; +class KInstance; +class QPixmap; +class KListViewItem; +class KoTemplatesPane; +class KoDetailsPaneBase; + +class KoOpenPane : public KoOpenPaneBase +{ + Q_OBJECT + + public: + /** + * Constructor + * @param parent the parent widget + * @param instance the KInstance to be used for KConfig data + * @param templateType the template-type (group) that should be selected on creation. + */ + KoOpenPane(QWidget *parent, KInstance* instance, const QString& templateType = QString::null); + virtual ~KoOpenPane(); + + QListViewItem* addPane(const QString& title, const QString& icon, QWidget* widget, int sortWeight); + QListViewItem* addPane(const QString& title, const QPixmap& icon, QWidget* widget, int sortWeight); + + /** + * If the application has a way to create a document not based on a template, but on user + * provided settings, the widget showing these gets set here. + * @see KoDocument::createCustomDocumentWidget() + * @param widget the widget. + */ + void setCustomDocumentWidget(QWidget *widget); + + protected slots: + void showOpenFileDialog(); + + void selectionChanged(QListViewItem* item); + void itemClicked(QListViewItem* item); + + /// Saves the splitter sizes for KoDetailsPaneBase based panes + void saveSplitterSizes(KoDetailsPaneBase* sender, const QValueList<int>& sizes); + + signals: + void openExistingFile(const QString&); + void openTemplate(const QString&); + + /// Emitted when the always use template has changed + void alwaysUseChanged(KoTemplatesPane* sender, const QString& alwaysUse); + + /// Emitted when one of the detail panes have changed it's splitter + void splitterResized(KoDetailsPaneBase* sender, const QValueList<int>& sizes); + + protected: + void initRecentDocs(); + /** + * Populate the list with all templates the user can choose. + * @param templateType the template-type (group) that should be selected on creation. + */ + void initTemplates(const QString& templateType); + + private: + KoOpenPanePrivate* d; +}; + +#endif //KOOPENPANE_H diff --git a/lib/kofficecore/KoPageLayout.cpp b/lib/kofficecore/KoPageLayout.cpp new file mode 100644 index 00000000..36e447e9 --- /dev/null +++ b/lib/kofficecore/KoPageLayout.cpp @@ -0,0 +1,242 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright 2002, 2003 David Faure <faure@kde.org> + Copyright 2003 Nicolas GOUTTE <goutte@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 "KoPageLayout.h" +#include <KoUnit.h> +#include <klocale.h> +#include <kprinter.h> +#include <kdebug.h> +#include <kglobal.h> +#include <KoDom.h> +#include <KoXmlNS.h> +#include <qdom.h> + +// paper formats ( mm ) +#define PG_A3_WIDTH 297.0 +#define PG_A3_HEIGHT 420.0 +#define PG_A4_WIDTH 210.0 +#define PG_A4_HEIGHT 297.0 +#define PG_A5_WIDTH 148.0 +#define PG_A5_HEIGHT 210.0 +#define PG_B5_WIDTH 182.0 +#define PG_B5_HEIGHT 257.0 +#define PG_US_LETTER_WIDTH 216.0 +#define PG_US_LETTER_HEIGHT 279.0 +#define PG_US_LEGAL_WIDTH 216.0 +#define PG_US_LEGAL_HEIGHT 356.0 +#define PG_US_EXECUTIVE_WIDTH 191.0 +#define PG_US_EXECUTIVE_HEIGHT 254.0 + +KoGenStyle KoPageLayout::saveOasis() const +{ + KoGenStyle style(KoGenStyle::STYLE_PAGELAYOUT); + style.addPropertyPt("fo:page-width", ptWidth); + style.addPropertyPt("fo:page-height", ptHeight); + style.addPropertyPt("fo:margin-left", ptLeft); + style.addPropertyPt("fo:margin-right", ptRight); + style.addPropertyPt("fo:margin-top", ptTop); + style.addPropertyPt("fo:margin-bottom", ptBottom); + style.addProperty("style:print-orientation", (orientation == PG_LANDSCAPE ? "landscape" : "portrait")); + return style; +} + +void KoPageLayout::loadOasis(const QDomElement &style) +{ + QDomElement properties( KoDom::namedItemNS( style, KoXmlNS::style, "page-layout-properties" ) ); + if ( !properties.isNull() ) + { + ptWidth = KoUnit::parseValue(properties.attributeNS( KoXmlNS::fo, "page-width", QString::null ) ); + ptHeight = KoUnit::parseValue(properties.attributeNS( KoXmlNS::fo, "page-height", QString::null ) ); + if (properties.attributeNS( KoXmlNS::style, "print-orientation", QString::null)=="portrait") + orientation=PG_PORTRAIT; + else + orientation=PG_LANDSCAPE; + ptRight = KoUnit::parseValue( properties.attributeNS( KoXmlNS::fo, "margin-right", QString::null ) ); + ptBottom = KoUnit::parseValue( properties.attributeNS( KoXmlNS::fo, "margin-bottom", QString::null ) ); + ptLeft = KoUnit::parseValue( properties.attributeNS( KoXmlNS::fo, "margin-left", QString::null ) ); + ptTop = KoUnit::parseValue( properties.attributeNS( KoXmlNS::fo, "margin-top", QString::null ) ); + // guessFormat takes millimeters + if ( orientation == PG_LANDSCAPE ) + format = KoPageFormat::guessFormat( POINT_TO_MM(ptHeight), POINT_TO_MM(ptWidth) ); + else + format = KoPageFormat::guessFormat( POINT_TO_MM(ptWidth), POINT_TO_MM(ptHeight) ); + } +} + +KoPageLayout KoPageLayout::standardLayout() +{ + KoPageLayout layout; + layout.format = KoPageFormat::defaultFormat(); + layout.orientation = PG_PORTRAIT; + layout.ptWidth = MM_TO_POINT( KoPageFormat::width( layout.format, layout.orientation ) ); + layout.ptHeight = MM_TO_POINT( KoPageFormat::height( layout.format, layout.orientation ) ); + layout.ptLeft = MM_TO_POINT( 20.0 ); + layout.ptRight = MM_TO_POINT( 20.0 ); + layout.ptTop = MM_TO_POINT( 20.0 ); + layout.ptBottom = MM_TO_POINT( 20.0 ); + layout.ptPageEdge = -1; + layout.ptBindingSide = -1; + return layout; +} + +struct PageFormatInfo +{ + KoFormat format; + KPrinter::PageSize kprinter; + const char* shortName; // Short name + const char* descriptiveName; // Full name, which will be translated + double width; // in mm + double height; // in mm +}; + +// NOTES: +// - the width and height of non-ISO formats are rounded +// http://en.wikipedia.org/wiki/Paper_size can help +// - the comments "should be..." indicates the exact values if the inch sizes would be multiplied by 25.4 mm/inch + +const PageFormatInfo pageFormatInfo[]= +{ + { PG_DIN_A3, KPrinter::A3, "A3", I18N_NOOP("ISO A3"), 297.0, 420.0 }, + { PG_DIN_A4, KPrinter::A4, "A4", I18N_NOOP("ISO A4"), 210.0, 297.0 }, + { PG_DIN_A5, KPrinter::A5, "A5", I18N_NOOP("ISO A5"), 148.0, 210.0 }, + { PG_US_LETTER, KPrinter::Letter, "Letter", I18N_NOOP("US Letter"), 215.9, 279.4 }, + { PG_US_LEGAL, KPrinter::Legal, "Legal", I18N_NOOP("US Legal"), 215.9, 355.6 }, + { PG_SCREEN, KPrinter::A4, "Screen", I18N_NOOP("Screen"), PG_A4_HEIGHT, PG_A4_WIDTH }, // Custom, so fall back to A4 + { PG_CUSTOM, KPrinter::A4, "Custom", I18N_NOOP2("Custom size", "Custom"), PG_A4_WIDTH, PG_A4_HEIGHT }, // Custom, so fall back to A4 + { PG_DIN_B5, KPrinter::B5, "B5", I18N_NOOP("ISO B5"), 182.0, 257.0 }, + // Hmm, wikipedia says 184.15 * 266.7 for executive ! + { PG_US_EXECUTIVE, KPrinter::Executive, "Executive", I18N_NOOP("US Executive"), 191.0, 254.0 }, // should be 190.5 mm x 254.0 mm + { PG_DIN_A0, KPrinter::A0, "A0", I18N_NOOP("ISO A0"), 841.0, 1189.0 }, + { PG_DIN_A1, KPrinter::A1, "A1", I18N_NOOP("ISO A1"), 594.0, 841.0 }, + { PG_DIN_A2, KPrinter::A2, "A2", I18N_NOOP("ISO A2"), 420.0, 594.0 }, + { PG_DIN_A6, KPrinter::A6, "A6", I18N_NOOP("ISO A6"), 105.0, 148.0 }, + { PG_DIN_A7, KPrinter::A7, "A7", I18N_NOOP("ISO A7"), 74.0, 105.0 }, + { PG_DIN_A8, KPrinter::A8, "A8", I18N_NOOP("ISO A8"), 52.0, 74.0 }, + { PG_DIN_A9, KPrinter::A9, "A9", I18N_NOOP("ISO A9"), 37.0, 52.0 }, + { PG_DIN_B0, KPrinter::B0, "B0", I18N_NOOP("ISO B0"), 1030.0, 1456.0 }, + { PG_DIN_B1, KPrinter::B1, "B1", I18N_NOOP("ISO B1"), 728.0, 1030.0 }, + { PG_DIN_B10, KPrinter::B10, "B10", I18N_NOOP("ISO B10"), 32.0, 45.0 }, + { PG_DIN_B2, KPrinter::B2, "B2", I18N_NOOP("ISO B2"), 515.0, 728.0 }, + { PG_DIN_B3, KPrinter::B3, "B3", I18N_NOOP("ISO B3"), 364.0, 515.0 }, + { PG_DIN_B4, KPrinter::B4, "B4", I18N_NOOP("ISO B4"), 257.0, 364.0 }, + { PG_DIN_B6, KPrinter::B6, "B6", I18N_NOOP("ISO B6"), 128.0, 182.0 }, + { PG_ISO_C5, KPrinter::C5E, "C5", I18N_NOOP("ISO C5"), 163.0, 229.0 }, // Some sources tells: 162 mm x 228 mm + { PG_US_COMM10, KPrinter::Comm10E, "Comm10", I18N_NOOP("US Common 10"), 105.0, 241.0 }, // should be 104.775 mm x 241.3 mm + { PG_ISO_DL, KPrinter::DLE, "DL", I18N_NOOP("ISO DL"), 110.0, 220.0 }, + { PG_US_FOLIO, KPrinter::Folio, "Folio", I18N_NOOP("US Folio"), 210.0, 330.0 }, // should be 209.54 mm x 330.2 mm + { PG_US_LEDGER, KPrinter::Ledger, "Ledger", I18N_NOOP("US Ledger"), 432.0, 279.0 }, // should be 431.8 mm x 297.4 mm + { PG_US_TABLOID, KPrinter::Tabloid, "Tabloid", I18N_NOOP("US Tabloid"), 279.0, 432.0 } // should be 297.4 mm x 431.8 mm +}; + +int KoPageFormat::printerPageSize( KoFormat format ) +{ + if ( format == PG_SCREEN ) + { + kdWarning() << "You use the page layout SCREEN. Printing in DIN A4 LANDSCAPE." << endl; + return KPrinter::A4; + } + else if ( format == PG_CUSTOM ) + { + kdWarning() << "The used page layout (CUSTOM) is not supported by KPrinter. Printing in A4." << endl; + return KPrinter::A4; + } + else if ( format <= PG_LAST_FORMAT ) + return pageFormatInfo[ format ].kprinter; + else + return KPrinter::A4; +} + +double KoPageFormat::width( KoFormat format, KoOrientation orientation ) +{ + if ( orientation == PG_LANDSCAPE ) + return height( format, PG_PORTRAIT ); + if ( format <= PG_LAST_FORMAT ) + return pageFormatInfo[ format ].width; + return PG_A4_WIDTH; // should never happen +} + +double KoPageFormat::height( KoFormat format, KoOrientation orientation ) +{ + if ( orientation == PG_LANDSCAPE ) + return width( format, PG_PORTRAIT ); + if ( format <= PG_LAST_FORMAT ) + return pageFormatInfo[ format ].height; + return PG_A4_HEIGHT; +} + +KoFormat KoPageFormat::guessFormat( double width, double height ) +{ + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + // We have some tolerance. 1pt is a third of a mm, this is + // barely noticeable for a page size. + if ( i != PG_CUSTOM + && kAbs( width - pageFormatInfo[i].width ) < 1.0 + && kAbs( height - pageFormatInfo[i].height ) < 1.0 ) + return static_cast<KoFormat>(i); + } + return PG_CUSTOM; +} + +QString KoPageFormat::formatString( KoFormat format ) +{ + if ( format <= PG_LAST_FORMAT ) + return QString::fromLatin1( pageFormatInfo[ format ].shortName ); + return QString::fromLatin1( "A4" ); +} + +KoFormat KoPageFormat::formatFromString( const QString & string ) +{ + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + if (string == QString::fromLatin1( pageFormatInfo[ i ].shortName )) + return pageFormatInfo[ i ].format; + } + // We do not know the format name, so we have a custom format + return PG_CUSTOM; +} + +KoFormat KoPageFormat::defaultFormat() +{ + int kprinter = KGlobal::locale()->pageSize(); + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + if ( pageFormatInfo[ i ].kprinter == kprinter ) + return static_cast<KoFormat>(i); + } + return PG_DIN_A4; +} + +QString KoPageFormat::name( KoFormat format ) +{ + if ( format <= PG_LAST_FORMAT ) + return i18n( pageFormatInfo[ format ].descriptiveName ); + return i18n( pageFormatInfo[ PG_DIN_A4 ].descriptiveName ); +} + +QStringList KoPageFormat::allFormats() +{ + QStringList lst; + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + lst << i18n( pageFormatInfo[ i ].descriptiveName ); + } + return lst; +} diff --git a/lib/kofficecore/KoPageLayout.h b/lib/kofficecore/KoPageLayout.h new file mode 100644 index 00000000..f9f98fc6 --- /dev/null +++ b/lib/kofficecore/KoPageLayout.h @@ -0,0 +1,260 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@kde.org> + Copyright 2002, 2003 David Faure <faure@kde.org> + Copyright 2003 Nicolas GOUTTE <goutte@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. +*/ + +#ifndef KOPAGELAYOUT_H +#define KOPAGELAYOUT_H + +#include <KoGenStyles.h> +#include <qstringlist.h> +#include <koffice_export.h> +class QDomElement; + +/** + * @brief Represents the paper format a document shall be printed on. + * + * For compatibility reasons, and because of screen and custom, + * this enum doesn't map to QPrinter::PageSize but KoPageFormat::printerPageSize + * does the conversion. + * + * @todo convert DIN to ISO in the names + */ +enum KoFormat { + PG_DIN_A3 = 0, + PG_DIN_A4 = 1, + PG_DIN_A5 = 2, + PG_US_LETTER = 3, + PG_US_LEGAL = 4, + PG_SCREEN = 5, + PG_CUSTOM = 6, + PG_DIN_B5 = 7, + PG_US_EXECUTIVE = 8, + PG_DIN_A0 = 9, + PG_DIN_A1 = 10, + PG_DIN_A2 = 11, + PG_DIN_A6 = 12, + PG_DIN_A7 = 13, + PG_DIN_A8 = 14, + PG_DIN_A9 = 15, + PG_DIN_B0 = 16, + PG_DIN_B1 = 17, + PG_DIN_B10 = 18, + PG_DIN_B2 = 19, + PG_DIN_B3 = 20, + PG_DIN_B4 = 21, + PG_DIN_B6 = 22, + PG_ISO_C5 = 23, + PG_US_COMM10 = 24, + PG_ISO_DL = 25, + PG_US_FOLIO = 26, + PG_US_LEDGER = 27, + PG_US_TABLOID = 28, + // update the number below and the static arrays if you add more values to the enum + PG_LAST_FORMAT = PG_US_TABLOID // used by koPageLayout.cpp +}; + +/** + * Represents the orientation of a printed document. + */ +enum KoOrientation { + PG_PORTRAIT = 0, + PG_LANDSCAPE = 1 +}; + +namespace KoPageFormat +{ + /** + * @brief Convert a KoFormat into a KPrinter::PageSize. + * + * If format is 'screen' it will use A4 landscape. + * If format is 'custom' it will use A4 portrait. + * (you may want to take care of those cases separately). + * Usually passed to KPrinter::setPageSize(). + * + * @note We return int instead of the enum to avoid including kprinter.h + */ + KOFFICECORE_EXPORT int /*KPrinter::PageSize*/ printerPageSize( KoFormat format ); + + /** + * Returns the width (in mm) for a given page format and orientation + * 'Custom' isn't supported by this function, obviously. + */ + KOFFICECORE_EXPORT double width( KoFormat format, KoOrientation orientation ); + + /** + * Returns the height (in mm) for a given page format and orientation + * 'Custom' isn't supported by this function, obviously. + */ + KOFFICECORE_EXPORT double height( KoFormat format, KoOrientation orientation ); + + /** + * Returns the internal name of the given page format. + * Use for saving. + */ + KOFFICECORE_EXPORT QString formatString( KoFormat format ); + + /** + * Convert a format string (internal name) to a page format value. + * Use for loading. + */ + KOFFICECORE_EXPORT KoFormat formatFromString( const QString & string ); + + /** + * Returns the default format (based on the KControl settings) + */ + KOFFICECORE_EXPORT KoFormat defaultFormat(); + + /** + * Returns the translated name of the given page format. + * Use for showing the user. + */ + KOFFICECORE_EXPORT QString name( KoFormat format ); + + /** + * Lists the translated names of all the available formats + */ + KOFFICECORE_EXPORT QStringList allFormats(); + + /** + * Try to find the paper format for the given width and height (in mm). + * Useful to some import filters. + */ + KOFFICECORE_EXPORT KoFormat guessFormat( double width, double height ); +} + + +/** + * @brief Header/Footer type. + * + * @note Yes, this should have been a bitfield, but there was only 0, 2, 3 in koffice-1.0. Don't ask why. + * In the long run this should be replaced with a more flexible repetition/section concept. + */ +enum KoHFType { + HF_SAME = 0, ///< 0: Header/Footer is the same on all pages + HF_FIRST_EO_DIFF = 1, ///< 1: Header/Footer is different on first, even and odd pages (2&3) + HF_FIRST_DIFF = 2, ///< 2: Header/Footer for the first page differs + HF_EO_DIFF = 3 ///< 3: Header/Footer for even - odd pages are different +}; + +/** + * This structure defines the page layout, including + * its size in pt, its format (e.g. A4), orientation, unit, margins etc. + */ +struct KoPageLayout +{ + /** Page format */ + KoFormat format; + /** Page orientation */ + KoOrientation orientation; + + /** Page width in pt */ + double ptWidth; + /** Page height in pt */ + double ptHeight; + /** Left margin in pt */ + double ptLeft; + /** Right margin in pt */ + double ptRight; + /** Top margin in pt */ + double ptTop; + /** Bottom margin in pt */ + double ptBottom; + double ptPageEdge; + double ptBindingSide; + + bool operator==( const KoPageLayout& l ) const { + return ( ptWidth == l.ptWidth && + ptHeight == l.ptHeight && + ptLeft == l.ptLeft && + ptRight == l.ptRight && + ptTop == l.ptTop && + ptBottom == l.ptBottom && + ptPageEdge == l.ptPageEdge && + ptBindingSide == l.ptBindingSide); + } + bool operator!=( const KoPageLayout& l ) const { + return !( (*this) == l ); + } + + /** + * Save this page layout to OASIS. + */ + KOFFICECORE_EXPORT KoGenStyle saveOasis() const; + + /** + * Load this page layout from OASIS + */ + KOFFICECORE_EXPORT void loadOasis(const QDomElement &style); + + /** + * @return a page layout with the default page size depending on the locale settings, + * default margins (2 cm), and portrait orientation. + * @since 1.4 + */ + static KOFFICECORE_EXPORT KoPageLayout standardLayout(); +}; + +/** structure for header-footer */ +struct KoHeadFoot +{ + QString headLeft; + QString headMid; + QString headRight; + QString footLeft; + QString footMid; + QString footRight; +}; + +/** structure for columns */ +struct KoColumns +{ + int columns; + double ptColumnSpacing; + bool operator==( const KoColumns& rhs ) const { + return columns == rhs.columns && + QABS(ptColumnSpacing - rhs.ptColumnSpacing) <= 1E-10; + } + bool operator!=( const KoColumns& rhs ) const { + return columns != rhs.columns || + QABS(ptColumnSpacing - rhs.ptColumnSpacing) > 1E-10; + } +}; + +/** structure for KWord header-footer */ +struct KoKWHeaderFooter +{ + KoHFType header; + KoHFType footer; + double ptHeaderBodySpacing; + double ptFooterBodySpacing; + double ptFootNoteBodySpacing; + bool operator==( const KoKWHeaderFooter& rhs ) const { + return header == rhs.header && footer == rhs.footer && + QABS(ptHeaderBodySpacing - rhs.ptHeaderBodySpacing) <= 1E-10 && + QABS(ptFooterBodySpacing - rhs.ptFooterBodySpacing) <= 1E-10 && + QABS(ptFootNoteBodySpacing - rhs.ptFootNoteBodySpacing) <= 1E-10; + } + bool operator!=( const KoKWHeaderFooter& rhs ) const { + return !( *this == rhs ); + } +}; + +#endif /* KOPAGELAYOUT_H */ + diff --git a/lib/kofficecore/KoPicture.cpp b/lib/kofficecore/KoPicture.cpp new file mode 100644 index 00000000..11bb89dd --- /dev/null +++ b/lib/kofficecore/KoPicture.cpp @@ -0,0 +1,290 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003 Nicolas GOUTTE <goutte@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 <qpainter.h> +#include <qfile.h> + +#include <kdebug.h> +#include <kurl.h> +#include <kio/netaccess.h> + +#include "KoPictureKey.h" +#include "KoPictureBase.h" +#include "KoPictureShared.h" +#include "KoPicture.h" + +uint KoPicture::uniqueValue = 0; + + +KoPicture::KoPicture(void) : m_sharedData(NULL) +{ + m_uniqueName = "Pictures"+ QString::number(uniqueValue++); +} + +KoPicture::~KoPicture(void) +{ + unlinkSharedData(); +} + +QString KoPicture::uniqueName() const +{ + return m_uniqueName; +} + +KoPicture::KoPicture(const KoPicture &other) +{ + m_sharedData=NULL; + (*this)=other; +} + +void KoPicture::assignPictureId( uint _id) +{ + if ( m_sharedData ) + m_sharedData->assignPictureId(_id); +} + +QString KoPicture::uniquePictureId() const +{ + if ( m_sharedData ) + return m_sharedData->uniquePictureId(); + else + return QString::null; +} + +KoPicture& KoPicture::operator=( const KoPicture &other ) +{ + //kdDebug(30003) << "KoPicture::= before" << endl; + if (other.m_sharedData) + other.linkSharedData(); + if (m_sharedData) + unlinkSharedData(); + m_sharedData=other.m_sharedData; + m_key=other.m_key; + //kdDebug(30003) << "KoPicture::= after" << endl; + return *this; +} + +void KoPicture::unlinkSharedData(void) +{ + if (m_sharedData && m_sharedData->deref()) + delete m_sharedData; + + m_sharedData=NULL; +} + +void KoPicture::linkSharedData(void) const +{ + if (m_sharedData) + m_sharedData->ref(); +} + +void KoPicture::createSharedData(void) +{ + if (!m_sharedData) + { + m_sharedData=new KoPictureShared(); + // Do not call m_sharedData->ref() + } +} + +KoPictureType::Type KoPicture::getType(void) const +{ + if (m_sharedData) + return m_sharedData->getType(); + return KoPictureType::TypeUnknown; +} + +KoPictureKey KoPicture::getKey(void) const +{ + return m_key; +} + +void KoPicture::setKey(const KoPictureKey& key) +{ + m_key=key; +} + + +bool KoPicture::isNull(void) const +{ + if (m_sharedData) + return m_sharedData->isNull(); + return true; +} + +void KoPicture::draw(QPainter& painter, int x, int y, int width, int height, int sx, int sy, int sw, int sh, bool fastMode) +{ + if (m_sharedData) + m_sharedData->draw(painter, x, y, width, height, sx, sy, sw, sh, fastMode); + else + { + // Draw a white box + kdWarning(30003) << "Drawing white rectangle! (KoPicture::draw)" << endl; + painter.save(); + painter.setBrush(QColor(255, 255, 255)); + painter.drawRect(x,y,width,height); + painter.restore(); + } +} + +bool KoPicture::loadXpm(QIODevice* io) +{ + kdDebug(30003) << "KoPicture::loadXpm" << endl; + if (!io) + { + kdError(30003) << "No QIODevice!" << endl; + return false; + } + createSharedData(); + return m_sharedData->loadXpm(io); +} + +bool KoPicture::save(QIODevice* io) const +{ + if (!io) + return false; + if (m_sharedData) + return m_sharedData->save(io); + return false; +} + +bool KoPicture::saveAsBase64( KoXmlWriter& writer ) const +{ + if ( m_sharedData ) + return m_sharedData->saveAsBase64( writer ); + return false; +} + +void KoPicture::clear(void) +{ + unlinkSharedData(); +} + +void KoPicture::clearAndSetMode(const QString& newMode) +{ + createSharedData(); + m_sharedData->clearAndSetMode(newMode); +} + +QString KoPicture::getExtension(void) const +{ + if (m_sharedData) + return m_sharedData->getExtension(); + return "null"; // Just a dummy +} + +QString KoPicture::getMimeType(void) const +{ + if (m_sharedData) + return m_sharedData->getMimeType(); + return QString(NULL_MIME_TYPE); +} + +bool KoPicture::load(QIODevice* io, const QString& extension) +{ + kdDebug(30003) << "KoPicture::load(QIODevice*, const QString&) " << extension << endl; + createSharedData(); + + return m_sharedData->load(io,extension); +} + +bool KoPicture::loadFromFile(const QString& fileName) +{ + kdDebug(30003) << "KoPicture::loadFromFile " << fileName << endl; + createSharedData(); + return m_sharedData->loadFromFile(fileName); +} + +bool KoPicture::loadFromBase64( const QCString& str ) +{ + createSharedData(); + return m_sharedData->loadFromBase64( str ); +} + +QSize KoPicture::getOriginalSize(void) const +{ + if (m_sharedData) + return m_sharedData->getOriginalSize(); + return QSize(0,0); +} + +QPixmap KoPicture::generatePixmap(const QSize& size, bool smoothScale) +{ + if (m_sharedData) + return m_sharedData->generatePixmap(size, smoothScale); + return QPixmap(); +} + +bool KoPicture::setKeyAndDownloadPicture(const KURL& url, QWidget *window) +{ + bool result=false; + + QString tmpFileName; + if ( KIO::NetAccess::download(url, tmpFileName, window) ) + { + KoPictureKey key; + key.setKeyFromFile( tmpFileName ); + setKey( key ); + result=loadFromFile( tmpFileName ); + KIO::NetAccess::removeTempFile( tmpFileName ); + } + + return result; +} + +QDragObject* KoPicture::dragObject( QWidget *dragSource, const char *name ) +{ + if (m_sharedData) + return m_sharedData->dragObject( dragSource, name ); + return 0L; +} + +QImage KoPicture::generateImage(const QSize& size) +{ + if (m_sharedData) + return m_sharedData->generateImage( size ); + return QImage(); +} + +bool KoPicture::hasAlphaBuffer() const +{ + if (m_sharedData) + return m_sharedData->hasAlphaBuffer(); + return false; +} + +void KoPicture::setAlphaBuffer(bool enable) +{ + if (m_sharedData) + m_sharedData->setAlphaBuffer(enable); +} + +QImage KoPicture::createAlphaMask(int conversion_flags) const +{ + if (m_sharedData) + return m_sharedData->createAlphaMask(conversion_flags); + return QImage(); +} + +void KoPicture::clearCache(void) +{ + if (m_sharedData) + m_sharedData->clearCache(); +} diff --git a/lib/kofficecore/KoPicture.h b/lib/kofficecore/KoPicture.h new file mode 100644 index 00000000..e4ed1d5d --- /dev/null +++ b/lib/kofficecore/KoPicture.h @@ -0,0 +1,259 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003, 2004 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPicture_h__ +#define __koPicture_h__ + +#include <qstring.h> +#include <qiodevice.h> +#include <qpixmap.h> +#include <koffice_export.h> + +#include "KoPictureKey.h" + +class KoXmlWriter; +class QPainter; +class QSize; +class QDragObject; +class KURL; + +class KoPictureShared; + +/** + * KoPicture is a container class for various types of pictures supported by %KOffice. + * + * @short A picture container class + */ +class KOFFICECORE_EXPORT KoPicture +{ +public: + /** + * Default constructor. + */ + KoPicture(void); + + /** + * Destructor. + */ + ~KoPicture(void); + + /** + * Copy constructor + */ + KoPicture(const KoPicture &other); + + /** + * Assignment operator + */ + KoPicture& operator=(const KoPicture& other); + + KoPictureType::Type getType(void) const; + + /** + * Retrieve the key structure describing the picture in a unique way. + */ + KoPictureKey getKey(void) const; + + /** + * Set the key structure describing the picture in a unique way + */ + void setKey(const KoPictureKey& key); + + /** + * Returns true if the picture is null. + */ + bool isNull(void) const; + + /** + * @brief Draw the picture in a painter. + * + * The parameter @p fastMode allows the picture to be re-sized and drawn quicker if possible + * + * The parameters @p width, @p height define the desired size for the picture + * + * The other parameters are very similar to QPainter::drawPixmap : + * (@p x, @p y) define the position in the painter, + * (@p sx, @p sy) specify the top-left point in picture that is to be drawn. The default is (0, 0). + * (@p sw, @p sh) specify the size of the picture that is to be drawn. The default, (-1, -1), means all the way to the bottom + * right of the pixmap. + * + */ + void draw(QPainter& painter, int x, int y, int width, int height, int sx = 0, int sy = 0, + int sw = -1, int sh = -1, bool fastMode = false); + + /** + * Create a dragobject containing this picture. + * @param dragSource must be 0 when copying to the clipboard + * @param name name for the QDragObject + * @return 0L if the picture is null, or if a dragobject for it isn't implemented [yet] + */ + QDragObject* dragObject( QWidget *dragSource = 0L, const char *name = 0L ); + + bool load(QIODevice* io, const QString& extension); + + /** + * Save picture into a QIODevice + * @param io QIODevice used for saving + */ + bool save(QIODevice* io) const; + + /** + * OASIS FlatXML support: + * Save picture as base64-encoded data into an XML writer. + * The caller will usually do something like + * @code + * writer.startElement( "office:binary-data" ); + * m_picture.saveAsBase64( writer ); + * writer.endElement(); + * @endcode + */ + bool saveAsBase64( KoXmlWriter& writer ) const; + + /** + * @return the image extension (e.g. png) + */ + QString getExtension(void) const; + + /** + * @return the image MIME type + */ + QString getMimeType(void) const; + + /** + * @return the original image size + */ + QSize getOriginalSize(void) const; + + /** + * Clear and set the mode of this KoPicture + * + * @param newMode a file extension (like "png") giving the wanted mode + */ + void clearAndSetMode(const QString& newMode); + + /** + * Reset the KoPicture (but not the key!) + */ + void clear(void); + + /** + * Load the picture from a file named @p fileName + */ + bool loadFromFile(const QString& fileName); + + /** + * Load the picture from base64-encoded data + */ + bool loadFromBase64(const QCString& str); + + /** + * Load a potentially broken XPM file (for old files of KPresenter) + */ + bool loadXpm(QIODevice* io); + + /** + * @deprecated To be replaced by @ref KoPicture::draw + * + * Returns a QPixmap from an image + * Returns an empty QPixmap if the KoPicture is not an image. + */ + QPixmap generatePixmap(const QSize& size, bool smoothScale = false); + + /** + * Download and set the key for a possibly remote file. + * + * @param url the url to download from + * @param window the parent widget for the download. You can pass + * NULL (0) if you absolutely cannot find a parent + * widget to use. + */ + bool setKeyAndDownloadPicture(const KURL& url, QWidget *window); + + /** + * Generate a QImage + * (always in slow mode) + * + * @param size the wanted size for the QImage + */ + QImage generateImage(const QSize& size); + + /** + * @return TRUE if the alpha channel processing has been enabled + */ + bool hasAlphaBuffer() const; + + /** + * Respect the image alpha buffer + */ + void setAlphaBuffer(bool enable); + + /** + * Creates an alpha mask for the picture + * (first you have to call @ref #setAlphaBuffer). + * + * @see hasAlphaBuffer() setAlphaBuffer() + */ + QImage createAlphaMask(int conversion_flags = 0) const; + + /** + * @brief Clear any cache + * + * This is used to avoid using too much memory + * especially if the application somehow also caches the KoPicture's output + */ + void clearCache(void); + + QString uniquePictureId() const; + void assignPictureId( uint _id); + +protected: + /** + * @internal + * Unregister shared data + */ + void unlinkSharedData(void); + /** + * @internal + * Register shared data + */ + void linkSharedData(void) const; + /** + * @internal + * Creare the shared data if needed + */ + void createSharedData(void); + QString uniqueName() const; + +protected: + /** + * @internal + * The key + */ + KoPictureKey m_key; + /** + * @internal + * The shared data + */ + KoPictureShared* m_sharedData; + static uint uniqueValue; + + QString m_uniqueName; +}; + +#endif /* __koPicture_h__ */ diff --git a/lib/kofficecore/KoPictureBase.cpp b/lib/kofficecore/KoPictureBase.cpp new file mode 100644 index 00000000..e608abc2 --- /dev/null +++ b/lib/kofficecore/KoPictureBase.cpp @@ -0,0 +1,142 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003 Nicolas GOUTTE <goutte@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 "KoPictureBase.h" + +#include <KoXmlWriter.h> + +#include <kdebug.h> +#include <kconfig.h> +#include <kglobal.h> + +#include <kmdcodec.h> +#include <qpainter.h> +#include <qpicture.h> +#include <qpixmap.h> +#include <qdragobject.h> + +static int s_useSlowResizeMode = -1; // unset + +KoPictureBase::KoPictureBase(void) +{ + // Slow mode can be very slow, especially at high zoom levels -> configurable + if ( s_useSlowResizeMode == -1 ) + { + KConfigGroup group( KGlobal::config(), "KOfficeImage" ); + s_useSlowResizeMode = group.readNumEntry( "HighResolution", 1 ); + kdDebug(30003) << "HighResolution = " << s_useSlowResizeMode << endl; + } +} + +KoPictureBase::~KoPictureBase(void) +{ +} + +KoPictureBase* KoPictureBase::newCopy(void) const +{ + return new KoPictureBase(*this); +} + +KoPictureType::Type KoPictureBase::getType(void) const +{ + return KoPictureType::TypeUnknown; +} + +bool KoPictureBase::isNull(void) const +{ + return true; // A KoPictureBase is always null. +} + +void KoPictureBase::draw(QPainter& painter, int x, int y, int width, int height, int, int, int, int, bool /*fastMode*/) +{ + // Draw a light red box (easier DEBUG) + kdWarning(30003) << "Drawing light red rectangle! (KoPictureBase::draw)" << endl; + painter.save(); + painter.setBrush(QColor(128,0,0)); + painter.drawRect(x,y,width,height); + painter.restore(); +} + +bool KoPictureBase::load(QIODevice* io, const QString& extension) +{ + return loadData(io->readAll(), extension); +} + +bool KoPictureBase::loadData(const QByteArray&, const QString&) +{ + // Nothing to load! + return false; +} + +bool KoPictureBase::save(QIODevice*) const +{ + // Nothing to save! + return false; +} + +bool KoPictureBase::saveAsBase64( KoXmlWriter& writer ) const +{ + QBuffer buffer; + buffer.open(IO_ReadWrite); + if ( !save( &buffer ) ) + return false; + QCString encoded = KCodecs::base64Encode( buffer.buffer() ); + writer.addTextNode( encoded ); + return true; +} + +QSize KoPictureBase::getOriginalSize(void) const +{ + return QSize(0,0); +} + +QPixmap KoPictureBase::generatePixmap(const QSize&, bool /*smoothScale*/) +{ + return QPixmap(); +} + +QString KoPictureBase::getMimeType(const QString&) const +{ + return QString(NULL_MIME_TYPE); +} + +bool KoPictureBase::isSlowResizeModeAllowed(void) const +{ + return s_useSlowResizeMode != 0; +} + +QDragObject* KoPictureBase::dragObject( QWidget * dragSource, const char * name ) +{ + QImage image (generateImage(getOriginalSize())); + if (image.isNull()) + return 0L; + else + return new QImageDrag( image, dragSource, name ); +} + +QImage KoPictureBase::generateImage(const QSize& size) +{ + return generatePixmap(size,true).convertToImage(); +} + +void KoPictureBase::clearCache(void) +{ + // Nothign to do! +} diff --git a/lib/kofficecore/KoPictureBase.h b/lib/kofficecore/KoPictureBase.h new file mode 100644 index 00000000..17a7df96 --- /dev/null +++ b/lib/kofficecore/KoPictureBase.h @@ -0,0 +1,128 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003, 2004 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPictureBase_h__ +#define __koPictureBase_h__ + +#include "KoPictureKey.h" // for KoPictureType + +#include <qstring.h> +#include <qimage.h> + +class KoXmlWriter; +class QPainter; +class QSize; +class QIODevice; +class QDragObject; + +const char NULL_MIME_TYPE[]="application/x-zerosize"; + +// TODO: fix documentation + +/** + * @internal + * Base class for KoPictureImage, KoPictureClipart, KoPictureEps, KoPictureWmf... + */ +class KoPictureBase +{ +public: + /** + * Default constructor. + */ + KoPictureBase(); + + /** + * Destructor. + */ + virtual ~KoPictureBase(); + + virtual KoPictureType::Type getType(void) const; + + virtual KoPictureBase* newCopy(void) const; + + /** + * Returns true if the picture is null. + */ + virtual bool isNull(void) const; + + /** + * @brief Draw the picture in a painter. + * + * The parameter @p fastMode allows the picture to be re-sized and drawn quicker if possible + * + * The parameters @p width, @p height define the desired size for the image + * Note that the image is being scaled to that size using scale() - except when printing. + * This avoids scaling the image at each paint event. + * + * The other parameters are very similar to QPainter::drawPixmap : + * (@p x, @p y) define the position in the painter, + * (@p sx, @p sy) specify the top-left point in pixmap that is to be drawn. The default is (0, 0). + * (@p sw, @p sh) specify the size of the pixmap that is to be drawn. The default, (-1, -1), means all the way to the bottom + * right of the pixmap. + */ + virtual void draw(QPainter& painter, int x, int y, int width, int height, int sx = 0, int sy = 0, int sw = -1, int sh = -1, bool fastMode = false); + + /** + * Create a dragobject containing this picture. + * @param dragSource must be 0 when copying to the clipboard + * @return 0L if the picture is null! + */ + virtual QDragObject* dragObject( QWidget *dragSource = 0L, const char *name = 0L ); + + virtual bool load(QIODevice* io, const QString& extension); + + virtual bool loadData(const QByteArray& array, const QString& extension); + + /** + * save file + * @param io QIODevice used for saving + */ + virtual bool save(QIODevice* io) const; + + /** + * OASIS FlatXML support: + * Save picture as base64-encoded data into an XML writer. + */ + virtual bool saveAsBase64( KoXmlWriter& writer ) const; + + virtual QSize getOriginalSize(void) const; + + virtual QPixmap generatePixmap(const QSize& size, bool smoothScale = false); + + virtual QString getMimeType(const QString& extension) const; + + bool isSlowResizeModeAllowed(void) const; + + /** + * Generate a QImage + * (always in slow mode) + */ + virtual QImage generateImage(const QSize& size); + + virtual bool hasAlphaBuffer() const + { return false; } + virtual void setAlphaBuffer(bool /*enable*/) + { } + virtual QImage createAlphaMask(int /*conversion_flags*/ = 0) const + { return QImage(); } + + virtual void clearCache(void); +}; + +#endif /* __koPictureBase_h__ */ diff --git a/lib/kofficecore/KoPictureClipart.cpp b/lib/kofficecore/KoPictureClipart.cpp new file mode 100644 index 00000000..f4d94fd2 --- /dev/null +++ b/lib/kofficecore/KoPictureClipart.cpp @@ -0,0 +1,150 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (c) 2001 David Faure <faure@kde.org> + Copyright (C) 2002 Nicolas GOUTTE <goutte@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 <qbuffer.h> +#include <qpainter.h> +#include <qpicture.h> +#include <qpixmap.h> + +#include <kdebug.h> +#include <kdeversion.h> +#if ! KDE_IS_VERSION( 3,1,90 ) +#include <kdebugclasses.h> +#endif + +#include "KoPictureKey.h" +#include "KoPictureBase.h" +#include "KoPictureClipart.h" + +KoPictureClipart::KoPictureClipart(void) : m_clipart(KoPictureType::formatVersionQPicture) +{ +} + +KoPictureClipart::~KoPictureClipart(void) +{ +} + +KoPictureBase* KoPictureClipart::newCopy(void) const +{ + return new KoPictureClipart(*this); +} + +KoPictureType::Type KoPictureClipart::getType(void) const +{ + return KoPictureType::TypeClipart; +} + +bool KoPictureClipart::isNull(void) const +{ + return m_clipart.isNull(); +} + +void KoPictureClipart::drawQPicture(QPicture& clipart, QPainter& painter, + int x, int y, int width, int height, int sx, int sy, int sw, int sh) +{ + kdDebug(30003) << "Drawing KoPictureClipart " << this << endl; + kdDebug(30003) << " x=" << x << " y=" << y << " width=" << width << " height=" << height << endl; + kdDebug(30003) << " sx=" << sx << " sy=" << sy << " sw=" << sw << " sh=" << sh << endl; + painter.save(); + // Thanks to Harri, Qt3 makes it much easier than Qt2 ;) + QRect br = clipart.boundingRect(); + kdDebug(30003) << " Bounding rect. " << br << endl; + + painter.translate(x,y); // Translating must be done before scaling! + if ( br.width() && br.height() ) + painter.scale(double(width)/double(br.width()),double(height)/double(br.height())); + else + kdWarning(30003) << "Null bounding rectangle: " << br.width() << " x " << br.height() << endl; + painter.drawPicture(0,0,clipart); + painter.restore(); +} + +void KoPictureClipart::draw(QPainter& painter, int x, int y, int width, int height, int sx, int sy, int sw, int sh, bool /*fastMode*/) +{ + drawQPicture(m_clipart, painter, x, y, width, height, sx, sy, sw, sh); +} + +bool KoPictureClipart::loadData(const QByteArray& array, const QString& extension) +{ + // Second, create the original clipart + kdDebug(30003) << "Trying to load clipart... (Size:" << m_rawData.size() << ")" << endl; + m_rawData=array; + QBuffer buffer(m_rawData); + buffer.open(IO_ReadOnly); + bool check = true; + if (extension=="svg") + { + if (!m_clipart.load(&buffer, "svg")) + { + kdWarning(30003) << "Loading SVG has failed! (KoPictureClipart::load)" << endl; + check = false; + } + } + else + { + if (!m_clipart.load(&buffer, NULL)) + { + kdWarning(30003) << "Loading QPicture has failed! (KoPictureClipart::load)" << endl; + check = false; + } + } + buffer.close(); + return check; +} + +bool KoPictureClipart::save(QIODevice* io) const +{ + // We save the raw data, as the SVG supposrt in QPicture is poor + Q_ULONG size=io->writeBlock(m_rawData); // WARNING: writeBlock returns Q_LONG but size() Q_ULONG! + return (size==m_rawData.size()); +} + +QSize KoPictureClipart::getOriginalSize(void) const +{ + return m_clipart.boundingRect().size(); +} + +QPixmap KoPictureClipart::generatePixmap(const QSize& size, bool /*smoothScale*/) +{ + // Not sure if it works, but it worked for KoPictureFilePreviewWidget::setClipart + QPixmap pixmap(size); + QPainter p; + + p.begin( &pixmap ); + p.setBackgroundColor( Qt::white ); + pixmap.fill( Qt::white ); + + QRect br = m_clipart.boundingRect(); + if ( br.width() && br.height() ) + p.scale( (double)pixmap.width() / (double)br.width(), (double)pixmap.height() / (double)br.height() ); + p.drawPicture( m_clipart ); + p.end(); + return pixmap; +} + +QString KoPictureClipart::getMimeType(const QString& extension) const +{ + if (extension=="svg") + return "image/svg+xml"; + else + return "image/x-vnd.trolltech.qpicture"; +} + diff --git a/lib/kofficecore/KoPictureClipart.h b/lib/kofficecore/KoPictureClipart.h new file mode 100644 index 00000000..9784c673 --- /dev/null +++ b/lib/kofficecore/KoPictureClipart.h @@ -0,0 +1,102 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2004 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPictureClipart_h__ +#define __koPictureClipart_h__ + +#include <qstring.h> +#include <qpicture.h> + +class QPainter; +class QSize; + +// TODO: fix documentation + +/** + * @internal + * KoPictureClipart is a container class for cliparts + */ +class KoPictureClipart : public KoPictureBase +{ +public: + /** + * Default constructor. + */ + KoPictureClipart(); + + /** + * Destructor. + */ + virtual ~KoPictureClipart(); + + virtual KoPictureType::Type getType(void) const; + + virtual KoPictureBase* newCopy(void) const; + + + /** + * Returns true if the picture is null. + */ + virtual bool isNull(void) const; + + /** + * @brief Draw the image in a painter. + * + * The parameter @p fastMode allows the picture to be re-sized and drawn quicker if possible + * + * The parameters @p width, @p height define the desired size for the image + * Note that the image is being scaled to that size using scale() - except when printing. + * This avoids scaling the image at each paint event. + * + * The other parameters are very similar to QPainter::drawPixmap : + * (@p x, @p y) define the position in the painter, + * (@p sx, @p sy) specify the top-left point in pixmap that is to be drawn. The default is (0, 0). + * (@p sw, @p sh) specify the size of the pixmap that is to be drawn. The default, (-1, -1), means all the way to the bottom + * right of the pixmap. + */ + virtual void draw(QPainter& painter, int x, int y, int width, int height, int sx = 0, int sy = 0, int sw = -1, int sh = -1, bool fastMode = false); + + virtual bool loadData(const QByteArray& array, const QString& extension); + + /** + * save file + * @param io QIODevice used for saving + */ + virtual bool save(QIODevice* io) const; + + virtual QSize getOriginalSize(void) const; + + virtual QPixmap generatePixmap(const QSize& size, bool smoothScale = false); + + virtual QString getMimeType(const QString& extension) const; + +protected: + QPixmap getPixmap(QImage& image); + /** + * @internal + * Draw a QPicture + */ + void drawQPicture(QPicture& clipart, QPainter& painter, + int x, int y, int width, int height, int sx, int sy, int sw, int sh); +protected: + QPicture m_clipart; ///< The clipart as QPicture + QByteArray m_rawData; ///< Copy of the loaded image file +}; + +#endif /* __koPictureClipart_h__ */ diff --git a/lib/kofficecore/KoPictureCollection.cpp b/lib/kofficecore/KoPictureCollection.cpp new file mode 100644 index 00000000..8941f32e --- /dev/null +++ b/lib/kofficecore/KoPictureCollection.cpp @@ -0,0 +1,324 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Reginald Stadlbauer <reggie@kde.org> + Copyright (c) 2001, 2004 David Faure <faure@kde.org> + Copyright (C) 2002, 2004 Nicolas GOUTTE <goutte@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 <qdom.h> + +#include <kdebug.h> +#include <kurl.h> + +#include <KoStoreDevice.h> +#include <KoXmlWriter.h> + +#include "KoPicture.h" +#include "KoPictureCollection.h" + +//#define DEBUG_PICTURES + +KoPicture KoPictureCollection::findPicture(const KoPictureKey& key) const +{ +#ifdef DEBUG_PICTURES + kdDebug(30003) << "KoPictureCollection::findPicture " << key.toString() << endl; +#endif + ConstIterator it = find( key ); + if ( it == end() ) + { + KoPicture picture; + picture.setKey(key); + return picture; + } + + return *it; +} + + +KoPicture KoPictureCollection::insertPicture(const KoPictureKey& key, const KoPicture& picture) +{ +#ifdef DEBUG_PICTURES + kdDebug(30003) << "KoPictureCollection::insertPicture " << key.toString() << endl; +#endif + KoPicture c = findPicture(key); + if (c.isNull()) + { +#ifdef DEBUG_PICTURES + kdDebug(30003) << "KoPictureCollection::insertPicture not found -> inserting" << endl; +#endif + c=picture; + c.setKey(key); // Be sure that the key is correctly set in the KoPicture! + insert(key, c); + } + return c; +} + +KoPicture KoPictureCollection::insertPicture(const KoPicture& picture) +{ + return insertPicture(picture.getKey(), picture); +} + +KoPicture KoPictureCollection::downloadPicture(const KURL& url, QWidget *window) +{ +#ifdef DEBUG_PICTURES + kdDebug(30003) << "KoPictureCollection::downloadPicture " << url.prettyURL() << endl; +#endif + + // If it is a local file, we can check the last modification date, so we should better use loadPicture + if (url.isLocalFile()) + return loadPicture(url.path()); + + + // We have really a remote file, so we cannot check the last modification date + // Therefore we have to always download the file + + KoPicture pic; +#ifdef DEBUG_PICTURES + kdDebug(30003) << "Trying to download picture from file " << url.prettyURL() << endl; +#endif + + if (pic.setKeyAndDownloadPicture(url, window)) + insertPicture(pic.getKey(), pic); + else + kdWarning(30003) << "Could not download KoPicture from " << url.prettyURL() << endl; + + return pic; +} + +KoPicture KoPictureCollection::loadPicture(const QString& fileName) +{ +#ifdef DEBUG_PICTURES + kdDebug(30003) << "KoPictureCollection::loadPicture " << fileName << endl; +#endif + // Check the last modified date of the file, as it is now. + KoPictureKey key; + key.setKeyFromFile(fileName); + + KoPicture c = findPicture(key); + if (c.isNull()) + { +#ifdef DEBUG_PICTURES + kdDebug(30003) << "Trying to load picture from file " << fileName << endl; +#endif + if (c.loadFromFile(fileName)) + insertPicture(key, c); + else + kdWarning(30003) << "Could not load KoPicture from " << fileName << endl; + } + return c; +} + +QString KoPictureCollection::getFileName(const Type pictureType, KoPicture& picture, int& counter) +{ + QString storeURL; + // ### TODO: remove "cliparts" when KPresenter is ready for it + if (pictureType==CollectionClipart) + storeURL="cliparts/clipart"; + else + storeURL="pictures/picture"; + storeURL+=QString::number(++counter); + storeURL+='.'; + storeURL+=picture.getExtension(); + return storeURL; +} + +QString KoPictureCollection::getOasisFileName(const KoPicture& picture) const +{ + QString storeURL( "Pictures/"); + if ( !picture.uniquePictureId().isEmpty() ) + storeURL+=picture.uniquePictureId(); + else + storeURL+=picture.getKey().toString(); + storeURL+='.'; + storeURL+=picture.getExtension(); + return storeURL; +} + +bool KoPictureCollection::saveToStore(const Type pictureType, KoStore *store, const QValueList<KoPictureKey>& keys) +{ + int counter=0; + QValueList<KoPictureKey>::const_iterator it = keys.begin(); + for ( ; it != keys.end(); ++it ) + { + KoPicture c = findPicture( *it ); + if (c.isNull()) + kdWarning(30003) << "Picture " << (*it).toString() << " not found in collection !" << endl; + else + { + QString storeURL=getFileName(pictureType, c, counter); + + if (store->open(storeURL)) + { + KoStoreDevice dev(store); + if ( ! c.save(&dev) ) + return false; // e.g. bad image? + if ( !store->close() ) + return false; // e.g. disk full + } + } + } + return true; +} + +bool KoPictureCollection::saveOasisToStore( KoStore *store, QValueList<KoPictureKey> keys, KoXmlWriter* manifestWriter ) +{ + QValueList<KoPictureKey>::Iterator it = keys.begin(); + for ( ; it != keys.end(); ++it ) + { + KoPicture c = findPicture( *it ); + if (c.isNull()) + kdWarning(30003) << "Picture " << (*it).toString() << " not found in collection !" << endl; + else + { + QString storeURL( getOasisFileName(c) ); + if (store->open(storeURL)) + { + KoStoreDevice dev(store); + if ( ! c.save(&dev) ) + return false; // e.g. bad image? + if ( !store->close() ) + return false; // e.g. disk full + manifestWriter->addManifestEntry( storeURL, c.getMimeType() ); + } + } + } + return true; +} + +QDomElement KoPictureCollection::saveXML(const Type pictureType, QDomDocument &doc, QValueList<KoPictureKey> keys) +{ + QString strElementName("PICTURES"); + if (pictureType==CollectionImage) + strElementName="PIXMAPS"; + else if (pictureType==CollectionClipart) + strElementName="CLIPARTS"; + QDomElement cliparts = doc.createElement( strElementName ); + int counter=0; + QValueList<KoPictureKey>::Iterator it = keys.begin(); + for ( ; it != keys.end(); ++it ) + { + KoPicture picture = findPicture( *it ); + if ( picture.isNull() ) + kdWarning(30003) << "Picture " << (*it).toString() << " not found in collection !" << endl; + else + { + QString pictureName=getFileName(pictureType, picture, counter); + QDomElement keyElem = doc.createElement( "KEY" ); + cliparts.appendChild(keyElem); + (*it).saveAttributes(keyElem); + keyElem.setAttribute("name", pictureName); + } + } + return cliparts; +} + +void KoPictureCollection::readXML( QDomElement& pixmapsElem, QMap <KoPictureKey, QString>& map ) +{ + for( QDomNode n = pixmapsElem.firstChild(); + !n.isNull(); + n = n.nextSibling() ) + { + QDomElement keyElement = n.toElement(); + if (keyElement.isNull()) continue; + if (keyElement.tagName()=="KEY") + { + KoPictureKey key; + key.loadAttributes(keyElement); + map.insert(key, keyElement.attribute("name")); + } + } +} + + +KoPictureCollection::StoreMap KoPictureCollection::readXML( QDomElement& pixmapsElem ) +{ + StoreMap map; + readXML(pixmapsElem, map); + return map; +} + +void KoPictureCollection::readFromStore( KoStore * store, const StoreMap & storeMap ) +{ +#ifdef DEBUG_PICTURES + kdDebug(30003) << "KoPictureCollection::readFromStore " << store << endl; +#endif + StoreMap::ConstIterator it = storeMap.begin(); + for ( ; it != storeMap.end(); ++it ) + { + KoPicture c = findPicture(it.key()); + if (!c.isNull()) + { + // Do not load a file that we already have! + //kdDebug(30003) << store << endl; + continue; + } + QString u(it.data()); + if (u.isEmpty()) + { + // old naming I suppose ? + u=it.key().toString(); + } + + KoPicture picture; + + if (!store->open( u )) + { + u.prepend( "file:" ); + if (!store->open( u )) + { + kdWarning(30003) << "Could load neither from store nor from file: " << it.data() << endl; + return; + } + } + + const int pos=u.findRev('.'); + if (pos==-1) + { + kdError(30003) << "File with no extension! Not supported!" << endl; + return; + } + const QString extension(u.mid(pos+1)); + + KoStoreDevice dev(store); + picture.load(&dev, extension); + store->close(); + + if (!picture.isNull()) + insertPicture(it.key(), picture); + } +} + +KoPicture KoPictureCollection::findOrLoad(const QString& fileName, const QDateTime& dateTime) +{ + // As now all KoPictureKey objects have a valid QDateTime, we must do it without a date/time check. + ConstIterator it = find( KoPictureKey ( fileName, dateTime ) ); + if ( it == end() ) + { + return loadPicture( fileName ); + } + return *it; +} + +void KoPictureCollection::assignUniqueIds() +{ + uint idx = 0; + Iterator it; + for ( it = begin(); it != end(); ++it, ++idx ) + { + it.data().assignPictureId(idx); + } +} diff --git a/lib/kofficecore/KoPictureCollection.h b/lib/kofficecore/KoPictureCollection.h new file mode 100644 index 00000000..22277bdd --- /dev/null +++ b/lib/kofficecore/KoPictureCollection.h @@ -0,0 +1,171 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Reginald Stadlbauer <reggie@kde.org> + Copyright (c) 2001 David Faure <faure@kde.org> + Copyright (C) 2002 Nicolas GOUTTE <goutte@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. +*/ + +#ifndef koPictureCollection_h +#define koPictureCollection_h + +#include <qmap.h> +#include <qdom.h> +#include <qvaluelist.h> + +#include "KoPicture.h" + +class KURL; + +class KoStore; +class KoXmlWriter; + +/** + * A collection of pictures (a key and the picture itself). + * + */ +class KOFFICECORE_EXPORT KoPictureCollection : public QMap<KoPictureKey, KoPicture> +{ +public: + enum Type { + /// collection with mixed pictures + CollectionPicture=0, + /// collection with images only + CollectionImage, + /// collection with cliparts only + CollectionClipart + }; + + /** + * Looks for a clipart in the collection, returns a new KoPicture with that key if not found. + */ + KoPicture findPicture( const KoPictureKey &key ) const; + + /** + * Inserts a picture into the collection, if not already there + */ + KoPicture insertPicture( const KoPictureKey& key, const KoPicture& picture ); + + /** + * Inserts a picture into the collection, if not already there + * Same as above, but takes the key from the @p picture + */ + KoPicture insertPicture( const KoPicture& picture ); + + /** + * @brief Download a possibly remote file + * + * If this file is really remote, it is always downloaded. + * If the file is local, it acts as @ref #loadPicture. + * @param url the URL to download from + * @param window the parent window for the download. You can pass NULL (0) + * if you absolutely cannot find anything to use. + */ + KoPicture downloadPicture(const KURL& url, QWidget *window); + + /** + * @brief Load a clipart from a file (and insert into the collection). + * + * The modification date of the file is checked, to create the key + * for this clipart. If this key maps to an existing clipart in the + * collection, then this picture is returned, otherwise the file is loaded. + */ + KoPicture loadPicture( const QString &fileName ); + + /** + * Save the used picturess from the collection into the store + * Usually called from completeSaving(). + * + * @param pictureType type for the stored picture + * @param store the store in which to save the pictures + * @param keys the list of keys corresponding to the pictures to save + * @return true on success, false on failure (e.g. disk full) + * + * @todo Reduce lameness of dox for pictureType. + */ + bool saveToStore(const Type pictureType, KoStore * store, const QValueList<KoPictureKey>& keys ); + + /** + * Generate the <PICTURES>, <PIXMAPS> or <CLIPARTS> tag, that saves the key and the related + * relative path in the store (e.g. pictures/picture1.png) for each picture. + * + * @param pictureType the type of the collection + * @param doc the DOM document in which the tags are to be generated + * @param keys the list of keys + */ + QDomElement saveXML(const Type pictureType, QDomDocument &doc, + QValueList<KoPictureKey> keys ); + + bool saveOasisToStore( KoStore *store, QValueList<KoPictureKey> keys, KoXmlWriter* manifestWriter ); + + + typedef QMap<KoPictureKey, QString> StoreMap; + /** + * Read the <PICTURES>, <PIXMAPS> or <CLIPARTS> tag, and save the result (key<->store-filename associations) + * into the QMap. You may want to 'new' a QMap in loadXML, and to use and then delete + * it in completeLoading (to save memory). + * + * @param pixmapsElem the <PICTURES>, <PIXMAPS> or <CLIPARTS> element + */ + StoreMap readXML( QDomElement &pixmapsElem ); + + /** + * Helper method for @ref #readFromStore + */ + void readXML( QDomElement& pixmapsElem, QMap <KoPictureKey, QString>& map ); + + /** + * Read all pictures from the store, into this collection + * The map comes from @ref #readXML, and is used to find which pictures + * to load, and which key to associate them. + */ + void readFromStore( KoStore * store, const StoreMap & storeMap ); + + /** + * @deprecated + * KPresenter needs to use the same code for loading images from a collection and + * for loading images from disk. + * + * @param fileName the name of the file to read from disk if needed + * @param dateTime the date and time + * + * Formerly, an invalid date/time meant to read the file from disk. This is not the case anymore. + */ + KoPicture findOrLoad(const QString& fileName, const QDateTime& dateTime); + + /** + * Return filename as url for picture + * + */ + QString getOasisFileName(const KoPicture& picture) const; + + /** + * Call it before to save Oasis file + */ + void assignUniqueIds(); + + +private: + /** + * @internal + */ + QString getFileName(const Type pictureType, KoPicture& picture, int& counter); + + class Private; + Private* d; +}; + +#endif /* __koPictureCollection_h_- */ diff --git a/lib/kofficecore/KoPictureEps.cpp b/lib/kofficecore/KoPictureEps.cpp new file mode 100644 index 00000000..752e2a45 --- /dev/null +++ b/lib/kofficecore/KoPictureEps.cpp @@ -0,0 +1,447 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003, 2004 Nicolas GOUTTE <goutte@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 <unistd.h> +#include <stdio.h> + +#include <qbuffer.h> +#include <qpainter.h> +#include <qpaintdevicemetrics.h> +#include <qfile.h> +#include <qtextstream.h> +#include <qregexp.h> +#include <qimage.h> +#include <qpixmap.h> +#include <qapplication.h> +#include <qdragobject.h> + +#include <kglobal.h> +#include <kdebug.h> +#include <kdeversion.h> +#if ! KDE_IS_VERSION( 3,1,90 ) +#include <kdebugclasses.h> +#endif +#include <ktempfile.h> +#include <kprocess.h> + +#include "KoPictureKey.h" +#include "KoPictureBase.h" +#include "KoPictureEps.h" + + +KoPictureEps::KoPictureEps(void) : m_psStreamStart(0), m_psStreamLength(0), m_cacheIsInFastMode(true) +{ + // Forbid QPixmap to cache the X-Window resources (Yes, it is slower!) + m_cachedPixmap.setOptimization(QPixmap::MemoryOptim); +} + +KoPictureEps::~KoPictureEps(void) +{ +} + +KoPictureBase* KoPictureEps::newCopy(void) const +{ + return new KoPictureEps(*this); +} + +KoPictureType::Type KoPictureEps::getType(void) const +{ + return KoPictureType::TypeEps; +} + +bool KoPictureEps::isNull(void) const +{ + return m_rawData.isNull(); +} + +QImage KoPictureEps::scaleWithGhostScript(const QSize& size, const int resolutionx, const int resolutiony ) +{ + if (!m_boundingBox.width() || !m_boundingBox.height()) + { + kdDebug(30003) << "EPS image has a null size! (in KoPictureEps::scaleWithGhostScript)" << endl; + return QImage(); + } + + // ### TODO: do not call GhostScript up to three times for each re-scaling (one call of GhostScript should be enough to know which device is available: gs --help) + // png16m is better, but not always available -> fallback to bmp16m, then fallback to ppm (256 colors) + // ### TODO: pcx24b is also a true colour format + // ### TODO: support alpha (other gs devices needed) + + const char* deviceTable[] = { "png16m", "bmp16m", "ppm", 0 }; + + QImage img; + + for ( int i = 0; deviceTable[i]; ++i) + { + if ( tryScaleWithGhostScript( img, size, resolutionx, resolutiony, deviceTable[i] ) != -1 ) + { + return img; + } + + } + + kdError(30003) << "Image from GhostScript cannot be loaded (in KoPictureEps::scaleWithGhostScript)" << endl; + return img; +} + +// Helper method for scaleWithGhostScript. Returns 1 on success, 0 on error, -1 if nothing generated +// (in which case another 'output device' can be tried) +int KoPictureEps::tryScaleWithGhostScript(QImage &image, const QSize& size, const int resolutionx, const int resolutiony, const char* device ) +// Based on the code of the file kdelibs/kimgio/eps.cpp +{ + kdDebug(30003) << "Sampling with GhostScript, using device \"" << device << "\" (in KoPictureEps::tryScaleWithGhostScript)" << endl; + + KTempFile tmpFile; + tmpFile.setAutoDelete(true); + + if ( tmpFile.status() ) + { + kdError(30003) << "No KTempFile! (in KoPictureEps::tryScaleWithGhostScript)" << endl; + return 0; // error + } + + const int wantedWidth = size.width(); + const int wantedHeight = size.height(); + const double xScale = double(size.width()) / double(m_boundingBox.width()); + const double yScale = double(size.height()) / double(m_boundingBox.height()); + + // create GS command line + + QString cmdBuf ( "gs -sOutputFile=" ); + cmdBuf += KProcess::quote(tmpFile.name()); + cmdBuf += " -q -g"; + cmdBuf += QString::number( wantedWidth ); + cmdBuf += "x"; + cmdBuf += QString::number( wantedHeight ); + + if ( ( resolutionx > 0) && ( resolutiony > 0) ) + { +#if 0 + // Do not play with resolution for now. + // It brings more problems at print than solutions + cmdBuf += " -r"; + cmdBuf += QString::number( resolutionx ); + cmdBuf += "x"; + cmdBuf += QString::number( resolutiony ); +#endif + } + + cmdBuf += " -dSAFER -dPARANOIDSAFER -dNOPAUSE -sDEVICE="; + cmdBuf += device; + //cmdBuf += " -c 255 255 255 setrgbcolor fill 0 0 0 setrgbcolor"; + cmdBuf += " -"; + cmdBuf += " -c showpage quit"; + + // run ghostview + + FILE* ghostfd = popen (QFile::encodeName(cmdBuf), "w"); + + if ( ghostfd == 0 ) + { + kdError(30003) << "No connection to GhostScript (in KoPictureEps::tryScaleWithGhostScript)" << endl; + return 0; // error + } + + // The translation is needed as GhostScript (7.07) cannot handle negative values in the bounding box otherwise. + fprintf (ghostfd, "\n%d %d translate\n", -qRound(m_boundingBox.left()*xScale), -qRound(m_boundingBox.top()*yScale)); + fprintf (ghostfd, "%g %g scale\n", xScale, yScale); + + // write image to gs + + fwrite( m_rawData.data() + m_psStreamStart, sizeof(char), m_psStreamLength, ghostfd); + + pclose ( ghostfd ); + + // load image + if( !image.load (tmpFile.name()) ) + { + // It failed - maybe the device isn't supported by gs + return -1; + } + if ( image.size() != size ) // this can happen due to rounding problems + { + //kdDebug(30003) << "fixing size to " << size.width() << "x" << size.height() + // << " (was " << image.width() << "x" << image.height() << ")" << endl; + image = image.scale( size ); // hmm, smoothScale instead? + } + kdDebug(30003) << "Image parameters: " << image.width() << "x" << image.height() << "x" << image.depth() << endl; + return 1; // success +} + +void KoPictureEps::scaleAndCreatePixmap(const QSize& size, bool fastMode, const int resolutionx, const int resolutiony ) +{ + kdDebug(30003) << "KoPictureEps::scaleAndCreatePixmap " << size << " " << (fastMode?QString("fast"):QString("slow")) + << " resolutionx: " << resolutionx << " resolutiony: " << resolutiony << endl; + if ((size==m_cachedSize) + && ((fastMode) || (!m_cacheIsInFastMode))) + { + // The cached pixmap has already the right size + // and: + // - we are in fast mode (We do not care if the re-size was done slowly previously) + // - the re-size was already done in slow mode + kdDebug(30003) << "Already cached!" << endl; + return; + } + + // Slow mode can be very slow, especially at high zoom levels -> configurable + if ( !isSlowResizeModeAllowed() ) + { + kdDebug(30003) << "User has disallowed slow mode!" << endl; + fastMode = true; + } + + // We cannot use fast mode, if nothing was ever cached. + if ( fastMode && !m_cachedSize.isEmpty()) + { + kdDebug(30003) << "Fast scaling!" << endl; + // Slower than caching a QImage, but faster than re-sampling! + QImage image( m_cachedPixmap.convertToImage() ); + m_cachedPixmap=image.scale( size ); + m_cacheIsInFastMode=true; + m_cachedSize=size; + } + else + { + QTime time; + time.start(); + + QApplication::setOverrideCursor( Qt::waitCursor ); + m_cachedPixmap = scaleWithGhostScript( size, resolutionx, resolutiony ); + QApplication::restoreOverrideCursor(); + m_cacheIsInFastMode=false; + m_cachedSize=size; + + kdDebug(30003) << "Time: " << (time.elapsed()/1000.0) << " s" << endl; + } + kdDebug(30003) << "New size: " << size << endl; +} + +void KoPictureEps::draw(QPainter& painter, int x, int y, int width, int height, int sx, int sy, int sw, int sh, bool fastMode) +{ + if ( !width || !height ) + return; + + QSize screenSize( width, height ); + //kdDebug() << "KoPictureEps::draw screenSize=" << screenSize.width() << "x" << screenSize.height() << endl; + + QPaintDeviceMetrics metrics (painter.device()); + kdDebug(30003) << "Metrics: X: " << metrics.logicalDpiX() << " x Y: " << metrics.logicalDpiX() << " (in KoPictureEps::draw)" << endl; + + if ( painter.device()->isExtDev() ) // Is it an external device (i.e. printer) + { + kdDebug(30003) << "Drawing for a printer (in KoPictureEps::draw)" << endl; + // For printing, always re-sample the image, as a printer has never the same resolution than a display. + QImage image( scaleWithGhostScript( screenSize, metrics.logicalDpiX(), metrics.logicalDpiY() ) ); + // sx,sy,sw,sh is meant to be used as a cliprect on the pixmap, but drawImage + // translates it to the (x,y) point -> we need (x+sx, y+sy). + painter.drawImage( x + sx, y + sy, image, sx, sy, sw, sh ); + } + else // No, it is simply a display + { + scaleAndCreatePixmap(screenSize, fastMode, metrics.logicalDpiX(), metrics.logicalDpiY() ); + + // sx,sy,sw,sh is meant to be used as a cliprect on the pixmap, but drawPixmap + // translates it to the (x,y) point -> we need (x+sx, y+sy). + painter.drawPixmap( x + sx, y + sy, m_cachedPixmap, sx, sy, sw, sh ); + } +} + +bool KoPictureEps::extractPostScriptStream( void ) +{ + kdDebug(30003) << "KoPictureEps::extractPostScriptStream" << endl; + QDataStream data( m_rawData, IO_ReadOnly ); + data.setByteOrder( QDataStream::LittleEndian ); + Q_UINT32 magic, offset, length; + data >> magic; + data >> offset; + data >> length; + if ( !length ) + { + kdError(30003) << "Length of PS stream is zero!" << endl; + return false; + } + if ( offset+length>m_rawData.size() ) + { + kdError(30003) << "Data stream of the EPSF file is longer than file: " << offset << "+" << length << ">" << m_rawData.size() << endl; + return false; + } + m_psStreamStart = offset; + m_psStreamLength = length; + return true; +} + +QString KoPictureEps::readLine( const QByteArray& array, const uint start, const uint length, uint& pos, bool& lastCharWasCr ) +{ + QString strLine; + const uint finish = kMin( start + length, array.size() ); + for ( ; pos < finish; ++pos ) // We are starting at pos + { + const char ch = array[ pos ]; // Read one character + if ( ch == '\n' ) + { + if ( lastCharWasCr ) + { + // We have a line feed following a Carriage Return + // As the Carriage Return has already ended the previous line, + // discard this Line Feed. + lastCharWasCr = false; + } + else + { + // We have a normal Line Feed, therefore we end the line + break; + } + } + else if ( ch == '\r' ) + { + // We have a Carriage Return, therefore we end the line + lastCharWasCr = true; + break; + } + else if ( ch == char(12) ) // Form Feed + { // ### TODO: can a FF happen in PostScript? + // Ignore the form feed + continue; + } + else + { + strLine += ch; + lastCharWasCr = false; + } + } + return strLine; +} + + +bool KoPictureEps::loadData(const QByteArray& array, const QString& /* extension */ ) +{ + + kdDebug(30003) << "KoPictureEps::load" << endl; + // First, read the raw data + m_rawData=array; + + if (m_rawData.isNull()) + { + kdError(30003) << "No data was loaded!" << endl; + return false; + } + + if ( ( m_rawData[0]==char(0xc5) ) && ( m_rawData[1]==char(0xd0) ) + && ( m_rawData[2]==char(0xd3) ) && ( m_rawData[3]==char(0xc6) ) ) + { + // We have a so-called "MS-DOS EPS file", we have to extract the PostScript stream + if (!extractPostScriptStream()) // Changes m_rawData + return false; + } + else + { + m_psStreamStart = 0; + m_psStreamLength = m_rawData.size(); + } + + QString lineBox; // Line with the bounding box + bool lastWasCr = false; // Was the last character of the line a carriage return? + uint pos = m_psStreamStart; // We start to search the bounding box at the start of the PostScript stream + QString line( readLine( m_rawData, m_psStreamStart, m_psStreamLength, pos, lastWasCr ) ); + kdDebug(30003) << "Header: " << line << endl; + if (!line.startsWith("%!")) + { + kdError(30003) << "Not a PostScript file!" << endl; + return false; + } + QRect rect; + bool lineIsBoundingBox = false; // Does "line" has a %%BoundingBox line? + for(;;) + { + ++pos; // Get over the previous line end (CR or LF) + line = readLine( m_rawData, m_psStreamStart, m_psStreamLength, pos, lastWasCr ); + kdDebug(30003) << "Checking line: " << line << endl; + // ### TODO: it seems that the bounding box can be delayed with "(atend)" in the trailer (GhostScript 7.07 does not support it either.) + if (line.startsWith("%%BoundingBox:")) + { + lineIsBoundingBox = true; + break; + } + // ### TODO: also abort on %%EndComments + // ### TODO: %? , where ? is non-white-space printable, does not end the comment! + else if (!line.startsWith("%%")) + break; // Not a EPS comment anymore, so abort as we are not in the EPS header anymore + } + if ( !lineIsBoundingBox ) + { + kdError(30003) << "KoPictureEps::load: could not find a bounding box!" << endl; + return false; + } + // Floating point values are not allowed in a Bounding Box, but ther are many such files out there... + QRegExp exp("(\\-?[0-9]+\\.?[0-9]*)\\s(\\-?[0-9]+\\.?[0-9]*)\\s(\\-?[0-9]+\\.?[0-9]*)\\s(\\-?[0-9]+\\.?[0-9]*)"); + if ( exp.search(line) == -1 ) + { + // ### TODO: it might be an "(atend)" and the bounding box is in the trailer + // (but GhostScript 7.07 does not support a bounding box in the trailer.) + // Note: in Trailer, it is the last BoundingBox that counts not the first! + kdError(30003) << "Not standard bounding box: " << line << endl; + return false; + } + kdDebug(30003) << "Reg. Exp. Found: " << exp.capturedTexts() << endl; + rect.setLeft((int)exp.cap(1).toDouble()); + rect.setTop((int)exp.cap(2).toDouble()); + rect.setRight((int)exp.cap(3).toDouble()); + rect.setBottom((int)exp.cap(4).toDouble()); + m_boundingBox=rect; + m_originalSize=rect.size(); + kdDebug(30003) << "Rect: " << rect << " Size: " << m_originalSize << endl; + return true; +} + +bool KoPictureEps::save(QIODevice* io) const +{ + // We save the raw data, to avoid damaging the file by many load/save cycles + Q_ULONG size=io->writeBlock(m_rawData); // WARNING: writeBlock returns Q_LONG but size() Q_ULONG! + return (size==m_rawData.size()); +} + +QSize KoPictureEps::getOriginalSize(void) const +{ + return m_originalSize; +} + +QPixmap KoPictureEps::generatePixmap(const QSize& size, bool smoothScale) +{ + scaleAndCreatePixmap(size,!smoothScale, 0, 0); + return m_cachedPixmap; +} + +QString KoPictureEps::getMimeType(const QString&) const +{ + return "image/x-eps"; +} + +QImage KoPictureEps::generateImage(const QSize& size) +{ + // 0, 0 == resolution unknown + return scaleWithGhostScript(size, 0, 0); +} + +void KoPictureEps::clearCache(void) +{ + m_cachedPixmap.resize(0, 0); + m_cacheIsInFastMode=true; + m_cachedSize=QSize(); +} diff --git a/lib/kofficecore/KoPictureEps.h b/lib/kofficecore/KoPictureEps.h new file mode 100644 index 00000000..ba2a4cb9 --- /dev/null +++ b/lib/kofficecore/KoPictureEps.h @@ -0,0 +1,136 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003, 2004 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPictureEps_h__ +#define __koPictureEps_h__ + +#include <qstring.h> + +class QPainter; +class QSize; +class KoPictureEpsPrivate; + +// TODO: fix documentation + +/** + * @internal + * KoPictureEps is a container class for a EPS picture + */ +class KoPictureEps : public KoPictureBase +{ +public: + /** + * Default constructor. + */ + KoPictureEps(); + + /** + * Destructor. + */ + virtual ~KoPictureEps(); + + KoPictureType::Type getType(void) const; + + KoPictureBase* newCopy(void) const; + + /** + * Returns true if the picture is null. + */ + virtual bool isNull(void) const; + + /** + * @brief Draw the image in a painter. + * + * No, this isn't as simple as painter.drawPixmap(). + * This method ensures that the best quality is used when printing, scaling the painter. + * + * The parameter @p fastMode allows the picture to be re-sized and drawn quicker if possible + * + * The parameters @p width, @p height define the desired size for the image + * Note that the image is being scaled to that size using scale() - except when printing. + * This avoids scaling the image at each paint event. + * + * The other parameters are very similar to QPainter::drawPixmap : + * (@p x, @p y) define the position in the painter, + * (@p sx, @p sy) specify the top-left point in pixmap that is to be drawn. The default is (0, 0). + * (@p sw, @p sh) specify the size of the pixmap that is to be drawn. The default, (-1, -1), means all the way to the bottom + * right of the pixmap. + */ + virtual void draw(QPainter& painter, int x, int y, int width, int height, int sx = 0, int sy = 0, int sw = -1, int sh = -1, bool fastMode = false); + + virtual bool loadData(const QByteArray& array, const QString& extension); + + virtual bool save(QIODevice* io) const; + + virtual QSize getOriginalSize(void) const; + + virtual QPixmap generatePixmap(const QSize& size, bool smoothScale = false); + + virtual QString getMimeType(const QString& extension) const; + + /** + * Generate a QImage + * (always in slow mode) + */ + virtual QImage generateImage(const QSize& size); + + virtual void clearCache(void); +protected: + QPixmap getPixmap(QImage& image); + void scaleAndCreatePixmap(const QSize& size, bool fastMode, const int resolutionx, const int resolutiony ); + QImage scaleWithGhostScript( const QSize& size, const int resolutionx, const int resolutiony ); + bool extractPostScriptStream( void ); + +private: + int tryScaleWithGhostScript(QImage &image, const QSize& size, const int resolutionx, const int resolutiony, const char* device ); + /** + * @brief Read a line from a PostScript stream in a QByteArray + * + * The PostScript stream is somewhere in the QByteArray and therefore has to be + * determined by the start and length of this stream and the current position + * + * @param array the array which is read + * @param start start position of the PostScript stream + * @param lenght lenght of the PostScript stream + * @param pos current position (which is also returned) + * @param lastCharWasCr defines if the previous line ended with a Carriage Return, + * to skip an eventual Line Feed at the start of this line + * @return the read line + */ + QString readLine( const QByteArray& array, const uint start, const uint length, uint& pos, bool& lastCharWasCr ); + +private: + /** + * Copy of the loaded EPS file + */ + QByteArray m_rawData; + QPixmap m_cachedPixmap; ///< Cached pixmap + QSize m_originalSize; ///< Original size of the EPS picture + QSize m_cachedSize; ///< size of the currently cached pixmap @see m_cachedPixmap + QRect m_boundingBox; ///< Bounding box, as read from the EPS file + uint m_psStreamStart; ///< Start position of the PostScript stream (like if it was a MS-DOS EPS file) + uint m_psStreamLength; ///< Lenght of the PostScript stream (like if it was a MS-DOS EPS file) + /** + * true, if the last cached image was done using fast mode. + * false, if the last cached image was done using slow mode. + */ + bool m_cacheIsInFastMode; +}; + +#endif /* __koPictureEps_h__ */ diff --git a/lib/kofficecore/KoPictureImage.cpp b/lib/kofficecore/KoPictureImage.cpp new file mode 100644 index 00000000..02110760 --- /dev/null +++ b/lib/kofficecore/KoPictureImage.cpp @@ -0,0 +1,194 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003 Nicolas GOUTTE <goutte@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 "KoPictureImage.h" +#include "KoPictureKey.h" + +#include <kdebug.h> +#include <kmimetype.h> + +#include <qbuffer.h> +#include <qpainter.h> +#include <qimage.h> +#include <qpixmap.h> +#include <qapplication.h> +#include <qdragobject.h> + +KoPictureImage::KoPictureImage(void) : m_cacheIsInFastMode(true) +{ + // Forbid QPixmap to cache the X-Window resources (Yes, it is slower!) + m_cachedPixmap.setOptimization(QPixmap::MemoryOptim); +} + +KoPictureImage::~KoPictureImage(void) +{ +} + +KoPictureBase* KoPictureImage::newCopy(void) const +{ + return new KoPictureImage(*this); +} + +KoPictureType::Type KoPictureImage::getType(void) const +{ + return KoPictureType::TypeImage; +} + +bool KoPictureImage::isNull(void) const +{ + return m_originalImage.isNull(); +} + +void KoPictureImage::scaleAndCreatePixmap(const QSize& size, bool fastMode) +{ + if ((size==m_cachedSize) + && ((fastMode) || (!m_cacheIsInFastMode))) + { + // The cached pixmap has already the right size + // and: + // - we are in fast mode (We do not care if the re-size was done slowly previously) + // - the re-size was already done in slow mode + return; + } + + // Slow mode can be very slow, especially at high zoom levels -> configurable + if ( !isSlowResizeModeAllowed() ) + { + kdDebug(30003) << "User has disallowed slow mode!" << endl; + fastMode = true; + } + + // Use QImage::scale if we have fastMode==true + if ( fastMode ) + { + m_cachedPixmap.convertFromImage(m_originalImage.scale( size ), QPixmap::Color); // Always color or else B/W can be reversed + m_cacheIsInFastMode=true; + } + else + { + m_cachedPixmap.convertFromImage(m_originalImage.smoothScale( size ), QPixmap::Color); // Always color or else B/W can be reversed + m_cacheIsInFastMode=false; + } + m_cachedSize=size; +} + +void KoPictureImage::draw(QPainter& painter, int x, int y, int width, int height, int sx, int sy, int sw, int sh, bool fastMode) +{ + //kdDebug() << "KoImage::draw currentSize:" << currentSize.width() << "x" << currentSize.height() << endl; + if ( !width || !height ) + return; + QSize origSize = getOriginalSize(); + const bool scaleImage = painter.device()->isExtDev() // we are printing + && ((width <= origSize.width()) || (height <= origSize.height())); + if( scaleImage ) + { + // use full resolution of image + double xScale = double(width) / double(origSize.width()); + double yScale = double(height) / double(origSize.height()); + + painter.save(); + painter.translate( x, y ); + painter.scale( xScale, yScale ); + // Note that sx, sy, sw and sh are unused in this case. Not a problem, since it's about printing. + // Note 2: we do not cache the QPixmap. As we are printing, the next time we will probably + // need again the screen version. + painter.drawImage(0, 0, m_originalImage); + painter.restore(); + } + else + { + QSize screenSize( width, height ); + //kdDebug() << "KoPictureImage::draw screenSize=" << screenSize.width() << "x" << screenSize.height() << endl; + + scaleAndCreatePixmap(screenSize, fastMode); + + // sx,sy,sw,sh is meant to be used as a cliprect on the pixmap, but drawPixmap + // translates it to the (x,y) point -> we need (x+sx, y+sy). + painter.drawPixmap( x + sx, y + sy, m_cachedPixmap, sx, sy, sw, sh ); + } +} + +bool KoPictureImage::loadData(const QByteArray& array, const QString& /* extension*/ ) +{ + m_rawData=array; + // Second, create the original image + QBuffer buffer(m_rawData); + buffer.open(IO_ReadWrite); + QImageIO imageIO(&buffer,NULL); + + if (!imageIO.read()) + { + buffer.close(); + kdError(30003) << "Image could not be loaded!" << endl; + return false; + } + buffer.close(); + m_originalImage=imageIO.image(); + + return true; +} + +bool KoPictureImage::save(QIODevice* io) const +{ + kdDebug() << k_funcinfo << "writing raw data. size=" << m_rawData.size() << endl; + // We save the raw data, to avoid damaging the file by many load/save cycles (especially for JPEG) + Q_ULONG size=io->writeBlock(m_rawData); // WARNING: writeBlock returns Q_LONG but size() Q_ULONG! + return (size==m_rawData.size()); +} + +QSize KoPictureImage::getOriginalSize(void) const +{ + return m_originalImage.size(); +} + +QPixmap KoPictureImage::generatePixmap(const QSize& size, bool smoothScale) +{ + scaleAndCreatePixmap(size,!smoothScale); + return m_cachedPixmap; +} + +QString KoPictureImage::getMimeType(const QString& extension) const +{ + QString fileName("/tmp/temp."); + fileName+=extension; + // Find the mimetype only by the extension, not by file content (as the file is empty!) + const QString mimetype( KMimeType::findByPath( fileName, 0 ,true )->name() ); + // ### TODO: use KMimeType::findByContent (but then the mimetype probably need to be cached) + kdDebug(30003) << "Image is mime type: " << mimetype << endl; + return mimetype; +} + +QDragObject* KoPictureImage::dragObject( QWidget *dragSource, const char *name ) +{ + return new QImageDrag( m_originalImage, dragSource, name ); +} + +QImage KoPictureImage::generateImage(const QSize& size) +{ + // We do not cache the image, as we will seldom need it again. + return m_originalImage.smoothScale( size ); +} + +void KoPictureImage::clearCache(void) +{ + m_cachedPixmap.resize(0, 0); + m_cacheIsInFastMode=true; + m_cachedSize=QSize(); +} diff --git a/lib/kofficecore/KoPictureImage.h b/lib/kofficecore/KoPictureImage.h new file mode 100644 index 00000000..98d7d588 --- /dev/null +++ b/lib/kofficecore/KoPictureImage.h @@ -0,0 +1,123 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003, 2004 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPictureImage_h__ +#define __koPictureImage_h__ + +#include "KoPictureBase.h" +#include <qstring.h> + +class KoPictureImagePrivate; +// TODO: fix documentation + +/** + * @internal + * KoPictureImage is a container class for a QImage-based picture + * \todo remove private class, as the header is not installed + */ +class KoPictureImage : public KoPictureBase +{ +public: + /** + * Default constructor. + */ + KoPictureImage(); + + /** + * Destructor. + */ + virtual ~KoPictureImage(); + + KoPictureType::Type getType(void) const; + + KoPictureBase* newCopy(void) const; + + /** + * Returns true if the picture is null. + */ + virtual bool isNull(void) const; + + /** + * @brief Draw the image in a painter. + * + * No, this isn't as simple as painter.drawPixmap(). + * This method ensures that the best quality is used when printing, scaling the painter. + * + * The parameter @p fastMode allows the picture to be re-sized and drawn quicker if possible + * + * The parameters @p width, @p height define the desired size for the image + * Note that the image is being scaled to that size using scale() - except when printing. + * This avoids scaling the image at each paint event. + * + * The other parameters are very similar to QPainter::drawPixmap : + * (@p x, @p y) define the position in the painter, + * (@p sx, @p sy) specify the top-left point in pixmap that is to be drawn. The default is (0, 0). + * (@p sw, @p sh) specify the size of the pixmap that is to be drawn. The default, (-1, -1), means all the way to the bottom + * right of the pixmap. + */ + virtual void draw(QPainter& painter, int x, int y, int width, int height, int sx = 0, int sy = 0, int sw = -1, int sh = -1, bool fastMode = false); + + virtual QDragObject* dragObject( QWidget *dragSource = 0L, const char *name = 0L ); + + virtual bool loadData(const QByteArray& array, const QString& extension); + + virtual bool save(QIODevice* io) const; + + virtual QSize getOriginalSize(void) const; + + virtual QPixmap generatePixmap(const QSize& size, bool smoothScale = false); + + virtual QString getMimeType(const QString& extension) const; + + /** + * Generate a QImage + * (always in slow mode) + */ + virtual QImage generateImage(const QSize& size); + + virtual bool hasAlphaBuffer() const + { return m_originalImage.hasAlphaBuffer(); } + + virtual void setAlphaBuffer(bool enable) + { m_originalImage.setAlphaBuffer(enable); } + + virtual QImage createAlphaMask(int conversion_flags = 0) const + { return m_originalImage.createAlphaMask(conversion_flags); } + + virtual void clearCache(void); + +protected: + QPixmap getPixmap(QImage& image); + void scaleAndCreatePixmap(const QSize& size, bool fastMode=false); + +private: + QImage m_originalImage; ///< Image as Qimage + QByteArray m_rawData; ///< Binary copy of the loaded image file + QPixmap m_cachedPixmap; ///< Cached pixmap + QSize m_cachedSize; ///< size of the currently cached pixmap @see m_cachedPixmap + /** + * true, if the last cached image was done using fast mode. + * false, if the last cached image was done using slow mode. + */ + bool m_cacheIsInFastMode; + class Private; + Private* d; +}; + +#endif /* __koPictureImage_h__ */ diff --git a/lib/kofficecore/KoPictureKey.cpp b/lib/kofficecore/KoPictureKey.cpp new file mode 100644 index 00000000..ef60cfed --- /dev/null +++ b/lib/kofficecore/KoPictureKey.cpp @@ -0,0 +1,151 @@ + +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright 2002 Nicolas GOUTTE <goutte@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 <qdatetime.h> +#include <qfileinfo.h> +#include <qdom.h> + +#include <kdebug.h> + +#include "KoPictureKey.h" + +static void resetDateTimeToEpoch(QDateTime& dt) +{ + // set the time point to 1970-01-01 + dt.setDate(QDate(1970,1,1)); + dt.setTime(QTime(0,0)); + // Note: we cannot use QDateTime;;setTime_t as it makes a local time correction! (### TODO: not true anymore with recent Qt versions) +} + +KoPictureKey::KoPictureKey() +{ + resetDateTimeToEpoch(m_lastModified); +} + +KoPictureKey::KoPictureKey( const QString &fn, const QDateTime &mod ) + : m_filename( fn ), m_lastModified( mod ) +{ + if (!m_lastModified.isValid()) + { + // As we have an invalid date, set the time point to 1970-01-01 + resetDateTimeToEpoch(m_lastModified); + } +} + +KoPictureKey::KoPictureKey( const QString &fn ) + : m_filename( fn ) +{ + resetDateTimeToEpoch(m_lastModified); +} + +KoPictureKey::KoPictureKey( const KoPictureKey &key ) + : m_filename( key.m_filename ), m_lastModified( key.m_lastModified ) +{ +} + +KoPictureKey& KoPictureKey::operator=( const KoPictureKey &key ) +{ + m_filename = key.m_filename; + m_lastModified = key.m_lastModified; + return *this; +} + +bool KoPictureKey::operator==( const KoPictureKey &key ) const +{ + return ( key.m_filename == m_filename && + key.m_lastModified == m_lastModified ); +} + +bool KoPictureKey::operator<( const KoPictureKey &key ) const +{ + return key.toString() < toString(); +} + +void KoPictureKey::saveAttributes( QDomElement &elem ) const +{ + QDate date = m_lastModified.date(); + QTime time = m_lastModified.time(); + elem.setAttribute( "filename", m_filename ); + elem.setAttribute( "year", date.year() ); + elem.setAttribute( "month", date.month() ); + elem.setAttribute( "day", date.day() ); + elem.setAttribute( "hour", time.hour() ); + elem.setAttribute( "minute", time.minute() ); + elem.setAttribute( "second", time.second() ); + elem.setAttribute( "msec", time.msec() ); +} + +void KoPictureKey::loadAttributes( const QDomElement &elem ) +{ + // Default date/time is the *nix epoch: 1970-01-01 00:00:00,000 + int year=1970, month=1, day=1; + int hour=0, minute=0, second=0, msec=0; // We must initialize to zero, as not all compilers are C99-compliant + + if( elem.hasAttribute( "key" ) ) + { + // Note: the old KWord format (up to 1.1-beta2) has no date/time + m_filename=elem.attribute( "key" ); + } + else + { + // ### TODO: document which format is this? + m_filename=elem.attribute( "filename" ); + } + + if( elem.hasAttribute( "year" ) ) + year=elem.attribute( "year" ).toInt(); + if( elem.hasAttribute( "month" ) ) + month=elem.attribute( "month" ).toInt(); + if( elem.hasAttribute( "day" ) ) + day=elem.attribute( "day" ).toInt(); + if( elem.hasAttribute( "hour" ) ) + hour=elem.attribute( "hour" ).toInt(); + if( elem.hasAttribute( "minute" ) ) + minute=elem.attribute( "minute" ).toInt(); + if( elem.hasAttribute( "second" ) ) + second=elem.attribute( "second" ).toInt(); + if( elem.hasAttribute( "msec" ) ) + msec=elem.attribute( "msec" ).toInt(); + + m_lastModified.setDate( QDate( year, month, day ) ); + m_lastModified.setTime( QTime( hour, minute, second, msec ) ); + + if (!m_lastModified.isValid()) + { + // If the date/time is not valid, make it valid by force! + kdWarning(30003) << "Correcting invalid date/time: " << toString() << " (in KoPictureKey::loadAttributes)" << endl; + resetDateTimeToEpoch(m_lastModified); + } +} + +QString KoPictureKey::toString() const +{ + // We do not use the default QDateTime::toString has it does not show microseconds + return QString::fromLatin1( "%1 %2" ) + .arg( m_filename, m_lastModified.toString("yyyy-MM-dd hh:mm:ss.zzz") ); +} + +void KoPictureKey::setKeyFromFile (const QString& filename) +{ + QFileInfo inf(filename); + m_filename = filename; + m_lastModified = inf.lastModified(); +} diff --git a/lib/kofficecore/KoPictureKey.h b/lib/kofficecore/KoPictureKey.h new file mode 100644 index 00000000..d336215b --- /dev/null +++ b/lib/kofficecore/KoPictureKey.h @@ -0,0 +1,154 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2004 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPictureKey_h__ +#define __koPictureKey_h__ + +#include <qstring.h> +#include <qdatetime.h> +#include <koffice_export.h> +/** + * \file koPictureKey.h + * \todo correct documentation (for example: sed "s/image/picture/g") + */ + +class QDomElement; + +namespace KoPictureType +{ + /** + * QPicture version used by KoPictureClipart + * + * Possible values: + * \li 3 for Qt 2.1.x and later Qt 2.x + * \li 4 for Qt 3.0 + * \li 5 for Qt 3.1 and later Qt 3.x + * \li -1 for current Qt + */ + const int formatVersionQPicture=-1; + + enum Type + { + TypeUnknown = 0, ///< Unknown or not-an-image @see KoPictureBase + TypeImage, ///< Image, QImage-based @see KoPictureImage + TypeEps, ///< Encapsulated Postscript @see KoPictureEps + TypeClipart, ///< Clipart, QPicture-based @see KoPictureClipart + TypeWmf ///< WMF (Windows Meta File) @see KoPictureWmf + }; +} + +/** + * KoPictureKey is the structure describing a picture in a unique way. + * It currently includes the original path to the picture and the modification + * date. + * + * @short Structure describing a picture on disk + * + * @note We use the *nix epoch (1970-01-01) as a time base because it is a valid date. + * That way we do not depend on a behaviour of the current QDateTime that might change in future versions of Qt + * and we are also nice to non-Qt programs wanting to read KOffice's files. + * + * @note This behaviour is also needed for re-saving KWord files having \<FORMAT id="2"\>. When saving again, + * these files get a \<KEY\> element as child of \<PIXMAPS\> but not one as child of \<FORMAT\> and \<IMAGE\>. + * Therefore we need to be careful that the key remains compatible to default values + * (another good reason for the *NIX epoch) + * + * @note In case of a remote path, the "original path" is the name of the temporary file that was + * used to download the file. + */ +class KOFFICEUI_EXPORT KoPictureKey +{ +public: + /** + * Default constructor. Creates a null key + */ + KoPictureKey(); + + /** + * @brief Constructs a key, from a filename and a modification date + * + * Storing the modification date as part of the key allows the user + * to update the file and import it into the application again, without + * the application reusing the old copy from the collection. + */ + KoPictureKey( const QString &fn, const QDateTime &mod ); + + /** + * Constructs a key from a filename + * @note The modification date is set to 1970-01-01 + */ + KoPictureKey( const QString &fn ); + + /** + * Copy constructor + */ + KoPictureKey( const KoPictureKey &key ); + + /** + * Assignment operator + */ + KoPictureKey &operator=( const KoPictureKey &key ); + + /** + * Comparison operator + */ + bool operator==( const KoPictureKey &key ) const; + + /** + * Comparison operator + * @note Used for sorting in the collection's map + */ + bool operator<( const KoPictureKey &key ) const; + + /** + * Convert this key into a string representation of it + */ + QString toString() const; + + /** + * Save this key in XML (as %KOffice 1.3) + */ + void saveAttributes( QDomElement &elem ) const; + + /** + * Load this key from XML (as %KOffice 1.3) + */ + void loadAttributes( const QDomElement &elem ); + + /** + * First part of the key: the filename + */ + QString filename() const { return m_filename; } + + /** + * Second part of the key: the modification date + */ + QDateTime lastModified() const { return m_lastModified; } + + /** + * Sets the key according to @p filename, including modification time + */ + void setKeyFromFile (const QString& filename); + +protected: + QString m_filename; + QDateTime m_lastModified; +}; + +#endif /* __koPictureKey_h__ */ diff --git a/lib/kofficecore/KoPictureShared.cpp b/lib/kofficecore/KoPictureShared.cpp new file mode 100644 index 00000000..5c2ff2a2 --- /dev/null +++ b/lib/kofficecore/KoPictureShared.cpp @@ -0,0 +1,535 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003, 2004 Nicolas GOUTTE <goutte@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 <qpainter.h> +#include <qfile.h> + +#include <kdebug.h> +#include <kurl.h> +#include <kfilterdev.h> +#include <kio/netaccess.h> + +#include "KoPictureKey.h" +#include "KoPictureBase.h" +#include "KoPictureImage.h" +#include "KoPictureEps.h" +#include "KoPictureClipart.h" +#include "KoPictureWmf.h" +#include "KoPictureShared.h" +#include <kmdcodec.h> + + +KoPictureShared::KoPictureShared(void) : m_base(NULL) +{ +} + +void KoPictureShared::assignPictureId( uint _id) +{ + m_pictureId = _id; +} + +QString KoPictureShared::uniquePictureId() const +{ + return "Pictures"+ QString::number(m_pictureId); +} + +KoPictureShared::~KoPictureShared(void) +{ + delete m_base; +} + +KoPictureShared::KoPictureShared(const KoPictureShared &other) + : QShared() // Some compilers want it explicitly! +{ + // We need to use newCopy, because we want a real copy, not just a copy of the part of KoPictureBase + if (other.m_base) + m_base=other.m_base->newCopy(); + else + m_base=NULL; +} + +KoPictureShared& KoPictureShared::operator=( const KoPictureShared &other ) +{ + clear(); + kdDebug(30003) << "KoPictureShared::= before" << endl; + if (other.m_base) + m_base=other.m_base->newCopy(); + kdDebug(30003) << "KoPictureShared::= after" << endl; + return *this; +} + +KoPictureType::Type KoPictureShared::getType(void) const +{ + if (m_base) + return m_base->getType(); + return KoPictureType::TypeUnknown; +} + +bool KoPictureShared::isNull(void) const +{ + if (m_base) + return m_base->isNull(); + return true; +} + +void KoPictureShared::draw(QPainter& painter, int x, int y, int width, int height, int sx, int sy, int sw, int sh, bool fastMode) +{ + if (m_base) + m_base->draw(painter, x, y, width, height, sx, sy, sw, sh, fastMode); + else + { + // Draw a red box (easier DEBUG) + kdWarning(30003) << "Drawing red rectangle! (KoPictureShared::draw)" << endl; + painter.save(); + painter.setBrush(QColor(255,0,0)); + painter.drawRect(x,y,width,height); + painter.restore(); + } +} + +bool KoPictureShared::loadWmf(QIODevice* io) +{ + kdDebug(30003) << "KoPictureShared::loadWmf" << endl; + if (!io) + { + kdError(30003) << "No QIODevice!" << endl; + return false; + } + + clear(); + + // The extension .wmf was used (KOffice 1.1.x) for QPicture files + // For an extern file or in the storage, .wmf can mean a real Windows Meta File. + + QByteArray array ( io->readAll() ); + + if ((array[0]=='Q') && (array[1]=='P') &&(array[2]=='I') && (array[3]=='C')) + { + m_base=new KoPictureClipart(); + setExtension("qpic"); + } + else + { + m_base=new KoPictureWmf(); + setExtension("wmf"); + } + return m_base->loadData(array, m_extension); +} + +bool KoPictureShared::loadTmp(QIODevice* io) +// We have a temp file, probably from a downloaded file +// We must check the file type +{ + kdDebug(30003) << "KoPictureShared::loadTmp" << endl; + if (!io) + { + kdError(30003) << "No QIODevice!" << endl; + return false; + } + + // The extension .wmf was used (KOffice 1.1.x) for QPicture files + // For an extern file or in the storage, .wmf can mean a real Windows Meta File. + + QByteArray array ( io->readAll() ); + return identifyAndLoad( array ); +} + +bool KoPictureShared::identifyAndLoad( QByteArray array ) +{ + if ( array.size() < 5 ) + { + kdError(30003) << "Picture is less than 5 bytes long!" << endl; + return false; + } + + QString strExtension; + bool flag=false; + + // Try to find the file type by comparing magic on the first few bytes! + // ### TODO: could not QImageIO::imageFormat do it too? (At least most of them?) + if ((array[0]==char(0x89)) && (array[1]=='P') &&(array[2]=='N') && (array[3]=='G')) + { + strExtension="png"; + } + else if ((array[0]==char(0xff)) && (array[1]==char(0xd8)) &&(array[2]==char(0xff)) && (array[3]==char(0xe0))) + { + strExtension="jpeg"; + } + else if ((array[0]=='B') && (array[1]=='M')) + { + strExtension="bmp"; + } + else if ((array[0]==char(0xd7)) && (array[1]==char(0xcd)) &&(array[2]==char(0xc6)) && (array[3]==char(0x9a))) + { + strExtension="wmf"; + } + else if ((array[0]=='<') && (array[1]=='?') && ( array[2]=='x' ) && (array[3]=='m') && ( array[4]=='l' ) ) + { + strExtension="svg"; + } + else if ((array[0]=='Q') && (array[1]=='P') &&(array[2]=='I') && (array[3]=='C')) + { + strExtension="qpic"; + } + else if ((array[0]=='%') && (array[1]=='!') &&(array[2]=='P') && (array[3]=='S')) + { + strExtension="eps"; + } + else if ((array[0]==char(0xc5)) && (array[1]==char(0xd0)) && (array[2]==char(0xd3)) && (array[3]==char(0xc6))) + { + // So called "MS-DOS EPS file" + strExtension="eps"; + } + else if ((array[0]=='G') && (array[1]=='I') && (array[2]=='F') && (array[3]=='8')) + { + // GIF (87a or 89a) + strExtension="gif"; + } + else if ( ( array[0] == char( 0037 ) ) && ( array[1] == char( 0213 ) ) ) + { + // Gzip + QBuffer buffer(array); + buffer.open(IO_ReadOnly); + + const bool flag = loadCompressed( &buffer, "application/x-gzip", "tmp" ); + buffer.close(); + return flag; + } + else if ( ( array[0] == 'B' ) && ( array[1] == 'Z' ) && ( array[2] == 'h') ) + { + // BZip2 + QBuffer buffer(array); + buffer.open(IO_ReadOnly); + const bool flag = loadCompressed( &buffer, "application/x-bzip2", "tmp" ); + buffer.close(); + return flag; + } + else + { + kdDebug(30003) << "Cannot identify the type of temp file!" + << " Trying to convert to PNG! (in KoPictureShared::loadTmp" << endl; + + // Do not trust QBuffer and do not work directly on the QByteArray array + // DF: It would be faster to work on array here, and to create a completely + // different QBuffer for the writing code! + QBuffer buf( array.copy() ); + if (!buf.open(IO_ReadOnly)) + { + kdError(30003) << "Could not open read buffer!" << endl; + return false; + } + + QImageIO imageIO(&buf,NULL); + + if (!imageIO.read()) + { + kdError(30003) << "Could not read image!" << endl; + return false; + } + + buf.close(); + + if ( !buf.open( IO_WriteOnly | IO_Truncate ) ) + { + kdError(30003) << "Could not open write buffer!" << endl; + return false; + } + + imageIO.setIODevice(&buf); + imageIO.setFormat("PNG"); + + if (!imageIO.write()) + { + kdError(30003) << "Could not write converted image!" << endl; + return false; + } + buf.close(); + + array = buf.buffer(); + + strExtension="png"; + } + + kdDebug(30003) << "Temp file considered to be " << strExtension << endl; + + clearAndSetMode(strExtension); + if (m_base) + flag = m_base->loadData(array,strExtension); + setExtension(strExtension); + + return flag; +} + + + +bool KoPictureShared::loadXpm(QIODevice* io) +{ + kdDebug(30003) << "KoPictureShared::loadXpm" << endl; + if (!io) + { + kdError(30003) << "No QIODevice!" << endl; + return false; + } + + clear(); + + // Old KPresenter XPM files have char(1) instead of some " + // Therefore we need to treat XPM separately + + QByteArray array=io->readAll(); + + // As XPM files are normally only ASCII files, we can replace it without problems + + int pos=0; + + while ((pos=array.find(char(1),pos))!=-1) + { + array[pos]='"'; + } + + // Now that the XPM file is corrected, we need to load it. + + m_base=new KoPictureImage(); + + QBuffer buffer(array); + bool check = m_base->load(&buffer,"xpm"); + setExtension("xpm"); + return check; +} + +bool KoPictureShared::save(QIODevice* io) const +{ + if (!io) + return false; + if (m_base) + return m_base->save(io); + return false; +} + +bool KoPictureShared::saveAsBase64( KoXmlWriter& writer ) const +{ + if ( m_base ) + m_base->saveAsBase64( writer ); + return false; +} + +void KoPictureShared::clear(void) +{ + // Clear does not reset the key m_key! + delete m_base; + m_base=NULL; +} + +void KoPictureShared::clearAndSetMode(const QString& newMode) +{ + delete m_base; + m_base=NULL; + + const QString mode=newMode.lower(); + + if ((mode=="svg") || (mode=="qpic")) + { + m_base=new KoPictureClipart(); + } + else if (mode=="wmf") + { + m_base=new KoPictureWmf(); + } + else if ( (mode=="eps") || (mode=="epsi") || (mode=="epsf") ) + { + m_base=new KoPictureEps(); + } + else + { // TODO: test if QImageIO really knows the file format + m_base=new KoPictureImage(); + } +} + +QString KoPictureShared::getExtension(void) const +{ + return m_extension; +} + +void KoPictureShared::setExtension(const QString& extension) +{ + m_extension = extension; +} + +QString KoPictureShared::getMimeType(void) const +{ + if (m_base) + return m_base->getMimeType(m_extension); + return QString(NULL_MIME_TYPE); +} + + +bool KoPictureShared::loadFromBase64( const QCString& str ) +{ + clear(); + QByteArray data; + KCodecs::base64Decode( str, data ); + return identifyAndLoad( data ); +} + +bool KoPictureShared::load(QIODevice* io, const QString& extension) +{ + kdDebug(30003) << "KoPictureShared::load(QIODevice*, const QString&) " << extension << endl; + bool flag=false; + QString ext(extension.lower()); + if (ext=="wmf") + flag=loadWmf(io); + else if (ext=="tmp") // ### TODO: also remote scripts need this, don't they? + flag=loadTmp(io); + else if ( ext == "bz2" ) + { + flag = loadCompressed( io, "application/x-bzip2", "tmp" ); + } + else if ( ext == "gz" ) + { + flag = loadCompressed( io, "application/x-gzip", "tmp" ); + } + else if ( ext == "svgz" ) + { + flag = loadCompressed( io, "application/x-gzip", "svg" ); + } + else + { + clearAndSetMode(ext); + if (m_base) + flag = m_base->load(io, ext); + setExtension(ext); + } + if (!flag) + { + kdError(30003) << "File was not loaded! (KoPictureShared::load)" << endl; + } + return flag; +} + +bool KoPictureShared::loadFromFile(const QString& fileName) +{ + kdDebug(30003) << "KoPictureShared::loadFromFile " << fileName << endl; + if ( fileName.isEmpty() ) + { + kdError(30003) << "Cannot load file with empty name!" << endl; + return false; + } + QFile file(fileName); + if (!file.open(IO_ReadOnly)) + return false; + + bool flag = false; + const int pos=fileName.findRev('.'); + if (pos==-1) + { + kdDebug(30003) << "File with no extension!" << endl; + // As we have no extension, consider it like a temporary file + flag = loadTmp( &file ); + } + else + { + const QString extension( fileName.mid( pos+1 ) ); + // ### TODO: check if the extension if gz or bz2 and find the previous extension + flag = load( &file, extension ); + } + file.close(); + return flag; +} + +QSize KoPictureShared::getOriginalSize(void) const +{ + if (m_base) + return m_base->getOriginalSize(); + return QSize(0,0); +} + +QPixmap KoPictureShared::generatePixmap(const QSize& size, bool smoothScale) +{ + if (m_base) + return m_base->generatePixmap(size, smoothScale); + return QPixmap(); +} + +QDragObject* KoPictureShared::dragObject( QWidget *dragSource, const char *name ) +{ + if (m_base) + return m_base->dragObject( dragSource, name ); + return 0L; +} + +QImage KoPictureShared::generateImage(const QSize& size) +{ + if (m_base) + return m_base->generateImage( size ); + return QImage(); +} + +bool KoPictureShared::hasAlphaBuffer() const +{ + if (m_base) + return m_base->hasAlphaBuffer(); + return false; +} + +void KoPictureShared::setAlphaBuffer(bool enable) +{ + if (m_base) + m_base->setAlphaBuffer(enable); +} + +QImage KoPictureShared::createAlphaMask(int conversion_flags) const +{ + if (m_base) + return m_base->createAlphaMask(conversion_flags); + return QImage(); +} + +void KoPictureShared::clearCache(void) +{ + if (m_base) + m_base->clearCache(); +} + +bool KoPictureShared::loadCompressed( QIODevice* io, const QString& mimeType, const QString& extension ) +{ + // ### TODO: check that we do not have an endless recursion + QIODevice* in = KFilterDev::device( io, mimeType, false); + + if ( !in ) + { + kdError(30003) << "Cannot create device for uncompressing! Aborting!" << endl; + return false; + } + + + if ( !in->open( IO_ReadOnly ) ) + { + kdError(30003) << "Cannot open file for uncompressing! Aborting!" << endl; + delete in; + return false; + } + + const bool flag = load( in, extension ); + + in->close(); + delete in; + + return flag; +} diff --git a/lib/kofficecore/KoPictureShared.h b/lib/kofficecore/KoPictureShared.h new file mode 100644 index 00000000..da286368 --- /dev/null +++ b/lib/kofficecore/KoPictureShared.h @@ -0,0 +1,214 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002, 2003, 2004 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPictureShared_h__ +#define __koPictureShared_h__ + +#include <qshared.h> +#include <qstring.h> +#include <qiodevice.h> +#include <qpixmap.h> + +#include "KoPictureKey.h" + +class KoXmlWriter; +class QPainter; +class QSize; + +class KURL; + +class KoPictureBase; + +/** + * @internal + * KoPictureShared is the class that contains the shared part for KoPicture + * + * @warning As with all QShared objects, the sharing is neither automatic nor transparent! + */ +class KoPictureShared : public QShared +{ +public: + /** + * Default constructor. + */ + KoPictureShared(void); + + /** + * Destructor. + */ + ~KoPictureShared(void); + + /** + * @brief Copy constructor + * + * This makes a deep copy. Do not use if you want to share! + */ + KoPictureShared(const KoPictureShared &other); + + /** + * @brief Assignment operator + * + * This makes a deep copy. Do not use if you want to share! + */ + KoPictureShared& operator=(const KoPictureShared& other); + + KoPictureType::Type getType(void) const; + + /** + * Returns true if the picture is null. + */ + bool isNull(void) const; + + /** + * @brief Draw the image in a painter. + * + * The parameter @p fastMode allows the picture to be re-sized and drawn quicker if possible + * + * The parameters @p width, @p height define the desired size for the picture + * + * The other parameters are very similar to QPainter::drawPixmap : + * (@p x, @p y) define the position in the painter, + * (@p sx, @p sy) specify the top-left point in pixmap that is to be drawn. The default is (0, 0). + * (@p sw, @p sh) specify the size of the pixmap that is to be drawn. The default, (-1, -1), means all the way to the bottom + * right of the pixmap. + */ + void draw(QPainter& painter, int x, int y, int width, int height, int sx = 0, int sy = 0, int sw = -1, int sh = -1, bool fastMode = false); + + /** + * Create a dragobject containing this picture. + * @param dragSource must be 0 when copying to the clipboard + * @return 0L if the picture is null! + */ + QDragObject* dragObject( QWidget *dragSource = 0L, const char *name = 0L ); + + bool load(QIODevice* io, const QString& extension); + bool loadFromBase64( const QCString& str ); + + /** + * Save picture into a QIODevice + * @param io QIODevice used for saving + */ + bool save(QIODevice* io) const; + + /** + * OASIS FlatXML support: + * Save picture as base64-encoded data into an XML writer. + */ + bool saveAsBase64( KoXmlWriter& writer ) const; + + void setExtension(const QString& extension); + + QString getExtension(void) const; + + QSize getOriginalSize(void) const; + + /** + * Clear and set the mode of this KoPictureShared + * + * @param newMode file extension (like "png") giving the wanted mode + */ + void clearAndSetMode(const QString& newMode); + + /** + * Reset the KoPictureShared (but not the key!) + */ + void clear(void); + + /** + * Load a file + * + * @param fileName the name of the file to load + */ + bool loadFromFile(const QString& fileName); + + /** + * Load a potentially broken XPM file (for KPresenter) + */ + bool loadXpm(QIODevice* io); + + /** + * @deprecated + * Returns a QPixmap from an image + * + * @param size the wanted size for the QPixmap + */ + QPixmap generatePixmap(const QSize& size, bool smoothScale = false); + + QString getMimeType(void) const; + + /** + * Generate a QImage + * + * (always in slow mode) + * + * @param size the wanted size for the QImage + */ + QImage generateImage(const QSize& size); + + bool hasAlphaBuffer() const; + void setAlphaBuffer(bool enable); + QImage createAlphaMask(int conversion_flags = 0) const; + + /** + * Clear any cache + * + * It is used to avoid using too much memory + * especially if the application somehow caches the KoPicture too. + */ + void clearCache(void); + + QString uniquePictureId() const; + void assignPictureId( uint _id); + +protected: + /** + * @internal + * Load a WMF file + * \note In %KOffice 1.1, a .wmf file was a QPicture file + */ + bool loadWmf(QIODevice* io); + + /** + * @internal + * Loads a temporary file, probably from a downloaded file + */ + bool loadTmp(QIODevice* io); + + /// Find type of image, create base accordingly, and load data + bool identifyAndLoad( QByteArray data ); + + /** + * @internal + * Loads a compressed file + * + * @warning risk of endless recurision, be careful when it is called from @see load + * + * @param io QIODevice of the compressed file/strea, + * @param mimeType mimetype of the (de-)compressor + * @param extension extension of the uncompressed file + */ + bool loadCompressed( QIODevice* io, const QString& mimeType, const QString& extension ); + +protected: + KoPictureBase* m_base; + QString m_extension; + uint m_pictureId; +}; + +#endif /* __koPictureShared_h__ */ diff --git a/lib/kofficecore/KoPictureWmf.cpp b/lib/kofficecore/KoPictureWmf.cpp new file mode 100644 index 00000000..96872af7 --- /dev/null +++ b/lib/kofficecore/KoPictureWmf.cpp @@ -0,0 +1,137 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (c) 2001 David Faure <faure@kde.org> + Copyright (C) 2002 Nicolas GOUTTE <goutte@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 <qbuffer.h> +#include <qpainter.h> +#include <qpicture.h> +#include <qpixmap.h> + +#include <kdebug.h> +#include <kdeversion.h> +#if ! KDE_IS_VERSION( 3,1,90 ) +#include <kdebugclasses.h> +#endif + +#include <kowmfpaint.h> +#include "KoPictureKey.h" +#include "KoPictureBase.h" +#include "KoPictureWmf.h" + +KoPictureWmf::KoPictureWmf(void) : m_clipart(KoPictureType::formatVersionQPicture) +{ +} + +KoPictureWmf::~KoPictureWmf(void) +{ +} + +KoPictureBase* KoPictureWmf::newCopy(void) const +{ + return new KoPictureWmf(*this); +} + +KoPictureType::Type KoPictureWmf::getType(void) const +{ + return KoPictureType::TypeWmf; +} + +bool KoPictureWmf::isNull(void) const +{ + return m_clipart.isNull(); +} + +void KoPictureWmf::drawQPicture(QPicture& clipart, QPainter& painter, + int x, int y, int width, int height, int sx, int sy, int sw, int sh) +{ + kdDebug(30003) << "Drawing KoPictureWmf " << this << endl; + kdDebug(30003) << " x=" << x << " y=" << y << " width=" << width << " height=" << height << endl; + kdDebug(30003) << " sx=" << sx << " sy=" << sy << " sw=" << sw << " sh=" << sh << endl; + painter.save(); + // Thanks to Harri, Qt3 makes it much easier than Qt2 ;) + QRect br = clipart.boundingRect(); + kdDebug(30003) << " Bounding rect. " << br << endl; + + painter.translate(x,y); // Translating must be done before scaling! + if ( br.width() && br.height() ) + painter.scale(double(width)/double(br.width()),double(height)/double(br.height())); + else + kdWarning(30003) << "Null bounding rectangle: " << br.width() << " x " << br.height() << endl; + painter.drawPicture(0,0,clipart); + painter.restore(); +} + +void KoPictureWmf::draw(QPainter& painter, int x, int y, int width, int height, int sx, int sy, int sw, int sh, bool /*fastMode*/) +{ + drawQPicture(m_clipart, painter, x, y, width, height, sx, sy, sw, sh); +} + +bool KoPictureWmf::loadData(const QByteArray& array, const QString& /* extension */) +{ + // Second, create the original clipart + kdDebug(30003) << "Trying to load clipart... (Size:" << array.size() << ")" << endl; + m_rawData=array; + + KoWmfPaint wmf; + if (!wmf.load(array)) + { + kdWarning(30003) << "Loading WMF has failed! (KoPictureWmf::load)" << endl; + return false; + } + m_originalSize = wmf.boundingRect().size(); + // draw wmf file with relative coordinate + wmf.play(m_clipart, true); + + return true; +} + +bool KoPictureWmf::save(QIODevice* io) const +{ + // We save the raw data, as the SVG supposrt in QPicture is poor + Q_ULONG size=io->writeBlock(m_rawData); // WARNING: writeBlock returns Q_LONG but size() Q_ULONG! + return (size==m_rawData.size()); +} + +QSize KoPictureWmf::getOriginalSize(void) const +{ + return m_originalSize; +} + +QPixmap KoPictureWmf::generatePixmap(const QSize& size, bool /*smoothScale*/) +{ + // Not sure if it works, but it worked for KoPictureFilePreviewWidget::setClipart + QPixmap pixmap(size); + QPainter p; + + p.begin( &pixmap ); + p.setBackgroundColor( Qt::white ); + pixmap.fill( Qt::white ); + + if ( m_originalSize.width() && m_originalSize.height() ) + p.scale( (double)pixmap.width() / (double)m_originalSize.width(), (double)pixmap.height() / (double)m_originalSize.height() ); + p.drawPicture( m_clipart ); + p.end(); + return pixmap; +} + +QString KoPictureWmf::getMimeType(const QString& /* extension */) const +{ + return "image/x-wmf"; +} diff --git a/lib/kofficecore/KoPictureWmf.h b/lib/kofficecore/KoPictureWmf.h new file mode 100644 index 00000000..f1924d9c --- /dev/null +++ b/lib/kofficecore/KoPictureWmf.h @@ -0,0 +1,108 @@ +/* This file is part of the KDE project + Copyright (c) 2001 Simon Hausmann <hausmann@kde.org> + Copyright (C) 2002 Nicolas GOUTTE <goutte@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. +*/ +#ifndef __koPictureWmf_h__ +#define __koPictureWmf_h__ + +#include <qstring.h> +#include <qpicture.h> +#include <koffice_export.h> + +class QPainter; +class QSize; + +/** + * \file koPictureWmf.h + * \todo fix documentation + */ + +/** + * @internal + * KoPictureWmf is a container class for WMF pictures. + */ +class KOFFICECORE_EXPORT KoPictureWmf : public KoPictureBase +{ +public: + /** + * Default constructor. + */ + KoPictureWmf(); + + /** + * Destructor. + */ + virtual ~KoPictureWmf(); + + virtual KoPictureType::Type getType(void) const; + + virtual KoPictureBase* newCopy(void) const; + + + /** + * Returns true if the picture is null. + */ + virtual bool isNull(void) const; + + /** + * @brief Draw the image in a painter. + * + * The parameter @p fastMode allows the picture to be re-sized and drawn quicker if possible + * + * The parameters @p width, @p height define the desired size for the image + * Note that the image is being scaled to that size using scale() - except when printing. + * This avoids scaling the image at each paint event. + * + * The other parameters are very similar to QPainter::drawPixmap : + * (@p x, @p y) define the position in the painter, + * (@p sx, @p sy) specify the top-left point in pixmap that is to be drawn. The default is (0, 0). + * (@p sw, @p sh) specify the size of the pixmap that is to be drawn. The default, (-1, -1), means all the way to the bottom + * right of the pixmap. + */ + virtual void draw(QPainter& painter, int x, int y, int width, int height, int sx = 0, int sy = 0, int sw = -1, int sh = -1, bool fastMode = false); + + virtual bool loadData(const QByteArray& array, const QString& extension); + + /** + * save file + * @param io QIODevice used for saving + */ + virtual bool save(QIODevice* io) const; + + virtual QSize getOriginalSize(void) const; + + virtual QPixmap generatePixmap(const QSize& size, bool smoothScale = false); + + virtual QString getMimeType(const QString& extension) const; + +protected: + QPixmap getPixmap(QImage& image); + /** + * @internal + * Draw a QPicture + */ + void drawQPicture(QPicture& clipart, QPainter& painter, + int x, int y, int width, int height, int sx, int sy, int sw, int sh); +protected: + QPicture m_clipart; + QByteArray m_rawData; + QSize m_size; + QSize m_originalSize; +}; + +#endif /* __koPictureWmf_h__ */ diff --git a/lib/kofficecore/KoPoint.h b/lib/kofficecore/KoPoint.h new file mode 100644 index 00000000..5f39e6c6 --- /dev/null +++ b/lib/kofficecore/KoPoint.h @@ -0,0 +1,128 @@ +/* This file is part of the KDE project + Copyright (C) 2001 David Faure <faure@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. +*/ + +#ifndef koPoint_h +#define koPoint_h + +#include <qwmatrix.h> +#include <math.h> + +/** + * A point whose coordinates are floating-point values ( "double"s ). + * The API isn't documented, it's a perfect mirror of QPoint. + */ +class KoPoint { + +public: + KoPoint() { m_x = 0; m_y = 0; } + KoPoint(const double &x, const double &y) : m_x(x), m_y(y) {} + explicit KoPoint(const QPoint & p) : m_x(p.x()), m_y(p.y()) {} + ~KoPoint() {} + + bool operator==(const KoPoint &rhs) const { return QABS(m_x-rhs.x()) < 1E-10 && QABS(m_y-rhs.y()) < 1E-10; } + bool operator!=(const KoPoint &rhs) const { return QABS(m_x-rhs.x()) > 1E-10 || QABS(m_y-rhs.y()) > 1E-10; } + + bool isNull() const { return m_x == 0 && m_y == 0; } + + double x() const { return m_x; } + double y() const { return m_y; } + void setX(const double &x) { m_x = x; } + void setY(const double &y) { m_y = y; } + + double &rx() { return m_x; } + double &ry() { return m_y; } + + KoPoint &operator=(const KoPoint &rhs) { m_x = rhs.x(); m_y = rhs.y(); return *this; } + KoPoint &operator+=( const KoPoint &rhs ) { m_x += rhs.x(); m_y += rhs.y(); return *this; } + KoPoint &operator-=( const KoPoint &rhs ) { m_x -= rhs.x(); m_y -= rhs.y(); return *this; } + KoPoint &operator*=( const double &c ) { m_x *= c; m_y *= c; return *this; } + + friend inline KoPoint operator+( const KoPoint &, const KoPoint & ); + friend inline KoPoint operator-( const KoPoint &, const KoPoint & ); + friend inline KoPoint operator*( const KoPoint &, const double & ); + friend inline KoPoint operator*( const double &, const KoPoint & ); + friend inline double operator*( const KoPoint &a, const KoPoint &b ); + + // Not in QPoint: + void setCoords(const double &x, const double &y) { m_x = x; m_y = y; } + KoPoint transform (const QWMatrix &m) const + { + double x, y; + m.map(m_x, m_y, &x, &y); + return KoPoint(x, y); + }; + + bool isNear(const KoPoint &p, double range) const + { + return (p.x() >= m_x - range && p.x() <= m_x + range && p.y() >= m_y - range && p.y() <= m_y + range); + } + + static double getAngle( const KoPoint& p1, const KoPoint& p2 ) { + double a = atan2( p2.x() - p1.x(), p2.y() - p1.y() ) + M_PI; + return ( ( - ( a * 360 ) / ( 2 * M_PI ) - 90 ) - 180 ); + } + + double manhattanLength() const + { + return QABS( m_x ) + QABS( m_y ); + } + + /// Convert to a QPoint - with precision loss! + QPoint toQPoint() const + { + return QPoint( qRound( m_x ), qRound( m_y ) ); + } + +private: + double m_x, m_y; +}; + +inline KoPoint operator+( const KoPoint &p1, const KoPoint &p2 ) +{ return KoPoint( p1.m_x+p2.m_x, p1.m_y+p2.m_y ); } + +inline KoPoint operator-( const KoPoint &p1, const KoPoint &p2 ) +{ return KoPoint( p1.m_x-p2.m_x, p1.m_y-p2.m_y ); } + +inline KoPoint operator*( const KoPoint &p, const double &c ) +{ return KoPoint( p.m_x*c, p.m_y*c ); } + +inline KoPoint operator*( const double &c, const KoPoint &p ) +{ return KoPoint( p.m_x*c, p.m_y*c ); } + +inline double operator*( const KoPoint &a, const KoPoint &b ) +{ return a.m_x * b.m_x + a.m_y * b.m_y; } + +/****************************** + kdDebug support +*******************************/ + +#include <kdebug.h> + +/** Show a floating point value with great precision (use within kdDebug) */ +#define DEBUGDOUBLE(d) QString::number( (d), 'g', 20 ) + +inline kdbgstream operator<<( kdbgstream str, const KoPoint & r ) { + // should this use DEBUGDOUBLE? + str << "(" << r.x() << ", " << r.y() << ")"; + return str; +} + +inline kndbgstream operator<<( kndbgstream str, const KoPoint & ) { return str; } + +#endif diff --git a/lib/kofficecore/KoQueryTrader.cpp b/lib/kofficecore/KoQueryTrader.cpp new file mode 100644 index 00000000..00f4379b --- /dev/null +++ b/lib/kofficecore/KoQueryTrader.cpp @@ -0,0 +1,214 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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 <kparts/factory.h> + +#include <KoQueryTrader.h> +#include <KoDocument.h> +#include <KoFilter.h> +#include <ktrader.h> +#include <kservicetype.h> +#include <kdebug.h> + +#include <qfile.h> + +#include <limits.h> // UINT_MAX + +/** + * Port from KOffice Trader to KTrader/KActivator (kded) by Simon Hausmann + * (c) 1999 Simon Hausmann <hausmann@kde.org> + * Port to KService and simplifications by David Faure <faure@kde.org> + */ + + +/******************************************************************* + * + * KoDocumentEntry + * + *******************************************************************/ + +KoDocumentEntry::KoDocumentEntry( KService::Ptr service ) + : m_service( service ) +{ +} + +KoDocument* KoDocumentEntry::createDoc( KoDocument* parent, const char* name ) const +{ + KLibFactory* factory = KLibLoader::self()->factory( QFile::encodeName(m_service->library()) ); + + if( !factory ) { + kdWarning(30003) << KLibLoader::self()->lastErrorMessage() << endl; + return 0; + } + + QObject* obj; + if ( factory->inherits( "KParts::Factory" ) ) + obj = static_cast<KParts::Factory *>(factory)->createPart( 0L, "", parent, name, "KoDocument" ); + else { + kdWarning(30003) << "factory doesn't inherit KParts::Factory ! It is a " << factory->className() << endl; // This shouldn't happen... + obj = factory->create( parent, name, "KoDocument" ); + } + + if ( !obj || !obj->inherits( "KoDocument" ) ) + { + delete obj; + return 0; + } + + return static_cast<KoDocument*>(obj); +} + +KoDocumentEntry KoDocumentEntry::queryByMimeType( const QString & mimetype ) +{ + QString constr = QString::fromLatin1( "[X-KDE-NativeMimeType] == '%1' or '%2' in [X-KDE-ExtraNativeMimeTypes]" ).arg( mimetype ).arg( mimetype ); + + QValueList<KoDocumentEntry> vec = query( false,constr ); + if ( vec.isEmpty() ) + { + kdWarning(30003) << "Got no results with " << constr << endl; + // Fallback to the old way (which was probably wrong, but better be safe) + QString constr = QString::fromLatin1( "'%1' in ServiceTypes" ).arg( mimetype ); + vec = query( constr ); + if ( vec.isEmpty() ) + { + // Still no match. Either the mimetype itself is unknown, or we have no service for it. + // Help the user debugging stuff by providing some more diagnostics + if ( KServiceType::serviceType( mimetype ) == 0L ) + { + kdError(30003) << "Unknown KOffice MimeType " << mimetype << "." << endl; + kdError(30003) << "Check your installation (for instance, run 'kde-config --path mime' and check the result)." << endl; + } else + { + kdError(30003) << "Found no KOffice part able to handle " << mimetype << "!" << endl; + kdError(30003) << "Check your installation (does the desktop file have X-KDE-NativeMimeType and KOfficePart, did you install KOffice in a different prefix than KDE, without adding the prefix to /etc/kderc ?)" << endl; + } + return KoDocumentEntry(); + } + } + + return vec[0]; +} + +QValueList<KoDocumentEntry> KoDocumentEntry::query( const QString & _constr ) +{ + return query(true,_constr); +} + +QValueList<KoDocumentEntry> KoDocumentEntry::query( bool _onlyDocEmb, const QString & _constr ) +{ + + QValueList<KoDocumentEntry> lst; + QString constr; + if ( !_constr.isEmpty() ) { + constr = "("; + constr += _constr; + constr += ") and "; + } + constr += " exist Library"; + + // Query the trader + KTrader::OfferList offers = KTrader::self()->query( "KOfficePart", constr ); + + KTrader::OfferList::ConstIterator it = offers.begin(); + unsigned int max = offers.count(); + for( unsigned int i = 0; i < max; i++, ++it ) + { + //kdDebug(30003) << " desktopEntryPath=" << (*it)->desktopEntryPath() + // << " library=" << (*it)->library() << endl; + + if ( (*it)->noDisplay() ) + continue; + // Maybe this could be done as a trader constraint too. + if ( (!_onlyDocEmb) || ((*it)->property("X-KDE-NOTKoDocumentEmbeddable").toString()!="1") ) + { + KoDocumentEntry d( *it ); + // Append converted offer + lst.append( d ); + // Next service + } + } + + if ( lst.count() > 1 && !_constr.isEmpty() ) + kdWarning(30003) << "KoDocumentEntry::query " << constr << " got " << max << " offers!" << endl; + + return lst; +} + + + + +/******************************************************************* + * + * KoFilterEntry + * + *******************************************************************/ + +KoFilterEntry::KoFilterEntry( KService::Ptr service ) + : m_service( service ) +{ + import = service->property( "X-KDE-Import" ).toStringList(); + export_ = service->property( "X-KDE-Export" ).toStringList(); + int w = service->property( "X-KDE-Weight" ).toInt(); + weight = w < 0 ? UINT_MAX : static_cast<unsigned int>( w ); + available = service->property( "X-KDE-Available" ).toString(); +} + +QValueList<KoFilterEntry::Ptr> KoFilterEntry::query( const QString & _constr ) +{ + kdDebug(30500) << "KoFilterEntry::query( " << _constr << " )" << endl; + QValueList<KoFilterEntry::Ptr> lst; + + KTrader::OfferList offers = KTrader::self()->query( "KOfficeFilter", _constr ); + + KTrader::OfferList::ConstIterator it = offers.begin(); + unsigned int max = offers.count(); + //kdDebug(30500) << "Query returned " << max << " offers" << endl; + for( unsigned int i = 0; i < max; i++ ) + { + //kdDebug(30500) << " desktopEntryPath=" << (*it)->desktopEntryPath() + // << " library=" << (*it)->library() << endl; + // Append converted offer + lst.append( new KoFilterEntry( *it ) ); + // Next service + it++; + } + + return lst; +} + +KoFilter* KoFilterEntry::createFilter( KoFilterChain* chain, QObject* parent, const char* name ) +{ + KLibFactory* factory = KLibLoader::self()->factory( QFile::encodeName( m_service->library() ) ); + + if ( !factory ) { + kdWarning(30003) << KLibLoader::self()->lastErrorMessage() << endl; + return 0; + } + + QObject* obj = factory->create( parent, name, "KoFilter" ); + if ( !obj || !obj->inherits( "KoFilter" ) ) + { + delete obj; + return 0; + } + + KoFilter* filter = static_cast<KoFilter*>( obj ); + filter->m_chain = chain; + return filter; +} diff --git a/lib/kofficecore/KoQueryTrader.h b/lib/kofficecore/KoQueryTrader.h new file mode 100644 index 00000000..55ab8dbe --- /dev/null +++ b/lib/kofficecore/KoQueryTrader.h @@ -0,0 +1,178 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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. +*/ + +#ifndef __ko_query_trader_h__ +#define __ko_query_trader_h__ + +#include <kservice.h> +#include <ksharedptr.h> +#include <qvaluelist.h> +#include <koffice_export.h> + +class QObject; +class QStringList; +class KoDocument; +class KoFilter; +class KoFilterChain; + +/** + * Represents an available KOffice component + * that supports the document interface. + */ +class KOFFICECORE_EXPORT KoDocumentEntry +{ + +public: + KoDocumentEntry() { m_service = 0L; } // for QValueList + KoDocumentEntry( KService::Ptr service ); + ~KoDocumentEntry() { } + + KService::Ptr service() const { return m_service; } + + /** + * @return TRUE if the service pointer is null + */ + bool isEmpty() const { return m_service == 0L; } + + /** + * @return name of the associated service + */ + QString name() const { return m_service->name(); } + + /** + * Mimetypes (and other service types) which this document can handle. + */ + QStringList mimeTypes() const { return m_service->serviceTypes(); } + + /** + * @return TRUE if the document can handle the requested mimetype. + */ + bool supportsMimeType( const QString & _mimetype ) const + { return mimeTypes().contains( _mimetype ); } + + /** + * Uses the factory of the component to create + * a document. If that is not possible, 0 is returned. + */ + KoDocument* createDoc( KoDocument* parent = 0, const char* name = 0 ) const; + + /** + * This function will query ksycoca to find all available components. + * The result will only contain parts, which are embeddable into a document + * + * @param _constr is a constraint expression as used by KTrader. + * You can use it to set additional restrictions on the available + * components. + */ + static QValueList<KoDocumentEntry> query( const QString & _constr = QString::null ); + + /** + * This function will query the system to find all available filters. + * + * @param _onlyDocEmb specifies if only KOffice Parts should be listed which are + * embeddable into other koDocuments, or all (if false) + * (eg.: it makes no sense to embed Kexi into KWord, + * but it makes sense to embed it into KoShell) + * @param _constr is a constraint expression as used by KDEDs trader interface. + * You can use it to set additional restrictions on the available + * components. + */ + // ### TODO: MERGE WITH ABOVE METHODE WHEN BIC+SIC CHANGES ARE ALLOWED + static QValueList<KoDocumentEntry> query( bool _onlyDocEmb,const QString& _constr); + /* this is how the signature should be looking after merging + static QValueList<KoDocumentEntry> query( bool _onlyDocEmb =true, const QString& _constr = QString::null ); + or better: use an enum for the first arg. + */ + + + /** + * This is a convenience function. + * + * @return a document entry for the KOffice component that supports + * the requested mimetype and fits the user best. + */ + static KoDocumentEntry queryByMimeType( const QString & mimetype ); + +private: + KService::Ptr m_service; +}; + +/** + * Represents an available filter. + */ +class KoFilterEntry : public KShared +{ + +public: + typedef KSharedPtr<KoFilterEntry> Ptr; + + KoFilterEntry() : weight( 0 ) { m_service = 0L; } // for QValueList + KoFilterEntry( KService::Ptr service ); + ~KoFilterEntry() { } + + KoFilter* createFilter( KoFilterChain* chain, QObject* parent = 0, const char* name = 0 ); + + /** + * The imported mimetype(s). + */ + QStringList import; + + /** + * The exported mimetype(s). + */ + QStringList export_; + + /** + * The "weight" of this filter path. Has to be > 0 to be valid. + */ + unsigned int weight; + + /** + * Do we have to check during runtime? + */ + QString available; + + /** + * @return TRUE if the filter can import the requested mimetype. + */ + bool imports( const QString& _mimetype ) const + { return ( import.contains( _mimetype ) ); } + + /** + * @return TRUE if the filter can export the requested mimetype. + */ + bool exports( const QString& _m ) const + { return ( export_.contains( _m ) ); } + + /** + * This function will query KDED to find all available filters. + * + * @param _constr is a constraint expression as used by KDEDs trader interface. + * You can use it to set additional restrictions on the available + * components. + */ + static QValueList<KoFilterEntry::Ptr> query( const QString& _constr = QString::null ); + + KService::Ptr service() const { return m_service; } + +private: + KService::Ptr m_service; +}; + +#endif diff --git a/lib/kofficecore/KoRect.cpp b/lib/kofficecore/KoRect.cpp new file mode 100644 index 00000000..b93a577d --- /dev/null +++ b/lib/kofficecore/KoRect.cpp @@ -0,0 +1,274 @@ +/* This file is part of the KDE project + Copyright (C) 2001 David Faure <faure@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 version 2, as published by the Free Software Foundation. + + 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 "KoRect.h" + +KoRect KoRect::normalize() const +{ + KoRect r; + if ( right() < left() ) { // swap bad x values + r.m_tl.setX( right() ); + r.m_br.setX( left() ); + } else { + r.m_tl.setX( left() ); + r.m_br.setX( right() ); + } + if ( bottom() < top() ) { // swap bad y values + r.m_tl.setY( bottom() ); + r.m_br.setY( top() ); + } else { + r.m_tl.setY( top() ); + r.m_br.setY( bottom() ); + } + return r; +} + +void KoRect::setTopLeft(const KoPoint &topleft) +{ + m_tl = topleft; +} + +void KoRect::setBottomRight(const KoPoint &bottomright) +{ + m_br = bottomright; +} + +void KoRect::setTopRight(const KoPoint &topright) +{ + m_br.rx() = topright.x(); + m_tl.ry() = topright.y(); +} + +void KoRect::setBottomLeft(const KoPoint &bottomleft) +{ + m_tl.rx() = bottomleft.x(); + m_br.ry() = bottomleft.y(); +} + + +KoPoint KoRect::center() const +{ + return KoPoint((left() + right()) / 2, (top() + bottom()) / 2); +} + +void KoRect::moveTopLeft(const KoPoint &topleft) +{ + m_br.rx() += topleft.x() - m_tl.x(); + m_br.ry() += topleft.y() - m_tl.y(); + m_tl = topleft; +} + +void KoRect::moveBottomRight(const KoPoint &bottomright) +{ + m_tl.rx() += bottomright.x() - m_br.x(); + m_tl.ry() += bottomright.y() - m_br.y(); + m_br = bottomright; +} + +void KoRect::moveTopRight(const KoPoint &topright) +{ + m_tl.rx() += topright.x() - m_br.x(); + m_br.ry() += topright.y() - m_tl.y(); + m_br.rx() = topright.x(); + m_tl.ry() = topright.y(); +} + +void KoRect::moveBottomLeft(const KoPoint &bottomleft) +{ + m_br.rx() += bottomleft.x() - m_tl.x(); + m_tl.ry() += bottomleft.y() - m_br.y(); + m_tl.rx() = bottomleft.x(); + m_br.ry() = bottomleft.y(); +} + +void KoRect::moveBy(const double &dx, const double &dy) +{ + m_tl.rx() += dx; + m_tl.ry() += dy; + m_br.rx() += dx; + m_br.ry() += dy; +} + +void KoRect::setRect(const double &x, const double &y, const double &width, const double &height) +{ + m_tl.setCoords( x, y ); + m_br.setCoords( x + width, y + height ); +} + +void KoRect::setRect(const KoRect &rect) +{ + m_tl = rect.m_tl; + m_br = rect.m_br; +} + +void KoRect::setCoords(const double &x1, const double &y1, const double &x2, const double &y2) +{ + m_tl.setCoords( x1, y1 ); + m_br.setCoords( x2, y2 ); +} + +void KoRect::setSize(const KoSize &size) +{ + setWidth(size.width()); + setHeight(size.height()); +} + +KoSize KoRect::size() const +{ + return KoSize(width(), height()); +} + +KoRect &KoRect::operator|=(const KoRect &rhs) { + + if(rhs.isEmpty()) + return *this; + if(isEmpty()) + { + *this = rhs; + return *this; + } + if(m_tl.x() > rhs.left()) + m_tl.setX(rhs.left()); + if(m_tl.y() > rhs.top()) + m_tl.setY(rhs.top()); + if(m_br.x() < rhs.right()) + m_br.setX(rhs.right()); + if(m_br.y() < rhs.bottom()) + m_br.setY(rhs.bottom()); + return *this; +} + +KoRect &KoRect::operator&=(const KoRect &rhs) { + + if(m_tl.x() < rhs.left()) + m_tl.setX(rhs.left()); + if(m_tl.y() < rhs.top()) + m_tl.setY(rhs.top()); + if(m_br.x() > rhs.right()) + m_br.setX(rhs.right()); + if(m_br.y() > rhs.bottom()) + m_br.setY(rhs.bottom()); + return *this; +} + +bool KoRect::contains(const KoPoint &p) const { + return (p.x() >= m_tl.x() && p.x() <= m_br.x() && p.y() >= m_tl.y() && p.y() <= m_br.y()); +} + +bool KoRect::contains(const double &x, const double &y) const { + return (x >= m_tl.x() && x <= m_br.x() && y >= m_tl.y() && y <= m_br.y()); +} + +bool KoRect::contains(const KoRect &r) const { + return (r.left() >= m_tl.x() && r.right() <= m_br.x() && r.top() >= m_tl.y() && r.bottom() <= m_br.y()); +} + + +KoRect KoRect::unite(const KoRect &r) const { + return *this | r; +} + +KoRect KoRect::intersect(const KoRect &r) const { + return *this & r; +} + +bool KoRect::intersects(const KoRect &r) const { + return ( QMAX(m_tl.x(), r.left()) <= QMIN(m_br.x(), r.right()) && + QMAX(m_tl.y(), r.top()) <= QMIN(m_br.y(), r.bottom()) ); +} + +KoRect operator|(const KoRect &lhs, const KoRect &rhs) { + + if(lhs.isEmpty()) + return rhs; + if(rhs.isEmpty()) + return lhs; + KoRect tmp; + tmp.setCoords( (lhs.left() < rhs.left() ? lhs.left() : rhs.left()), + (lhs.top() < rhs.top() ? lhs.top() : rhs.top()), + (lhs.right() > rhs.right() ? lhs.right() : rhs.right()), + (lhs.bottom() > rhs.bottom() ? lhs.bottom() : rhs.bottom()) ); + return tmp; +} + +KoRect operator&(const KoRect &lhs, const KoRect &rhs) { + + KoRect tmp; + tmp.setCoords( (lhs.left() > rhs.left() ? lhs.left() : rhs.left()), + (lhs.top() > rhs.top() ? lhs.top() : rhs.top()), + (lhs.right() < rhs.right() ? lhs.right() : rhs.right()), + (lhs.bottom() < rhs.bottom() ? lhs.bottom() : rhs.bottom()) ); + return tmp; +} + +bool operator==(const KoRect &lhs, const KoRect &rhs) { + return ( lhs.topLeft()==rhs.topLeft() && + lhs.bottomRight()==rhs.bottomRight() ); +} + +bool operator!=(const KoRect &lhs, const KoRect &rhs) { + return ( lhs.topLeft()!=rhs.topLeft() || + lhs.bottomRight()!=rhs.bottomRight() ); +} + +KoRect KoRect::transform(const QWMatrix &m) const +{ + KoRect result; + if(m.m12() == 0.0F && m.m21() == 0.0F) + { + result = KoRect(topLeft().transform(m), bottomRight().transform(m)); + } + else + { + int i; + KoPoint p[4] = { KoPoint(m_tl.x(), m_tl.y()), KoPoint(m_tl.x(), m_br.x()), + KoPoint(m_br.x(), m_br.x()), KoPoint(m_br.x(), m_tl.y()) }; + for(i = 0; i < 4; i++) + p[i] = p[i].transform(m); + + result.setLeft(p[0].x()); + result.setTop(p[0].y()); + result.setRight(p[0].x()); + result.setBottom(p[0].y()); + + for(int i = 1; i < 4; i++) + { + result.setLeft(QMIN(p[i].x(), result.left())); + result.setTop(QMIN(p[i].y(), result.top())); + result.setRight(QMAX(p[i].x(), result.right())); + result.setBottom(QMAX(p[i].y(), result.bottom())); + } + } + return result; +} + +KoRect KoRect::translate(double dx, double dy) const +{ + return KoRect(left() + dx, top() + dy, width(), height()); +} + +QRect KoRect::toQRect() const +{ + return QRect( qRound( left() ), qRound( top() ), qRound( width() ), qRound( height() ) ); +} + +//static +KoRect KoRect::fromQRect( const QRect &rect ) +{ + return KoRect( rect.left(), rect.top(), rect.width(), rect.height() ); +} diff --git a/lib/kofficecore/KoRect.h b/lib/kofficecore/KoRect.h new file mode 100644 index 00000000..dfc63946 --- /dev/null +++ b/lib/kofficecore/KoRect.h @@ -0,0 +1,148 @@ +/* This file is part of the KDE project + Copyright (C) 2001 David Faure <faure@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 version 2, as published by the Free Software Foundation. + + 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. +*/ + +#ifndef koRect_h +#define koRect_h + +#include "KoPoint.h" +#include "KoSize.h" +#include <qrect.h> +#include "koffice_export.h" + +/** + * A rect whose coordinates are floating-point values ( "double"s ). + * The API isn't documented, it's a perfect mirror of QRect. + */ +class KOFFICEUI_EXPORT KoRect { + +public: + KoRect() + : m_tl(), m_br() {} + KoRect(const KoPoint &topleft, const KoPoint &bottomright) + : m_tl(topleft), m_br(bottomright) {} + KoRect(const KoPoint &topleft, const KoSize &size) + {m_tl = topleft; setSize(size);} + KoRect(const double &left, const double &top, const double &width, const double &height) + : m_tl(left,top), m_br(left+width,top+height) {} + ~KoRect() {} + + bool isNull() const { return m_tl == m_br; } + // Like QRect, a null KoRect is empty. + bool isEmpty() const { return m_tl.x() > m_br.x() || m_tl.y() > m_br.y() || isNull(); } + // Unlike QRect, a null KoRect is valid (0-sized). + bool isValid() const { return m_tl.x() <= m_br.x() && m_tl.y() <= m_br.y(); } + KoRect normalize() const; + + double left() const { return m_tl.x(); } + double top() const { return m_tl.y(); } + double right() const { return m_br.x(); } + double bottom() const { return m_br.y(); } + + double& rLeft() { return m_tl.rx(); } + double& rTop() { return m_tl.ry(); } + double& rRight() { return m_br.rx(); } + double& rBottom() { return m_br.ry(); } + + double x() const { return left(); } + double y() const { return top(); } + + void setLeft(const double &left) { m_tl.setX(left); } + void setTop(const double &top) { m_tl.setY(top); } + void setRight(const double &right) { m_br.setX(right); } + void setBottom(const double &bottom) { m_br.setY(bottom); } + + void setX(const double &x) { m_tl.setX(x); } //same as setLeft() + void setY(const double &y) { m_tl.setY(y); } //same as setTop() + + KoPoint topLeft() const { return m_tl; } + KoPoint bottomRight() const { return m_br; } + KoPoint topRight() const { return KoPoint(m_br.x(), m_tl.y()); } + KoPoint bottomLeft() const { return KoPoint(m_tl.x(), m_br.y()); } + KoPoint center() const; + + void setTopLeft(const KoPoint &topleft); + void setBottomRight(const KoPoint &bottomright); + void setTopRight(const KoPoint &topright); + void setBottomLeft(const KoPoint &bottomleft); + + void moveTopLeft(const KoPoint &topleft); + void moveBottomRight(const KoPoint &bottomright); + void moveTopRight(const KoPoint &topright); + void moveBottomLeft(const KoPoint &bottomleft); + //void moveCenter(const KoPoint ¢er); + void moveBy(const double &dx, const double &dy); + + void setRect(const double &x, const double &y, const double &width, const double &height); + void setRect(const KoRect &rect); + void setCoords(const double &x1, const double &y1, const double &x2, const double &y2); + + KoSize size() const; + double width() const { return m_br.x()-m_tl.x(); } + double height() const { return m_br.y()-m_tl.y(); } + void setWidth(const double &width) { m_br.setX(m_tl.x()+width); } + void setHeight(const double &height) { m_br.setY(m_tl.y()+height); } + void setSize(const KoSize &size); + + KoRect &operator|=(const KoRect &rhs); + KoRect &operator&=(const KoRect &rhs); + /** + * Returns if the point is contained in the rect. + * @param p the point to test + * Will return true if the point is contained in the 2d area this rect represents, this means + * that it will return true on everthing from the topleft() to the bottomright(); + * Note that for KoRect(0, 0, 100, 100) the KoPoint(0, 0) as well as KoPoint(100, 100) are + * mathmatically contained in the rect, this in contrary to pixel based rectangles. + */ + bool contains(const KoPoint &p) const; + /// Helper function for the above function. + bool contains(const double &x, const double &y) const; + /// Helper function for the above function. + bool contains(const KoRect &r) const; + KoRect unite(const KoRect &r) const; + KoRect intersect(const KoRect &r) const; + bool intersects(const KoRect &r) const; + + KoRect transform(const QWMatrix &m) const; + KoRect translate(double dx, double dy) const; + + QRect toQRect() const; + static KoRect fromQRect( const QRect &rect ); + +private: + KoPoint m_tl, m_br; +}; + +KoRect operator|(const KoRect &lhs, const KoRect &rhs); +KoRect operator&(const KoRect &lhs, const KoRect &rhs); +KOFFICEUI_EXPORT bool operator==(const KoRect &lhs, const KoRect &rhs); +KOFFICEUI_EXPORT bool operator!=(const KoRect &lhs, const KoRect &rhs); + + +/** Show the position and size of a rectangle (use within kdDebug) */ +#define DEBUGRECT(rc) (rc).x() << "," << (rc).y() << " " << (rc).width() << "x" << (rc).height() + +//inline kdbgstream operator<<( kdbgstream str, const KoRect & r ) { str << "[" << r.left() << ", " << r.top() << " - " << r.right() << ", " << r.bottom() << "]"; return str; } +inline kdbgstream operator<<( kdbgstream str, const KoRect & r ) { str << "[" << r.left() << "," << r.top() << " " << r.width() << "x" << r.height() << "]"; return str; } +inline kndbgstream operator<<( kndbgstream str, const KoRect & ) { return str; } + +/** Show the rectangles that form a region (use as a standalone statement) */ +#define DEBUGREGION(reg) { QMemArray<QRect>rs=reg.rects(); for (int i=0;i<rs.size();++i) \ + kdDebug()<<" "<<DEBUGRECT(rs[i] )<<endl; } +// You can now use kdDebug() << theregion << endl; (kdebug.h) + +#endif diff --git a/lib/kofficecore/KoSetPropCommand.h b/lib/kofficecore/KoSetPropCommand.h new file mode 100644 index 00000000..53ee73bf --- /dev/null +++ b/lib/kofficecore/KoSetPropCommand.h @@ -0,0 +1,86 @@ +/* This file is part of the KDE project + Copyright (C) 2000 Werner Trobin <trobin@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. +*/ + +#ifndef KOGENCOMMAND_H +#define KOGENCOMMAND_H + +#include <kcommand.h> + +/** + * Generic command to set a property on an object. + * This variant is for simple types, where the setter method takes a value. + */ +template<class Property, class Object, void (Object::* Function) (Property)> class KoSetBasicPropCommand : public KNamedCommand { + +public: + KoSetBasicPropCommand(Object *object, const QString &name) : KNamedCommand(name), m_object(object) {} + KoSetBasicPropCommand(Object *object, const QString &name, + const Property &oldProperty, const Property &newProperty) : KNamedCommand(name), + m_object(object), m_oldProperty(oldProperty), m_newProperty(newProperty) {} + virtual ~KoSetBasicPropCommand() {} + + virtual void execute() { if(m_object) (m_object->*Function)(m_newProperty); } + virtual void unexecute() { if(m_object) (m_object->*Function)(m_oldProperty); } + + void setOldProperty(const Property &oldProperty) { m_oldProperty=oldProperty; } + const Property &oldProperty() const { return m_oldProperty; } + void setNewProperty(const Property &newProperty) { m_newProperty=newProperty; } + const Property &newProperty() const { return m_newProperty; } + +private: + // Don't copy or assign this stuff + KoSetBasicPropCommand(const KoSetBasicPropCommand<Property, Object, Function> &rhs); + KoSetBasicPropCommand &operator=(const KoSetBasicPropCommand<Property, Object, Function> &rhs); + + Object *m_object; + Property m_oldProperty, m_newProperty; +}; + +/** + * Generic command to set a property on an object. + * This variant is for non-trivial types, where the setter method takes a const reference. + */ +template<class Property, class Object, void (Object::* Function) (const Property &)> class KoSetPropCommand : public KNamedCommand { + +public: + KoSetPropCommand(Object *object, const QString &name) : KNamedCommand(name), m_object(object) {} + KoSetPropCommand(Object *object, const QString &name, + const Property &oldProperty, const Property &newProperty) : KNamedCommand(name), + m_object(object), m_oldProperty(oldProperty), m_newProperty(newProperty) {} + virtual ~KoSetPropCommand() {} + + virtual void execute() { if(m_object) (m_object->*Function)(m_newProperty); } + virtual void unexecute() { if(m_object) (m_object->*Function)(m_oldProperty); } + + void setOldProperty(const Property &oldProperty) { m_oldProperty=oldProperty; } + const Property &oldProperty() const { return m_oldProperty; } + void setNewProperty(const Property &newProperty) { m_newProperty=newProperty; } + const Property &newProperty() const { return m_newProperty; } + +private: + // Don't copy or assign this stuff + KoSetPropCommand(const KoSetPropCommand<Property, Object, Function> &rhs); + KoSetPropCommand &operator=(const KoSetPropCommand<Property, Object, Function> &rhs); + + Object *m_object; + Property m_oldProperty, m_newProperty; +}; + +#endif /* KOGENCOMMAND_H */ + diff --git a/lib/kofficecore/KoSize.h b/lib/kofficecore/KoSize.h new file mode 100644 index 00000000..3cdc30bc --- /dev/null +++ b/lib/kofficecore/KoSize.h @@ -0,0 +1,230 @@ +/* This file is part of the KDE project + Copyright (C) 2001 Lukas Tinkl <lukas@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. + + This file borrows from the QSize class; + Copyright (C) 1992-2000 Trolltech AS. All rights reserved. +*/ + +#ifndef koSize_h +#define koSize_h + +#include <qsize.h> +#include <qtl.h> + +/** + * A size whose coordinates are floating-point values ( "double"s ). + * The API isn't documented, it's a perfect mirror of QSize. + */ +class KoSize +{ +public: + KoSize(); + KoSize( double w, double h ); + + bool isNull() const; + bool isEmpty() const; + bool isValid() const; + + double width() const; + double height() const; + void setWidth( double w ); + void setHeight( double h ); + + KoSize expandedTo( const KoSize & ) const; + KoSize boundedTo( const KoSize & ) const; + + double &rwidth(); + double &rheight(); + + KoSize &operator+=( const KoSize & ); + KoSize &operator-=( const KoSize & ); + KoSize &operator*=( int c ); + KoSize &operator*=( double c ); + KoSize &operator/=( int c ); + KoSize &operator/=( double c ); + + friend inline bool operator==( const KoSize &, const KoSize & ); + friend inline bool operator!=( const KoSize &, const KoSize & ); + friend inline const KoSize operator+( const KoSize &, const KoSize & ); + friend inline const KoSize operator-( const KoSize &, const KoSize & ); + friend inline const KoSize operator*( const KoSize &, int ); + friend inline const KoSize operator*( int, const KoSize & ); + friend inline const KoSize operator*( const KoSize &, double ); + friend inline const KoSize operator*( double, const KoSize & ); + friend inline const KoSize operator/( const KoSize &, int ); + friend inline const KoSize operator/( const KoSize &, double ); + + inline QSize toQSize() const; + static KoSize fromQSize( const QSize &size ) + { + return KoSize(size.width(), size.height()); + } + + void transpose() + { + qSwap(wd, ht); + } + +private: + static void warningDivByZero() + { +#if defined(QT_CHECK_MATH) + qWarning( "KoSize: Division by zero error" ); +#endif + } + + double wd; + double ht; +}; + + +/***************************************************************************** + KoSize inline functions + *****************************************************************************/ + +inline KoSize::KoSize() +{ wd = ht = -1.0; } + +inline KoSize::KoSize( double w, double h ) +{ wd=w; ht=h; } + +inline bool KoSize::isNull() const +{ return wd==0.0 && ht==0.0; } + +inline bool KoSize::isEmpty() const +{ return wd<=0.0 || ht<=0.0; } + +inline bool KoSize::isValid() const +{ return wd>=0.0 && ht>=0.0; } + +inline double KoSize::width() const +{ return wd; } + +inline double KoSize::height() const +{ return ht; } + +inline void KoSize::setWidth( double w ) +{ wd=w; } + +inline void KoSize::setHeight( double h ) +{ ht=h; } + +inline double &KoSize::rwidth() +{ return wd; } + +inline double &KoSize::rheight() +{ return ht; } + +inline KoSize &KoSize::operator+=( const KoSize &s ) +{ wd+=s.wd; ht+=s.ht; return *this; } + +inline KoSize &KoSize::operator-=( const KoSize &s ) +{ wd-=s.wd; ht-=s.ht; return *this; } + +inline KoSize &KoSize::operator*=( int c ) +{ wd*=c; ht*=c; return *this; } + +inline KoSize &KoSize::operator*=( double c ) +{ wd=wd*c; ht=ht*c; return *this; } + +inline bool operator==( const KoSize &s1, const KoSize &s2 ) +{ return s1.wd == s2.wd && s1.ht == s2.ht; } + +inline bool operator!=( const KoSize &s1, const KoSize &s2 ) +{ return s1.wd != s2.wd || s1.ht != s2.ht; } + +inline const KoSize operator+( const KoSize & s1, const KoSize & s2 ) +{ return KoSize(s1.wd+s2.wd, s1.ht+s2.ht); } + +inline const KoSize operator-( const KoSize &s1, const KoSize &s2 ) +{ return KoSize(s1.wd-s2.wd, s1.ht-s2.ht); } + +inline const KoSize operator*( const KoSize &s, int c ) +{ return KoSize(s.wd*c, s.ht*c); } + +inline const KoSize operator*( int c, const KoSize &s ) +{ return KoSize(s.wd*c, s.ht*c); } + +inline const KoSize operator*( const KoSize &s, double c ) +{ return KoSize(s.wd*c, s.ht*c); } + +inline const KoSize operator*( double c, const KoSize &s ) +{ return KoSize(s.wd*c, s.ht*c); } + +inline KoSize &KoSize::operator/=( int c ) +{ +#if defined(QT_CHECK_MATH) + if ( c == 0 ) + warningDivByZero(); +#endif + wd/=c; ht/=c; + return *this; +} + +inline KoSize &KoSize::operator/=( double c ) +{ +#if defined(QT_CHECK_MATH) + if ( c == 0.0 ) + warningDivByZero(); +#endif + wd=wd/c; ht=ht/c; + return *this; +} + +inline const KoSize operator/( const KoSize &s, int c ) +{ +#if defined(QT_CHECK_MATH) + if ( c == 0 ) + KoSize::warningDivByZero(); +#endif + return KoSize(s.wd/c, s.ht/c); +} + +inline const KoSize operator/( const KoSize &s, double c ) +{ +#if defined(QT_CHECK_MATH) + if ( c == 0.0 ) + KoSize::warningDivByZero(); +#endif + return KoSize(s.wd/c, s.ht/c); +} + +inline KoSize KoSize::expandedTo( const KoSize & otherSize ) const +{ + return KoSize( QMAX(wd,otherSize.wd), QMAX(ht,otherSize.ht) ); +} + +inline KoSize KoSize::boundedTo( const KoSize & otherSize ) const +{ + return KoSize( QMIN(wd,otherSize.wd), QMIN(ht,otherSize.ht) ); +} + +inline QSize KoSize::toQSize() const +{ + return QSize(qRound(wd), qRound(ht)); +} + +/****************************** + kdDebug support +*******************************/ +#include <kdebug.h> + +inline kdbgstream operator<<( kdbgstream str, const KoSize & sz ) { str << "[" << sz.width() << "x" << sz.height() << "]"; return str; } +inline kndbgstream operator<<( kndbgstream str, const KoSize & ) { return str; } + +#endif diff --git a/lib/kofficecore/KoSpeaker.cpp b/lib/kofficecore/KoSpeaker.cpp new file mode 100644 index 00000000..db975623 --- /dev/null +++ b/lib/kofficecore/KoSpeaker.cpp @@ -0,0 +1,602 @@ +/* +* This file is part of the KDE/KOffice project. +* Copyright (C) 2005, Gary Cramblitt <garycramblitt@comcast.net> +* +* @author Gary Cramblitt <garycramblitt@comcast.net> +* @since KOffice 1.5 +* +* 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. +*/ + +// Qt includes. +#include <qtimer.h> +#include <qcursor.h> +#include <qtooltip.h> +#include <qwhatsthis.h> +#include <qmenubar.h> +#include <qlabel.h> +#include <qbutton.h> +#include <qcombobox.h> +#include <qtabbar.h> +#include <qgroupbox.h> +#include <qlineedit.h> +#include <qtextedit.h> +#include <qlistview.h> +#include <qlistbox.h> +#include <qiconview.h> +#include <qtable.h> +#include <qgridview.h> +#include <qregexp.h> +#include <qstylesheet.h> + +// KDE includes. +#include <kapplication.h> +#include <klocale.h> +#include <kglobal.h> +#include <dcopclient.h> +#include <kconfig.h> +#include <ktrader.h> +#include <kdebug.h> + +// KoSpeaker includes. +#include "KoSpeaker.h" +#include "KoSpeaker.moc" + +// ------------------ KoSpeakerPrivate ------------------------ + +class KoSpeakerPrivate +{ +public: + KoSpeakerPrivate() : + m_versionChecked(false), + m_enabled(false), + m_speakFlags(0), + m_timeout(600), + m_timer(0), + m_prevPointerWidget(0), + m_prevPointerId(-1), + m_prevFocusWidget(0), + m_prevFocusId(-1), + m_prevWidget(0), + m_prevId(-1), + m_cancelSpeakWidget(false) + {} + + // List of text jobs. + QValueList<uint> m_jobNums; + // Whether the version of KTTSD has been requested from the daemon. + bool m_versionChecked; + // KTTSD version string. + QString m_kttsdVersion; + // Language code of last spoken text. + QString m_langCode; + // Word used before speaking an accelerator letter. + QString m_acceleratorPrefix; + // Whether TTS service is available or not. + bool m_enabled; + // TTS options. + uint m_speakFlags; + // Timer which implements the polling interval. + int m_timeout; + QTimer* m_timer; + // Widget and part of widget for 1) last widget under mouse pointer, 2) last widget with focus, and + // last widget spoken. + QWidget* m_prevPointerWidget; + int m_prevPointerId; + QWidget* m_prevFocusWidget; + int m_prevFocusId; + QWidget* m_prevWidget; + int m_prevId; + // True when cancelSpeakWidget has been called in response to customSpeakWidget signal. + bool m_cancelSpeakWidget; +}; + +// ------------------ KoSpeaker ------------------------------- + +KoSpeaker* KoSpeaker::KSpkr = 0L; + +KoSpeaker::KoSpeaker() +{ + Q_ASSERT(!KSpkr); + KSpkr = this; + d = new KoSpeakerPrivate(); + readConfig(KGlobal::config()); +} + +KoSpeaker::~KoSpeaker() +{ + if (d->m_jobNums.count() > 0) { + for (int i = d->m_jobNums.count() - 1; i >= 0; i--) + removeText(d->m_jobNums[i]); + d->m_jobNums.clear(); + } + delete d; + KSpkr = 0; +} + +bool KoSpeaker::isEnabled() const { return d->m_enabled; } + +void KoSpeaker::probe() +{ + d->m_timer->stop(); + QWidget* w; + QPoint pos; + bool spoke = false; + if ( d->m_speakFlags & SpeakFocusWidget ) { + w = kapp->focusWidget(); + if (w) { + spoke = maybeSayWidget(w); + if (!spoke) + emit customSpeakWidget(w, pos, d->m_speakFlags); + } + } + if ( !spoke && d->m_speakFlags & SpeakPointerWidget ) { + pos = QCursor::pos(); + w = kapp->widgetAt(pos, true); + if (w) { + if (!maybeSayWidget(w, pos)) + emit customSpeakWidget(w, pos, d->m_speakFlags); + } + } + d->m_timer->start(d->m_timeout); +} + +void KoSpeaker::queueSpeech(const QString& msg, const QString& langCode /*= QString()*/, bool first /*= true*/) +{ + if (!startKttsd()) return; + int jobCount = d->m_jobNums.count(); + if (first && jobCount > 0) { + for (int i = jobCount - 1; i >= 0; i--) + removeText(d->m_jobNums[i]); + d->m_jobNums.clear(); + jobCount = 0; + } + QString s = msg.stripWhiteSpace(); + if (s.isEmpty()) return; + // kdDebug() << "KoSpeaker::queueSpeech: s = [" << s << "]" << endl; + // If no language code given, assume desktop setting. + QString languageCode = langCode; + if (langCode.isEmpty()) + languageCode = KGlobal::locale()->language(); + // kdDebug() << "KoSpeaker::queueSpeech:languageCode = " << languageCode << endl; + // If KTTSD version is 0.3.5 or later, we can use the appendText method to submit a + // single, multi-part text job. Otherwise, must submit separate text jobs. + // If language code changes, then must also start a new text job so that it will + // be spoken in correct talker. + if (getKttsdVersion().isEmpty()) + d->m_jobNums.append(setText(s, languageCode)); + else { + if ((jobCount == 0) || (languageCode != d->m_langCode)) + d->m_jobNums.append(setText(s, languageCode)); + else + appendText(s, d->m_jobNums[jobCount-1]); + } + d->m_langCode = languageCode; +} + +void KoSpeaker::startSpeech() +{ + for (uint i = 0; i < d->m_jobNums.count(); i++) + startText(d->m_jobNums[i]); +} + +void KoSpeaker::readConfig(KConfig* config) +{ + delete d->m_timer; + d->m_timer = 0; + config->setGroup("TTS"); + d->m_speakFlags = 0; + if (config->readBoolEntry("SpeakPointerWidget", false)) d->m_speakFlags |= SpeakPointerWidget; + if (config->readBoolEntry("SpeakFocusWidget", false)) d->m_speakFlags |= SpeakFocusWidget; + if (config->readBoolEntry("SpeakTooltips", true)) d->m_speakFlags |= SpeakTooltip; + if (config->readBoolEntry("SpeakWhatsThis", false)) d->m_speakFlags |= SpeakWhatsThis; + if (config->readBoolEntry("SpeakDisabled", true)) d->m_speakFlags |= SpeakDisabled; + if (config->readBoolEntry("SpeakAccelerators", true)) d->m_speakFlags |= SpeakAccelerator; + d->m_timeout = config->readNumEntry("PollingInterval", 600); + d->m_acceleratorPrefix = config->readEntry("AcceleratorPrefixWord", i18n("Accelerator")); + if (d->m_speakFlags & (SpeakPointerWidget | SpeakFocusWidget)) { + if (startKttsd()) { + d->m_timer = new QTimer( this ); + connect( d->m_timer, SIGNAL(timeout()), this, SLOT(probe()) ); + d->m_timer->start( d->m_timeout ); + } + } +} + +bool KoSpeaker::maybeSayWidget(QWidget* w, const QPoint& pos /*=QPoint()*/) +{ + if (!w) return false; + + int id = -1; + QString text; + + if (w->inherits("QViewportWidget")) { + w = w->parentWidget(); + if (!w) return false; + } + + // Handle widgets that have multiple parts. + + if ( w->inherits("QMenuBar") ) { + QMenuBar* menuBar = dynamic_cast<QMenuBar *>(w); + if (pos == QPoint()) { + for (uint i = 0; i < menuBar->count(); ++i) + if (menuBar->isItemActive(menuBar->idAt(i))) { + id = menuBar->idAt(i); + break; + } + } + // TODO: This doesn't work. Need way to figure out the QMenuItem underneath mouse pointer. + // id = menuBarItemAt(menuBar, pos); + if ( id != -1 ) + text = menuBar->text(id); + } + else + if (w->inherits("QPopupMenu")) { + QPopupMenu* popupMenu = dynamic_cast<QPopupMenu *>(w); + if (pos == QPoint()) { + for (uint i = 0; i < popupMenu->count(); ++i) + if (popupMenu->isItemActive(popupMenu->idAt(i))) { + id = popupMenu->idAt(i); + break; + } + } else + id = popupMenu->idAt(popupMenu->mapFromGlobal(pos)); + if ( id != -1 ) + text = popupMenu->text(id); + } + else + if (w->inherits("QTabBar")) { + QTabBar* tabBar = dynamic_cast<QTabBar *>(w); + QTab* tab = 0; + if (pos == QPoint()) + tab = tabBar->tabAt(tabBar->currentTab()); + else + tab = tabBar->selectTab(tabBar->mapFromGlobal(pos)); + if (tab) { + id = tab->identifier(); + text = tab->text(); + } + } + else + if (w->inherits("QListView")) { + QListView* lv = dynamic_cast<QListView *>(w); + QListViewItem* item = 0; + if (pos == QPoint()) + item = lv->currentItem(); + else + item = lv->itemAt(lv->viewport()->mapFromGlobal(pos)); + if (item) { + id = lv->itemPos(item); + text = item->text(0); + for (int col = 1; col < lv->columns(); ++col) + if (!item->text(col).isEmpty()) text += ". " + item->text(col); + } + } + else + if (w->inherits("QListBox")) { + QListBox* lb = dynamic_cast<QListBox *>(w); + // qt docs say coordinates are in "on-screen" coordinates. What does that mean? + QListBoxItem* item = 0; + if (pos == QPoint()) + item = lb->item(lb->currentItem()); + else + item = lb->itemAt(lb->mapFromGlobal(pos)); + if (item) { + id = lb->index(item); + text = item->text(); + } + } + else + if (w->inherits("QIconView")) { + QIconView* iv = dynamic_cast<QIconView *>(w); + QIconViewItem* item = 0; + if (pos == QPoint()) + item = iv->currentItem(); + else + item = iv->findItem(iv->viewportToContents(iv->viewport()->mapFromGlobal(pos))); + if (item) { + id = item->index(); + text = item->text(); + } + } + else + if (w->inherits("QTable")) { + QTable* tbl = dynamic_cast<QTable *>(w); + int row = -1; + int col = -1; + if (pos == QPoint()) { + row = tbl->currentRow(); + col = tbl->currentColumn(); + } else { + QPoint p = tbl->viewportToContents(tbl->viewport()->mapFromGlobal(pos)); + row = tbl->rowAt(p.y()); + col = tbl->columnAt(p.x()); + } + if (row >= 0 && col >= 0) { + id = (row * tbl->numCols()) + col; + text = tbl->text(row, col); + } + } + else + if (w->inherits("QGridView")) { + QGridView* gv = dynamic_cast<QGridView *>(w); + // TODO: QGridView does not have a "current" row or column. Don't think they can even get focus? + int row = -1; + int col = -1; + if (pos != QPoint()) { + QPoint p = gv->viewportToContents(gv->viewport()->mapFromGlobal(pos)); + row = gv->rowAt(p.y()); + col = gv->columnAt(p.x()); + } + if (row >= 0 && col >= 0) + id = (row * gv->numCols()) + col; + } + + if (pos == QPoint()) { + if ( w == d->m_prevFocusWidget && id == d->m_prevFocusId) return false; + d->m_prevFocusWidget = w; + d->m_prevFocusId = id; + } else { + if ( w == d->m_prevPointerWidget && id == d->m_prevPointerId) return false; + d->m_prevPointerWidget = w; + d->m_prevPointerId = id; + } + if (w == d->m_prevWidget && id == d->m_prevId) return false; + d->m_prevWidget = w; + d->m_prevId = id; + + // kdDebug() << " w = " << w << endl; + + d->m_cancelSpeakWidget = false; + emit customSpeakNewWidget(w, pos, d->m_speakFlags); + if (d->m_cancelSpeakWidget) return true; + + // Handle simple, single-part widgets. + if ( w->inherits("QButton") ) + text = dynamic_cast<QButton *>(w)->text(); + else + if (w->inherits("QComboBox")) + text = dynamic_cast<QComboBox *>(w)->currentText(); + else + if (w->inherits("QLineEdit")) + text = dynamic_cast<QLineEdit *>(w)->text(); + else + if (w->inherits("QTextEdit")) + text = dynamic_cast<QTextEdit *>(w)->text(); + else + if (w->inherits("QLabel")) + text = dynamic_cast<QLabel *>(w)->text(); + else + if (w->inherits("QGroupBox")) { + // TODO: Should calculate this number from font size? + if (w->mapFromGlobal(pos).y() < 30) + text = dynamic_cast<QGroupBox *>(w)->title(); + } +// else +// if (w->inherits("QWhatsThat")) { +// text = dynamic_cast<QWhatsThat *>(w)->text(); +// } + + text = text.stripWhiteSpace(); + if (!text.isEmpty()) { + if (text.right(1) == ".") + text += " "; + else + text += ". "; + } + if (d->m_speakFlags & SpeakTooltip || text.isEmpty()) { + // kdDebug() << "pos = " << pos << endl; + // QPoint p = w->mapFromGlobal(pos); + // kdDebug() << "p = " << p << endl; + QString t = QToolTip::textFor(w, pos); + t = t.stripWhiteSpace(); + if (!t.isEmpty()) { + if (t.right(1) != ".") t += "."; + text += t + " "; + } + } + + if (d->m_speakFlags & SpeakWhatsThis || text.isEmpty()) { + QString t = QWhatsThis::textFor(w, pos); + t = t.stripWhiteSpace(); + if (!t.isEmpty()) { + if (t.right(1) != ".") t += "."; + text += t + " "; + } + } + + if (d->m_speakFlags & SpeakDisabled) { + if (!w->isEnabled()) + text += i18n("A grayed widget", "Disabled. "); + } + + return sayWidget(text); +} + +bool KoSpeaker::sayWidget(const QString& msg) +{ + QString s = msg; + if (d->m_speakFlags & SpeakAccelerator) { + int amp = s.find("&"); + if (amp >= 0) { + QString acc = s.mid(++amp,1); + acc = acc.stripWhiteSpace(); + if (!acc.isEmpty()) + s += ". " + d->m_acceleratorPrefix + " " + acc + "."; + } + } + s.remove("&"); + if (QStyleSheet::mightBeRichText(s)) { + // kdDebug() << "richtext" << endl; + s.replace(QRegExp("</?[pbius]>"), ""); + s.replace(QRegExp("</?h\\d>"), ""); + s.replace(QRegExp("<(br|hr)>"), " "); + s.replace(QRegExp( + "</?(qt|center|li|pre|div|span|em|strong|big|small|sub|sup|code|tt|font|nobr|ul|ol|dl|dt)>"), ""); + s.replace(QRegExp("</?(table|tr|th|td).*>"), ""); + s.replace(QRegExp("</?a\\s.+>"), ""); + // Replace <img source="small|frame_text"> with "small frame_text image. " + s.replace(QRegExp("<img\\s.*(?:source=|src=)\"([^|\"]+)[|]?([^|\"]*)\">"), "\\1 \\2 image. "); + } + if (s.isEmpty()) return false; + s.replace("Ctrl+", i18n("control plus ")); + s.replace("Alt+", i18n("alt plus ")); + s.replace("+", i18n(" plus ")); + sayScreenReaderOutput(s, ""); + return true; +} + +// This doesn't work. Anybody know how to find the menu item underneath mouse pointer +// in a QMenuBar? +// int KoSpeaker::menuBarItemAt(QMenuBar* m, const QPoint& p) +// { +// for (uint i = 0; i < m->count(); i++) { +// int id = m->idAt(i); +// QMenuItem* mi = m->findItem(id); +// QWidget* w = mi->widget(); +// if (w->rect().contains(w->mapFromGlobal(p))) return id; +// } +// return -1; +// } + +/*static*/ bool KoSpeaker::isKttsdInstalled() +{ + KTrader::OfferList offers = KTrader::self()->query("DCOP/Text-to-Speech", "Name == 'KTTSD'"); + return (offers.count() > 0); +} + +bool KoSpeaker::startKttsd() +{ + DCOPClient *client = kapp->dcopClient(); + // If KTTSD not running, start it. + if (!client->isApplicationRegistered("kttsd")) + { + QString error; + if (kapp->startServiceByDesktopName("kttsd", QStringList(), &error)) { + kdDebug() << "KoSpeaker::startKttsd: error starting KTTSD service: " << error << endl; + d->m_enabled = false; + } else + d->m_enabled = true; + } else + d->m_enabled = true; + return d->m_enabled; +} + +QString KoSpeaker::getKttsdVersion() +{ + // Determine which version of KTTSD is running. Note that earlier versions of KSpeech interface + // did not support version() method, so we must manually marshall this call ourselves. + if (d->m_enabled) { + if (!d->m_versionChecked) { + DCOPClient *client = kapp->dcopClient(); + QByteArray data; + QCString replyType; + QByteArray replyData; + if ( client->call("kttsd", "KSpeech", "version()", data, replyType, replyData, true) ) { + QDataStream arg(replyData, IO_ReadOnly); + arg >> d->m_kttsdVersion; + kdDebug() << "KoSpeaker::startKttsd: KTTSD version = " << d->m_kttsdVersion << endl; + } + d->m_versionChecked = true; + } + } + return d->m_kttsdVersion; +} + +void KoSpeaker::sayScreenReaderOutput(const QString &msg, const QString &talker) +{ + if (msg.isEmpty()) return; + DCOPClient *client = kapp->dcopClient(); + QByteArray data; + QCString replyType; + QByteArray replyData; + QDataStream arg(data, IO_WriteOnly); + arg << msg << talker; + if ( !client->call("kttsd", "KSpeech", "sayScreenReaderOutput(QString,QString)", + data, replyType, replyData, true) ) { + kdDebug() << "KoSpeaker::sayScreenReaderOutput: failed" << endl; + } +} + +uint KoSpeaker::setText(const QString &text, const QString &talker) +{ + if (text.isEmpty()) return 0; + DCOPClient *client = kapp->dcopClient(); + QByteArray data; + QCString replyType; + QByteArray replyData; + QDataStream arg(data, IO_WriteOnly); + arg << text << talker; + uint jobNum = 0; + if ( !client->call("kttsd", "KSpeech", "setText(QString,QString)", + data, replyType, replyData, true) ) { + kdDebug() << "KoSpeaker::sayText: failed" << endl; + } else { + QDataStream arg2(replyData, IO_ReadOnly); + arg2 >> jobNum; + } + return jobNum; +} + +int KoSpeaker::appendText(const QString &text, uint jobNum /*=0*/) +{ + if (text.isEmpty()) return 0; + DCOPClient *client = kapp->dcopClient(); + QByteArray data; + QCString replyType; + QByteArray replyData; + QDataStream arg(data, IO_WriteOnly); + arg << text << jobNum; + int partNum = 0; + if ( !client->call("kttsd", "KSpeech", "appendText(QString,uint)", + data, replyType, replyData, true) ) { + kdDebug() << "KoSpeaker::appendText: failed" << endl; + } else { + QDataStream arg2(replyData, IO_ReadOnly); + arg2 >> partNum; + } + return partNum; +} + +void KoSpeaker::startText(uint jobNum /*=0*/) +{ + DCOPClient *client = kapp->dcopClient(); + QByteArray data; + QCString replyType; + QByteArray replyData; + QDataStream arg(data, IO_WriteOnly); + arg << jobNum; + if ( !client->call("kttsd", "KSpeech", "startText(uint)", + data, replyType, replyData, true) ) { + kdDebug() << "KoSpeaker::startText: failed" << endl; + } +} + +void KoSpeaker::removeText(uint jobNum /*=0*/) +{ + DCOPClient *client = kapp->dcopClient(); + QByteArray data; + QCString replyType; + QByteArray replyData; + QDataStream arg(data, IO_WriteOnly); + arg << jobNum; + if ( !client->call("kttsd", "KSpeech", "removeText(uint)", + data, replyType, replyData, true) ) { + kdDebug() << "KoSpeaker::removeText: failed" << endl; + } +} + diff --git a/lib/kofficecore/KoSpeaker.h b/lib/kofficecore/KoSpeaker.h new file mode 100644 index 00000000..6f8b430f --- /dev/null +++ b/lib/kofficecore/KoSpeaker.h @@ -0,0 +1,204 @@ +/** @file +* This file is part of the KDE/KOffice project. +* Copyright (C) 2005, Gary Cramblitt <garycramblitt@comcast.net> +* +* @author Gary Cramblitt <garycramblitt@comcast.net> +* @since KOffice 1.5 +* +* 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. +*/ + +#ifndef KOSPEAKER_H +#define KOSPEAKER_H + +// Qt includes. +#include <qobject.h> +#include <qstring.h> + +// KDE includes. +#include <ksharedptr.h> + +// KOffice includes. +#include <koffice_export.h> + +class QWidget; +class QPoint; +class KConfig; +class KoSpeakerPrivate; + +#define kospeaker KoSpeaker::koSpeaker() + +/** KoSpeaker is a singleton object that provides Text-to-Speech services for KOffice applications. + * When activated, it will speak the text of widgets under the mouse pointer and/or the + * the widget with focus. + * + * It also provides some methods for speaking text from documents. + * + * IMPORTANT: This class will be removed from KOffice when KOffice is converted to KDE4. + * It will be replaced with a proper screen reading capability using the AT-SPI. + * + * This is quite a hack and doesn't work reliably. The following are current problems: + * 1. Cannot speak menu items in a QMenuBar (top menu of app). + * 2. Doesn't understand every possible widget. + * + * This capability is @em not intended for completely blind users. Such users cannot use + * KDE 3.x anyway, since it lacks a screen reader. Instead, this capability is intended as + * an aid to users with other vision disabilities. + * + * KOffice applications can access this object using the kospeaker global. + */ +class KOFFICECORE_EXPORT KoSpeaker : public QObject, public KShared +{ + Q_OBJECT +public: + KoSpeaker(); + ~KoSpeaker(); + + /** Speech Options */ + enum SpeakFlags { + SpeakFocusWidget = 0x0001 /**< Speak widget with focus */, + SpeakPointerWidget = 0x0002 /**< Speak widget under mouse pointer */, + SpeakWhatsThis = 0x0004 /**< Speak Whats This if available */, + SpeakTooltip = 0x0008 /**< Speak tooltip if available */, + SpeakAccelerator = 0x0010 /**< Speak accelerator */, + SpeakDisabled = 0x0020 /**< Say 'disabled' if not enabled */ + }; + + /** + * Returns true if TTS services are available. If KTTSD daemon is not running, it is started. + * Will return false if: + * -- KTTSD daemon is not installed, or + * -- Was not able to start KTTSD daemon for some reason. + */ + bool isEnabled() const; + + /** + * Reads configuration options from @p config object and starts TTS if screen reader + * capability is requested. + * If KTTSD daemon is not installed, @ref isEnabled will return false. + * If screen reader is requested and KTTSD is installed, but not running, it will be started. + */ + void readConfig(KConfig* config); + + /** + * Given a widget @p w and its @p pos screen coordinates, tries to extract the text of the widget + * and speak it. If @p pos is not specified, and the widget has multiple parts (such as + * a QListView), uses the current part. + * Call @ref isEnabled to ensure TTS is available before calling this method. + */ + bool maybeSayWidget(QWidget* w, const QPoint& pos = QPoint()); + + /** + * Speak a @p msg that came from a widget, such as the widget's text label, tool tip, etc. + * Speaks using ScreenReaderOutput, which has highest priority, and therefore, should only be + * be used in very time-sensitive contexts and for short messages. + * Certain standard substitutions are performed on the message. For example, "Ctrl+" becomes + * "control plus". "Qt" markup is stripped. + * @returns true if anything is actually spoken. + * Call @ref isEnabled to ensure TTS is available before calling this method. + */ + bool sayWidget(const QString& msg); + + /** + * Cancels speaking of widget. Usually called by slots that receive @ref customSpeakNewWidget + * signal when they wish to speak the widget themselves. + */ + void cancelSpeakWidget(); + + /** + * Queue a @p msg as a speech text job. The text is encoded in the @p langCode language. + * Examples "en", "es", "en_US". If not specified, defaults to current desktop setting. + * If @p first is true and a job is already speaking, cancel it. + * If @p first is false, appends to the already queued job. + * If the KTTSD daemon is not already running, it is started. + */ + void queueSpeech(const QString& msg, const QString& langCode = QString(), bool first = true); + + /** + * Start speaking queued text job (if any). + */ + void startSpeech(); + + /** + * Returns whether the KTTSD deamon is installed in the system. If not, apps should disable + * or hide options/commands to speak. + */ + static bool isKttsdInstalled(); + + /** + * Returns the KoSpeaker object singleton. Apps should use "kospeaker" rather than this function + * directly. + */ + static KoSpeaker* koSpeaker() { return KSpkr; } + +signals: + /** + * This signal is emitted whenever a new widget has received focus or the mouse pointer + * has moved to a new widget. If a receiver wishes to handle speaking of the widget itself, + * it should call @ref cancelSpeakWidget() . + * @param w The widget. + * @param p Mouse pointer global coordinates, or in the case of a focus change (0,0). + * @param flags Speech options. @ref SpeakFlags. + * + * IMPORTANT: This signal is emitted from the @ref maybeSayWidget method. Slots who + * call maybeSayWidget should take care to avoid infinite recursion. + */ + void customSpeakNewWidget(QWidget* w, const QPoint& p, uint flags); + + /** + * This signal is emitted each polling interval when KoSpeaker did not speak the widget + * (either because it did not think the widget was a new one or because it did not + * understand the widget). If both mouse pointer and focus flags are set, it may + * emit twice per polling interval. + * @param w The widget. + * @param p Mouse pointer global coordinates, or in the case of a focus change (0,0). + * @param flags Speech options. @ref SpeakFlags. + * + * IMPORTANT: This signal is emitted frequently. Receivers should be coded efficiently. + */ + void customSpeakWidget(QWidget* w, const QPoint& p, uint flags); + +protected: + static KoSpeaker* KSpkr; + +private slots: + /** + * Tells the class to do it's stuff - ie. figure out + * which widget is under the mouse pointer or which has focus and speak it. + */ + void probe(); + +private: + // int menuBarItemAt(QMenuBar* m, const QPoint& p); + + // Start the KTTSD daemon if not already running. + bool startKttsd(); + // Return the KTTSD daemon version string. + QString getKttsdVersion(); + + // These methods correspond to dcop interface in kdelibs/interfaces/kspeech/kspeech.h. + // They use manual marshalling, instead of using kspeech_stub, because KOffice + // supports KDE 3.3 and above and kspeech.h didn't appear until 3.4. + void sayScreenReaderOutput(const QString &msg, const QString &talker); + uint setText(const QString &text, const QString &talker); + int appendText(const QString &text, uint jobNum=0); + void startText(uint jobNum=0); + void removeText(uint jobNum=0); + + KoSpeakerPrivate* d; +}; + +#endif // H_KOSPEAKER diff --git a/lib/kofficecore/KoStyleStack.cpp b/lib/kofficecore/KoStyleStack.cpp new file mode 100644 index 00000000..4b5b807b --- /dev/null +++ b/lib/kofficecore/KoStyleStack.cpp @@ -0,0 +1,296 @@ +/* This file is part of the KDE project + Copyright (c) 2003 Lukas Tinkl <lukas@kde.org> + Copyright (c) 2003 David Faure <faure@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 "KoStyleStack.h" +#include "KoUnit.h" +#include "KoDom.h" +#include "KoXmlNS.h" + +#include <kdebug.h> + +//#define DEBUG_STYLESTACK + +KoStyleStack::KoStyleStack() + : m_styleNSURI( KoXmlNS::style ), m_foNSURI( KoXmlNS::fo ) +{ + clear(); +} + +KoStyleStack::KoStyleStack( const char* styleNSURI, const char* foNSURI ) + : m_propertiesTagName( "properties" ), m_styleNSURI( styleNSURI ), m_foNSURI( foNSURI ) +{ + clear(); +} + +KoStyleStack::~KoStyleStack() +{ +} + +void KoStyleStack::clear() +{ + m_stack.clear(); +#ifdef DEBUG_STYLESTACK + kdDebug(30003) << "clear!" << endl; +#endif +} + +void KoStyleStack::save() +{ + m_marks.push( m_stack.count() ); +#ifdef DEBUG_STYLESTACK + kdDebug(30003) << "save (level " << m_marks.count() << ") -> index " << m_stack.count() << endl; +#endif +} + +void KoStyleStack::restore() +{ + Q_ASSERT( !m_marks.isEmpty() ); + int toIndex = m_marks.pop(); +#ifdef DEBUG_STYLESTACK + kdDebug(30003) << "restore (level " << m_marks.count()+1 << ") -> to index " << toIndex << endl; +#endif + Q_ASSERT( toIndex > -1 ); + Q_ASSERT( toIndex <= (int)m_stack.count() ); // If equal, nothing to remove. If greater, bug. + for ( int index = (int)m_stack.count() - 1; index >= toIndex; --index ) + m_stack.pop_back(); +} + +void KoStyleStack::pop() +{ + Q_ASSERT( !m_stack.isEmpty() ); + m_stack.pop_back(); +#ifdef DEBUG_STYLESTACK + kdDebug(30003) << "pop -> count=" << m_stack.count() << endl; +#endif +} + +void KoStyleStack::push( const QDomElement& style ) +{ + m_stack.append( style ); +#ifdef DEBUG_STYLESTACK + kdDebug(30003) << "pushed " << style.attributeNS( m_styleNSURI, "name", QString::null ) << " -> count=" << m_stack.count() << endl; +#endif +} + +bool KoStyleStack::hasAttribute( const QString& name, const QString& detail ) const +{ + QString fullName( name ); + if ( !detail.isEmpty() ) + { + fullName += '-'; + fullName += detail; + } + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = (*it).namedItem( "style:"+m_propertiesTagName ).toElement(); + if ( properties.hasAttribute( name ) || + ( !detail.isEmpty() && properties.hasAttribute( fullName ) ) ) + return true; + } + return false; +} + +QString KoStyleStack::attribute( const QString& name, const QString& detail ) const +{ + QString fullName( name ); + if ( !detail.isEmpty() ) + { + fullName += '-'; + fullName += detail; + } + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = (*it).namedItem( "style:"+m_propertiesTagName ).toElement(); + if ( properties.hasAttribute( name ) ) + return properties.attribute( name ); + if ( !detail.isEmpty() && properties.hasAttribute( fullName ) ) + return properties.attribute( fullName ); + } + return QString::null; +} + +QString KoStyleStack::attributeNS( const char* nsURI, const char* name, const char* detail ) const +{ + QString fullName( name ); + if ( detail ) + { + fullName += '-'; + fullName += detail; + } + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = KoDom::namedItemNS( *it, m_styleNSURI, m_propertiesTagName ); + if ( properties.hasAttributeNS( nsURI, name ) ) + return properties.attributeNS( nsURI, name, QString::null ); + if ( detail && properties.hasAttributeNS( nsURI, fullName ) ) + return properties.attributeNS( nsURI, fullName, QString::null ); + } + return QString::null; +} + +bool KoStyleStack::hasAttributeNS( const char* nsURI, const char* name, const char* detail ) const +{ + QString fullName( name ); + if ( detail ) + { + fullName += '-'; + fullName += detail; + } + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = KoDom::namedItemNS( *it, m_styleNSURI, m_propertiesTagName ); + if ( properties.hasAttributeNS( nsURI, name ) || + ( detail && properties.hasAttributeNS( nsURI, fullName ) ) ) + return true; + } + return false; +} + +// Font size is a bit special. "115%" applies to "the fontsize of the parent style". +// This can be generalized though (hasAttributeThatCanBePercentOfParent() ? :) +// Although, if we also add support for fo:font-size-rel here then it's not general anymore. +double KoStyleStack::fontSize() const +{ + const QString name = "font-size"; + double percent = 1; + QValueList<QDomElement>::ConstIterator it = m_stack.end(); // reverse iterator + + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = KoDom::namedItemNS( *it, m_styleNSURI, m_propertiesTagName ).toElement(); + if ( properties.hasAttributeNS( m_foNSURI, name ) ) { + const QString value = properties.attributeNS( m_foNSURI, name, QString::null ); + if ( value.endsWith( "%" ) ) + percent *= value.left( value.length() - 1 ).toDouble() / 100.0; + else + return percent * KoUnit::parseValue( value ); // e.g. 12pt + } + } + return 0; +} + +bool KoStyleStack::hasChildNode(const QString & name) const +{ + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = (*it).namedItem( "style:"+m_propertiesTagName ).toElement(); + if ( !properties.namedItem( name ).isNull() ) + return true; + } + + return false; +} + +QDomElement KoStyleStack::childNode(const QString & name) const +{ + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = (*it).namedItem( "style:"+m_propertiesTagName ).toElement(); + if ( !properties.namedItem( name ).isNull() ) + return properties.namedItem( name ).toElement(); + } + + return QDomElement(); // a null element +} + +bool KoStyleStack::hasChildNodeNS( const char* nsURI, const char* localName ) const +{ + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = KoDom::namedItemNS( *it, m_styleNSURI, m_propertiesTagName ); + if ( !KoDom::namedItemNS( properties, nsURI, localName ).isNull() ) + return true; + } + + return false; +} + +QDomElement KoStyleStack::childNodeNS( const char* nsURI, const char* localName) const +{ + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + + while ( it != m_stack.begin() ) + { + --it; + QDomElement properties = KoDom::namedItemNS( *it, m_styleNSURI, m_propertiesTagName ); + QDomElement e = KoDom::namedItemNS( properties, nsURI, localName ); + if ( !e.isNull() ) + return e; + } + + return QDomElement(); // a null element +} + +bool KoStyleStack::isUserStyle( const QDomElement& e, const QString& family ) const +{ + if ( e.attributeNS( m_styleNSURI, "family", QString::null ) != family ) + return false; + const QDomElement parent = e.parentNode().toElement(); + //kdDebug(30003) << k_funcinfo << "tagName=" << e.tagName() << " parent-tagName=" << parent.tagName() << endl; + return parent.localName() == "styles" /*&& parent.namespaceURI() == KoXmlNS::office*/; +} + +QString KoStyleStack::userStyleName( const QString& family ) const +{ + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + //kdDebug(30003) << k_funcinfo << (*it).attributeNS( m_styleNSURI, "name", QString::null) << endl; + if ( isUserStyle( *it, family ) ) + return (*it).attributeNS( m_styleNSURI, "name", QString::null ); + } + // Can this ever happen? + return "Standard"; +} + +QString KoStyleStack::userStyleDisplayName( const QString& family ) const +{ + QValueList<QDomElement>::ConstIterator it = m_stack.end(); + while ( it != m_stack.begin() ) + { + --it; + //kdDebug(30003) << k_funcinfo << (*it).attributeNS( m_styleNSURI, "display-name") << endl; + if ( isUserStyle( *it, family ) ) + return (*it).attributeNS( m_styleNSURI, "display-name", QString::null ); + } + return QString::null; // no display name, this can happen since it's optional +} + +void KoStyleStack::setTypeProperties( const char* typeProperties ) +{ + m_propertiesTagName = typeProperties == 0 ? QCString( "properties" ) : ( QCString( typeProperties ) + "-properties" ); +} diff --git a/lib/kofficecore/KoStyleStack.h b/lib/kofficecore/KoStyleStack.h new file mode 100644 index 00000000..f3c5faf1 --- /dev/null +++ b/lib/kofficecore/KoStyleStack.h @@ -0,0 +1,201 @@ +/* This file is part of the KDE project + Copyright (c) 2003 Lukas Tinkl <lukas@kde.org> + Copyright (c) 2003 David Faure <faure@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. +*/ + +#ifndef KOSTYLESTACK_H +#define KOSTYLESTACK_H + + +#include <qvaluelist.h> +#include <qdom.h> +#include <qvaluestack.h> + +#include <kdemacros.h> + +#include "koffice_export.h" + +/** + * @brief This class implements a stack for the different styles of an object. + * + * There can be several styles that are valid for one object. For example + * a textobject on a page has styles 'pr3' and 'P7' and a paragraph in + * that textobject has styles 'P1' and 'T3'. And some styles even have + * parent-styles... + * + * If you want to know if there is, for example, the attribute 'fo:font-family' + * for this paragraph, you have to look into style 'T3', 'P1', 'P7' and 'pr3'. + * When you find this attribute in one style you have to stop processing the list + * and take the found attribute for this object. + * + * This is what this class does. You can push styles on the stack while walking + * through the xml-tree to your object and then ask the stack if any of the styles + * provides a certain attribute. The stack will search from top to bottom, i.e. + * in our example from 'T3' to 'pr3' and return the first occurrence of the wanted + * attribute. + * + * So this is some sort of inheritance where the styles on top of the stack overwrite + * the same attribute of a lower style on the stack. + * + * In general though, you wouldn't use push/pop directly, but KoOasisLoadingContext::fillStyleStack + * or KoOasisLoadingContext::addStyles to automatically push a style and all its + * parent styles onto the stack. + */ +class KOFFICECORE_EXPORT KoStyleStack +{ +public: + /** + * Create a OASIS style stack + */ + KoStyleStack(); + /** + * Create a style stack based on other namespaces than OASIS - used for OOo-1.1 import. + */ + KoStyleStack( const char* styleNSURI, const char* foNSURI ); + virtual ~KoStyleStack(); + + /** + * Clears the complete stack. + */ + void clear(); + + /** + * Save the current state of the stack. Any items added between + * this call and its corresponding restore() will be removed when calling restore(). + */ + void save(); + + /** + * Restore the stack to the state it was at the corresponding save() call. + */ + void restore(); + + /** + * Removes the style on top of the stack. + */ + void pop(); + + /** + * Pushes the new style onto the stack. + */ + void push( const QDomElement& style ); + + /** + * Check if any of the styles on the stack has an attribute called 'name'-'detail' + * where detail is e.g. left, right, top or bottom. + * This allows to also find 'name' alone (e.g. padding implies padding-left, padding-right etc.) + */ + bool hasAttribute( const QString& name, const QString& detail = QString::null ) const KDE_DEPRECATED; + + /** + * Search for the attribute called 'name', starting on top of the stack, + * and return it. + */ + QString attribute( const QString& name, const QString& detail = QString::null ) const KDE_DEPRECATED; + + /** + * Check if any of the styles on the stack has an attribute called 'name'-'detail' + * where detail is e.g. left, right, top or bottom. + * This allows to also find 'name' alone (e.g. padding implies padding-left, padding-right etc.) + */ + bool hasAttributeNS( const char* nsURI, const char* localName, const char* detail = 0 ) const; + + /** + * Search for the attribute called 'name', starting on top of the stack, + * and return it. + */ + QString attributeNS( const char* nsURI, const char* localName, const char* detail = 0 ) const; + + /** + * Check if any of the styles on the stack has a child node called 'name'. + */ + bool hasChildNode( const QString & name ) const KDE_DEPRECATED; + + /** + * Search for a child node called 'name', starting on top of the stack, + * and return it. + */ + QDomElement childNode( const QString & name ) const KDE_DEPRECATED; + + /** + * Check if any of the styles on the stack has a child element called 'localName' in the namespace 'nsURI'. + */ + bool hasChildNodeNS( const char* nsURI, const char* localName ) const; + + /** + * Search for a child element which has a child element called 'localName' + * in the namespace 'nsURI' starting on top of the stack, + * and return it. + */ + QDomElement childNodeNS( const char* nsURI, const char* localName ) const; + + /** + * Special case for the current font size, due to special handling of fo:font-size="115%". + */ + double fontSize() const; + + /** + * Return the name of the style specified by the user, + * i.e. not an auto style. + * This is used to know e.g. which user-style is associated with the current paragraph. + * There could be none though. + */ + QString userStyleName( const QString& family ) const; + + /** + * Return the display name of the style specified by the user, + * i.e. not an auto style + */ + QString userStyleDisplayName( const QString& family ) const; + + /** + * Set the type of properties that will be looked for. + * For instance setTypeProperties("paragraph") will make hasAttribute() and attribute() + * look into "paragraph-properties". + * If @p typeProperties is 0, the stylestack is resetted to look for "properties" + * as it does by default. + */ + void setTypeProperties( const char* typeProperties ); + +private: + bool isUserStyle( const QDomElement& e, const QString& family ) const; + +private: + /// For save/restore: stack of "marks". Each mark is an index in m_stack. + QValueStack<int> m_marks; + + /** + * We use QValueList instead of QValueStack because we need access to all styles + * not only the top one. + */ + QValueList<QDomElement> m_stack; + + QCString m_propertiesTagName; + + const char* m_styleNSURI; + const char* m_foNSURI; + + class KoStyleStackPrivate; + KoStyleStackPrivate *d; + + // forbidden + void operator=( const KoStyleStack& ); + KoStyleStack( const KoStyleStack& ); +}; + +#endif /* KOSTYLESTACK_H */ diff --git a/lib/kofficecore/KoTemplates.cpp b/lib/kofficecore/KoTemplates.cpp new file mode 100644 index 00000000..7fbf3a99 --- /dev/null +++ b/lib/kofficecore/KoTemplates.cpp @@ -0,0 +1,379 @@ +/* This file is part of the KDE project + Copyright (C) 2000 Werner Trobin <trobin@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 <KoTemplates.h> + +#include <qdir.h> +#include <qimage.h> +#include <qprinter.h> + +#include <kdesktopfile.h> +#include <ksimpleconfig.h> +#include <kdebug.h> +#include <kdeversion.h> +#include <kinstance.h> +#include <ksavefile.h> +#include <kstandarddirs.h> +#include <kiconloader.h> +#include <kio/netaccess.h> +#include <klocale.h> + +#include <stdlib.h> + + +KoTemplate::KoTemplate(const QString &name, const QString &description, const QString &file, + const QString &picture, const QString &fileName, const QString &_measureSystem, + bool hidden, bool touched) : + m_name(name), m_descr(description), m_file(file), m_picture(picture), m_fileName(fileName), + m_hidden(hidden), m_touched(touched), m_cached(false), m_measureSystem(_measureSystem) +{ +} + +const QPixmap &KoTemplate::loadPicture( KInstance* instance ) { + + if(m_cached) + return m_pixmap; + m_cached=true; + if ( m_picture[ 0 ] == '/' ) + { + // ### TODO: use the class KoPicture instead of QImage to support non-image pictures + QImage img( m_picture ); + if (img.isNull()) { + kdWarning() << "Couldn't find icon " << m_picture << endl; + m_pixmap=QPixmap(); + return m_pixmap; + } + const int maxHeightWidth = 128; // ### TODO: some people would surely like to have 128x128 + if (img.width() > maxHeightWidth || img.height() > maxHeightWidth) { + img = img.smoothScale( maxHeightWidth, maxHeightWidth, QImage::ScaleMax ); + } + m_pixmap.convertFromImage(img); + return m_pixmap; + } else { // relative path + m_pixmap = instance->iconLoader()->loadIcon( m_picture, KIcon::Desktop, 128 ); + return m_pixmap; + } +} + + +KoTemplateGroup::KoTemplateGroup(const QString &name, const QString &dir, + int _sortingWeight, bool touched) : + m_name(name), m_touched(touched), m_sortingWeight(_sortingWeight) +{ + m_dirs.append(dir); + m_templates.setAutoDelete(true); +} + +bool KoTemplateGroup::isHidden() const { + + QPtrListIterator<KoTemplate> it(m_templates); + bool hidden=true; + while(it.current()!=0L && hidden) { + hidden=it.current()->isHidden(); + ++it; + } + return hidden; +} + +void KoTemplateGroup::setHidden(bool hidden) const { + + QPtrListIterator<KoTemplate> it(m_templates); + for( ; it.current()!=0L; ++it) + it.current()->setHidden(hidden); + m_touched=true; +} + +bool KoTemplateGroup::add(KoTemplate *t, bool force, bool touch) { + + KoTemplate *myTemplate=find(t->name()); + if(myTemplate==0L) { + m_templates.append(t); + m_touched=touch; + return true; + } + else if(myTemplate && force) { + //kdDebug() << "removing :" << myTemplate->fileName() << endl; + QFile::remove( myTemplate->fileName() ); + QFile::remove( myTemplate->picture() ); + QFile::remove( myTemplate->file() ); + m_templates.removeRef(myTemplate); + m_templates.append(t); + m_touched=touch; + return true; + } + return false; +} + +KoTemplate *KoTemplateGroup::find(const QString &name) const { + + QPtrListIterator<KoTemplate> it(m_templates); + while(it.current() && it.current()->name()!=name) + ++it; + return it.current(); +} + + +KoTemplateTree::KoTemplateTree(const QCString &templateType, + KInstance *instance, bool readTree) : + m_templateType(templateType), m_instance(instance), m_defaultGroup(0L), + m_defaultTemplate(0L) { + + m_groups.setAutoDelete(true); + if(readTree) + readTemplateTree(); +} + +void KoTemplateTree::readTemplateTree() { + + readGroups(); + readTemplates(); +} + +void KoTemplateTree::writeTemplateTree() { + QString localDir=m_instance->dirs()->saveLocation(m_templateType); + + for(KoTemplateGroup *group=m_groups.first(); group!=0L; group=m_groups.next()) { + //kdDebug() << "---------------------------------" << endl; + //kdDebug() << "group: " << group->name() << endl; + + bool touched=false; + for(KoTemplate *t=group->first(); t!=0L && !touched && !group->touched(); t=group->next()) + touched=t->touched(); + + if(group->touched() || touched) { + //kdDebug() << "touched" << endl; + if(!group->isHidden()) { + //kdDebug() << "not hidden" << endl; + KStandardDirs::makeDir(localDir+group->name()); // create the local group dir + } + else { + //kdDebug() << "hidden" << endl; + if(group->dirs().count()==1 && !group->dirs().grep(localDir).isEmpty()) { + //kdDebug() << "local only" << endl; + KIO::NetAccess::del(group->dirs().first(), 0); + //kdDebug() << "removing: " << group->dirs().first() << endl; + } + else { + //kdDebug() << "global" << endl; + KStandardDirs::makeDir(localDir+group->name()); + } + } + } + for(KoTemplate *t=group->first(); t!=0L; t=group->next()) { + if(t->touched()) { + //kdDebug() << "++template: " << t->name() << endl; + writeTemplate(t, group, localDir); + } + if(t->isHidden() && t->touched() ) { + //kdDebug() << "+++ delete local template ##############" << endl; + writeTemplate(t, group, localDir); + QFile::remove(t->file()); + QFile::remove(t->picture()); + } + } + } +} + +void KoTemplateTree::add(KoTemplateGroup *g) { + + KoTemplateGroup *group=find(g->name()); + if(group==0L) + m_groups.append(g); + else + group->addDir(g->dirs().first()); // "...there can be only one..." (Queen) +} + +KoTemplateGroup *KoTemplateTree::find(const QString &name) const { + + QPtrListIterator<KoTemplateGroup> it(m_groups); + while(it.current() && it.current()->name()!=name) + ++it; + return it.current(); +} + +void KoTemplateTree::readGroups() { + + QStringList dirs = m_instance->dirs()->resourceDirs(m_templateType); + for(QStringList::ConstIterator it=dirs.begin(); it!=dirs.end(); ++it) { + //kdDebug() << "dir: " << *it << endl; + QDir dir(*it); + // avoid the annoying warning + if(!dir.exists()) + continue; + dir.setFilter(QDir::Dirs); + QStringList templateDirs=dir.entryList(); + for(QStringList::ConstIterator tdirIt=templateDirs.begin(); tdirIt!=templateDirs.end(); ++tdirIt) { + if(*tdirIt=="." || *tdirIt=="..") // we don't want to check those dirs :) + continue; + QDir templateDir(*it+*tdirIt); + QString name=*tdirIt; + QString defaultTab; + int sortingWeight = 1000; + if(templateDir.exists(".directory")) { + KSimpleConfig config(templateDir.absPath()+"/.directory", true); + config.setDesktopGroup(); + name=config.readEntry("Name"); + defaultTab=config.readEntry("X-KDE-DefaultTab"); + sortingWeight=config.readNumEntry("X-KDE-SortingWeight", 1000); + //kdDebug() << "name: " << name <<endl; + } + KoTemplateGroup *g=new KoTemplateGroup(name, *it+*tdirIt+QChar('/'), sortingWeight); + add(g); + if(defaultTab=="true") + m_defaultGroup=g; + } + } +} + +void KoTemplateTree::readTemplates() { + QString dontShow = "imperial"; + + if(KGlobal::locale()->pageSize() == QPrinter::Letter) { + dontShow = "metric"; + } + + QPtrListIterator<KoTemplateGroup> groupIt(m_groups); + for( ; groupIt.current()!=0L; ++groupIt) { + QStringList dirs=groupIt.current()->dirs(); + for(QStringList::ConstIterator it=dirs.begin(); it!=dirs.end(); ++it) { + QDir d(*it); + if( !d.exists() ) + continue; + QStringList files=d.entryList( QDir::Files | QDir::Readable, QDir::Name ); + for(unsigned int i=0; i<files.count(); ++i) { + QString filePath = *it + files[i]; + //kdDebug() << "filePath: " << filePath << endl; + QString icon; + QString text; + QString description; + QString hidden_str; + QString fileName; + bool hidden=false; + bool defaultTemplate = false; + QString templatePath; + QString measureSystem; + // If a desktop file, then read the name from it. + // Otherwise (or if no name in it?) use file name + if (KDesktopFile::isDesktopFile(filePath)) { + KSimpleConfig config(filePath, true); + config.setDesktopGroup(); + if (config.readEntry("Type")=="Link") { + text=config.readEntry("Name"); + fileName=filePath; + description=config.readEntry("Comment"); + //kdDebug() << "name: " << text << endl; + icon=config.readEntry("Icon"); + if(icon[0]!='/' && // allow absolute paths for icons + QFile::exists(*it+icon)) // allow icons from icontheme + icon=*it+icon; + //kdDebug() << "icon2: " << icon << endl; + hidden=config.readBoolEntry("X-KDE-Hidden", false); + defaultTemplate = config.readBoolEntry("X-KDE-DefaultTemplate", false); + measureSystem=config.readEntry("X-KDE-MeasureSystem").lower(); + + // Don't add a template that is for the wrong measure system + if(measureSystem == dontShow) + continue; + + //kdDebug() << "hidden: " << hidden_str << endl; + templatePath=config.readPathEntry("URL"); + //kdDebug() << "Link to : " << templatePath << endl; + if(templatePath[0]!='/') { + if(templatePath.left(6)=="file:/") // I doubt this will happen + templatePath=templatePath.right(templatePath.length()-6); + //else + // kdDebug() << "dirname=" << *it << endl; + templatePath=*it+templatePath; + //kdDebug() << "templatePath: " << templatePath << endl; + } + } else + continue; // Invalid + } + // The else if and the else branch are here for compat. with the old system + else if ( files[i].right(4) != ".png" ) + // Ignore everything that is not a PNG file + continue; + else { + // Found a PNG file - the template must be here in the same dir. + icon = filePath; + QFileInfo fi(filePath); + text = fi.baseName(); + templatePath = filePath; // Note that we store the .png file as the template ! + // That's the way it's always been done. Then the app replaces the extension... + } + KoTemplate *t=new KoTemplate(text, description, templatePath, icon, fileName, + measureSystem, hidden); + groupIt.current()->add(t, false, false); // false -> we aren't a "user", false -> don't + // "touch" the group to avoid useless + // creation of dirs in .kde/blah/... + if ( defaultTemplate ) + m_defaultTemplate = t; + } + } + } +} + +void KoTemplateTree::writeTemplate(KoTemplate *t, KoTemplateGroup *group, + const QString &localDir) { + QString fileName; + if ( t->isHidden() ) + { + fileName = t->fileName(); + // try to remove the file + if ( QFile::remove(fileName) || !QFile::exists(fileName) ) + { + QFile::remove( t->name() ); + QFile::remove( t->picture() ); + return; + } + } + // be sure that the template's file name is unique so we don't overwrite an other + QString const path = localDir + group->name() + '/'; + QString const name = KoTemplates::stripWhiteSpace( t->name() ); + fileName = path + name + ".desktop"; + if ( t->isHidden() && QFile::exists(fileName) ) + return; + QString fill; + while ( KIO::NetAccess::exists( fileName, true, 0 ) ) + { + fill += '_'; + fileName = path + fill + name + ".desktop"; + } + + KSimpleConfig config( fileName ); + config.setDesktopGroup(); + config.writeEntry("Type", "Link"); + config.writePathEntry("URL", t->file()); + config.writeEntry("Name", t->name()); + config.writeEntry("Icon", t->picture()); + config.writeEntry("X-KDE-Hidden", t->isHidden()); +} + +namespace KoTemplates { +QString stripWhiteSpace(const QString &string) { + + QString ret; + for(unsigned int i=0; i<string.length(); ++i) { + QChar tmp(string[i]); + if(!tmp.isSpace()) + ret+=tmp; + } + return ret; +} +} diff --git a/lib/kofficecore/KoTemplates.h b/lib/kofficecore/KoTemplates.h new file mode 100644 index 00000000..4c9e3133 --- /dev/null +++ b/lib/kofficecore/KoTemplates.h @@ -0,0 +1,150 @@ +/* + This file is part of the KDE project + Copyright (C) 2000 Werner Trobin <trobin@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. +*/ + +#ifndef koTemplates_h +#define koTemplates_h + +#include <qptrlist.h> +#include <qstringlist.h> +#include <qpixmap.h> +#include <koffice_export.h> + +class KInstance; + +/** @internal */ +class KOFFICECORE_EXPORT KoTemplate { + +public: + KoTemplate(const QString &name, + const QString &description=QString::null, + const QString &file=QString::null, + const QString &picture=QString::null, + const QString &fileName=QString::null, + const QString &_measureSystem=QString::null, + bool hidden=false, bool touched=false); + ~KoTemplate() {} + + QString name() const { return m_name; } + QString description() const { return m_descr; }; + QString file() const { return m_file; } + QString picture() const { return m_picture; } + QString fileName() const { return m_fileName; } + const QPixmap &loadPicture( KInstance* instance ); + + bool isHidden() const { return m_hidden; } + void setHidden(bool hidden=true) { m_hidden=hidden; m_touched=true; } + + bool touched() const { return m_touched; } + + QString measureSystem() const { return m_measureSystem; } + void setMeasureSystem(const QString& system) { m_measureSystem = system; } + +private: + QString m_name, m_descr, m_file, m_picture, m_fileName; + bool m_hidden; + mutable bool m_touched; + bool m_cached; + QPixmap m_pixmap; + QString m_measureSystem; +}; + + +class KOFFICECORE_EXPORT KoTemplateGroup { + +public: + KoTemplateGroup(const QString &name, + const QString &dir=QString::null, + int _sortingWeight=0, + bool touched=false); + ~KoTemplateGroup() {} + + QString name() const { return m_name; } + QStringList dirs() const { return m_dirs; } + void addDir(const QString &dir) { m_dirs.append(dir); m_touched=true; } + int sortingWeight() const { return m_sortingWeight; } + void setSortingWeight(int weight) { m_sortingWeight = weight; } + /// If all children are hidden, we are hidden too + bool isHidden() const; + /// if we should hide, we hide all the children + void setHidden(bool hidden=true) const; + + KoTemplate *first() { return m_templates.first(); } + KoTemplate *next() { return m_templates.next(); } + KoTemplate *last() { return m_templates.last(); } + KoTemplate *prev() { return m_templates.prev(); } + KoTemplate *current() { return m_templates.current(); } + + bool add(KoTemplate *t, bool force=false, bool touch=true); + KoTemplate *find(const QString &name) const; + + bool touched() const { return m_touched; } + +private: + QString m_name; + QStringList m_dirs; + QPtrList<KoTemplate> m_templates; + mutable bool m_touched; + int m_sortingWeight; +}; + + +class KoTemplateTree { + +public: + KoTemplateTree(const QCString &templateType, KInstance *instance, + bool readTree=false); + ~KoTemplateTree() {} + + QCString templateType() const { return m_templateType; } + KInstance *instance() const { return m_instance; } + void readTemplateTree(); + void writeTemplateTree(); + + KoTemplateGroup *first() { return m_groups.first(); } + KoTemplateGroup *next() { return m_groups.next(); } + KoTemplateGroup *last() { return m_groups.last(); } + KoTemplateGroup *prev() { return m_groups.prev(); } + KoTemplateGroup *current() { return m_groups.current(); } + + void add(KoTemplateGroup *g); + KoTemplateGroup *find(const QString &name) const; + + KoTemplateGroup *defaultGroup() const { return m_defaultGroup; } + KoTemplate *defaultTemplate() const { return m_defaultTemplate; } + +private: + void readGroups(); + void readTemplates(); + void writeTemplate(KoTemplate *t, KoTemplateGroup *group, + const QString &localDir); + + QCString m_templateType; + KInstance *m_instance; + QPtrList<KoTemplateGroup> m_groups; + KoTemplateGroup *m_defaultGroup; + KoTemplate *m_defaultTemplate; +}; + + +namespace KoTemplates { +QString stripWhiteSpace(const QString &string); +} + +#endif diff --git a/lib/kofficecore/KoUnit.cpp b/lib/kofficecore/KoUnit.cpp new file mode 100644 index 00000000..fc459c51 --- /dev/null +++ b/lib/kofficecore/KoUnit.cpp @@ -0,0 +1,215 @@ +/* This file is part of the KDE project + Copyright (C) 2001 David Faure <faure@kde.org> + Copyright (C) 2004, Nicolas GOUTTE <goutte@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 <KoGlobal.h> +#include "KoUnit.h" +#include <KoXmlWriter.h> + +#include <klocale.h> +#include <kglobal.h> +#include <kdebug.h> + +#include <qregexp.h> +#include <qdom.h> + +QStringList KoUnit::listOfUnitName() +{ + QStringList lst; + for ( uint i = 0 ; i <= KoUnit::U_LASTUNIT ; ++i ) + { + KoUnit::Unit unit = static_cast<KoUnit::Unit>( i ); + lst.append( KoUnit::unitDescription( unit ) ); + } + return lst; +} + +QString KoUnit::unitDescription( Unit _unit ) +{ + switch ( _unit ) + { + case KoUnit::U_MM: + return i18n("Millimeters (mm)"); + case KoUnit::U_CM: + return i18n("Centimeters (cm)"); + case KoUnit::U_DM: + return i18n("Decimeters (dm)"); + case KoUnit::U_INCH: + return i18n("Inches (in)"); + case KoUnit::U_PI: + return i18n("Pica (pi)"); + case KoUnit::U_DD: + return i18n("Didot (dd)"); + case KoUnit::U_CC: + return i18n("Cicero (cc)"); + case KoUnit::U_PT: + return i18n("Points (pt)" ); + default: + return i18n("Error!"); + } +} + +double KoUnit::toUserValue( double ptValue, Unit unit ) +{ + switch ( unit ) { + case U_MM: + return toMM( ptValue ); + case U_CM: + return toCM( ptValue ); + case U_DM: + return toDM( ptValue ); + case U_INCH: + return toInch( ptValue ); + case U_PI: + return toPI( ptValue ); + case U_DD: + return toDD( ptValue ); + case U_CC: + return toCC( ptValue ); + case U_PT: + default: + return toPoint( ptValue ); + } +} + +double KoUnit::ptToUnit( const double ptValue, const Unit unit ) +{ + switch ( unit ) + { + case U_MM: + return POINT_TO_MM( ptValue ); + case U_CM: + return POINT_TO_CM( ptValue ); + case U_DM: + return POINT_TO_DM( ptValue ); + case U_INCH: + return POINT_TO_INCH( ptValue ); + case U_PI: + return POINT_TO_PI( ptValue ); + case U_DD: + return POINT_TO_DD( ptValue ); + case U_CC: + return POINT_TO_CC( ptValue ); + case U_PT: + default: + return ptValue; + } +} + +QString KoUnit::toUserStringValue( double ptValue, Unit unit ) +{ + return KGlobal::locale()->formatNumber( toUserValue( ptValue, unit ) ); +} + +double KoUnit::fromUserValue( double value, Unit unit ) +{ + switch ( unit ) { + case U_MM: + return MM_TO_POINT( value ); + case U_CM: + return CM_TO_POINT( value ); + case U_DM: + return DM_TO_POINT( value ); + case U_INCH: + return INCH_TO_POINT( value ); + case U_PI: + return PI_TO_POINT( value ); + case U_DD: + return DD_TO_POINT( value ); + case U_CC: + return CC_TO_POINT( value ); + case U_PT: + default: + return value; + } +} + +double KoUnit::fromUserValue( const QString& value, Unit unit, bool* ok ) +{ + return fromUserValue( KGlobal::locale()->readNumber( value, ok ), unit ); +} + +double KoUnit::parseValue( QString value, double defaultVal ) +{ + value.simplifyWhiteSpace(); + value.remove( ' ' ); + + if( value.isEmpty() ) + return defaultVal; + + int index = value.find( QRegExp( "[a-z]+$" ) ); + if ( index == -1 ) + return value.toDouble(); + + QString unit = value.mid( index ); + value.truncate ( index ); + double val = value.toDouble(); + + if ( unit == "pt" ) + return val; + + bool ok; + Unit u = KoUnit::unit( unit, &ok ); + if( ok ) + return fromUserValue( val, u ); + + if( unit == "m" ) + return fromUserValue( val * 10.0, U_DM ); + else if( unit == "km" ) + return fromUserValue( val * 10000.0, U_DM ); + kdWarning() << "KoUnit::parseValue: Unit " << unit << " is not supported, please report." << endl; + + // TODO : add support for mi/ft ? + return defaultVal; +} + +KoUnit::Unit KoUnit::unit( const QString &_unitName, bool* ok ) +{ + if ( ok ) + *ok = true; + if ( _unitName == QString::fromLatin1( "mm" ) ) return U_MM; + if ( _unitName == QString::fromLatin1( "cm" ) ) return U_CM; + if ( _unitName == QString::fromLatin1( "dm" ) ) return U_DM; + if ( _unitName == QString::fromLatin1( "in" ) + || _unitName == QString::fromLatin1("inch") /*compat*/ ) return U_INCH; + if ( _unitName == QString::fromLatin1( "pi" ) ) return U_PI; + if ( _unitName == QString::fromLatin1( "dd" ) ) return U_DD; + if ( _unitName == QString::fromLatin1( "cc" ) ) return U_CC; + if ( _unitName == QString::fromLatin1( "pt" ) ) return U_PT; + if ( ok ) + *ok = false; + return U_PT; +} + +QString KoUnit::unitName( Unit _unit ) +{ + if ( _unit == U_MM ) return QString::fromLatin1( "mm" ); + if ( _unit == U_CM ) return QString::fromLatin1( "cm" ); + if ( _unit == U_DM ) return QString::fromLatin1( "dm" ); + if ( _unit == U_INCH ) return QString::fromLatin1( "in" ); + if ( _unit == U_PI ) return QString::fromLatin1( "pi" ); + if ( _unit == U_DD ) return QString::fromLatin1( "dd" ); + if ( _unit == U_CC ) return QString::fromLatin1( "cc" ); + return QString::fromLatin1( "pt" ); +} + +void KoUnit::saveOasis(KoXmlWriter* settingsWriter, Unit _unit) +{ + settingsWriter->addConfigItem( "unit", unitName(_unit) ); +} diff --git a/lib/kofficecore/KoUnit.h b/lib/kofficecore/KoUnit.h new file mode 100644 index 00000000..07638215 --- /dev/null +++ b/lib/kofficecore/KoUnit.h @@ -0,0 +1,175 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Reginald Stadlbauer <reggie@kde.org>, Torben Weis <weis@kde.org> + Copyright (C) 2004, Nicolas GOUTTE <goutte@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. +*/ + +#ifndef kounit_h +#define kounit_h +#include <qstring.h> +#include <qstringlist.h> +#include <math.h> // for floor +#include <koffice_export.h> + +class KoXmlWriter; +class QDomElement; + +// 1 inch ^= 72 pt +// 1 inch ^= 25.399956 mm (-pedantic ;p) +// 1 pt = 1/12 pi +// 1 pt ^= 0.0077880997 cc +// 1 cc = 12 dd +// Note: I don't use division but multiplication with the inverse value +// because it's faster ;p (Werner) +#define POINT_TO_MM(px) ((px)*0.352777167) +#define MM_TO_POINT(mm) ((mm)*2.83465058) +#define POINT_TO_CM(px) ((px)*0.0352777167) +#define CM_TO_POINT(cm) ((cm)*28.3465058) +#define POINT_TO_DM(px) ((px)*0.00352777167) +#define DM_TO_POINT(dm) ((dm)*283.465058) +#define POINT_TO_INCH(px) ((px)*0.01388888888889) +#define INCH_TO_POINT(inch) ((inch)*72.0) +#define MM_TO_INCH(mm) ((mm)*0.039370147) +#define INCH_TO_MM(inch) ((inch)*25.399956) +#define POINT_TO_PI(px)((px)*0.083333333) +#define POINT_TO_DD(px)((px)*0.006490083) +#define POINT_TO_CC(px)((px)*0.077880997) +#define PI_TO_POINT(pi)((pi)*12) +#define DD_TO_POINT(dd)((dd)*154.08124) +#define CC_TO_POINT(cc)((cc)*12.840103) +/** + * %KOffice stores everything in pt (using "double") internally. + * When displaying a value to the user, the value is converted to the user's unit + * of choice, and rounded to a reasonable precision to avoid 0.999999 + */ +class KOFFICECORE_EXPORT KoUnit +{ +public: + /** Length units supported by KOffice. */ + enum Unit { + U_MM = 0, + U_PT = 1, + U_INCH = 2, + U_CM = 3, + U_DM = 4, + U_PI = 5, // pica + U_DD = 6, // didot + U_CC = 7, // cicero + U_LASTUNIT = U_CC // update when adding a new unit + // when adding a new unit, make sure to implement support for it in koRuler too + }; + + /// Prepare ptValue to be displayed in pt + static double toPoint( double ptValue ) { + // No conversion, only rounding (to 0.001 precision) + return floor( ptValue * 1000.0 ) / 1000.0; + } + + /// Prepare ptValue to be displayed in mm + static double toMM( double ptValue ) { + // "mm" values are rounded to 0.0001 millimeters + return floor( POINT_TO_MM( ptValue ) * 10000.0 ) / 10000.0; + } + + /// Prepare ptValue to be displayed in cm + static double toCM( double ptValue ) { + return floor( POINT_TO_CM( ptValue ) * 10000.0 ) / 10000.0; + } + + /// Prepare ptValue to be displayed in dm + static double toDM( double ptValue ) { + return floor( POINT_TO_DM( ptValue ) * 10000.0 ) / 10000.0; + } + + /// Prepare ptValue to be displayed in inch + static double toInch( double ptValue ) { + // "in" values are rounded to 0.00001 inches + return floor( POINT_TO_INCH( ptValue ) * 100000.0 ) / 100000.0; + } + + /// Prepare ptValue to be displayed in pica + static double toPI( double ptValue ) { + // "pi" values are rounded to 0.00001 inches + return floor( POINT_TO_PI( ptValue ) * 100000.0 ) / 100000.0; + } + + /// Prepare ptValue to be displayed in didot + static double toDD( double ptValue ) { + // "dd" values are rounded to 0.00001 inches + return floor( POINT_TO_DD( ptValue ) * 100000.0 ) / 100000.0; + } + + /// Prepare ptValue to be displayed in cicero + static double toCC( double ptValue ) { + // "cc" values are rounded to 0.00001 inches + return floor( POINT_TO_CC( ptValue ) * 100000.0 ) / 100000.0; + } + + /** + * This method is the one to use to display a value in a dialog + * \return the value @p ptValue converted to @p unit and rounded, ready to be displayed + * Old name: ptToUnit + */ + static double toUserValue( double ptValue, Unit unit ); + + /** + * Convert the value @p ptValue to a given unit @p unit + * Unlike KoUnit::ptToUnit the return value remains unrounded, so that it can be used in complex calculation + * \return the converted value + * Old name: ptToUnitUnrounded + */ + static double ptToUnit( const double ptValue, const Unit unit ); + + /// This method is the one to use to display a value in a dialog + /// @return the value @p ptValue converted to @p unit and rounded, ready to be displayed + /// Old name: userValue + static QString toUserStringValue( double ptValue, Unit unit ); + + /// This method is the one to use to read a value from a dialog + /// @return the value in @p unit, converted to points for internal use + /// Old name: ptFromUnit + static double fromUserValue( double value, Unit unit ); + + /// This method is the one to use to read a value from a dialog + /// @param value value entered by the user + /// @param unit unit type selected by the user + /// @param ok if set, the pointed bool is set to true if the value could be + /// converted to a double, and to false otherwise. + /// @return the value in @p unit, converted to points for internal use + static double fromUserValue( const QString& value, Unit unit, bool* ok = 0 ); + + /// Convert a unit name into a Unit enum + /// @param _unitName name to convert + /// @param ok if set, it will be true if the unit was known, false if unknown + static Unit unit( const QString &_unitName, bool* ok = 0 ); + /// Get the name of a unit + static QString unitName( Unit _unit ); + /// Get the full (translated) description of a unit + static QString unitDescription( Unit _unit ); + static QStringList listOfUnitName(); + + /// parse common %KOffice and OO values, like "10cm", "5mm" to pt + static double parseValue( QString value, double defaultVal = 0.0 ); + // Note: the above method doesn't take a const ref, since it modifies the arg. + + /// Save a unit in OASIS format + static void saveOasis(KoXmlWriter* settingsWriter, Unit _unit); + +}; + + +#endif diff --git a/lib/kofficecore/KoView.cpp b/lib/kofficecore/KoView.cpp new file mode 100644 index 00000000..9650192c --- /dev/null +++ b/lib/kofficecore/KoView.cpp @@ -0,0 +1,868 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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 <KoView.h> +#include <KoDocument.h> +#include <KoMainWindow.h> +#include <KoFrame.h> +#include <KoViewIface.h> +#include <KoDocumentChild.h> + +#include <klocale.h> +#include <kglobal.h> +#include <kdebug.h> +#include <kparts/partmanager.h> +#include <kparts/event.h> +#include <kcursor.h> +#include <assert.h> +#include <kstatusbar.h> +#include <kapplication.h> +#include <qtimer.h> + +class KoViewPrivate +{ +public: + KoViewPrivate() + { + m_inOperation = false; + m_zoom = 1.0; + m_children.setAutoDelete( true ); + m_manager = 0L; + m_tempActiveWidget = 0L; + m_dcopObject = 0; + m_registered=false; + m_documentDeleted=false; + } + ~KoViewPrivate() + { + } + + QGuardedPtr<KoDocument> m_doc; // our KoDocument + QGuardedPtr<KParts::PartManager> m_manager; + double m_zoom; + QPtrList<KoViewChild> m_children; + QWidget *m_tempActiveWidget; + KoViewIface *m_dcopObject; + bool m_registered; // are we registered at the part manager? + bool m_documentDeleted; // true when m_doc gets deleted [can't use m_doc==0 + // since this only happens in ~QObject, and views + // get deleted by ~KoDocument]. + QTimer *m_scrollTimer; + + // Hmm sorry for polluting the private class with such a big inner class. + // At the beginning it was a little struct :) + class StatusBarItem { + public: + StatusBarItem() // for QValueList + : m_widget(0), m_visible(false) + {} + StatusBarItem( QWidget * widget, int stretch, bool permanent ) + : m_widget(widget), m_stretch(stretch), m_permanent(permanent), m_visible(false) + {} + + QWidget * widget() const { return m_widget; } + + void ensureItemShown( KStatusBar * sb ) + { + if ( !m_visible ) + { + sb->addWidget( m_widget, m_stretch, m_permanent ); + m_visible = true; + m_widget->show(); + } + } + void ensureItemHidden( KStatusBar * sb ) + { + if ( m_visible ) + { + sb->removeWidget( m_widget ); + m_visible = false; + m_widget->hide(); + } + } + private: + QWidget * m_widget; + int m_stretch; + bool m_permanent; + bool m_visible; // true when the item has been added to the statusbar + }; + QValueList<StatusBarItem> m_statusBarItems; // Our statusbar items + bool m_inOperation; //in the middle of an operation (no screen refreshing)? +}; + +KoView::KoView( KoDocument *document, QWidget *parent, const char *name ) + : QWidget( parent, name ) +{ + Q_ASSERT( document ); + + //kdDebug(30003) << "KoView::KoView " << this << endl; + d = new KoViewPrivate; + d->m_doc = document; + KParts::PartBase::setPartObject( this ); + + setFocusPolicy( StrongFocus ); + + setMouseTracking( true ); + + connect( d->m_doc, SIGNAL( childChanged( KoDocumentChild * ) ), + this, SLOT( slotChildChanged( KoDocumentChild * ) ) ); + + connect( d->m_doc, SIGNAL( sigBeginOperation() ), + this, SLOT( beginOperation() ) ); + + connect( d->m_doc, SIGNAL( sigEndOperation() ), + this, SLOT( endOperation() ) ); + + + actionCollection()->setWidget( this ); + setupGlobalActions(); + KActionCollection *coll = actionCollection(); + /**** not needed anymore, according to David (Werner) + QValueList<KAction*> docActions = document->actionCollection()->actions(); + QValueList<KAction*>::ConstIterator it = docActions.begin(); + QValueList<KAction*>::ConstIterator end = docActions.end(); + for (; it != end; ++it ) + coll->insert( *it ); + */ + + KStatusBar * sb = statusBar(); + if ( sb ) // No statusbar in e.g. konqueror + { + coll->setHighlightingEnabled( true ); + connect( coll, SIGNAL( actionStatusText( const QString & ) ), + this, SLOT( slotActionStatusText( const QString & ) ) ); + connect( coll, SIGNAL( clearStatusText() ), + this, SLOT( slotClearStatusText() ) ); + + connect( d->m_doc, SIGNAL( sigStatusBarMessage( const QString& ) ), + this, SLOT( slotActionStatusText( const QString& ) ) ); + connect( d->m_doc, SIGNAL( sigClearStatusBarMessage() ), + this, SLOT( slotClearStatusText() ) ); + } + d->m_doc->setCurrent(); + + d->m_scrollTimer = new QTimer( this ); + connect (d->m_scrollTimer, SIGNAL( timeout() ), this, SLOT( slotAutoScroll() ) ); +} + +KoView::~KoView() +{ + kdDebug(30003) << "KoView::~KoView " << this << endl; + delete d->m_scrollTimer; + delete d->m_dcopObject; + if (!d->m_documentDeleted) + { + if ( koDocument() && !koDocument()->isSingleViewMode() ) + { + if ( d->m_manager && d->m_registered ) // if we aren't registered we mustn't unregister :) + d->m_manager->removePart( koDocument() ); + d->m_doc->removeView(this); + d->m_doc->setCurrent( false ); + } + } + delete d; +} + +KoDocument *KoView::koDocument() const +{ + return d->m_doc; +} + +void KoView::setDocumentDeleted() +{ + d->m_documentDeleted = true; +} + +bool KoView::documentDeleted() const +{ + return d->m_documentDeleted; +} + +bool KoView::hasDocumentInWindow( KoDocument *doc ) +{ + return child( doc ) != 0L; +} + +void KoView::setPartManager( KParts::PartManager *manager ) +{ + d->m_manager = manager; + if ( !koDocument()->isSingleViewMode() && + manager->parts()->containsRef( koDocument() ) == 0 ) // is there another view registered? + { + d->m_registered = true; // no, so we have to register now and ungregister again in the DTOR + manager->addPart( koDocument(), false ); + } + else + d->m_registered = false; // There is already another view registered for that part... +} + +KParts::PartManager *KoView::partManager() const +{ + return d->m_manager; +} + +KAction *KoView::action( const QDomElement &element ) const +{ + static const QString &attrName = KGlobal::staticQString( "name" ); + QString name = element.attribute( attrName ); + + KAction *act = KXMLGUIClient::action( name.utf8() ); + + if ( !act ) + act = d->m_doc->KXMLGUIClient::action( name.utf8() ); + + return act; +} + +KoDocument *KoView::hitTest( const QPoint &viewPos ) +{ + KoViewChild *viewChild; + + QPoint pos = reverseViewTransformations( viewPos ); + + KoDocumentChild *docChild = selectedChild(); + if ( docChild ) + { + if ( ( viewChild = child( docChild->document() ) ) ) + { + if ( viewChild->frameRegion().contains( pos ) ) + return 0; + } + else + if ( docChild->frameRegion().contains( pos ) ) + return 0; + } + + docChild = activeChild(); + if ( docChild ) + { + if ( ( viewChild = child( docChild->document() ) ) ) + { + if ( viewChild->frameRegion().contains( pos ) ) + return 0; + } + else + if ( docChild->frameRegion().contains( pos ) ) + return 0; + } + + return koDocument()->hitTest( pos ); +} + +int KoView::leftBorder() const +{ + return 0; +} + +int KoView::rightBorder() const +{ + return 0; +} + +int KoView::topBorder() const +{ + return 0; +} + +int KoView::bottomBorder() const +{ + return 0; +} + +void KoView::setZoom( double zoom ) +{ + d->m_zoom = zoom; + update(); +} + +double KoView::zoom() const +{ + return d->m_zoom; +} + +QWidget *KoView::canvas() const +{ + //dfaure: since the view plays two roles in this method (the const means "you can modify the canvas + // but not the view", it's just coincidence that the view is the canvas by default ;) + return const_cast<KoView *>(this); +} + +int KoView::canvasXOffset() const +{ + return 0; +} + +int KoView::canvasYOffset() const +{ + return 0; +} + +void KoView::canvasAddChild( KoViewChild * ) +{ +} + +void KoView::customEvent( QCustomEvent *ev ) +{ + if ( KParts::PartActivateEvent::test( ev ) ) + partActivateEvent( (KParts::PartActivateEvent *)ev ); + else if ( KParts::PartSelectEvent::test( ev ) ) + partSelectEvent( (KParts::PartSelectEvent *)ev ); + else if( KParts::GUIActivateEvent::test( ev ) ) + guiActivateEvent( (KParts::GUIActivateEvent*)ev ); +} + +void KoView::partActivateEvent( KParts::PartActivateEvent *event ) +{ + if ( event->part() != (KParts::Part *)koDocument() ) + { + assert( event->part()->inherits( "KoDocument" ) ); + + KoDocumentChild *child = koDocument()->child( (KoDocument *)event->part() ); + if ( child && event->activated() ) + { + if ( child->isRectangle() && !child->isTransparent() ) + { + KoViewChild *viewChild = new KoViewChild( child, this ); + d->m_children.append( viewChild ); + + QApplication::setOverrideCursor(waitCursor); + // This is the long operation (due to toolbar layout stuff) + d->m_manager->setActivePart( child->document(), viewChild->frame()->view() ); + QApplication::restoreOverrideCursor(); + + // Now we can move the frame to the right place + viewChild->setInitialFrameGeometry(); + } + else + { + emit regionInvalidated( child->frameRegion( matrix() ), true ); + } + emit childActivated( child ); + } + else if ( child ) + { + emit regionInvalidated( child->frameRegion( matrix() ), true ); + emit childDeactivated( child ); + } + else + emit invalidated(); + } + else + emit activated( event->activated() ); +} + +void KoView::partSelectEvent( KParts::PartSelectEvent *event ) +{ + if ( event->part() != (KParts::Part *)koDocument() ) + { + assert( event->part()->inherits( "KoDocument" ) ); + + KoDocumentChild *child = koDocument()->child( (KoDocument *)event->part() ); + + if ( child && event->selected() ) + { + QRegion r = child->frameRegion( matrix() ); + r.translate( - canvasXOffset(), - canvasYOffset() ); + emit regionInvalidated( r, true ); + emit childSelected( child ); + } + else if ( child ) + { + QRegion r = child->frameRegion( matrix() ); + r.translate( - canvasXOffset(), - canvasYOffset() ); + emit regionInvalidated( r, true ); + emit childUnselected( child ); + } + else + emit invalidated(); + } + else + emit selected( event->selected() ); +} + +void KoView::guiActivateEvent( KParts::GUIActivateEvent * ev ) +{ + showAllStatusBarItems( ev->activated() ); +} + +void KoView::showAllStatusBarItems( bool show ) +{ + KStatusBar * sb = statusBar(); + if ( !sb ) + return; + QValueListIterator<KoViewPrivate::StatusBarItem> it = d->m_statusBarItems.begin(); + for ( ; it != d->m_statusBarItems.end() ; ++it ) + if ( show ) + (*it).ensureItemShown( sb ); + else + (*it).ensureItemHidden( sb ); +} + +void KoView::addStatusBarItem( QWidget * widget, int stretch, bool permanent ) +{ + KoViewPrivate::StatusBarItem item( widget, stretch, permanent ); + d->m_statusBarItems.append(item); + QValueListIterator<KoViewPrivate::StatusBarItem> it = d->m_statusBarItems.fromLast(); + KStatusBar * sb = statusBar(); + Q_ASSERT(sb); + if (sb) + (*it).ensureItemShown( sb ); +} + +void KoView::removeStatusBarItem( QWidget * widget ) +{ + KStatusBar * sb = statusBar(); + QValueListIterator<KoViewPrivate::StatusBarItem> it = d->m_statusBarItems.begin(); + for ( ; it != d->m_statusBarItems.end() ; ++it ) + if ( (*it).widget() == widget ) + { + if ( sb ) + (*it).ensureItemHidden( sb ); + d->m_statusBarItems.remove( it ); + break; + } + if ( it == d->m_statusBarItems.end() ) + kdWarning() << "KoView::removeStatusBarItem. Widget not found : " << widget << endl; +} + +KoDocumentChild *KoView::selectedChild() +{ + if ( !d->m_manager ) + return 0L; + + KParts::Part *selectedPart = d->m_manager->selectedPart(); + + if ( !selectedPart || !selectedPart->inherits( "KoDocument" ) ) + return 0L; + + return koDocument()->child( (KoDocument *)selectedPart ); +} + +KoDocumentChild *KoView::activeChild() +{ + if ( !d->m_manager ) + return 0L; + + KParts::Part *activePart = d->m_manager->activePart(); + + if ( !activePart || !activePart->inherits( "KoDocument" ) ) + return 0L; + + return koDocument()->child( (KoDocument *)activePart ); +} + +void KoView::enableAutoScroll( ) +{ + d->m_scrollTimer->start( 50 ); +} + +void KoView::disableAutoScroll( ) +{ + d->m_scrollTimer->stop(); +} + +void KoView::paintEverything( QPainter &painter, const QRect &rect, bool transparent ) +{ + koDocument()->paintEverything( painter, rect, transparent, this ); +} + +KoViewChild *KoView::child( KoView *view ) +{ + QPtrListIterator<KoViewChild> it( d->m_children ); + for (; it.current(); ++it ) + if ( it.current()->frame()->view() == view ) + return it.current(); + + return 0L; +} + +KoViewChild *KoView::child( KoDocument *doc ) +{ + QPtrListIterator<KoViewChild> it( d->m_children ); + for (; it.current(); ++it ) + if ( it.current()->documentChild()->document() == doc ) + return it.current(); + + return 0L; +} + +QWMatrix KoView::matrix() const +{ + QWMatrix m; + m.scale( zoom(), zoom() ); + //m.translate( canvasXOffset() , canvasYOffset() ); + return m; +} + +void KoView::slotChildActivated( bool a ) +{ + // Only interested in deactivate events + if ( a ) + return; + + KoViewChild* ch = child( (KoView*)sender() ); + if ( !ch ) + return; + + KoView* view = ch->frame()->view(); + + QWidget *activeWidget = view->d->m_tempActiveWidget; + + if ( d->m_manager->activeWidget() ) + activeWidget = d->m_manager->activeWidget(); + + if ( !activeWidget || !activeWidget->inherits( "KoView" ) ) + return; + + // Is the new active view a child of this one ? + // In this case we may not delete! + // QObject *n = d->m_manager->activeWidget(); + QObject *n = activeWidget; + while( n ) + if ( n == (QObject *)view ) + return; + else + n = n->parent(); + + + d->m_tempActiveWidget = activeWidget; + QApplication::setOverrideCursor(waitCursor); + d->m_manager->setActivePart( 0L ); + + QGuardedPtr<KoDocumentChild> docChild = ch->documentChild(); + QGuardedPtr<KoFrame> chFrame = ch->frame(); + if ( docChild && chFrame && chFrame->view() ) + { + docChild->setContentsPos( chFrame->view()->canvasXOffset(), + chFrame->view()->canvasYOffset() ); + docChild->document()->setViewBuildDocument( chFrame->view(), chFrame->view()->xmlguiBuildDocument() ); + } + + d->m_children.remove( ch ); + + d->m_manager->addPart( docChild->document(), false ); // the destruction of the view removed the part from the partmanager. re-add it :) + + QApplication::restoreOverrideCursor(); + + // #### HACK + // We want to delete as many views as possible and this + // trick is used to go upwards in the view-tree. + emit activated( FALSE ); +} + +void KoView::slotChildChanged( KoDocumentChild *child ) +{ + QRegion region( child->oldPointArray( matrix() ) ); + emit regionInvalidated( child->frameRegion( matrix(), true ).unite( region ), true ); +} + +int KoView::autoScrollAcceleration( int offset ) const +{ + if(offset < 40) + return offset; + else + return offset*offset/40; +} + +void KoView::slotAutoScroll( ) +{ + QPoint scrollDistance; + bool actuallyDoScroll = false; + QPoint pos( mapFromGlobal( QCursor::pos() ) ); + + //Provide progressive scrolling depending on the mouse position + if ( pos.y() < topBorder() ) + { + scrollDistance.setY ((int) - autoScrollAcceleration( - pos.y() + topBorder() )); + actuallyDoScroll = true; + } + else if ( pos.y() > height() - bottomBorder() ) + { + scrollDistance.setY ((int) autoScrollAcceleration(pos.y() - height() + bottomBorder() )); + actuallyDoScroll = true; + } + + if ( pos.x() < leftBorder() ) + { + scrollDistance.setX ((int) - autoScrollAcceleration( - pos.x() + leftBorder() )); + actuallyDoScroll = true; + } + else if ( pos.x() > width() - rightBorder() ) + { + scrollDistance.setX ((int) autoScrollAcceleration( pos.x() - width() + rightBorder() )); + actuallyDoScroll = true; + } + + if ( actuallyDoScroll ) + { + int state=0; +#if KDE_IS_VERSION(3,4,0) + state = kapp->keyboardMouseState(); +#endif + + pos = canvas()->mapFrom(this, pos); + QMouseEvent * event = new QMouseEvent(QEvent::MouseMove, pos, 0, state); + + QApplication::postEvent( canvas(), event ); + emit autoScroll( scrollDistance ); + } +} + + +void KoView::setupGlobalActions() { + actionNewView = new KAction( i18n( "&New View" ), "window_new", 0, + this, SLOT( newView() ), + actionCollection(), "view_newview" ); +} + +void KoView::setupPrinter( KPrinter & ) +{ + kdDebug() << "KoView::setupPrinter not implemented by the application!" << endl; +} + +void KoView::print( KPrinter & ) +{ + kdDebug() << "KoView::print not implemented by the application!" << endl; +} + +void KoView::newView() { + assert( ( d!=0L && d->m_doc ) ); + + KoDocument *thisDocument = d->m_doc; + KoMainWindow *shell = new KoMainWindow( thisDocument->instance() ); + shell->setRootDocument(thisDocument); + shell->show(); +} + +bool KoView::isInOperation() const +{ + return d->m_inOperation; +} + +void KoView::beginOperation() +{ + d->m_inOperation = true; + canvas()->setUpdatesEnabled(FALSE); +} + +void KoView::endOperation() +{ + canvas()->setUpdatesEnabled(TRUE); + d->m_inOperation = false; + +// canvas()->update(); +} + +KoMainWindow * KoView::shell() const +{ + return dynamic_cast<KoMainWindow *>( topLevelWidget() ); +} + +KMainWindow * KoView::mainWindow() const +{ + return dynamic_cast<KMainWindow *>( topLevelWidget() ); +} + +KStatusBar * KoView::statusBar() const +{ + KoMainWindow *mw = shell(); + return mw ? mw->statusBar() : 0L; +} + +void KoView::slotActionStatusText( const QString &text ) +{ + KStatusBar *sb = statusBar(); + if ( sb ) + sb->message( text ); +} + +void KoView::slotClearStatusText() +{ + KStatusBar *sb = statusBar(); + if ( sb ) + sb->clear(); +} + +DCOPObject *KoView::dcopObject() +{ + if ( !d->m_dcopObject ) + d->m_dcopObject = new KoViewIface( this ); + return d->m_dcopObject; +} + +class KoViewChild::KoViewChildPrivate +{ +public: + KoViewChildPrivate() + { + } + ~KoViewChildPrivate() + { + } +}; + +KoViewChild::KoViewChild( KoDocumentChild *child, KoView *_parentView ) +{ + d = new KoViewChildPrivate; + m_parentView = _parentView; + m_child = child; + + m_frame = new KoFrame( parentView()->canvas() ); + KoView *view = child->document()->createView( m_frame ); + view->setXMLGUIBuildDocument( child->document()->viewBuildDocument( view ) ); + + view->setPartManager( parentView()->partManager() ); + + // hack? (Werner) + view->setZoom( parentView()->zoom() * QMAX(child->xScaling(), child->yScaling()) ); + + m_frame->setView( view ); + m_frame->show(); + m_frame->raise(); + + parentView()->canvasAddChild( this ); + + + /* + KoViewChild has basically three geometries to keep in sync. + - The KoDocumentChild geometry (i.e. the embedded object's geometry, unzoomed) + - Its own geometry (used for hit-test etc.) + - The KoFrame geometry (the graphical widget for moving the object when active) + + So we need to subtract the scrollview's offset for the frame geometry, since it's a widget. + + The rules are + (R1) frameGeometry = viewGeometry(childGeometry) "+" m_frame->{left|right|top|bottom}Border() - scrollview offset, + (R2) frameGeometry = myGeometry "+" active_frame_border - scrollview offset. + + So: (R3, unused) myGeometry = viewGeometry(childGeometry) "+" m_frame->{left|right|top|bottom}Border() "-" active_frame_border + + Notes: active_frame_border is m_frame->border() (0 when inactive, 5 when active). + {left|right|top|bottom}Border are the borders used in kspread (0 when inactive, big when active). + "+" border means we add a border, so it's a subtraction on x, y and an addition on width, height. + + viewGeometry() applies the zoom as well as any other translation the app might want to do + */ + + // Setting the frameGeometry is done in setInitialFrameGeometry, which is + // also called right after activation. + + connect( view, SIGNAL( activated( bool ) ), + parentView(), SLOT( slotChildActivated( bool ) ) ); +} + +KoViewChild::~KoViewChild() +{ + if ( m_frame ) + { + slotFrameGeometryChanged(); + delete static_cast<KoFrame *>( m_frame ); + } + delete d; +} + +void KoViewChild::slotFrameGeometryChanged() +{ + // Set our geometry from the frame geometry (R2 reversed) + QRect geom = m_frame->geometry(); + int b = m_frame->border(); + QRect borderRect( geom.x() + b + parentView()->canvasXOffset(), + geom.y() + b + parentView()->canvasYOffset(), + geom.width() - b * 2, + geom.height() - b * 2 ); + setGeometry( borderRect ); + + if(m_child) + { + // Set the child geometry from the frame geometry (R1 reversed) + QRect borderLessRect( geom.x() + m_frame->leftBorder() + parentView()->canvasXOffset(), + geom.y() + m_frame->topBorder() + parentView()->canvasYOffset(), + geom.width() - m_frame->leftBorder() - m_frame->rightBorder(), + geom.height() - m_frame->topBorder() - m_frame->bottomBorder() ); + + // We don't want to trigger slotDocGeometryChanged again + lock(); + QRect childGeom = parentView()->reverseViewTransformations( borderLessRect ); + kdDebug() << "KoChild::slotFrameGeometryChanged child geometry " + << ( geometry() == childGeom ? "already " : "set to " ) + << childGeom << endl; + m_child->setGeometry( childGeom ); + unlock(); + } +} + +void KoViewChild::slotDocGeometryChanged() +{ + if ( locked() ) + return; + // Set frame geometry from child geometry (R1) + // The frame's resizeEvent will call slotFrameGeometryChanged. + QRect geom = parentView()->applyViewTransformations( m_child->geometry() ); + QRect borderRect( geom.x() - m_frame->leftBorder() - parentView()->canvasXOffset(), + geom.y() - m_frame->topBorder() - parentView()->canvasYOffset(), + geom.width() + m_frame->leftBorder() + m_frame->rightBorder(), + geom.height() + m_frame->topBorder() + m_frame->bottomBorder() ); + kdDebug() << "KoViewChild::slotDocGeometryChanged frame geometry " + << ( m_frame->geometry() == borderRect ? "already " : "set to " ) + << borderRect << endl; + + m_frame->setGeometry( borderRect ); +} + +QPoint KoView::applyViewTransformations( const QPoint& p ) const +{ + return QPoint( qRound( p.x() * zoom() ), qRound( p.y() * zoom() ) ); +} + +QPoint KoView::reverseViewTransformations( const QPoint& v ) const +{ + return QPoint( qRound( v.x() / zoom() ), qRound( v.y() / zoom() ) ); +} + +QRect KoView::applyViewTransformations( const QRect& r ) const +{ + return QRect( applyViewTransformations( r.topLeft() ), + applyViewTransformations( r.bottomRight() ) ); +} + +QRect KoView::reverseViewTransformations( const QRect& r ) const +{ + return QRect( reverseViewTransformations( r.topLeft() ), + reverseViewTransformations( r.bottomRight() ) ); +} + +void KoViewChild::setInitialFrameGeometry() +{ + kdDebug() << k_funcinfo << endl; + + // Connect only now, so that the GUI building doesn't move us around. + connect( m_frame, SIGNAL( geometryChanged() ), + this, SLOT( slotFrameGeometryChanged() ) ); + connect( m_child, SIGNAL( changed( KoChild * ) ), + this, SLOT( slotDocGeometryChanged() ) ); + + // Set frameGeometry from childGeometry + slotDocGeometryChanged(); + // Set myGeometry from frameGeometry + slotFrameGeometryChanged(); + +} + +#include "KoView.moc" + diff --git a/lib/kofficecore/KoView.h b/lib/kofficecore/KoView.h new file mode 100644 index 00000000..c2be52ec --- /dev/null +++ b/lib/kofficecore/KoView.h @@ -0,0 +1,502 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <weis@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. +*/ +#ifndef __koView_h__ +#define __koView_h__ + +#include <qwidget.h> +#include <qguardedptr.h> + +#include <kparts/part.h> + +#include <KoChild.h> +#include <koffice_export.h> + +class KAction; +class KActionCollection; +class QCustomEvent; +class KoDocument; +class KoMainWindow; +class KMainWindow; +class KoViewPrivate; +class KoViewChild; +class KoFrame; +class KStatusBar; +class KInstance; +class KPrinter; +class KoDocumentChild; +class DCOPObject; + +namespace KParts +{ + class PartManager; + class PartActivateEvent; + class PartSelectEvent; +} + +/** + * This class is used to display a @ref KoDocument. + * + * Multiple views can be attached to one document at a time. + */ +class KOFFICECORE_EXPORT KoView : public QWidget, public KParts::PartBase +{ + Q_OBJECT +public: + /** + * Creates a new view for the document. Usually you don't create views yourself + * since the KOffice components come with their own view classes which inherit + * KoView. + * + * The standard way to retrieve a KoView is to call @ref KoDocument::createView. + * + * @param document is the document which should be displayed in this view. This pointer + * must not be zero. + * @param parent parent widget for this view. + * @param name Name of the view. The name is used in DCOP, so the name should + * match the pattern [A-Za-z_][A-Za-z_0-9]*. + * + */ + KoView( KoDocument *document, QWidget *parent = 0, const char *name = 0 ); + /** + * Destroys the view and unregisters at the document. + */ + virtual ~KoView(); + + /** + * Retrieves the document object of this view. + */ + KoDocument *koDocument() const; + + /** + * Tells this view that its document has got deleted (called internally) + */ + void setDocumentDeleted(); + /** + * @return true if the document has already got deleted. + * This can be useful for the view destructor to know if it can + * access the document or not. + */ + bool documentDeleted() const; + + virtual void setPartManager( KParts::PartManager *manager ); + virtual KParts::PartManager *partManager() const; + + /** + * Returns the action described action object. In fact only the "name" attribute + * of @p element is of interest here. The method searches in the + * KActionCollection of this view. + * + * Please notice that KoView indirectly inherits KXMLGUIClient. + * + * @see KXMLGUIClient + * @see KXMLGUIClient::actionCollection + * @see KoDocument::action + */ + virtual KAction *action( const QDomElement &element ) const; + + /** + * Retrieves the document that is hit. This can be an embedded document. + * + * The default implementation asks @ref KoDocument::hitTest. This + * will iterate over all child documents to detect a hit. + * + * If your koffice component has multiple pages, like for example KSpread, then the hittest + * may not succeed for a child that is not on the visible page. In those + * cases you need to reimplement this method. + */ + virtual KoDocument *hitTest( const QPoint &pos ); + + /** + * Retrieves the left border width that is displayed around the content if + * the view is active. + * + * In a spread sheet this border is for example used to display the + * rows, while a top border is used to display the names of the cells + * and a right and bottom border is used to display scrollbars. If the view + * becomes inactive, then this stuff is not displayed anymore. + * + * @ref KoFrame uses this border information. If an embedded document becomes active + * then it is resized so that it has enough space to display the borders and to + * display the same content as before the activation. + * So if for example all of your borders are 20 pixels, then activating the embedded + * document causes the KoView to move 20 pixels up/left and the size and width increases + * by 20+20 pixels each. + * + * The default border is 0. + */ + virtual int leftBorder() const; + /** + * @see #leftBorder + */ + virtual int rightBorder() const; + /** + * @see #leftBorder + */ + virtual int topBorder() const; + /** + * @see #leftBorder + */ + virtual int bottomBorder() const; + + /** + * Scales the view on the content. This does not affect the contents + * data structures. You can use this mechanism to implement a zoom + * for example. + * + * The method calls QWidget::update so that the scaled content + * is automatically displayed. + * + * The default scaling is 1.0 in both orientations. + */ + virtual void setZoom( double zoom ); + /** + * @return the current scaling factor (zoom level) + * + * @see #setZoom + */ + virtual double zoom() const; + + /** + * Overload this function if the content will be displayed + * on some child widget instead of the view directly. + * + * By default this function returns a pointer to the view. + */ + virtual QWidget *canvas() const; + + /** + * Overload this function if the content will be displayed + * with an offset relative to the upper left corner + * of the canvas widget. + * + * By default this function returns 0. + */ + virtual int canvasXOffset() const; + + /** + * Overload this function if the content will be displayed + * with an offset relative to the upper left corner + * of the canvas widget. + * + * By default this function returns 0. + */ + virtual int canvasYOffset() const; + + /** + * Overload this function if you need to perform some actions + * after KoView (the part widget) is inserted into canvas. + * You should call for example addChild(QWidget*) method + * of QScrollView here, if canvas is a viewport of QScrollView. + * + * By default this function does nothing. + */ + virtual void canvasAddChild( KoViewChild *child ); + + /** + * @return the selected child. The function returns 0 if + * no direct child is currently selected. + */ + virtual KoDocumentChild *selectedChild(); + + /** + * @return the active child. The function returns 0 if + * no direct child is currently active. + */ + virtual KoDocumentChild *activeChild(); + + /** + * Sets up so that autoScroll signals are emitted when the mouse pointer is outside the view + */ + void enableAutoScroll(); + + /** + * Stops the emitting of autoScroll signals + */ + void disableAutoScroll(); + + /** + * calls KoDocument::paintEverything() + */ + virtual void paintEverything( QPainter &painter, const QRect &rect, bool transparent = false ); + + /** + * @return TRUE if the document @p doc is represented in this view by + * some KoViewChild. + * + * This is just a convenience function for @ref #child. + */ + bool hasDocumentInWindow( KoDocument *doc ); + + /** + * Returns the matrix which is used by the view to transform the content. + * Currently only scaling is supported. + * + * The matrix changes when calling @ref #setZoom. + * + * @deprecated, use applyViewTransformations / reverseViewTransformations instead. + */ + virtual QWMatrix matrix() const KDE_DEPRECATED; + + /** + * Apply the transformations that the view makes to its contents. + * This is used for embedded objects. + * By default this simply applies the zoom(). + * Reimplement to add some translation if needed (e.g. to center the page) + */ + virtual QPoint applyViewTransformations( const QPoint& ) const; + + /** + * Reverse the transformations that the view makes to its contents, + * i.e. undo the transformations done by applyViewTransformations(). + * This is used for embedded objects. + * By default this simply unzooms the point. + * Reimplement to add some translation if needed (e.g. to center the page) + */ + virtual QPoint reverseViewTransformations( const QPoint& ) const; + + /** + * Overload for QRect, usually it's not needed to reimplement this one. + */ + virtual QRect applyViewTransformations( const QRect& ) const; + + /** + * Overload for QRect, usually it's not needed to reimplement this one. + */ + virtual QRect reverseViewTransformations( const QRect& ) const; + + /** + * @return the KoViewChild which is responsible for the @p view or 0. + * + * This method does no recursion. + */ + KoViewChild *child( KoView *view ); + /** + * A convenience function which returns the KoViewChild which in turn holds the + * @ref KoView that in turn holds the @p document. + */ + KoViewChild *child( KoDocument *document ); + + /** + * Return a DCOP interface for this view + * KOffice Applications are strongly recommended to reimplement this method, + * so that their dcop interface provides more functionality than the basic KoViewIface + */ + virtual DCOPObject * dcopObject(); + + /** + * Overload this method to setup KPrinter before the actual printing. + * + * @see #print + */ + virtual void setupPrinter( KPrinter &printer ); + + // BCI: make it return a bool, so that aborting doesn't still fire up the print preview afterwards + /** + * Overload this method with your own printing code. + */ + virtual void print( KPrinter &printer ); + + /** + * @return the KoMainWindow in which this view is currently. + * WARNING: this could be 0L, if the main window isn't a koffice main window. + * (e.g. it can be any KParts application). + */ + KoMainWindow * shell() const; + + /** + * @return the KMainWindow in which this view is currently. + * This one should never return 0L, in a KDE app. + */ + KMainWindow* mainWindow() const; + + /** + * @return the statusbar of the KoMainWindow in which this view is currently. + * WARNING: this could be 0L, if the main window isn't a koffice main window. + * (e.g. it can be any KParts application). + */ + KStatusBar * statusBar() const; + + /** + * This adds a widget to the statusbar for this view. + * If you use this method instead of using statusBar() directly, + * KoView will take care of removing the items when the view GUI is deactivated + * and readding them when it is reactivated. + * The parameters are the same as QStatusBar::addWidget(). + * + * Note that you can't use KStatusBar methods (inserting text items by id). + * But you can create a KStatusBarLabel with a dummy id instead, and use + * it directly, to get the same look and feel. + */ + void addStatusBarItem( QWidget * widget, int stretch = 0, bool permanent = false ); + + /** + * Remove a widget from the statusbar for this view. + */ + void removeStatusBarItem( QWidget * widget ); + + /** + * Show or hide all statusbar items. Used by KoMainWindow during saving. + */ + void showAllStatusBarItems( bool show ); + + /** + * You have to implement this method and disable/enable certain functionality (actions for example) in + * your view to allow/disallow editing of the document. + */ + virtual void updateReadWrite( bool readwrite ) = 0; + + /** + * Check to see if the view is currently in the middle of an operation which means + * that there will be no screen refreshes until a signal from the document hits + * the @ref #endOperation slot + */ + bool isInOperation() const; + +public slots: + /** + * Slot to create a new view around the contained @ref #koDocument. + */ + virtual void newView(); + + /** + * Slot to allow code to signal the beginning of an operation where the screen should + * not update until it is done. + * + * @see #endOperation + */ + virtual void beginOperation(); + + /** + * Slot to allow code to signal the end of an operation where the screen should + * not have been updating. So now it will update. + * + * @see #beginOperation + */ + virtual void endOperation(); + + /** + * Display a message in the status bar (calls QStatusBar::message()) + * @todo rename to something more generic + */ + void slotActionStatusText( const QString &text ); + + /** + * End of the message in the status bar (calls QStatusBar::clear()) + * @todo rename to something more generic + */ + void slotClearStatusText(); + +protected: + /** + * This method handles three events: KParts::PartActivateEvent, KParts::PartSelectEvent + * and KParts::GUIActivateEvent. + * The respective handlers are called if such an event is found. + */ + virtual void customEvent( QCustomEvent *ev ); + + /** + * Handles the event KParts::PartActivateEvent. + */ + virtual void partActivateEvent( KParts::PartActivateEvent *event ); + /** + * Handles the event KParts::PartSelectEvent. + */ + virtual void partSelectEvent( KParts::PartSelectEvent *event ); + /** + * Handles the event KParts::GUIActivateEvent. + */ + virtual void guiActivateEvent( KParts::GUIActivateEvent * ); + +signals: + void activated( bool active ); + void selected( bool select ); + + void autoScroll(const QPoint &scrollDistance); + + void childSelected( KoDocumentChild *child ); + void childUnselected( KoDocumentChild *child ); + + void childActivated( KoDocumentChild *child ); + void childDeactivated( KoDocumentChild *child ); + + void regionInvalidated( const QRegion ®ion, bool erase ); + + void invalidated(); + +// KDE invents public signals :) +#undef signals +#define signals public +signals: + + /** + * Make it possible for plugins to request + * the embedding of an image into the current + * document. Used e.g. by the scan-plugin + */ + void embeddImage(const QString &filename); + +#undef signals +#define signals protected + +protected slots: + virtual void slotChildActivated( bool a ); + virtual void slotChildChanged( KoDocumentChild *child ); + virtual void slotAutoScroll( ); + +private: + KAction *actionNewView; + virtual void setupGlobalActions( void ); + KoViewPrivate *d; + int autoScrollAcceleration( int offset ) const; +}; + +/** + * This class represents an active embedded document. + */ +class KoViewChild : public KoChild +{ + Q_OBJECT +public: + KoViewChild( KoDocumentChild *child, KoView *_parentView ); + virtual ~KoViewChild(); + + KoDocumentChild *documentChild() const { return m_child; } + KoView *parentView() const { return m_parentView; } + KoFrame *frame() const { return m_frame; } + + void setInitialFrameGeometry(); + +public slots: + + // Call this when the view transformations change + void reposition() { slotDocGeometryChanged(); } + +private slots: + void slotFrameGeometryChanged(); + void slotDocGeometryChanged(); + +private: + QGuardedPtr<KoDocumentChild> m_child; + QGuardedPtr<KoFrame> m_frame; + QGuardedPtr<KoView> m_parentView; + class KoViewChildPrivate; + KoViewChildPrivate *d; +}; + +#endif diff --git a/lib/kofficecore/KoViewIface.cc b/lib/kofficecore/KoViewIface.cc new file mode 100644 index 00000000..378db1e6 --- /dev/null +++ b/lib/kofficecore/KoViewIface.cc @@ -0,0 +1,76 @@ +/* This file is part of the KDE project + Copyright (c) 2000 Simon Hausmann <hausmann@kde.org> + + $Id: KoViewIface.cc 508787 2006-02-12 18:28:12Z ingwa $ + + 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 "KoViewIface.h" + +#include "KoView.h" + +#include <kapplication.h> +#include <dcopclient.h> +#include <kdcopactionproxy.h> +#include <kaction.h> + +//static +QCString KoViewIface::newIfaceName() +{ + static int s_viewIFNumber = 0; + QCString name; name.setNum( s_viewIFNumber++ ); name.prepend("View-"); + return name; +} + +KoViewIface::KoViewIface( KoView *view ) + : DCOPObject( newIfaceName() ) +{ + m_actionProxy = new KDCOPActionProxy( view->actionCollection(), this ); +} + +KoViewIface::KoViewIface( const char *name, KoView *view ) + : DCOPObject( name ) +{ + m_pView = view; + m_actionProxy = new KDCOPActionProxy( view->actionCollection(), this ); +} + +KoViewIface::~KoViewIface() +{ + delete m_actionProxy; +} + +DCOPRef KoViewIface::action( const QCString &name ) +{ + return DCOPRef( kapp->dcopClient()->appId(), m_actionProxy->actionObjectId( name ) ); +} + +QCStringList KoViewIface::actions() +{ + QCStringList res; + QValueList<KAction *> lst = m_actionProxy->actions(); + QValueList<KAction *>::ConstIterator it = lst.begin(); + QValueList<KAction *>::ConstIterator end = lst.end(); + for (; it != end; ++it ) + res.append( (*it)->name() ); + + return res; +} + +QMap<QCString,DCOPRef> KoViewIface::actionMap() +{ + return m_actionProxy->actionMap(); +} diff --git a/lib/kofficecore/KoViewIface.h b/lib/kofficecore/KoViewIface.h new file mode 100644 index 00000000..1d01af64 --- /dev/null +++ b/lib/kofficecore/KoViewIface.h @@ -0,0 +1,56 @@ +/* This file is part of the KDE project + Copyright (c) 2000 Simon Hausmann <hausmann@kde.org> + + $Id: KoViewIface.h 466447 2005-10-02 17:54:10Z zander $ + + 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. +*/ +#ifndef __KoViewIface_h__ +#define __KoViewIface_h__ + +#include <dcopobject.h> +#include <dcopref.h> + +#include "koffice_export.h" + +class KDCOPActionProxy; +class KoView; + +class KOFFICECORE_EXPORT KoViewIface : public DCOPObject +{ + K_DCOP +public: + KoViewIface( KoView *view ); + KoViewIface( const char *name, KoView *view ); + // TODO same args order as KoDocumentIface + + virtual ~KoViewIface(); + + // Generate a name for this interface. Automatically used if + // the first constructor is used. + static QCString newIfaceName(); + +k_dcop: + DCOPRef action( const QCString &name ); + QCStringList actions(); + QMap<QCString,DCOPRef> actionMap(); + +protected: + KoView *m_pView; + KDCOPActionProxy *m_actionProxy; +}; + +#endif diff --git a/lib/kofficecore/KoXmlNS.cpp b/lib/kofficecore/KoXmlNS.cpp new file mode 100644 index 00000000..769ee653 --- /dev/null +++ b/lib/kofficecore/KoXmlNS.cpp @@ -0,0 +1,44 @@ +/* This file is part of the KDE project + Copyright (C) 2004 David Faure <faure@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 "KoXmlNS.h" + +const char* const KoXmlNS::office = "urn:oasis:names:tc:opendocument:xmlns:office:1.0"; +const char* const KoXmlNS::meta = "urn:oasis:names:tc:opendocument:xmlns:meta:1.0"; +const char* const KoXmlNS::config = "urn:oasis:names:tc:opendocument:xmlns:config:1.0"; +const char* const KoXmlNS::text = "urn:oasis:names:tc:opendocument:xmlns:text:1.0"; +const char* const KoXmlNS::table = "urn:oasis:names:tc:opendocument:xmlns:table:1.0"; +const char* const KoXmlNS::draw = "urn:oasis:names:tc:opendocument:xmlns:drawing:1.0"; +const char* const KoXmlNS::presentation = "urn:oasis:names:tc:opendocument:xmlns:presentation:1.0"; +const char* const KoXmlNS::dr3d = "urn:oasis:names:tc:opendocument:xmlns:dr3d:1.0"; +const char* const KoXmlNS::chart = "urn:oasis:names:tc:opendocument:xmlns:chart:1.0"; +const char* const KoXmlNS::form = "urn:oasis:names:tc:opendocument:xmlns:form:1.0"; +const char* const KoXmlNS::script = "urn:oasis:names:tc:opendocument:xmlns:script:1.0"; +const char* const KoXmlNS::style = "urn:oasis:names:tc:opendocument:xmlns:style:1.0"; +const char* const KoXmlNS::number = "urn:oasis:names:tc:opendocument:xmlns:datastyle:1.0"; +const char* const KoXmlNS::manifest = "urn:oasis:names:tc:opendocument:xmlns:manifest:1.0"; + +const char* const KoXmlNS::math = "http://www.w3.org/1998/Math/MathML"; +const char* const KoXmlNS::svg = "urn:oasis:names:tc:opendocument:xmlns:svg-compatible:1.0"; +const char* const KoXmlNS::fo = "urn:oasis:names:tc:opendocument:xmlns:xsl-fo-compatible:1.0"; +const char* const KoXmlNS::dc = "http://purl.org/dc/elements/1.1/"; +const char* const KoXmlNS::xlink = "http://www.w3.org/1999/xlink"; +// const char* const KoXmlNS::smil = "urn:oasis:names:tc:opendocument:xmlns:smil-compatible:1.0"; + +const char* const KoXmlNS::koffice = "http://www.koffice.org/2005/"; diff --git a/lib/kofficecore/KoXmlNS.h b/lib/kofficecore/KoXmlNS.h new file mode 100644 index 00000000..410519b8 --- /dev/null +++ b/lib/kofficecore/KoXmlNS.h @@ -0,0 +1,55 @@ +/* This file is part of the KDE project + Copyright (C) 2004 David Faure <faure@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. +*/ + +#ifndef KOXMLNS_H +#define KOXMLNS_H + +#include <koffice_export.h> +/** + * Repository of XML namespaces used for OASIS documents. + * Note: if we have code which needs often those as QStrings, then maybe + * we need static const QString& versions of them too. Needs an init() though. + */ +class KOFFICECORE_EXPORT KoXmlNS { +public: + static const char* const office; + static const char* const meta; + static const char* const config; + static const char* const text; + static const char* const table; + static const char* const draw; + static const char* const presentation; + static const char* const dr3d; + static const char* const chart; + static const char* const form; + static const char* const script; + static const char* const style; + static const char* const number; + static const char* const manifest; + + static const char* const math; + static const char* const svg; + static const char* const fo; + static const char* const dc; + static const char* const xlink; + + static const char* const koffice; +}; + +#endif /* KOXMLNS_H */ diff --git a/lib/kofficecore/KoXmlReader.cpp b/lib/kofficecore/KoXmlReader.cpp new file mode 100644 index 00000000..ae1e2ae6 --- /dev/null +++ b/lib/kofficecore/KoXmlReader.cpp @@ -0,0 +1,1568 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Ariya Hidayat <ariya@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. +*/ + +/* + This is a memory-efficient DOM implementation for KOffice. See the API + documentation for details. + + IMPORTANT ! + + * When you change this stuff, make sure it DOES NOT BREAK the test suite. + Build tests/koxmlreadertest.cpp and verify it. Many sleepless nights + have been sacrificed for this piece of code, do not let those precious + hours wasted! + + * Run testdom.cpp WITH Valgrind's memcheck tool and make sure NO illegal + memory read/write and any type of leak occurs. If you are not familiar + with Valgrind then RTFM first and come back again later on. + + * The public API shall remain as compatible as QDom. + + * All QDom-compatible methods should behave the same. All QDom-compatible + functions should return the same result. In case of doubt, run + koxmlreadertest.cpp but uncomment KOXML_USE_QDOM in koxmlreader.h + so that the tests are performed with standard QDom. + + Some differences compared to QDom: + + - DOM tree in KoXmlDocument is read-only, you can not modify it. This is + sufficient for KOffice since the tree is only accessed when loading + a document to the application. For saving the document to XML file, + use KoXmlWriter. + + - Comment node (like QDomComment) is not implemented as comments are + simply ignored. + + - DTD, entity and entity reference are not handled. Thus, the associated + nodes (like QDomDocumentType, QDomEntity, QDomEntityReference) are also + not implemented. + + - Attribute mapping node is not implemented. But of course, functions to + query attributes of an element are available. + + + */ + +#include "KoXmlReader.h" + +#ifndef KOXML_USE_QDOM + +#include <qxml.h> +#include <qdom.h> + +#include <qmap.h> +#include <qcstring.h> + +// double QString, used for hashing against namespace and qualified name pair +class DQString +{ +public: + DQString() { } + DQString( const QString& s1, const QString& s2 ){ str1 = s1; str2 = s2; } + DQString( const DQString& b ) { str1 = b.str1; str2 = b.str2; } + DQString& operator=( const DQString& b ){ str1 = b.str1; str2 = b.str2; return *this; } + bool operator==( const DQString& b ) const { return (str1==b.str1) && (str2==b.str2); } + bool operator!=( const DQString& b ) const { return (str1!=b.str1) || (str2!=b.str2); } + bool operator<( const DQString& b ) const + { return ( str1 < b.str1 ) ? true : ( str1==b.str1 ) ? str2<b.str2 : false; } + QString s1() const { return str1; } + QString s2() const { return str2; } +private: + QString str1, str2; +}; + +// just for completeness, this is the self-test for DQString above +#if 0 + DQString b1; + DQString b2; + CHECK( b1==b2, true ); + CHECK( b1!=b2, false ); + + b1 = DQString( "sweet","princess" ); + b2 = DQString( "sweet","princess" ); + CHECK( b1==b2, true ); + CHECK( b1!=b2, false ); + + b1 = DQString( "sweet","princess" ); + b2 = DQString( "bad","prince" ); + CHECK( b1==b2, false ); + CHECK( b1!=b2, true ); +#endif + +class KoXmlStream +{ +public: + KoXmlStream(){ saveData = true; data.reserve(1024); pos = 0; } + QString stringData() const { return data; } + void setSaveData( bool s ){ saveData = s; } + int at() const { return pos; } + KoXmlStream& operator<<( const QString& str ) + { if(saveData) data.append(str); pos+=str.length(); return *this; } + KoXmlStream& appendEscape( const QString& str ); + +private: + bool saveData; + QString data; + int pos; +}; + +KoXmlStream& KoXmlStream::appendEscape( const QString& str ) +{ + unsigned len = str.length(); + + if( saveData ) + { + data.reserve( data.length() + len ); + for( unsigned c=0; c<len; c++ ) + if( str[c]=='<' ){ data.append( "<"); pos += 4; } else + if( str[c]=='>'){ data.append( ">"); pos+= 4; } else + if( str[c]=='"'){ data.append( """); pos += 6; } else + if( str[c]=='&'){ data.append( "&"); pos += 5; } else + { data.append( str[c] ); pos++; } + } + else + { + pos += len; + for( unsigned c=0; c<len; c++ ) + if( str[c]=='<' ) pos += 3; else // "<" + if( str[c]=='>') pos+= 3; else // ">" + if( str[c]=='"') pos += 5; else // """ + if( str[c]=='&') pos += 4; // "&" + } + + return *this; +} + +class KoXmlNodeData +{ +public: + + KoXmlNodeData(); + virtual ~KoXmlNodeData(); + + // generic properties + KoXmlNode::NodeType nodeType; + QString tagName; + QString namespaceURI; + QString prefix; + QString localName; + + // reference counting + unsigned long count; + void ref() { count++; } + void unref() { --count; if( !count ) delete this; } + + // type information + virtual const char* typeInfo() const { return "Node"; } + QString nodeName() const; + + // for tree and linked-list + KoXmlNodeData* parent; + KoXmlNodeData* prev; + KoXmlNodeData* next; + KoXmlNodeData* first; + KoXmlNodeData* last; + + QString text(); + + // node manipulation + void appendChild( KoXmlNodeData* child ); + virtual void clear(); + KoXmlNodeData* ownerDocument(); + + // attributes + void setAttribute( const QString& name, const QString& value ); + QString attribute( const QString& name ); + bool hasAttribute( const QString& name ); + void setAttributeNS( const QString& nsURI, const QString& name, const QString& value ); + QString attributeNS( const QString& nsURI, const QString& name ); + bool hasAttributeNS( const QString& nsURI, const QString& name ); + + // for text and CDATA + QString data() const; + void setData( const QString& data ); + + // for document node + QXmlSimpleReader* xmlReader; + QString buffer; + bool setContent( QXmlInputSource* source, QXmlReader* reader, + QString* errorMsg = 0, int* errorLine = 0, int* errorColumn = 0 ); + + // used when doing on-demand (re)parse + bool loaded; + unsigned startPos, endPos; + void loadChildren( int depth=1 ); + void unloadChildren(); + bool fastLoading; + +private: + QMap<QString,QString> attr; + QMap<DQString,QString> attrNS; + QString textData; + friend class KoXmlHandler; +}; + +class KoXmlHandler : public QXmlDefaultHandler +{ +public: + KoXmlHandler( KoXmlNodeData*, bool processNamespace ); + ~KoXmlHandler(); + + void setMaxDepth( int d ){ maxDepth = d; } + void setInitialOffset( int ofs ){ parseOffset = ofs; } + + // content handler + bool startDocument(); + bool endDocument(); + + bool startElement( const QString& nsURI, const QString& localName, + const QString& qName, const QXmlAttributes& atts ); + bool endElement( const QString& nsURI, const QString& localName, + const QString& qName ); + + bool characters( const QString& ch ); + bool processingInstruction( const QString& target, const QString& data ); + bool skippedEntity( const QString& name ); + + // lexical handler + bool startCDATA(); + bool endCDATA(); + bool startEntity( const QString & ); + bool endEntity( const QString & ); + bool startDTD( const QString& name, const QString& publicId, + const QString& systemId ); + bool comment( const QString& ch ); + + // decl handler + bool externalEntityDecl( const QString &name, const QString &publicId, + const QString &systemId ) ; + + // DTD handler + bool notationDecl( const QString & name, const QString & publicId, + const QString & systemId ); + bool unparsedEntityDecl( const QString &name, const QString &publicId, + const QString &systemId, const QString ¬ationName ) ; + + // error handler + bool fatalError( const QXmlParseException& exception ); + + QString errorMsg; + int errorLine; + int errorColumn; + +private: + bool processNamespace; + KoXmlNodeData* rootNode; + KoXmlNodeData* currentNode; + QString entityName; + bool cdata; + int parseOffset; + KoXmlStream bufferStream; + int elementDepth; + int maxDepth; +}; + +// ================================================================== +// +// KoXmlNodeData +// +// ================================================================== + +KoXmlNodeData::KoXmlNodeData() +{ + nodeType = KoXmlNode::NullNode; + + tagName = QString::null; + prefix = QString::null; + localName = QString::null; + namespaceURI = QString::null; + textData = QString::null; + + count = 1; + parent = 0; + prev = next = 0; + first = last = 0; + + xmlReader = 0; + startPos = endPos = 0; + + fastLoading = false; + + // assume true, it will be set to false by XML parser when this node + // apparently has children AND the children are not loaded + loaded = true; +} + +KoXmlNodeData::~KoXmlNodeData() +{ + clear(); +} + +void KoXmlNodeData::clear() +{ + if( first ) + for( KoXmlNodeData* node = first; node ; ) + { + KoXmlNodeData* next = node->next; + node->unref(); + node = next; + } + + nodeType = KoXmlNode::NullNode; + tagName = QString::null; + prefix = QString::null; + namespaceURI = QString::null; + textData = QString::null; + + attr.clear(); + attrNS.clear(); + + parent = 0; + prev = next = 0; + first = last = 0; + + delete xmlReader; + xmlReader = 0; + buffer = QString::null; +} + +QString KoXmlNodeData::text() +{ + QString t( "" ); + + loadChildren(); + + KoXmlNodeData* node = first; + while ( node ) + { + switch( node->nodeType ) + { + case KoXmlNode::ElementNode: + t += node->text(); break; + case KoXmlNode::TextNode: + t += node->data(); break; + case KoXmlNode::CDATASectionNode: + t += node->data(); break; + default: break; + } + node = node->next; + } + + return t; +} + +QString KoXmlNodeData::nodeName() const +{ + QString n; + + switch( nodeType ) + { + case KoXmlNode::ElementNode: + n = tagName; + if (!prefix.isEmpty()) n.prepend(":").prepend(prefix); break; + case KoXmlNode::TextNode: return QString("#text"); + case KoXmlNode::CDATASectionNode: return QString("#cdata-section"); + case KoXmlNode::DocumentNode: return QString("#document"); + default: break; + } + + return n; +} + +KoXmlNodeData* KoXmlNodeData::ownerDocument() +{ + KoXmlNodeData* owner = this; + + while( owner->parent ) + owner = owner->parent; + + return (owner->nodeType==KoXmlNode::DocumentNode) ? owner : 0; +} + +void KoXmlNodeData::appendChild( KoXmlNodeData* node ) +{ + node->parent = this; + if( !last ) + first = last = node; + else + { + last->next = node; + node->prev = last; + node->next = 0; + last = node; + } +} + +void KoXmlNodeData::setAttribute( const QString& name, const QString& value ) +{ + attr[ name ] = value; +} + +QString KoXmlNodeData::attribute( const QString& name ) +{ + return attr[ name ]; +} + +bool KoXmlNodeData::hasAttribute( const QString& name ) +{ + return attr.contains( name ); +} + +void KoXmlNodeData::setAttributeNS( const QString& nsURI, +const QString& name, const QString& value ) +{ + QString prefix; + QString localName = name; + int i = name.find( ':' ); + if( i != -1 ) + { + localName = name.mid( i + 1 ); + prefix = name.left( i ); + } + + if( prefix.isNull() ) return; + + DQString key( nsURI, localName ); + attrNS[ key ] = value; +} + +QString KoXmlNodeData::attributeNS( const QString& nsURI, const QString& name ) +{ + DQString key( nsURI, name ); + return attrNS[ key ]; +} + +bool KoXmlNodeData::hasAttributeNS( const QString& nsURI, const QString& name ) +{ + DQString key( nsURI, name ); + return attrNS.contains( key ); +} + +QString KoXmlNodeData::data() const +{ + return textData; +} + +void KoXmlNodeData::setData( const QString& d ) +{ + textData = d; +} + +bool KoXmlNodeData::setContent( QXmlInputSource* source, +QXmlReader* reader, QString* errorMsg, int* errorLine, int* errorColumn ) +{ + if( nodeType != KoXmlNode::DocumentNode ) + return false; + + clear(); + nodeType = KoXmlNode::DocumentNode; + + // sanity checks + if( !source ) return false; + if( !reader ) return false; + + // copy the reader for later on-demand loading + // FIXME this is a workaround because no copy is possible with QXmlReader + char* features[] = + { + "http://xml.org/sax/features/namespaces", + "http://xml.org/sax/features/namespace-prefixes", + "http://trolltech.com/xml/features/report-whitespace-only-CharData", + "http://trolltech.com/xml/features/report-start-end-entity" + }; + xmlReader = new QXmlSimpleReader; + for( int fi=0; fi<4; fi++ ) + xmlReader->setFeature( features[fi], reader->feature( features[fi] ) ); + + bool processNamespace = + reader->feature( "http://xml.org/sax/features/namespaces" ) && + !reader->feature( "http://xml.org/sax/features/namespace-prefixes" ); + + KoXmlHandler handler( this, processNamespace ); + reader->setContentHandler( &handler ); + reader->setErrorHandler( &handler ); + reader->setLexicalHandler( &handler ); + reader->setDeclHandler( &handler ); + reader->setDTDHandler( &handler ); + + if( !fastLoading ) + handler.setMaxDepth( 4 ); + + if( !reader->parse( source ) ) + { + // parsing error has occurred + if( errorMsg ) *errorMsg = handler.errorMsg; + if( errorLine ) *errorLine = handler.errorLine; + if( errorColumn ) *errorColumn = handler.errorColumn; + return false; + } + + return true; +} + +void KoXmlNodeData::loadChildren( int depth ) +{ + // for more than 1 level, force reloading anyway + if( ( depth== 1 ) && loaded ) return; + + KoXmlNodeData* doc = ownerDocument(); + if( !doc ) return; + + // fast loading? then this on-demand loading makes no sense + if( doc->fastLoading ) return; + + unloadChildren(); + + bool nsProcess = + doc->xmlReader->feature( "http://xml.org/sax/features/namespaces" ) && + !doc->xmlReader->feature( "http://xml.org/sax/features/namespace-prefixes" ); + + + // XML snippet for the children, including this element + QString snippet = doc->buffer.mid( startPos, endPos-startPos+1 ); + + // now parse all subnodes + KoXmlHandler handler( this, nsProcess ); + handler.setMaxDepth( depth ); + handler.setInitialOffset( startPos ); + doc->xmlReader->setContentHandler( &handler ); + doc->xmlReader->setErrorHandler( &handler ); + doc->xmlReader->setLexicalHandler( &handler ); + doc->xmlReader->setDeclHandler( &handler ); + doc->xmlReader->setDTDHandler( &handler ); + + QXmlInputSource source; + source.setData( snippet ); + if( !doc->xmlReader->parse( source ) ) + { + // parsing error has occurred, which should not happen + // nothing we can do except... + loaded = false; + qWarning( "On-demand loading triggers parse error!" ); + } + else + loaded = true; +} + +void KoXmlNodeData::unloadChildren() +{ + if( !loaded ) return; + + if( first ) + for( KoXmlNodeData* node = first; node ; ) + { + KoXmlNodeData* next = node->next; + node->unloadChildren(); + node->unref(); + node = next; + } + + loaded = false; + first = last = 0; +} + + +// ================================================================== +// +// KoXmlHandler +// +// ================================================================== + +KoXmlHandler::KoXmlHandler( KoXmlNodeData* n, bool ns ): +QXmlDefaultHandler() +{ + processNamespace = ns; + + rootNode = n; + currentNode = n; + cdata = false; + entityName = QString::null; + + errorMsg = QString::null; + errorLine = 0; + errorColumn = 0; + + parseOffset = 0; + elementDepth = -1; + maxDepth = 999; + + bufferStream.setSaveData( rootNode->nodeType == KoXmlNode::DocumentNode ); +} + +KoXmlHandler::~KoXmlHandler() +{ +} + +bool KoXmlHandler::startDocument() +{ + // just for sanity + currentNode = rootNode; + cdata = false; + entityName = QString::null; + elementDepth = -1; + + return true; +} + +bool KoXmlHandler::endDocument() +{ + // just for sanity + if( rootNode->nodeType == KoXmlNode::DocumentNode ) + if( currentNode!=rootNode ) + return false; + + if( rootNode->nodeType == KoXmlNode::DocumentNode ) + rootNode->buffer = bufferStream.stringData(); + + return true; +} + +bool KoXmlHandler::startDTD( const QString& name, const QString& publicId, +const QString& systemId ) +{ + Q_UNUSED( name ); + Q_UNUSED( publicId ); + Q_UNUSED( systemId ); + + // we skip DTD + return true; +} + +bool KoXmlHandler::startElement( const QString& nsURI, const QString& localName, +const QString& name, const QXmlAttributes& atts ) +{ + Q_UNUSED( localName ); + + // sanity check + if( !currentNode ) + return false; + + // we are going one level deeper + elementDepth++; + + QString nodePrefix, nodeLocalName, nodeTagName; + KoXmlNodeData* element = 0; + + if( processNamespace ) + { + // parse, using namespace + nodeTagName = name; + nodeLocalName = name; + nodePrefix = nsURI.isNull() ? QString::null : QString(""); + int i = name.find( ':' ); + if( i != -1 ) + { + nodeTagName = name.mid( i + 1 ); + nodeLocalName = nodeTagName; + nodePrefix = name.left( i ); + } + + if( elementDepth <= maxDepth ) + { + // construct a new element + element = new KoXmlNodeData; + element->nodeType = KoXmlNode::ElementNode; + element->parent = currentNode; + element->namespaceURI = nsURI; + element->prefix = nodePrefix; + element->localName = nodeLocalName; + element->tagName = nodeTagName; + + // Note: endPos will be later fixed in endElement + element->endPos = element->startPos = parseOffset + bufferStream.at(); + + // handle the attributes + for( int c=0; c<atts.length(); c++ ) + { + QString prefix; + QString qName; // with prefix + QString name; // without prefix, i.e. local name + + name = qName = atts.qName(c); + int i = qName.find( ':' ); + if( i != -1 ) prefix = qName.left( i ); + if( i != -1 ) name = qName.mid( i + 1 ); + element->setAttributeNS( atts.uri(c), qName, atts.value(c) ); + element->setAttribute( name, atts.value(c) ); + } + } + + // save in buffer for later on-demand loading + if( ( rootNode->nodeType != KoXmlNode::DocumentNode ) || !rootNode->fastLoading ) + { + bufferStream << "<"; + if( !nodePrefix.isEmpty() ) + bufferStream << nodePrefix << ":"; + bufferStream << localName; + bufferStream << " xmlns"; + if( !nodePrefix.isEmpty() ) + bufferStream << ":" << nodePrefix; + bufferStream << "=\""; + bufferStream.appendEscape( nsURI ); + bufferStream << "\""; + for( int c=0; c<atts.length(); c++ ) + { + QString prefix; + QString name = atts.qName(c); // qName contains the prefix + int i = name.find( ':' ); + if( i != -1 ) prefix = name.left( i ); + if( i != -1 ) name = atts.qName(c).mid( i + 1 ); + if( !atts.uri(c).isEmpty() ) + bufferStream << " xmlns:" << prefix << "=\"" << atts.uri(c) << "\""; + bufferStream << " "; + if( !prefix.isEmpty() ) bufferStream << prefix << ":"; + bufferStream << name << "=\""; + bufferStream.appendEscape( atts.value(c) ); + bufferStream << "\""; + } + bufferStream << ">"; + } + } + else + { + // parse, without using namespace + nodeTagName = name; + + if( elementDepth <= maxDepth ) + { + // construct a new element + element = new KoXmlNodeData; + element->nodeType = KoXmlNode::ElementNode; + element->parent = currentNode; + element->namespaceURI = QString::null; + element->prefix = QString::null; + element->localName = QString::null; + element->tagName = nodeTagName; + + if( rootNode->nodeType == KoXmlNode::DocumentNode ) + element->fastLoading = rootNode->fastLoading; + + // Note: endPos will be later fixed in endElement + element->endPos = element->startPos = parseOffset + bufferStream.at(); + + // handle the attributes + for( int c=0; c<atts.length(); c++ ) + element->setAttribute( atts.qName(c), atts.value(c) ); + } + + // save in buffer for later on-demand loading + if( ( rootNode->nodeType != KoXmlNode::DocumentNode ) || !rootNode->fastLoading ) + { + bufferStream << "<"; + bufferStream << nodeTagName; + for( int c=0; c<atts.length(); c++ ) + { + bufferStream << " " << atts.qName(c) << "=\""; + bufferStream.appendEscape( atts.value(c) ); + bufferStream << "\""; + } + bufferStream << ">"; + } + + } + + // if we do not parse a complete document, the first element is ignored + // this feature is used in on-demand loading + // e.g. "<ul><li><b>bold items</b></li></ul>", if root node points to + // <ul> tag, then the first <ul> is skipped so that next <li> element + // becomes the child of it. + if( elementDepth == 0 ) + if( rootNode->nodeType != KoXmlNode::DocumentNode ) + { + delete element; + return true; + } + + if( element ) + { + // add as the child and traverse to it + currentNode->loaded = true; + currentNode->appendChild( element ); + currentNode = element; + } + else + currentNode->loaded = false; + + return true; +} + +bool KoXmlHandler::endElement( const QString& nsURI, const QString& localName, +const QString& qName ) +{ + Q_UNUSED( nsURI ); + Q_UNUSED( localName ); + + // sanity check + if( !currentNode ) return false; + if( !currentNode->parent ) return false; + + // see comments in startElement about first element and on-demand loading + if( rootNode->nodeType == KoXmlNode::DocumentNode ) + if( currentNode == rootNode ) + return false; + + // buffer for on-demand loading + if( ( rootNode->nodeType != KoXmlNode::DocumentNode ) || !rootNode->fastLoading ) + bufferStream << "</" << qName << ">"; + + // fix up the pointer + currentNode->endPos = parseOffset + bufferStream.at() - 1; + + // go up one level + if( elementDepth <= maxDepth ) + currentNode = currentNode->parent; + + // we are going up one level + elementDepth--; + + return true; +} + +bool KoXmlHandler::characters( const QString& str ) +{ + // sanity check + if( rootNode->nodeType == KoXmlNode::DocumentNode ) + if( currentNode == rootNode ) + return false; + + // are we inside entity ? + if( !entityName.isEmpty() ) + { + // we do not handle entity but need to keep track of it + // because we want to skip it alltogether + return true; + } + + if( cdata ) + { + // are we inside CDATA section ? + if( elementDepth <= maxDepth ) + { + KoXmlNodeData* cdata = new KoXmlNodeData; + cdata->nodeType = KoXmlNode::CDATASectionNode; + cdata->parent = currentNode; + cdata->setData( str ); + currentNode->appendChild( cdata ); + } + + if( ( rootNode->nodeType != KoXmlNode::DocumentNode ) || !rootNode->fastLoading ) + bufferStream << "<![CDATA[" << str << "]]>"; // no escape for CDATA + } + else + { + // this must be normal text node + if( elementDepth <= maxDepth ) + { + KoXmlNodeData* text = new KoXmlNodeData; + text->nodeType = KoXmlNode::TextNode; + text->parent = currentNode; + text->setData( str ); + currentNode->appendChild( text ); + } + + // save it for later use + if( ( rootNode->nodeType != KoXmlNode::DocumentNode ) || !rootNode->fastLoading ) + bufferStream.appendEscape( str ); + } + + return true; +} + +bool KoXmlHandler::processingInstruction( const QString& target, +const QString& data ) +{ + // sanity check + if( !currentNode ) + return false; + + KoXmlNodeData* instruction = new KoXmlNodeData; + instruction->nodeType = KoXmlNode::ProcessingInstructionNode; + instruction->parent = currentNode; + instruction->tagName = target; + instruction->setData( data ); + + currentNode->appendChild( instruction ); + + return true; +} + +bool KoXmlHandler::skippedEntity( const QString& name ) +{ + Q_UNUSED( name ); + + // we skip entity + return true; +} + +bool KoXmlHandler::startCDATA() +{ + cdata = true; + return true; +} + +bool KoXmlHandler::endCDATA() +{ + cdata = false; + return true; +} + +bool KoXmlHandler::startEntity( const QString& name ) +{ + entityName = name; + return true; +} + +bool KoXmlHandler::endEntity( const QString& name ) +{ + Q_UNUSED( name ); + entityName = QString::null; + return true; +} + +bool KoXmlHandler::comment( const QString& comment ) +{ + Q_UNUSED( comment ); + + // we skip comment + return true; +} + +bool KoXmlHandler::unparsedEntityDecl( const QString &name, +const QString &publicId, const QString &systemId, const QString ¬ationName ) +{ + Q_UNUSED( name ); + Q_UNUSED( publicId ); + Q_UNUSED( systemId ); + Q_UNUSED( notationName ); + + // we skip entity + return true; +} + +bool KoXmlHandler::externalEntityDecl( const QString &name, +const QString &publicId, const QString &systemId ) +{ + Q_UNUSED( name ); + Q_UNUSED( publicId ); + Q_UNUSED( systemId ); + + // we skip entity + return true; +} + +bool KoXmlHandler::notationDecl( const QString & name, +const QString & publicId, const QString & systemId ) +{ + Q_UNUSED( name ); + Q_UNUSED( publicId ); + Q_UNUSED( systemId ); + + // we skip notation node + return true; +} + +bool KoXmlHandler::fatalError( const QXmlParseException& exception ) +{ + errorMsg = exception.message(); + errorLine = exception.lineNumber(); + errorColumn = exception.columnNumber(); + return QXmlDefaultHandler::fatalError( exception ); +} + +// ================================================================== +// +// KoXmlNode +// +// ================================================================== + +// Creates a null node +KoXmlNode::KoXmlNode() +{ + d = new KoXmlNodeData; +} + +// Destroys this node +KoXmlNode::~KoXmlNode() +{ + if( d ) d->unref(); +} + +// Creates a copy of another node +KoXmlNode::KoXmlNode( const KoXmlNode& node ) +{ + d = node.d; + d->ref(); +} + +// Creates a node for specific implementation +KoXmlNode::KoXmlNode( KoXmlNodeData* data ) +{ + d = data; + data->ref(); +} + +// Creates a shallow copy of another node +KoXmlNode& KoXmlNode::operator=( const KoXmlNode& node ) +{ + d->unref(); + d = node.d; + d->ref(); + return *this; +} + +// Note: two null nodes are always equal +bool KoXmlNode::operator==( const KoXmlNode& node ) const +{ + if( isNull() && node.isNull() ) return true; + return( d==node.d ); +} + +// Note: two null nodes are always equal +bool KoXmlNode::operator!=( const KoXmlNode& node ) const +{ + if( isNull() && !node.isNull() ) return true; + if( !isNull() && node.isNull() ) return true; + if( isNull() && node.isNull() ) return false; + return( d!=node.d ); +} + +KoXmlNode::NodeType KoXmlNode::nodeType() const +{ + return d->nodeType; +} + +bool KoXmlNode::isElement() const +{ + return d->nodeType == ElementNode; +} + +bool KoXmlNode::isText() const +{ + return (d->nodeType == TextNode) || isCDATASection(); +} + +bool KoXmlNode::isCDATASection() const +{ + return d->nodeType == CDATASectionNode; +} + +bool KoXmlNode::isDocument() const +{ + return d->nodeType == DocumentNode; +} + +bool KoXmlNode::isNull() const +{ + return d->nodeType == NullNode; +} + +void KoXmlNode::clear() +{ + d->unref(); + d = new KoXmlNodeData; +} + +QString KoXmlNode::nodeName() const +{ + return d->nodeName(); +} + +QString KoXmlNode::prefix() const +{ + return isElement() ? d->prefix : QString::null; +} + +QString KoXmlNode::namespaceURI() const +{ + return isElement() ? d->namespaceURI : QString::null; +} + +QString KoXmlNode::localName() const +{ + return isElement() ? d->localName : QString::null; +} + +KoXmlDocument KoXmlNode::ownerDocument() const +{ + KoXmlNodeData* node = d; + while( node->parent ) node = node->parent; + + if( node->nodeType != DocumentNode ) return KoXmlDocument(); + return KoXmlDocument( node ); +} + +KoXmlNode KoXmlNode::parentNode() const +{ + return d->parent ? KoXmlNode( d->parent ) : KoXmlNode(); +} + +bool KoXmlNode::hasChildNodes() const +{ + d->loadChildren(); + return d->first!=0 ; +} + +KoXmlNode KoXmlNode::firstChild() const +{ + if( !d->fastLoading ) + d->loadChildren(); + return d->first ? KoXmlNode( d->first ) : KoXmlNode(); +} + +KoXmlNode KoXmlNode::lastChild() const +{ + if( !d->fastLoading ) + d->loadChildren(); + return d->last ? KoXmlNode( d->last ) : KoXmlNode(); +} + +KoXmlNode KoXmlNode::nextSibling() const +{ + return d->next ? KoXmlNode( d->next ) : KoXmlNode(); +} + +KoXmlNode KoXmlNode::previousSibling() const +{ + return d->prev ? KoXmlNode( d->prev ) : KoXmlNode(); +} + +KoXmlNode KoXmlNode::namedItem( const QString& name ) const +{ + if( !d->fastLoading ) + d->loadChildren(); + + KoXmlNodeData* node = d->first; + while ( node ) + { + if( node->nodeName() == name ) + return KoXmlNode( node ); + node = node->next; + } + + // not found + return KoXmlNode(); +} + +KoXmlNode KoXmlNode::namedItemNS( const QString& nsURI, const QString& name ) const +{ + if( !d->fastLoading ) + d->loadChildren(); + + KoXmlNodeData* node = d->first; + while ( node ) + { + if( !node->prefix.isNull() ) + if( node->namespaceURI == nsURI ) + if( node->localName == name ) + return KoXmlNode( node ); + node = node->next; + } + + // not found + return KoXmlNode(); +} + +KoXmlElement KoXmlNode::toElement() +{ + return isElement() ? KoXmlElement( d ) : KoXmlElement(); +} + +KoXmlText KoXmlNode::toText() +{ + return isText() ? KoXmlText( d ) : KoXmlText(); +} + +KoXmlCDATASection KoXmlNode::toCDATASection() +{ + return isCDATASection() ? KoXmlCDATASection( (KoXmlNodeData*)d ) : + KoXmlCDATASection(); +} + +KoXmlDocument KoXmlNode::toDocument() +{ + return isDocument() ? KoXmlDocument( d ) : KoXmlDocument(); +} + +void KoXmlNode::load( int depth ) +{ + d->loadChildren( depth ); +} + +void KoXmlNode::unload() +{ + d->unloadChildren(); +} + +// ================================================================== +// +// KoXmlElement +// +// ================================================================== + +// Creates an empty element +KoXmlElement::KoXmlElement(): KoXmlNode() +{ + d->unref(); + d = new KoXmlNodeData; +} + +KoXmlElement::~KoXmlElement() +{ + d->unref(); + d = 0; +} + +// Creates a shallow copy of another element +KoXmlElement::KoXmlElement( const KoXmlElement& element ): KoXmlNode() +{ + d->unref(); + d = element.d; + d->ref(); +} + +KoXmlElement::KoXmlElement( KoXmlNodeData* data ): KoXmlNode() +{ + d->unref(); + d = data; + d->ref(); +} + +// Copies another element +KoXmlElement& KoXmlElement::operator=( const KoXmlElement& element ) +{ + KoXmlNode::operator=( element ); + return *this; +} + +bool KoXmlElement::operator== ( const KoXmlElement& element ) const +{ + if( isNull() || element.isNull() ) return false; + return (d==element.d); +} + +bool KoXmlElement::operator!= ( const KoXmlElement& element ) const +{ + if( isNull() && element.isNull() ) return false; + if( isNull() || element.isNull() ) return true; + return (d!=element.d); +} + +QString KoXmlElement::tagName() const +{ + return isElement() ? ((KoXmlNodeData*)d)->tagName: QString::null; +} + +QString KoXmlElement::text() const +{ + return d->text(); +} + +bool KoXmlElement::isElement() const +{ + return true; +} + +QString KoXmlElement::attribute( const QString& name ) const +{ + return attribute( name, QString::null ); +} + +QString KoXmlElement::attribute( const QString& name, +const QString& defaultValue ) const +{ + if( !isElement() ) + return defaultValue; + + if( !hasAttribute( name ) ) + return defaultValue; + + return ((KoXmlNodeData*)d)->attribute( name ); +} + +QString KoXmlElement::attributeNS( const QString& namespaceURI, +const QString& localName, const QString& defaultValue ) const +{ + if( !isElement() ) + return defaultValue; + + if( !hasAttributeNS( namespaceURI,localName ) ) + return defaultValue; + + return ((KoXmlNodeData*)d)->attributeNS( namespaceURI,localName ); +} + +bool KoXmlElement::hasAttribute( const QString& name ) const +{ + return isElement() ? ((KoXmlNodeData*)d)->hasAttribute( name ) : false; +} + +bool KoXmlElement::hasAttributeNS( const QString& namespaceURI, +const QString& localName ) const +{ + return isElement() ? ((KoXmlNodeData*)d)->hasAttributeNS( + namespaceURI, localName ) : false;; +} + +// ================================================================== +// +// KoXmlText +// +// ================================================================== + +KoXmlText::KoXmlText(): KoXmlNode() +{ + d->unref(); + d = new KoXmlNodeData; + d->nodeType = TextNode; +} + +KoXmlText::~KoXmlText() +{ + if( d ) d->unref(); + d = 0; +} + +KoXmlText::KoXmlText( const KoXmlText& text ): KoXmlNode() +{ + d->unref(); + d = (KoXmlNodeData*) text.d; + d->ref(); +} + +KoXmlText::KoXmlText( KoXmlNodeData* data ): KoXmlNode() +{ + d->unref(); + d = data; + d->ref(); +} + +bool KoXmlText::isText() const +{ + return true; +} + +QString KoXmlText::data() const +{ + return ((KoXmlNodeData*)d)->data(); +} + +KoXmlText& KoXmlText::operator=( const KoXmlText& element ) +{ + KoXmlNode::operator=( element ); + return *this; +} + +// ================================================================== +// +// KoXmlCDATASection +// +// ================================================================== + +KoXmlCDATASection::KoXmlCDATASection(): KoXmlText() +{ + d->unref(); + d = new KoXmlNodeData; + d->nodeType = KoXmlNode::CDATASectionNode; +} + +KoXmlCDATASection::~KoXmlCDATASection() +{ + d->unref(); + d = 0; +} + +KoXmlCDATASection::KoXmlCDATASection( const KoXmlCDATASection& cdata ): +KoXmlText() +{ + d->unref(); + d = (KoXmlNodeData*) cdata.d; + d->ref(); +} + +KoXmlCDATASection::KoXmlCDATASection( KoXmlNodeData* cdata ): +KoXmlText() +{ + d->unref(); + d = cdata; + d->ref(); +} + +bool KoXmlCDATASection::isCDATASection() const +{ + return true; +} + +KoXmlCDATASection& KoXmlCDATASection::operator=( const KoXmlCDATASection& cdata ) +{ + KoXmlNode::operator=( cdata ); + return *this; +} + +// ================================================================== +// +// KoXmlDocument +// +// ================================================================== + +KoXmlDocument::KoXmlDocument(): KoXmlNode() +{ + d->unref(); + d = new KoXmlNodeData; + d->nodeType = KoXmlNode::DocumentNode; +} + +KoXmlDocument::~KoXmlDocument() +{ + d->unref(); + d = 0; +} + +KoXmlDocument::KoXmlDocument( KoXmlNodeData* data ): KoXmlNode() +{ + d->unref(); + d = data; + d->ref(); +} + +// Creates a copy of another document +KoXmlDocument::KoXmlDocument( const KoXmlDocument& doc ): KoXmlNode() +{ + d->unref(); + d = doc.d; + d->ref(); +} + +// Creates a shallow copy of another document +KoXmlDocument& KoXmlDocument::operator=( const KoXmlDocument& doc ) +{ + KoXmlNode::operator=( doc ); + return *this; +} + +// Checks if this document and doc are equals +bool KoXmlDocument::operator==( const KoXmlDocument& doc ) const +{ + return( d==doc.d ); +} + +// Checks if this document and doc are not equals +bool KoXmlDocument::operator!=( const KoXmlDocument& doc ) const +{ + return( d!=doc.d ); +} + +bool KoXmlDocument::isDocument() const +{ + return true; +} + +KoXmlElement KoXmlDocument::documentElement() const +{ + for( KoXmlNodeData* node=d->first; node; ) + if( node->nodeType==KoXmlNode::ElementNode ) + return KoXmlElement( node ); + else node = node->next; + + return KoXmlElement(); +} + +void KoXmlDocument::setFastLoading( bool f ) +{ + d->fastLoading = f; +} + +bool KoXmlDocument::fastLoading() const +{ + return d->fastLoading; +} + +bool KoXmlDocument::setContent( QXmlInputSource *source, QXmlReader *reader, + QString* errorMsg, int* errorLine, int* errorColumn ) +{ + if( d->nodeType != KoXmlNode::DocumentNode ) + return false; + + return d->setContent( source, reader, errorMsg, errorLine, errorColumn ); +} + +// no namespace processing +bool KoXmlDocument::setContent( QIODevice* device, QString* errorMsg, +int* errorLine, int* errorColumn ) +{ + return setContent( device, false, errorMsg, errorLine, errorColumn ); +} + +bool KoXmlDocument::setContent( QIODevice* device, bool namespaceProcessing, +QString* errorMsg, int* errorLine, int* errorColumn ) +{ + if( d->nodeType != KoXmlNode::DocumentNode ) + return false; + + QXmlSimpleReader reader; + reader.setFeature( "http://xml.org/sax/features/namespaces", namespaceProcessing ); + reader.setFeature( "http://xml.org/sax/features/namespace-prefixes", !namespaceProcessing ); + reader.setFeature( "http://trolltech.com/xml/features/report-whitespace-only-CharData", false ); + + // FIXME this hack is apparently private + //reader.setUndefEntityInAttrHack(true); + + QXmlInputSource source( device ); + return d->setContent( &source, &reader, errorMsg, errorLine, errorColumn ); +} + +#endif + +KoXmlElement KoXml::namedItemNS( const KoXmlNode& node, const char* nsURI, +const char* localName ) +{ +#ifdef KOXML_USE_QDOM + // David's solution for namedItemNS, only for QDom stuff + KoXmlNode n = node.firstChild(); + for ( ; !n.isNull(); n = n.nextSibling() ) { + if ( n.isElement() && n.localName() == localName && + n.namespaceURI() == nsURI ) + return n.toElement(); + } + return KoXmlElement(); +#else + return node.namedItemNS( nsURI, localName).toElement(); +#endif +} + +void KoXml::load( KoXmlNode& node, int depth ) +{ +#ifdef KOXML_USE_QDOM + // do nothing, QDom has no on-demand loading + Q_UNUSED( node ); + Q_UNUSED( depth ); +#else + node.load( depth ); +#endif +} + + +void KoXml::unload( KoXmlNode& node ) +{ +#ifdef KOXML_USE_QDOM + // do nothing, QDom has no on-demand unloading + Q_UNUSED( node ); +#else + node.unload(); +#endif +} diff --git a/lib/kofficecore/KoXmlReader.h b/lib/kofficecore/KoXmlReader.h new file mode 100644 index 00000000..df2f10c9 --- /dev/null +++ b/lib/kofficecore/KoXmlReader.h @@ -0,0 +1,320 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Ariya Hidayat <ariya@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. +*/ + +#ifndef KOFFICE_XMLREADER +#define KOFFICE_XMLREADER + +// use standard QDom, useful to test KoXml classes against Qt's QDom +//#define KOXML_USE_QDOM + +#include <qdom.h> +#include <koffice_export.h> + +#ifdef KOXML_USE_QDOM + +#define KoXmlNode QDomNode +#define KoXmlElement QDomElement +#define KoXmlText QDomText +#define KoXmlCDATASection QDomCDATASection +#define KoXmlDocument QDomDocument + +#else + +class QIODevice; +class QString; +class QTextStream; +class QXmlReader; +class QXmlInputSource; + +class KoXmlElement; +class KoXmlText; +class KoXmlCDATASection; +class KoXmlDocument; +class KoXmlNodeData; + +/** + * KoXmlNode represents a node in a DOM tree. + * + * KoXmlNode is a base class for KoXmlElement, KoXmlText. + * Often, these subclasses are used for getting the data instead of KoXmlNode. + * However, as base class, KoXmlNode is very helpful when for example iterating + * all child nodes within one parent node. + * + * KoXmlNode implements an explicit sharing, a node shares its data with + * other copies (if exist). + * + * @author Ariya Hidayat <ariya@kde.org> + */ +class KOFFICECORE_EXPORT KoXmlNode +{ +public: + + enum NodeType + { + NullNode = 0, + ElementNode, + TextNode, + CDATASectionNode, + ProcessingInstructionNode, + DocumentNode + }; + + KoXmlNode(); + KoXmlNode( const KoXmlNode& node ); + KoXmlNode& operator=( const KoXmlNode& node ); + bool operator== ( const KoXmlNode& ) const; + bool operator!= ( const KoXmlNode& ) const; + virtual ~KoXmlNode(); + + virtual KoXmlNode::NodeType nodeType() const; + virtual bool isNull() const; + virtual bool isElement() const; + virtual bool isText() const; + virtual bool isCDATASection() const; + virtual bool isDocument() const; + + void clear(); + KoXmlElement toElement(); + KoXmlText toText(); + KoXmlCDATASection toCDATASection(); + KoXmlDocument toDocument(); + + virtual QString nodeName() const; + virtual QString namespaceURI() const; + virtual QString prefix() const; + virtual QString localName() const; + + KoXmlDocument ownerDocument() const; + KoXmlNode parentNode() const; + + bool hasChildNodes() const; + KoXmlNode firstChild() const; + KoXmlNode lastChild() const; + KoXmlNode nextSibling() const; + KoXmlNode previousSibling() const; + + KoXmlNode namedItem( const QString& name ) const; + KoXmlNode namedItemNS( const QString& nsURI, const QString& name ) const; + + /** + * Loads all child nodes (if any) of this node. Normally you do not need + * to call this function as the child nodes will be automatically + * loaded when necessary. + */ + void load( int depth=1 ); + + /** + * Releases all child nodes of this node. + */ + void unload(); + +protected: + KoXmlNodeData* d; + KoXmlNode( KoXmlNodeData* ); +}; + +/** + * KoXmlElement represents a tag element in a DOM tree. + * + * KoXmlElement holds information about an XML tag, along with its attributes. + * + * @author Ariya Hidayat <ariya@kde.org> + */ + +class KOFFICECORE_EXPORT KoXmlElement: public KoXmlNode +{ +public: + KoXmlElement(); + KoXmlElement( const KoXmlElement& element ); + KoXmlElement& operator=( const KoXmlElement& element ); + virtual ~KoXmlElement(); + bool operator== ( const KoXmlElement& ) const; + bool operator!= ( const KoXmlElement& ) const; + + QString tagName() const; + QString text() const; + virtual bool isElement() const; + + QString attribute( const QString& name ) const; + QString attribute( const QString& name, const QString& defaultValue ) const; + QString attributeNS( const QString& namespaceURI, const QString& localName, + const QString& defaultValue ) const; + bool hasAttribute( const QString& name ) const; + bool hasAttributeNS( const QString& namespaceURI, const QString& localName ) const; + +private: + friend class KoXmlNode; + friend class KoXmlDocument; + KoXmlElement( KoXmlNodeData* ); +}; + +/** + * KoXmlText represents a text in a DOM tree. + * @author Ariya Hidayat <ariya@kde.org> + */ +class KOFFICECORE_EXPORT KoXmlText: public KoXmlNode +{ +public: + KoXmlText(); + KoXmlText( const KoXmlText& text ); + KoXmlText& operator=( const KoXmlText& text ); + virtual ~KoXmlText(); + + QString data() const; + void setData( const QString& data ); + virtual bool isText() const; + +private: + friend class KoXmlNode; + friend class KoXmlDocument; + KoXmlText( KoXmlNodeData* ); +}; + +/** + * KoXmlCDATASection represents a CDATA section in a DOM tree. + * @author Ariya Hidayat <ariya@kde.org> + */ +class KOFFICECORE_EXPORT KoXmlCDATASection: public KoXmlText +{ +public: + KoXmlCDATASection(); + KoXmlCDATASection( const KoXmlCDATASection& cdata ); + KoXmlCDATASection& operator=( const KoXmlCDATASection& cdata ); + virtual ~KoXmlCDATASection(); + + virtual bool isCDATASection() const; + +private: + friend class KoXmlNode; + friend class KoXmlDocument; + KoXmlCDATASection( KoXmlNodeData* ); +}; + +/** + * KoXmlDocument represents an XML document, structured in a DOM tree. + * + * KoXmlDocument is designed to be memory efficient. Unlike QDomDocument from + * Qt's XML module, KoXmlDocument does not store all nodes in the DOM tree. + * Some nodes will be loaded and parsed on-demand only. + * + * KoXmlDocument is read-only, you can not modify its content. + * + * @author Ariya Hidayat <ariya@kde.org> + */ + +class KOFFICECORE_EXPORT KoXmlDocument: public KoXmlNode +{ +public: + KoXmlDocument(); + KoXmlDocument( const KoXmlDocument& node ); + KoXmlDocument& operator=( const KoXmlDocument& node ); + bool operator==( const KoXmlDocument& ) const; + bool operator!=( const KoXmlDocument& ) const; + virtual ~KoXmlDocument(); + + virtual bool isDocument() const; + + KoXmlElement documentElement() const; + + void setFastLoading( bool f ); + bool fastLoading() const; + + bool setContent( QIODevice* device, bool namespaceProcessing, + QString* errorMsg = 0, int* errorLine = 0, int* errorColumn = 0 ); + bool setContent( QIODevice* device, + QString* errorMsg = 0, int* errorLine = 0, int* errorColumn = 0 ); + bool setContent( QXmlInputSource *source, QXmlReader *reader, + QString* errorMsg = 0, int* errorLine = 0, int* errorColumn = 0 ); + +// bool setContent( const QCString& text, bool namespaceProcessing, QString *errorMsg=0, int *errorLine=0, int *errorColumn=0 ); +// bool setContent( const QByteArray& text, bool namespaceProcessing, QString *errorMsg=0, int *errorLine=0, int *errorColumn=0 ); +// bool setContent( const QString& text, bool namespaceProcessing, QString *errorMsg=0, int *errorLine=0, int *errorColumn=0 ); +// bool setContent( QIODevice* dev, bool namespaceProcessing, QString *errorMsg=0, int *errorLine=0, int *errorColumn=0 ); +// bool setContent( const QCString& text, QString *errorMsg=0, int *errorLine=0, int *errorColumn=0 ); +// bool setContent( const QByteArray& text, QString *errorMsg=0, int *errorLine=0, int *errorColumn=0 ); +// bool setContent( const QString& text, QString *errorMsg=0, int *errorLine=0, +// int *errorColumn=0 ); + + +private: + friend class KoXmlNode; + KoXmlDocument( KoXmlNodeData* ); +}; + +#endif // KOXML_USE_QDOM + +/** + * This namespace contains a few convenience functions to simplify code using QDom + * (when loading OASIS documents, in particular). + * + * To find the child element with a given name, use KoXml::namedItemNS. + * + * To find all child elements with a given name, use + * QDomElement e; + * forEachElement( e, parent ) + * { + * if ( e.localName() == "..." && e.namespaceURI() == KoXmlNS::... ) + * { + * ... + * } + * } + * Note that this means you don't ever need to use QDomNode nor toElement anymore! + * Also note that localName is the part without the prefix, this is the whole point + * of namespace-aware methods. + * + * To find the attribute with a given name, use QDomElement::attributeNS. + * + * Do not use getElementsByTagNameNS, it's recursive (which is never needed in KOffice). + * Do not use tagName() or nodeName() or prefix(), since the prefix isn't fixed. + * + * @author David Faure <faure@kde.org> + */ +namespace KoXml { + + /** + * A namespace-aware version of QDomNode::namedItem(), + * which also takes care of casting to a QDomElement. + * Use this when a domelement is known to have only *one* child element + * with a given tagname. + * + * Note: do *NOT* use getElementsByTagNameNS, it's recursive! + */ + KOFFICECORE_EXPORT KoXmlElement namedItemNS( const KoXmlNode& node, + const char* nsURI, const char* localName ); + + /** + * Explicitly load child nodes of specified node, up to given depth. + * This function has no effect if QDom is used. + */ + KOFFICECORE_EXPORT void load( KoXmlNode& node, int depth = 1 ); + + /** + * Unload child nodes of specified node. + * This function has no effect if QDom is used. + */ + KOFFICECORE_EXPORT void unload( KoXmlNode& node ); + +} + +#define forEachElement( elem, parent ) \ + for ( KoXmlNode _node = parent.firstChild(); !_node.isNull(); _node = _node.nextSibling() ) \ + if ( !( elem = _node.toElement() ).isNull() ) + + +#endif // KOFFICE_XMLREADER diff --git a/lib/kofficecore/Koversiondialog.cpp b/lib/kofficecore/Koversiondialog.cpp new file mode 100644 index 00000000..b21ae52f --- /dev/null +++ b/lib/kofficecore/Koversiondialog.cpp @@ -0,0 +1,143 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Laurent Montel <montel@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 <qlabel.h> +#include <qlayout.h> +#include <qmultilineedit.h> +#include <qpushbutton.h> +#include <qtoolbutton.h> +#include <qapplication.h> +#include <qlayout.h> +#include <kiconloader.h> +#include <kbuttonbox.h> +#include <kdebug.h> +#include <kmessagebox.h> +#include <klocale.h> +#include <klistview.h> +#include <kdebug.h> + +#include <qmultilineedit.h> + +#include "Koversiondialog.h" + + +KoVersionDialog::KoVersionDialog( QWidget* parent, const char* name ) + : KDialogBase( parent, name, true, i18n("Version"), Ok|Cancel ) +{ + QWidget* page = new QWidget( this ); + setMainWidget( page ); + + QGridLayout *grid1 = new QGridLayout( page,10,3,KDialog::marginHint(), KDialog::spacingHint()); + + list=new KListView(page, "versionlist"); + list->addColumn(i18n("Date & Time")); + list->addColumn(i18n("Saved By")); + list->addColumn(i18n("Comment")); + + grid1->addMultiCellWidget(list,0,8,0,0); + + m_pAdd=new QPushButton(i18n("&Add"),page); + grid1->addWidget(m_pAdd,1,2); + + m_pRemove=new QPushButton(i18n("&Remove"),page); + grid1->addWidget(m_pRemove,2,2); + + m_pModify=new QPushButton(i18n("&Modify"),page); + grid1->addWidget(m_pModify,3,2); + + m_pOpen=new QPushButton(i18n("&Open"),page); + grid1->addWidget(m_pOpen,4,2); + + + connect( m_pRemove, SIGNAL( clicked() ), this, SLOT( slotRemove() ) ); + connect( m_pAdd, SIGNAL( clicked() ), this, SLOT( slotAdd() ) ); + connect( m_pOpen, SIGNAL( clicked() ), this, SLOT( slotOpen() ) ); + connect( m_pModify, SIGNAL( clicked() ), this, SLOT( slotModify() ) ); + + updateButton(); + + resize( 600, 250 ); + +} + +KoVersionDialog::~KoVersionDialog() +{ +} + +void KoVersionDialog::updateButton() +{ +#if 0 + bool state = ( list->currentItem() >= 0 ); + m_pRemove->setEnabled( state ); +#endif +} + +void KoVersionDialog::slotAdd() +{ + //TODO create entry +} + +void KoVersionDialog::slotRemove() +{ + //TODO remove entry +} + +void KoVersionDialog::slotModify() +{ + KoVersionModifyDialog * dlg = new KoVersionModifyDialog( this /*, const QString &_comment*/ /*TODO add*/ ); + if ( dlg->exec() ) + { + //TODO + kdDebug()<<" comment :"<<dlg->comment()<<endl; + } + delete dlg; + +} + +void KoVersionDialog::slotOpen() +{ + //TODO open file +} + +void KoVersionDialog::slotOk() +{ + accept(); +} + +KoVersionModifyDialog::KoVersionModifyDialog( QWidget* parent, const QString &/*comment*/, const char* name ) + : KDialogBase( parent, name, true, i18n("Comment"), Ok|Cancel ) +{ + QWidget* page = new QWidget( this ); + setMainWidget( page ); + + QHBoxLayout *grid1 = new QHBoxLayout( page,KDialog::marginHint(), KDialog::spacingHint()); + + m_multiline=new QMultiLineEdit(page, "multiline"); + grid1->addWidget( m_multiline ); + +} + +QString KoVersionModifyDialog::comment() const +{ + return m_multiline->text(); +} + + +#include "Koversiondialog.moc" diff --git a/lib/kofficecore/Koversiondialog.h b/lib/kofficecore/Koversiondialog.h new file mode 100644 index 00000000..6a8c7fb6 --- /dev/null +++ b/lib/kofficecore/Koversiondialog.h @@ -0,0 +1,68 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Laurent Montel <montel@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. +*/ + +#ifndef __VERSION_DIALOG__ +#define __VERSION_DIALOG__ + +#include <kdialogbase.h> + +class QPushButton; +class QToolButton; +class KListView; +class QMultiLineEdit; + +class KoVersionDialog : public KDialogBase +{ + Q_OBJECT +public: + KoVersionDialog( QWidget* parent, const char* name = 0L ); + ~KoVersionDialog(); + +public slots: + virtual void slotOk(); + void slotRemove(); + void slotAdd(); + void slotOpen(); + void slotModify(); + +protected: + + void init(); + void updateButton(); + + KListView * list; + QPushButton* m_pRemove; + QPushButton* m_pAdd; + QPushButton* m_pOpen; + QPushButton* m_pModify; +}; + +class KoVersionModifyDialog : public KDialogBase +{ + Q_OBJECT +public: + KoVersionModifyDialog( QWidget* parent, const QString &_comment = QString::null , const char* name = 0L ); + + QString comment() const; + +private: + QMultiLineEdit *m_multiline; +}; + +#endif diff --git a/lib/kofficecore/Makefile.am b/lib/kofficecore/Makefile.am new file mode 100644 index 00000000..df7cdfa8 --- /dev/null +++ b/lib/kofficecore/Makefile.am @@ -0,0 +1,70 @@ +####### General stuff + +SUBDIRS = . tests +KDE_CXXFLAGS = $(USE_RTTI) $(WOVERLOADED_VIRTUAL) +INCLUDES= $(KSTORE_INCLUDES) $(KWMF_INCLUDES) $(all_includes) +libkofficecore_la_LIBADD = $(LIB_KSTORE) $(LIB_KOWMF) $(LIB_KPARTS) \ + $(LIB_KDEPRINT) $(LIB_KABC) $(LIB_KWMF) + +####### Files + +lib_LTLIBRARIES = libkofficecore.la +kde_module_LTLIBRARIES = kodocinfopropspage.la + +libkofficecore_la_SOURCES = KoDocument.cpp \ + KoGlobal.cpp KoUnit.cpp KoFilterManager.cpp \ + KoMainWindow.cpp \ + KoApplication.cpp KoQueryTrader.cpp \ + KoFilter.cpp KoFilterChain.cpp \ + KoPictureKey.cpp KoPictureBase.cpp KoPicture.cpp KoPictureShared.cpp \ + KoPictureImage.cpp KoPictureClipart.cpp KoPictureCollection.cpp \ + KoPictureEps.cpp KoPictureWmf.cpp \ + KoDocumentInfo.cpp \ + KoView.cpp KoFrame.cpp KoContainerHandler.cpp KoDocumentChild.cpp \ + KoDocumentInfoDlg.cpp KoFactory.cpp KoChild.cpp \ + koDocumentInfoAboutWidget.ui koDocumentInfoAuthorWidget.ui \ + koDocumentInfoUserMetadataWidget.ui \ + KoApplicationIface.cc KoApplicationIface.skel \ + KoDocumentIface.cc KoDocumentIface.skel KoViewIface.cc KoViewIface.skel \ + KoMainWindowIface.cc KoMainWindowIface.skel kofficeversion.cc KoOasisStyles.cpp \ + KoStyleStack.cpp KoGenStyles.cpp KoOasisSettings.cpp KoPageLayout.cpp \ + KoFileDialog.cpp KoXmlNS.cpp KoDom.cpp Koversiondialog.cpp KoOasisStore.cpp \ + kkbdaccessextensions.cpp koDetailsPaneBase.ui koOpenPaneBase.ui KoOpenPane.cpp \ + KoTemplates.cpp KoDetailsPane.cpp \ + KoSpeaker.cpp KoOasisLoadingContext.cpp \ + KoRect.cpp + +libkofficecore_la_LDFLAGS = $(all_libraries) -version-info 3:0:0 -no-undefined + +include_HEADERS = KoContainerHandler.h \ + KoFilter.h KoFilterChain.h \ + KoGlobal.h KoUnit.h KoDocument.h \ + KoMainWindow.h \ + KoApplication.h KoQueryTrader.h \ + KoFilterManager.h \ + KoDocumentInfo.h \ + KoView.h KoFrame.h KoDocumentChild.h \ + KoDocumentInfoDlg.h KoFactory.h KoChild.h \ + KoApplicationIface.h KoDocumentIface.h KoViewIface.h KoMainWindowIface.h \ + KoPictureKey.h KoPicture.h KoPictureCollection.h kofficeversion.h KoOasisStyles.h \ + KoStyleStack.h KoGenStyles.h KoOasisSettings.h KoPageLayout.h KoXmlNS.h KoDom.h Koversiondialog.h \ + kkbdaccessextensions.h \ + koffice_export.h KoOpenPane.h \ + KoSpeaker.h KoOasisLoadingContext.h \ + KoPoint.h + +noinst_HEADERS = KoDocument_p.h KoFilterManager_p.h \ + KoPictureShared.h KoPictureBase.h KoPictureImage.h KoPictureClipart.h \ + KoPictureEps.h KoPictureWmf.h + +kodocinfopropspage_la_SOURCES = KoDocInfoPropsFactory.cpp +kodocinfopropspage_la_LIBADD = ./libkofficecore.la +kodocinfopropspage_la_LDFLAGS = $(all_libraries) -module $(KDE_PLUGIN) + +METASOURCES = AUTO + +rcdir = $(kde_datadir)/koffice +rc_DATA = koffice_shell.rc + +service_DATA = kodocinfopropspage.desktop +servicedir = $(kde_servicesdir) diff --git a/lib/kofficecore/THOUGHTS b/lib/kofficecore/THOUGHTS new file mode 100644 index 00000000..f0121aa9 --- /dev/null +++ b/lib/kofficecore/THOUGHTS @@ -0,0 +1,164 @@ +------------------------------------------------------------------------------ +- Bugs and missing features in the KOffice Library - +- some initial brainstroming... comments? flames? - +------------------------------------------------------------------------------ + +1) Accurate Coordinates: +IMHO it's a must for KOffice applications to store reasonably accurate +coordinates. If zooming is supported I think we should try to make it work +up to zoom factors of ten (1000%) or so. This means you need some kind +of "spare precision". Another related issue is getting KOffice applications +support 1:1 sizes on the screen (i.e. that an A4 paper is A4 also on the +screen). Here the problem is that in X we can have a wide range of +resolutions (from 75 or so up to 120 dpi). We also can have different +resolutions in X and Y direction, but as Qt only honors one of them (the +Y resolution) I think it's safe to ignore the X resolution (see +QPaintDevice::x11AppDpiY()). +In lib/kofficeui we have some classes replacing QPoint/QRect (they use double +precision floating point numbers to store coordinates). + +2) Embedding: +2.1) "Plain" KOffice embedding (paintContent): +We finally added two double values for the X and Y zoom factor. This means +using these values and the QRect argument we can "ask" the part to draw +certain areas of itself. One problem I still see with this method is, that we +only have a QRect there. It's not that problematic there because we can +agree on some hacks if it's not accurate enough (e.g. values * 100.0), or we +just agree that this QRect represents screen pixels at the given zoom value(s) +and the app can calculate the requested area to be painted itself. This should +solve zooming and "scrolling" problems. +One ugly problem also is what we will do about performance problems on +repainting a part. Some time ago we got a mail on KOffice where some guy +asked what to do about making that embedded painting faster. The reason is +that we *always* repaint the whole area, no matter what. This can easily be +optimized to a WNorthWestGravity like behavior, because you can't edit those +embedded parts anyway, so even kchart and other non-WNG apps support that. +( we have to translate the painter, of course). + +### Zooming Special (copy from DESIGN_zooming in kofficecore): +There are two kinds of zooming: + +1) View Zooming: +This kind of zooming is only done by full KoViews, regardless +whether they are the root views (not embedded), or they are +*active* embedded views. There only has to be a KoView object. + +There is *one* zoom factor for both axes (x/y) and the whole +content (text, objects, children,...) is zoomed accordingly. + +This kind of zooming is like the zoom support you know from +other applications. + + +2) Child Zooming: +Imagine you'd like to show a certain part of an embedded +document, but you have to fill a specified area. At the +moment this is "impossible" (we know that it's kind of +possible, but it's not straightforward) because when you +resize the child's frame, the contents resize, too. Due +to that you see more of the child's content :) + +The solution will be: If you simply resize a frame, the +child document will resize, too, and show more contents. +If you press Alt/Meta during resizing, the child's +content will stay the same, but it will be zoomed to +fit the rectangle you specify. +Pressing Shift gives you a constant width/height ratio +during resizing. +Pressing Alt+Shift is allowed, too, if you can do that :p + +As you already know we'll support a zoom factor for +each axe here. (i.e. x/y zoom factors) + +Related to that is child-panning. Indepenent of the zoom factor +the child shouldn't just show the left/top corner, but it should +"remember" the position inside the document when it's repainted +as inactive child. +### + +If a part doesn't support zooming natively we have to fall back to WMartix +hacks (don't know how we can "ask" a part about that, but I'm sure we find +some BC way). + +2.2) Widget embedding: +That's a tough topic... but let's see... There are a few different possibilities +we have to treat differently: +a) plain KParts: +We can only embed the widgets obviously. Printing will look horribly, but +I think we should support that nonetheless. Imagine a KPresenter (screen) +presentation with an embedded video player part on some page :)) +I'd say when printing those files the parts aren't printed at all by default... +makes more sense to me than redirecting an ugly 96dpi printout to some +QPrinter... well. It should at least be possible to print that stuff and get +some crappy output (but at least some output, e.g. an empty frame with some +text information about the part in it). Of course the user should be warned +before embedding such parts :) +b) embedding special KOffice parts as widgets: +Here we can simply add one entry "X-KDE-EmbedAsWidget" or so to our +.desktop files for KOffice parts. This will guarantee then that this special +part wants to be embedded as a widget. This surely makes sense, but still +we have some problems on printing... no idea how to solve that one. Maybe +we should use the widget for viewing on the screen and fall back to a plain +paintContents when printing. + +General embedding stuff: +Do we support the "transparent" flag in paintContent with all parts we have? +IMHO it's a nice feature and we sould keep it, but if no part supports it... +well :] +Maybe we'll have to add a X-KDE-DoNotEmbed flag to the .desktop files +at some point. These parts should be excluded in the part select dia, then. + +3) Handling of embedded parts: +This is one of the most annoying things in KOffice. Every application +handles embedded parts different. Of course this makes sense for most +of the cases, but maybe we can make that a bit more consistent at least +among "object-based" applications like KPresenter, Kivio, and KIllustrator. + +4) Printing: +Here the problem is that even Qt has enough problems with printing :( +We definitely need some magic here, because right now we don't take +any advantage of the better resolution of the printer. This will be a hard +job (and an ugly hack) even if we have accurate coordinates and so +on. Lars told me that QPainter will have some setResolution call for +printing in 3.0... let's see. As a temporary solution he suggested to scale +the painter by 0.1 and print 10 times as big... don't know if that works. + +5) Shell menus: +The Help -> About entry should be the one for the active part. + +6) Image handling: +We need one class (IMVHO a part is too much overhead) which properly +handles all kinds of images. Internal ones, external ones, maybe thumbnails, +proper rescaling w/o any quality loss on rescaling again and again (read: +keep the original around),... Fortunately Simon already implemented most of it. +He also suggested that we need a very tiny KOffice part which just can +display images. That way we can finally support images in KSpread and other +non-object based apps. + +7) Colors: +What about RGB-CMYK-HSV-LAB-... should this be solved in the KOffice libs? +I see some code for this in krayon, and maybe Matthias Elter or gis can help +us with that one. + +8) General configuration options: +Do we need a KOffice kcm? What to configure there? +e.g.: +- start default template (which one :) (start without any template dialog) +- default page size +- default unit +- ... + +9) Make it possible for components to provide their own about dialog. + What's currently needed to acomplish this is an awful addShell() + hack, like in kivio_doc.cpp . + + Possible solution: + Make a slot in KoMainWindow call a virtual method in KoDocument which + calls back the real show-about method. That would give components the + ability to re-implement that very method and be done. + + Disadvantage: breaks binary compatibility. + Possible workaround for the BC breakage: Don't call a virtual method + but call/connect-to a slot in the document only if it exists and + call the default show-about method otherwise. + diff --git a/lib/kofficecore/document-info.dtd b/lib/kofficecore/document-info.dtd new file mode 100644 index 00000000..bff7a00b --- /dev/null +++ b/lib/kofficecore/document-info.dtd @@ -0,0 +1,80 @@ +<!-- $Id: document-info.dtd 466019 2005-10-01 11:07:55Z mlaurent $ + +This is an XML document type definition (DTD) for the documentinfo.xml files which +are used by all KOffice-1.1 and 1.1.1 programs, to store information about the +document (e.g. title and abstract) and its author. + +ChangeLog: +Initial version (for koffice-1.1.1) written by David Faure <faure@kde.org>. +--> + +<!-- + The document-info for a KOffice document contains one main element for each tab of +the "Document Info" dialog. As of KOffice-1.1, this includes: author and about. +A third page, "log" is planned for the future. +--> +<!ELEMENT document-info + (author?, about?, log?)> + +<!-- + The "author" element stores information about the author of the document. +--> +<!ELEMENT author + (full-name?, initial?, title?, company?, email?, telephone?, telephone-work?, fax?, country?, postal-code?, city?, street?, position?)> + +<!ELEMENT full-name + (#PCDATA)> +<!ELEMENT initial + (#PCDATA)> +<!ELEMENT title + (#PCDATA)> +<!ELEMENT company + (#PCDATA)> +<!ELEMENT email + (#PCDATA)> +<!ELEMENT telephone + (#PCDATA)> +<!ELEMENT telephone-work + (#PCDATA)> +<!ELEMENT fax + (#PCDATA)> +<!ELEMENT country + (#PCDATA)> +<!ELEMENT postal-code + (#PCDATA)> +<!ELEMENT city + (#PCDATA)> +<!ELEMENT street + (#PCDATA)> +<!ELEMENT position + (#PCDATA)> + +<!-- + The "about" element contains information about the document itself. + This includes an optional title for the document, and an optional abstract. +--> +<!ELEMENT about + (abstract?, title?, keyword?, subject?, initial-creator?, + editing-cycles?, creation-date?, date?)> +<!ELEMENT abstract + (#PCDATA)> +<!--title already defined above--> +<!ELEMENT keyword + (#PCDATA)> +<!ELEMENT subject + (#PCDATA)> +<!ELEMENT initial-creator + (#PCDATA)> +<!ELEMENT editing-cycles + (#PCDATA)> +<!ELEMENT creation-date + (#PCDATA)> +<!ELEMENT date + (#PCDATA)> + +<!-- + The "log" element is unused at the moment. +--> +<!ELEMENT log + EMPTY> + diff --git a/lib/kofficecore/kkbdaccessextensions.cpp b/lib/kofficecore/kkbdaccessextensions.cpp new file mode 100644 index 00000000..aae37203 --- /dev/null +++ b/lib/kofficecore/kkbdaccessextensions.cpp @@ -0,0 +1,667 @@ +/* This file is part of the KDE project + Copyright (C) 2005, Gary Cramblitt <garycramblitt@comcast.net> + + 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. +*/ + +// Qt includes +#include <qsplitter.h> +#include <qdockwindow.h> +#include <qdockarea.h> +#include <qevent.h> +#include <qcursor.h> +#include <qobjectlist.h> +#include <qwidgetlist.h> +#include <qlabel.h> +#include <qtooltip.h> + +// KDE includes +#include <klocale.h> +#include <kglobal.h> +#include <kapplication.h> +#include <kmainwindow.h> +#include <kaction.h> +#include <kdebug.h> + +// KKbdAccessExtensions includes +#include "kkbdaccessextensions.h" +// TODO: See eventFilter method. +//#include "kkbdaccessextensions.moc" + +class KPanelKbdSizerIcon : public QCursor +{ + public: + KPanelKbdSizerIcon() : + QCursor(Qt::SizeAllCursor), + isActive(false) + { + currentPos = QPoint(-1, -1); + } + + ~KPanelKbdSizerIcon() + { + hide(); + } + + void show(const QPoint p) { + if (!isActive) { + originalPos = QCursor::pos(); + kapp->setOverrideCursor(*this); + isActive = true; + } + if (p != pos()) + setPos(p); + currentPos = p; + } + + void hide() { + if (isActive) { + kapp->restoreOverrideCursor(); + QCursor::setPos(originalPos); + } + isActive = false; + } + + void setShape(int shayp) + { + if (shayp != shape()) { + // Must restore and override to get the icon to refresh. + if (isActive) kapp->restoreOverrideCursor(); + QCursor::setShape(shayp); + if (isActive) kapp->setOverrideCursor(*this); + } + } + + // Return the difference between a position and where icon is supposed to be. + QSize delta(const QPoint p) + { + QPoint d = p - currentPos; + return QSize(d.x(), d.y()); + } + + // Return the difference between where the icon is currently positioned and where + // it is supposed to be. + QSize delta() { return delta(pos()); } + + // True if the sizing icon is visible. + bool isActive; + + private: + // Icon's current position. + QPoint currentPos; + // Mouse cursor's original position when icon is shown. + QPoint originalPos; +}; + +class KKbdAccessExtensionsPrivate +{ + public: + KKbdAccessExtensionsPrivate() : + fwdAction(0), + revAction(0), + accessKeysAction(0), + panel(0), + handleNdx(0), + icon(0), + stepSize(10), + accessKeyLabels(0) {}; + + ~KKbdAccessExtensionsPrivate() + { + delete icon; + // TODO: This crashes, but should delete in the event that KMainWindow is not deleted. + if (accessKeyLabels) { + accessKeyLabels->setAutoDelete(false); + delete accessKeyLabels; + } + } + + // Action that starts panel sizing (defaults to F8), forward and reverse; + KAction* fwdAction; + KAction* revAction; + + // Action that starts access keys. + KAction* accessKeysAction; + + // The splitter or dockwindow currently being sized. If 0, sizing is not in progress. + QWidget* panel; + + // Index of current handle of the panel. When panel is a QDockWindow: + // 1 = size horizontally + // 2 = size vertically + uint handleNdx; + + // Sizing icon. + KPanelKbdSizerIcon* icon; + + // Sizing increment. + int stepSize; + + // List of the access key QLabels. If not 0, access keys are onscreen. + QPtrList<QLabel>* accessKeyLabels; + + // Pointer to the KMainWindow. + KMainWindow* mainWindow; +}; + +KKbdAccessExtensions::KKbdAccessExtensions(KMainWindow* parent, const char* name) : + QObject(parent, name) +{ + // kdDebug() << "KKbdAccessExtensions::KKbdAccessExtensions: running." << endl; + d = new KKbdAccessExtensionsPrivate; + d->mainWindow = parent; + d->fwdAction = new KAction(i18n("Resize Panel Forward"), KShortcut("F8"), + 0, 0, parent->actionCollection(), "resize_panel_forward"); + d->revAction = new KAction(i18n("Resize Panel Reverse"), KShortcut("Shift+F8"), + 0, 0, parent->actionCollection(), "resize_panel_reverse"); + d->accessKeysAction = new KAction(i18n("Access Keys"), KShortcut("Alt+F8"), + 0, 0, parent->actionCollection(), "access_keys"); + // "Disable" the shortcuts so we can see them in eventFilter. + d->fwdAction->setEnabled(false); + d->revAction->setEnabled(false); + d->accessKeysAction->setEnabled(false); + d->icon = new KPanelKbdSizerIcon(); + kapp->installEventFilter(this); +} + +KKbdAccessExtensions::~KKbdAccessExtensions() +{ + kapp->removeEventFilter(this); + if (d->panel) exitSizing(); + delete d; +} + +int KKbdAccessExtensions::stepSize() const { return d->stepSize; } + +void KKbdAccessExtensions::setStepSize(int s) { d->stepSize = s; } + +bool KKbdAccessExtensions::eventFilter( QObject *o, QEvent *e ) +{ + if ( e->type() == QEvent::KeyPress ) { + // TODO: This permits only a single-key shortcut. For example, Alt+S,R would not work. + // If user configures a multi-key shortcut, it is undefined what will happen here. + // It would be better to handle these as KShortcut activate() signals, but the problem + // is that once a QDockWindow is undocked and has focus, the KShortcut activate() signals + // don't fire anymore. + KShortcut fwdSc = d->fwdAction->shortcut(); + KShortcut revSc = d->revAction->shortcut(); + KShortcut accessKeysSc = d->accessKeysAction->shortcut(); + QKeyEvent* kev = dynamic_cast<QKeyEvent *>(e); + KKey k = KKey(kev); + KShortcut sc = KShortcut(k); + // kdDebug() << "KKbdAccessExtensions::eventFilter: Key press " << sc << endl; + if (!d->accessKeyLabels) { + if (sc == fwdSc) { + nextHandle(); + return true; + } + if (sc == revSc) { + prevHandle(); + return true; + } + } + if (d->panel) { + if (k == KKey(Key_Escape)) + exitSizing(); + else + resizePanelFromKey(kev->key(), kev->state()); + // Eat the key. + return true; + } + if (sc == accessKeysSc && !d->panel) { + if (d->accessKeyLabels) { + delete d->accessKeyLabels; + d->accessKeyLabels = 0; + } else + displayAccessKeys(); + return true; + } + if (d->accessKeyLabels) { + if (k == KKey(Key_Escape)) { + delete d->accessKeyLabels; + d->accessKeyLabels = 0; + } else + handleAccessKey(kev); + return true; + } + return false; + } + else if (d->icon->isActive && e->type() == QEvent::MouseButtonPress) { + exitSizing(); + return true; + } + else if (d->accessKeyLabels && e->type() == QEvent::MouseButtonPress) { + delete d->accessKeyLabels; + d->accessKeyLabels = 0; + return true; + } +/* else if (e->type() == QEvent::MouseMove && d->icon->isActive) { + // Lock mouse cursor down. + showIcon(); + dynamic_cast<QMouseEvent *>(e)->accept(); + return true; + }*/ + else if (e->type() == QEvent::MouseMove && d->icon->isActive && d->panel) { + // Resize according to mouse movement. + QMouseEvent* me = dynamic_cast<QMouseEvent *>(e); + QSize s = d->icon->delta(); + int dx = s.width(); + int dy = s.height(); + resizePanel(dx, dy, me->state()); + me->accept(); + showIcon(); + return true; + } + else if (e->type() == QEvent::Resize && d->panel && o == d->panel) { + // TODO: This doesn't always work. + showIcon(); + } + return false; +} + +QWidgetList* KKbdAccessExtensions::getAllPanels() +{ + QWidgetList* allWidgets = kapp->allWidgets(); + QWidgetList* allPanels = new QWidgetList; + QWidget* widget = allWidgets->first(); + while (widget) { + if (widget->isVisible()) { + if (::qt_cast<QSplitter*>( widget )) { + // Only size QSplitters with at least two handles (there is always one hidden). + if (dynamic_cast<QSplitter *>(widget)->sizes().count() >= 2) + allPanels->append(widget); + } else if (::qt_cast<QDockWindow*>( widget )) { + if (dynamic_cast<QDockWindow *>(widget)->isResizeEnabled()) { + // kdDebug() << "KKbdAccessExtensions::getAllPanels: QDockWindow = " << widget->name() << endl; + allPanels->append(widget); + } + } + } + widget = allWidgets->next(); + } + delete allWidgets; + return allPanels; +} + +void KKbdAccessExtensions::nextHandle() +{ + QWidget* panel = d->panel; + // See if current panel has another handle. If not, find next panel. + if (panel) { + bool advance = true; + d->handleNdx++; + if (::qt_cast<QSplitter*>( panel )) + advance = (d->handleNdx >= dynamic_cast<QSplitter *>(panel)->sizes().count()); + else + // Undocked windows have only one "handle" (center). + advance = (d->handleNdx > 2 || !dynamic_cast<QDockWindow *>(panel)->area()); + if (advance) { + QWidgetList* allWidgets = getAllPanels(); + allWidgets->findRef(panel); + panel = 0; + if (allWidgets->current()) panel = allWidgets->next(); + delete allWidgets; + d->handleNdx = 1; + } + } else { + // Find first panel. + QWidgetList* allWidgets = getAllPanels(); + panel = allWidgets->first(); + delete allWidgets; + d->handleNdx = 1; + } + d->panel = panel; + if (panel) + showIcon(); + else + exitSizing(); +} + +void KKbdAccessExtensions::prevHandle() +{ + QWidget* panel = d->panel; + // See if current panel has another handle. If not, find previous panel. + if (panel) { + bool rewind = true; + d->handleNdx--; + rewind = (d->handleNdx < 1); + if (rewind) { + QWidgetList* allWidgets = getAllPanels(); + allWidgets->findRef(panel); + panel = 0; + if (allWidgets->current()) panel = allWidgets->prev(); + delete allWidgets; + if (panel) { + if (::qt_cast<QSplitter*>( panel )) + d->handleNdx = dynamic_cast<QSplitter *>(panel)->sizes().count() - 1; + else { + if (dynamic_cast<QDockWindow *>(panel)->area()) + d->handleNdx = 2; + else + d->handleNdx = 1; + } + } + } + } else { + // Find last panel. + QWidgetList* allWidgets = getAllPanels(); + panel = allWidgets->last(); + delete allWidgets; + if (panel) { + if (::qt_cast<QSplitter*>( panel )) + d->handleNdx = dynamic_cast<QSplitter *>(panel)->sizes().count() - 1; + else { + if (dynamic_cast<QDockWindow *>(panel)->area()) + d->handleNdx = 2; + else + d->handleNdx = 1; + } + } + } + d->panel = panel; + if (panel) + showIcon(); + else + exitSizing(); +} + +void KKbdAccessExtensions::exitSizing() +{ + // kdDebug() << "KKbdAccessExtensions::exiting sizing mode." << endl; + hideIcon(); + d->handleNdx = 0; + d->panel = 0; +} + +void KKbdAccessExtensions::showIcon() +{ + if (!d->panel) return; + QPoint p; + // kdDebug() << "KKbdAccessExtensions::showIcon: topLevelWidget = " << d->panel->topLevelWidget()->name() << endl; + if (::qt_cast<QSplitter*>( d->panel )) { + QSplitter* splitter = dynamic_cast<QSplitter *>(d->panel); + int handleNdx = d->handleNdx - 1; + QValueList<int> sizes = splitter->sizes(); + // kdDebug() << "KKbdAccessExtensions::showIcon: sizes = " << sizes << endl; + if (splitter->orientation() == Qt::Horizontal) { + d->icon->setShape(Qt::SizeHorCursor); + p.setX(sizes[handleNdx] + (splitter->handleWidth() / 2)); + p.setY(splitter->height() / 2); + } else { + d->icon->setShape(Qt::SizeVerCursor); + p.setX(splitter->width() / 2); + p.setY(sizes[handleNdx] + (splitter->handleWidth() / 2)); + } + // kdDebug() << "KKbdAccessExtensions::showIcon: p = " << p << endl; + p = splitter->mapToGlobal(p); + // kdDebug() << "KKbdAccessExtensions::showIcon: mapToGlobal = " << p << endl; + } else { + QDockWindow* dockWindow = dynamic_cast<QDockWindow *>(d->panel); + p = dockWindow->pos(); + if (dockWindow->area()) { + // kdDebug() << "KKbdAccessExtensions::showIcon: pos = " << p << " of window = " << dockWindow->parentWidget()->name() << endl; + p = dockWindow->parentWidget()->mapTo(dockWindow->topLevelWidget(), p); + // kdDebug() << "KKbdAccessExtensions::showIcon: mapTo = " << p << " of window = " << dockWindow->topLevelWidget()->name() << endl; + // TODO: How to get the handle width? + if (d->handleNdx == 1) { + d->icon->setShape(Qt::SizeHorCursor); + if (dockWindow->area()->orientation() == Qt::Vertical) { + if (dockWindow->area()->handlePosition() == QDockArea::Normal) + // Handle is to the right of the dock window. + p.setX(p.x() + dockWindow->width()); + // else Handle is to the left of the dock window. + } else + // Handle is to the right of the dock window. + p.setX(p.x() + dockWindow->width()); + p.setY(p.y() + (dockWindow->height() / 2)); + } else { + d->icon->setShape(Qt::SizeVerCursor); + p.setX(p.x() + (dockWindow->width() / 2)); + if (dockWindow->area()->orientation() == Qt::Vertical) + // Handle is below the dock window. + p.setY(p.y() + dockWindow->height()); + else { + if (dockWindow->area()->handlePosition() == QDockArea::Normal) + // Handle is below the dock window. + p.setY(p.y() + dockWindow->height()); + // else Handle is above the dock window. + } + } + p = dockWindow->topLevelWidget()->mapToGlobal(p); + } else { + d->icon->setShape(Qt::SizeAllCursor); + p = QPoint(dockWindow->width() / 2, dockWindow->height() / 2); + p = dockWindow->mapToGlobal(p); // Undocked. Position in center of window. + } + } + // kdDebug() << "KKbdAccessExtensions::showIcon: show(p) = " << p << endl; + d->icon->show(p); +} + +void KKbdAccessExtensions::hideIcon() +{ + d->icon->hide(); +} + +void KKbdAccessExtensions::resizePanel(int dx, int dy, int state) +{ + int adj = dx + dy; + if (adj == 0) return; + // kdDebug() << "KKbdAccessExtensions::resizePanel: panel = " << d->panel->name() << endl; + if (::qt_cast<QSplitter*>( d->panel )) { + QSplitter* splitter = dynamic_cast<QSplitter *>(d->panel); + int handleNdx = d->handleNdx - 1; + QValueList<int> sizes = splitter->sizes(); + // kdDebug() << "KKbdAccessExtensions::resizePanel: before sizes = " << sizes << endl; + sizes[handleNdx] = sizes[handleNdx] + adj; + // kdDebug() << "KKbdAccessExtensions::resizePanel: setSizes = " << sizes << endl; + splitter->setSizes(sizes); + QApplication::postEvent(splitter, new QEvent(QEvent::LayoutHint)); + } else { + // TODO: How to get the handle width? + QDockWindow* dockWindow = dynamic_cast<QDockWindow *>(d->panel); + if (dockWindow->area()) { + // kdDebug() << "KKbdAccessExtensions::resizePanel: fixedExtent = " << dockWindow->fixedExtent() << endl; + QSize fe = dockWindow->fixedExtent(); + if (d->handleNdx == 1) { + // When vertically oriented and dock area is on right side of screen, pressing + // left arrow increases size. + if (dockWindow->area()->orientation() == Qt::Vertical && + dockWindow->area()->handlePosition() == QDockArea::Reverse) adj = -adj; + int w = fe.width(); + if (w < 0) w = dockWindow->width(); + w = w + adj; + if (w > 0 ) dockWindow->setFixedExtentWidth(w); + } else { + // When horizontally oriented and dock area is at bottom of screen, + // pressing up arrow increases size. + if (dockWindow->area()->orientation() == Qt::Horizontal && + dockWindow->area()->handlePosition() == QDockArea::Reverse) adj = -adj; + int h = fe.height(); + if (h < 0) h = dockWindow->height(); + h = h + adj; + if (h > 0) dockWindow->setFixedExtentHeight(h); + } + dockWindow->updateGeometry(); + QApplication::postEvent(dockWindow->area(), new QEvent(QEvent::LayoutHint)); + // kdDebug() << "KKbdAccessExtensions::resizePanel: fixedExtent = " << dockWindow->fixedExtent() << endl; + } else { + if (state == Qt::ShiftButton) { + QSize s = dockWindow->size(); + s.setWidth(s.width() + dx); + s.setHeight(s.height() + dy); + dockWindow->resize(s); + } else { + QPoint p = dockWindow->pos(); + p.setX(p.x() + dx); + p.setY(p.y() + dy); + dockWindow->move(p); + } + } + } +} + +void KKbdAccessExtensions::resizePanelFromKey(int key, int state) +{ + // kdDebug() << "KPanelKdbSizer::resizePanelFromKey: key = " << key << " state = " << state << endl; + if (!d->panel) return; + int dx = 0; + int dy = 0; + int stepSize = d->stepSize; + switch (key) { + case Qt::Key_Left: dx = -stepSize; break; + case Qt::Key_Right: dx = stepSize; break; + case Qt::Key_Up: dy = -stepSize; break; + case Qt::Key_Down: dy = stepSize; break; + case Qt::Key_Prior: dy = -5 * stepSize; break; + case Qt::Key_Next: dy = 5 * stepSize; break; + } + int adj = dx + dy; + // kdDebug() << "KKbdAccessExtensions::resizePanelFromKey: adj = " << adj << endl; + if (adj != 0) + resizePanel(dx, dy, state); + else { + if (key == Qt::Key_Enter && ::qt_cast<QDockWindow*>( d->panel )) { + QDockWindow* dockWindow = dynamic_cast<QDockWindow *>(d->panel); + if (dockWindow->area()) + dockWindow->undock(); + else + dockWindow->dock(); + } + } + showIcon(); +} + +void KKbdAccessExtensions::displayAccessKeys() +{ + // Build a list of valid access keys that don't collide with shortcuts. + QString availableAccessKeys = "ABCDEFGHIJKLMNOPQRSTUVWXYZ01234567890"; + QPtrList<KXMLGUIClient> allClients = d->mainWindow->factory()->clients(); + QPtrListIterator<KXMLGUIClient> it( allClients ); + KXMLGUIClient *client; + while( (client=it.current()) !=0 ) + { + ++it; + KActionPtrList actions = client->actionCollection()->actions(); + for (int j = 0; j < (int)actions.count(); j++) { + KAction* action = actions[j]; + KShortcut sc = action->shortcut(); + for (int i = 0; i < (int)sc.count(); i++) { + KKeySequence seq = sc.seq(i); + if (seq.count() == 1) { + QString s = seq.toString(); + if (availableAccessKeys.contains(s)) + availableAccessKeys.remove(s); + } + } + } + } + // Find all visible, focusable widgets and create a QLabel for each. Don't exceed + // available list of access keys. + QWidgetList* allWidgets = kapp->allWidgets(); + QWidget* widget = allWidgets->first(); + int accessCount = 0; + int maxAccessCount = availableAccessKeys.length(); + int overlap = 20; + QPoint prevGlobalPos = QPoint(-overlap, -overlap); + while (widget && (accessCount < maxAccessCount)) { + if (widget->isVisible() && widget->isFocusEnabled() ) { + QRect r = widget->rect(); + QPoint p(r.x(), r.y()); + // Don't display an access key if within overlap pixels of previous one. + QPoint globalPos = widget->mapToGlobal(p); + QPoint diffPos = globalPos - prevGlobalPos; + if (diffPos.manhattanLength() > overlap) { + accessCount++; + QLabel* lab=new QLabel(widget, "", widget, 0, Qt::WDestructiveClose); + lab->setPalette(QToolTip::palette()); + lab->setLineWidth(2); + lab->setFrameStyle(QFrame::Box | QFrame::Plain); + lab->setMargin(3); + lab->adjustSize(); + lab->move(p); + if (!d->accessKeyLabels) { + d->accessKeyLabels = new QPtrList<QLabel>; + d->accessKeyLabels->setAutoDelete(true); + } + d->accessKeyLabels->append(lab); + prevGlobalPos = globalPos; + } + } + widget = allWidgets->next(); + } + if (accessCount > 0) { + // Sort the access keys from left to right and down the screen. + QValueList<KSortedLabel> sortedLabels; + for (int i = 0; i < accessCount; i++) + sortedLabels.append(KSortedLabel(d->accessKeyLabels->at(i))); + qHeapSort( sortedLabels ); + // Assign access key labels. + for (int i = 0; i < accessCount; i++) { + QLabel* lab = sortedLabels[i].label(); + QChar s = availableAccessKeys[i]; + lab->setText(s); + lab->adjustSize(); + lab->show(); + } + } +} + +// Handling of the HTML accesskey attribute. +bool KKbdAccessExtensions::handleAccessKey( const QKeyEvent* ev ) +{ +// Qt interprets the keyevent also with the modifiers, and ev->text() matches that, +// but this code must act as if the modifiers weren't pressed + if (!d->accessKeyLabels) return false; + QChar c; + if( ev->key() >= Key_A && ev->key() <= Key_Z ) + c = 'A' + ev->key() - Key_A; + else if( ev->key() >= Key_0 && ev->key() <= Key_9 ) + c = '0' + ev->key() - Key_0; + else { + // TODO fake XKeyEvent and XLookupString ? + // This below seems to work e.g. for eacute though. + if( ev->text().length() == 1 ) + c = ev->text()[ 0 ]; + } + if( c.isNull()) + return false; + + QLabel* lab = d->accessKeyLabels->first(); + while (lab) { + if (lab->text() == c) { + lab->buddy()->setFocus(); + delete d->accessKeyLabels; + d->accessKeyLabels = 0; + return true; + } + lab = d->accessKeyLabels->next(); + } + return false; +} + +KSortedLabel::KSortedLabel(QLabel* l) : + m_l(l) { } + +KSortedLabel::KSortedLabel() : + m_l(0) { } + +bool KSortedLabel::operator<( KSortedLabel l ) +{ + QPoint p1 = m_l->mapToGlobal(m_l->pos()); + QPoint p2 = l.label()->mapToGlobal(l.label()->pos()); + return (p1.y() < p2.y() || (p1.y() == p2.y() && p1.x() < p2.x())); +} diff --git a/lib/kofficecore/kkbdaccessextensions.h b/lib/kofficecore/kkbdaccessextensions.h new file mode 100644 index 00000000..acca057c --- /dev/null +++ b/lib/kofficecore/kkbdaccessextensions.h @@ -0,0 +1,175 @@ +/** @file +* This file is part of the KDE/KOffice project. +* Copyright (C) 2005, Gary Cramblitt <garycramblitt@comcast.net> +* +* @author Gary Cramblitt <garycramblitt@comcast.net> +* @since KOffice 1.5 +* +* 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. +*/ + +#ifndef __KKBDACCESSEXTENSIONS_H__ +#define __KKBDACCESSEXTENSIONS_H__ + +// Qt includes. +#include <qobject.h> + +// KOffice includes. +#include <koffice_export.h> + +class KKbdAccessExtensionsPrivate; +class QWidgetList; +class KMainWindow; + +/** KKbdAccessExtensions is an object that improves accessibility for motor impaired users +* who may not be able to easily use a mouse. It adds two new capabilities using the keyboard: +* +* - Resizing and positioning of panel widgets derived from QSplitter and QDockWindow. +* - Setting focus to any widget that accepts focus. +* +* @section sizing_mode Sizing Mode +* +* Users may press F8 or Shift-F8 (defaults) to enter sizing mode. A sizing icon appears on the first +* QSplitter or QDockWindow handle found in the application (F8) or the last such handle (Shift+F8). +* (A "handle" is the divider bar that appears to the left, right, above, or below each panel +* of a QSplitter or QDockArea.) +* +* Once in sizing mode, the following functions are available via the keyboard: +* +* - F8 Moves to the next sizing handle. After the last handle, exits sizing mode. +* - Shift+F8 Moves to the previous sizing handle. After the first handle, exits sizing mode. +* - Esc Exits sizing mode. +* - LeftArrow When on a vertical sizing handle, moves the handle to the left. +* When on a horizontal sizing handle, moves the handle up. +* - RightArrow When on a vertical sizing handle, moves the handle to the right. +* When on a horizontal sizing handle, moves the handle down. +* - UpArrow When on a vertical sizing handle, moves the handle to the left. +* When on a horizontal sizing handle, moves the handle up. +* - DownArrow When on a vertical sizing handle, moves the handle to the right. +* When on a horizontal sizing handle, moves the handle down. +* - PgUp Like LeftArrow or UpArrow, but moves the handle 5X farther to the left or up. +* - PgDn Like RightArrow or DownArrow, but moves the handle 5X farther to the right or down. +* - Enter (On numeric keypad). When on the handle of a QDockWindow, undocks or docks +* the widget. Ignored when on the handle of a QSplitter. +* +* The default step size for each arrow key press is 10 pixels. +* +* When a QDockWindow is undocked, the sizing icon appears in the center of the window. +* The arrow keys and PgUp/PgDn move the undocked window on the screen. Shifted arrow keys +* and PgUp/PgDn decrease/increase the size of the undocked window. +* +* When the sizing icon is on a sizing handle, the mouse may also be used to move the handle +* without having to click and drag. When moving the mouse while sizing icon is on an undocked +* QDockWindow, the window moves with the mouse. Holding Shift down while moving the mouse +* sizes the QDockWindow. +* +* @note Users can also move and size undocked windows using the Window Operations Menu (Alt+F3). +* +* Clicking any mouse button exits sizing mode. +* +* When entering sizing mode, the position of the mouse cursor is saved and restored when +* exiting sizing mode. +* +* For a QSplitter or QDockWindow to be found, it must be in the kapp::allWidgets() list. +* +* @section focus_setting Focus Setting +* +* Users can press Alt-F8. A small box appears in the upperleft corner of each visible widget +* on the screen that can accept focus. Each box is assigned a single letter or digit. +* User can press the corresponding key to set focus to the widget. +* +* At most 36 such shortcuts are possible. If any application shortcuts are single letters +* or digits, those shortcuts are not in any of the boxes. +* +* Clicking any mouse button exits Focus Setting mode. +* +* @section notes Notes +* +* The F8, Shift+F8, and Alt+F8 keys are KShortcuts and therefore user may choose different keys in +* the application's Configure Shortcuts dialog. +* +* @note At present, these shortcuts may not be multi-key. If user sets multi-key +* shortcuts, they will not work. +* +* F8/Shift+F8 are the default shortcuts because these are the keys used for similar +* functionality in GNOME and Java SWT. +*/ +class KOFFICECORE_EXPORT KKbdAccessExtensions : public QObject +{ + // TODO: A .moc isn't really needed right now, but see TODO in eventFilter method. + // Q_PROPERTY(int stepSize READ stepSize WRITE setStepSize) + + public: + /** Constructor. + * @param parent KMainWindow of the application. Required. + * @param name (optional) Name of this object. + */ + KKbdAccessExtensions(KMainWindow* parent, const char* name = 0); + + /** Destructor. */ + virtual ~KKbdAccessExtensions(); + + /** Returns number of pixels panel is sized for each arrow key pressed. Default is 10. */ + int stepSize() const; + /** Sets number of pixels panel is sized for each arrow key pressed. */ + void setStepSize(int s); + + protected: + /** Event filter installed on kapp object. */ + bool eventFilter( QObject *o, QEvent *e ); + + /** Retrieves a list of all Splitter and DockArea widgets in the application. */ + QWidgetList* getAllPanels(); + /** Advances to the next Panel handle. If not currently in resizing mode, + turns it on. */ + void nextHandle(); + /** Moves to the previous Panel handle. If not currently in resizing mode, + turns it on. */ + void prevHandle(); + /** Exits Sizing mode. */ + void exitSizing(); + /** Moves panel handle based on key pressed. */ + void resizePanelFromKey(int key, int state); + /** Moves panel handle based on deltaX and deltaY and state of keyboard modifier keys. */ + void resizePanel(int dx, int dy, int state); + /** Displays the sizer icon. */ + void showIcon(); + /** Hides the sizer icon. */ + void hideIcon(); + + /** Displays the access keys. */ + void displayAccessKeys(); + /** Handles an access keypress. */ + bool handleAccessKey( const QKeyEvent* ev ); + + private: + KKbdAccessExtensionsPrivate* d; +}; + +/** Provides a way to sort QLabelss using a QValueList based on their screen position. */ +class KSortedLabel +{ +public: + KSortedLabel(QLabel* l); + KSortedLabel(); // default constructor + bool operator<( KSortedLabel l); + QLabel* label() { return m_l; } + +private: + QLabel* m_l; +}; + +#endif // __KKBDACCESSEXTENSIONS_H__ diff --git a/lib/kofficecore/koDetailsPaneBase.ui b/lib/kofficecore/koDetailsPaneBase.ui new file mode 100644 index 00000000..282e8bb6 --- /dev/null +++ b/lib/kofficecore/koDetailsPaneBase.ui @@ -0,0 +1,238 @@ +<!DOCTYPE UI><UI version="3.3" stdsetdef="1"> +<class>KoDetailsPaneBase</class> +<widget class="QWidget"> + <property name="name"> + <cstring>KoDetailsPaneBase</cstring> + </property> + <property name="geometry"> + <rect> + <x>0</x> + <y>0</y> + <width>528</width> + <height>549</height> + </rect> + </property> + <property name="caption"> + <string>DetailsPaneBase</string> + </property> + <grid> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <property name="margin"> + <number>0</number> + </property> + <widget class="QSplitter" row="0" column="0"> + <property name="name"> + <cstring>m_splitter</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <widget class="KListView"> + <column> + <property name="text"> + <string></string> + </property> + <property name="clickable"> + <bool>true</bool> + </property> + <property name="resizable"> + <bool>false</bool> + </property> + </column> + <property name="name"> + <cstring>m_documentList</cstring> + </property> + <property name="sizePolicy"> + <sizepolicy> + <hsizetype>7</hsizetype> + <vsizetype>7</vsizetype> + <horstretch>30</horstretch> + <verstretch>0</verstretch> + </sizepolicy> + </property> + <property name="resizeMode"> + <enum>LastColumn</enum> + </property> + </widget> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout2</cstring> + </property> + <grid> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <property name="margin"> + <number>0</number> + </property> + <widget class="KTextBrowser" row="5" column="0" rowspan="1" colspan="5"> + <property name="name"> + <cstring>m_detailsLabel</cstring> + </property> + <property name="frameShape"> + <enum>NoFrame</enum> + </property> + </widget> + <widget class="QLabel" row="0" column="0" rowspan="1" colspan="5"> + <property name="name"> + <cstring>m_previewLabel</cstring> + </property> + <property name="text"> + <string></string> + </property> + <property name="alignment"> + <set>AlignCenter</set> + </property> + </widget> + <spacer row="3" column="0"> + <property name="name"> + <cstring>spacer3_2</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>0</width> + <height>20</height> + </size> + </property> + </spacer> + <widget class="Line" row="4" column="0" rowspan="1" colspan="5"> + <property name="name"> + <cstring>line1</cstring> + </property> + <property name="frameShape"> + <enum>HLine</enum> + </property> + <property name="frameShadow"> + <enum>Sunken</enum> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + </widget> + <widget class="KPushButton" row="2" column="2"> + <property name="name"> + <cstring>m_openButton</cstring> + </property> + <property name="enabled"> + <bool>false</bool> + </property> + <property name="text"> + <string></string> + </property> + <property name="autoDefault"> + <bool>true</bool> + </property> + <property name="default"> + <bool>true</bool> + </property> + </widget> + <widget class="QCheckBox" row="3" column="1" rowspan="1" colspan="3"> + <property name="name"> + <cstring>m_alwaysUseCheckBox</cstring> + </property> + <property name="enabled"> + <bool>false</bool> + </property> + <property name="text"> + <string>Always use this template</string> + </property> + <property name="toolTip" stdset="0"> + <string>Always use this template at application start up</string> + </property> + </widget> + <spacer row="3" column="4"> + <property name="name"> + <cstring>spacer3</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>0</width> + <height>20</height> + </size> + </property> + </spacer> + <widget class="QLabel" row="1" column="0" rowspan="1" colspan="5"> + <property name="name"> + <cstring>m_titleLabel</cstring> + </property> + <property name="font"> + <font> + <bold>1</bold> + </font> + </property> + <property name="text"> + <string></string> + </property> + <property name="alignment"> + <set>AlignCenter</set> + </property> + </widget> + <spacer row="2" column="0" rowspan="1" colspan="2"> + <property name="name"> + <cstring>spacer8</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>0</width> + <height>20</height> + </size> + </property> + </spacer> + <spacer row="2" column="3" rowspan="1" colspan="2"> + <property name="name"> + <cstring>spacer8_2</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>0</width> + <height>20</height> + </size> + </property> + </spacer> + </grid> + </widget> + </widget> + </grid> +</widget> +<customwidgets> +</customwidgets> +<tabstops> + <tabstop>m_documentList</tabstop> + <tabstop>m_openButton</tabstop> + <tabstop>m_alwaysUseCheckBox</tabstop> + <tabstop>m_detailsLabel</tabstop> +</tabstops> +<layoutdefaults spacing="6" margin="11"/> +<layoutfunctions spacing="KDialog::spacingHint" margin="KDialog::marginHint"/> +<includehints> + <includehint>klistview.h</includehint> + <includehint>ktextbrowser.h</includehint> + <includehint>kpushbutton.h</includehint> +</includehints> +</UI> diff --git a/lib/kofficecore/koDocumentInfoAboutWidget.ui b/lib/kofficecore/koDocumentInfoAboutWidget.ui new file mode 100644 index 00000000..29070e54 --- /dev/null +++ b/lib/kofficecore/koDocumentInfoAboutWidget.ui @@ -0,0 +1,352 @@ +<!DOCTYPE UI><UI version="3.3" stdsetdef="1"> +<class>KoDocumentInfoAboutWidget</class> +<widget class="QWidget"> + <property name="name"> + <cstring>KoDocumentInfoAboutWidget</cstring> + </property> + <property name="geometry"> + <rect> + <x>0</x> + <y>0</y> + <width>448</width> + <height>557</height> + </rect> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <property name="margin"> + <number>0</number> + </property> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout8</cstring> + </property> + <hbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="QLabel"> + <property name="name"> + <cstring>pixmapLabel</cstring> + </property> + <property name="minimumSize"> + <size> + <width>56</width> + <height>56</height> + </size> + </property> + </widget> + <widget class="KLineEdit"> + <property name="name"> + <cstring>leDocFile</cstring> + </property> + <property name="text"> + <string></string> + </property> + <property name="readOnly"> + <bool>true</bool> + </property> + </widget> + </hbox> + </widget> + <widget class="Line"> + <property name="name"> + <cstring>line1</cstring> + </property> + <property name="frameShape"> + <enum>HLine</enum> + </property> + <property name="frameShadow"> + <enum>Sunken</enum> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + </widget> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout3</cstring> + </property> + <grid> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="KLineEdit" row="2" column="1"> + <property name="name"> + <cstring>leDocKeywords</cstring> + </property> + </widget> + <widget class="QLabel" row="1" column="0"> + <property name="name"> + <cstring>textLabel1_2</cstring> + </property> + <property name="text"> + <string>Subject:</string> + </property> + </widget> + <widget class="QLabel" row="2" column="0"> + <property name="name"> + <cstring>textLabel2</cstring> + </property> + <property name="text"> + <string>Keywords:</string> + </property> + </widget> + <widget class="QLayoutWidget" row="3" column="0"> + <property name="name"> + <cstring>layout14</cstring> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="QLabel"> + <property name="name"> + <cstring>textLabel3</cstring> + </property> + <property name="text"> + <string>Abstract:</string> + </property> + </widget> + <spacer> + <property name="name"> + <cstring>spacer2_2</cstring> + </property> + <property name="orientation"> + <enum>Vertical</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>20</width> + <height>31</height> + </size> + </property> + </spacer> + </vbox> + </widget> + <widget class="KLineEdit" row="1" column="1"> + <property name="name"> + <cstring>leDocSubject</cstring> + </property> + </widget> + <widget class="KLineEdit" row="0" column="1"> + <property name="name"> + <cstring>leDocTitle</cstring> + </property> + </widget> + <widget class="KTextEdit" row="3" column="1"> + <property name="name"> + <cstring>meDocAbstract</cstring> + </property> + <property name="tabChangesFocus"> + <bool>true</bool> + </property> + </widget> + <widget class="QLabel" row="0" column="0"> + <property name="name"> + <cstring>textLabel1_2_2</cstring> + </property> + <property name="text"> + <string>Title:</string> + </property> + </widget> + </grid> + </widget> + <widget class="Line"> + <property name="name"> + <cstring>line2</cstring> + </property> + <property name="frameShape"> + <enum>HLine</enum> + </property> + <property name="frameShadow"> + <enum>Sunken</enum> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + </widget> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout11</cstring> + </property> + <grid> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <spacer row="0" column="2" rowspan="4" colspan="2"> + <property name="name"> + <cstring>spacer3</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>90</width> + <height>70</height> + </size> + </property> + </spacer> + <spacer row="4" column="2"> + <property name="name"> + <cstring>spacer5</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>20</width> + <height>20</height> + </size> + </property> + </spacer> + <widget class="QLabel" row="2" column="0"> + <property name="name"> + <cstring>textLabel5</cstring> + </property> + <property name="text"> + <string>Modified:</string> + </property> + </widget> + <widget class="QLabel" row="3" column="1"> + <property name="name"> + <cstring>labelLastPrinted</cstring> + </property> + </widget> + <widget class="QLabel" row="2" column="1"> + <property name="name"> + <cstring>labelModified</cstring> + </property> + </widget> + <widget class="QLabel" row="1" column="0"> + <property name="name"> + <cstring>textLabel4</cstring> + </property> + <property name="text"> + <string>Created:</string> + </property> + </widget> + <widget class="QLabel" row="3" column="0"> + <property name="name"> + <cstring>textLabel4_2</cstring> + </property> + <property name="text"> + <string>Last printed:</string> + </property> + </widget> + <widget class="QLabel" row="5" column="1"> + <property name="name"> + <cstring>labelRevision</cstring> + </property> + </widget> + <widget class="QLabel" row="5" column="0"> + <property name="name"> + <cstring>textLabel7</cstring> + </property> + <property name="text"> + <string>Revision number:</string> + </property> + </widget> + <widget class="QLabel" row="4" column="1"> + <property name="name"> + <cstring>labelEditing</cstring> + </property> + </widget> + <widget class="QLabel" row="4" column="0"> + <property name="name"> + <cstring>textLabel6</cstring> + </property> + <property name="text"> + <string>Total editing time:</string> + </property> + </widget> + <widget class="KPushButton" row="4" column="3"> + <property name="name"> + <cstring>pbReset</cstring> + </property> + <property name="minimumSize"> + <size> + <width>0</width> + <height>24</height> + </size> + </property> + <property name="maximumSize"> + <size> + <width>32767</width> + <height>24</height> + </size> + </property> + <property name="text"> + <string>Reset</string> + </property> + </widget> + <widget class="QLabel" row="1" column="1"> + <property name="name"> + <cstring>labelCreated</cstring> + </property> + </widget> + <widget class="QLabel" row="0" column="1"> + <property name="name"> + <cstring>labelType</cstring> + </property> + </widget> + <widget class="QLabel" row="0" column="0"> + <property name="name"> + <cstring>textLabel1</cstring> + </property> + <property name="text"> + <string>Type:</string> + </property> + </widget> + <spacer row="5" column="3"> + <property name="name"> + <cstring>spacer4</cstring> + </property> + <property name="orientation"> + <enum>Vertical</enum> + </property> + <property name="sizeType"> + <enum>Minimum</enum> + </property> + <property name="sizeHint"> + <size> + <width>50</width> + <height>16</height> + </size> + </property> + </spacer> + </grid> + </widget> + </vbox> +</widget> +<tabstops> + <tabstop>leDocFile</tabstop> + <tabstop>leDocTitle</tabstop> + <tabstop>leDocSubject</tabstop> + <tabstop>leDocKeywords</tabstop> + <tabstop>meDocAbstract</tabstop> +</tabstops> +<layoutdefaults spacing="6" margin="11"/> +<layoutfunctions spacing="KDialog::spacingHint" margin="KDialog::marginHint"/> +<includehints> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>ktextedit.h</includehint> + <includehint>kpushbutton.h</includehint> +</includehints> +</UI> diff --git a/lib/kofficecore/koDocumentInfoAuthorWidget.ui b/lib/kofficecore/koDocumentInfoAuthorWidget.ui new file mode 100644 index 00000000..e442a1e8 --- /dev/null +++ b/lib/kofficecore/koDocumentInfoAuthorWidget.ui @@ -0,0 +1,311 @@ +<!DOCTYPE UI><UI version="3.3" stdsetdef="1"> +<class>KoDocumentInfoAuthorWidget</class> +<widget class="QWidget"> + <property name="name"> + <cstring>KoDocumentInfoAuthorWidget</cstring> + </property> + <property name="geometry"> + <rect> + <x>0</x> + <y>0</y> + <width>363</width> + <height>492</height> + </rect> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <property name="margin"> + <number>0</number> + </property> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout4</cstring> + </property> + <hbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="QLabel"> + <property name="name"> + <cstring>labelAuthor</cstring> + </property> + <property name="minimumSize"> + <size> + <width>56</width> + <height>56</height> + </size> + </property> + </widget> + <widget class="KLineEdit"> + <property name="name"> + <cstring>leFullName</cstring> + </property> + </widget> + </hbox> + </widget> + <widget class="Line"> + <property name="name"> + <cstring>line1</cstring> + </property> + <property name="frameShape"> + <enum>HLine</enum> + </property> + <property name="frameShadow"> + <enum>Sunken</enum> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + </widget> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout16</cstring> + </property> + <grid> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="QLabel" row="9" column="0"> + <property name="name"> + <cstring>textLabel10</cstring> + </property> + <property name="text"> + <string>Postal code:</string> + </property> + </widget> + <widget class="QLabel" row="6" column="0"> + <property name="name"> + <cstring>textLabel7</cstring> + </property> + <property name="text"> + <string>Telephone (work):</string> + </property> + </widget> + <widget class="QLabel" row="4" column="0"> + <property name="name"> + <cstring>textLabel5</cstring> + </property> + <property name="text"> + <string>Email:</string> + </property> + </widget> + <widget class="KLineEdit" row="3" column="1"> + <property name="name"> + <cstring>leCompany</cstring> + </property> + </widget> + <widget class="KLineEdit" row="9" column="1"> + <property name="name"> + <cstring>lePostalCode</cstring> + </property> + </widget> + <widget class="QLabel" row="10" column="0"> + <property name="name"> + <cstring>textLabel11</cstring> + </property> + <property name="text"> + <string>City:</string> + </property> + </widget> + <widget class="KLineEdit" row="2" column="1"> + <property name="name"> + <cstring>leAuthorPosition</cstring> + </property> + </widget> + <widget class="KLineEdit" row="8" column="1"> + <property name="name"> + <cstring>leStreet</cstring> + </property> + </widget> + <widget class="KLineEdit" row="7" column="1"> + <property name="name"> + <cstring>leFax</cstring> + </property> + </widget> + <widget class="KLineEdit" row="5" column="1"> + <property name="name"> + <cstring>leTelephoneHome</cstring> + </property> + </widget> + <widget class="QLabel" row="0" column="0"> + <property name="name"> + <cstring>textLabel1</cstring> + </property> + <property name="text"> + <string>Initials:</string> + </property> + </widget> + <widget class="QLabel" row="7" column="0"> + <property name="name"> + <cstring>textLabel8</cstring> + </property> + <property name="text"> + <string>Fax:</string> + </property> + </widget> + <widget class="KLineEdit" row="4" column="1"> + <property name="name"> + <cstring>leEmail</cstring> + </property> + </widget> + <widget class="KLineEdit" row="1" column="1"> + <property name="name"> + <cstring>leAuthorTitle</cstring> + </property> + </widget> + <widget class="QLabel" row="3" column="0"> + <property name="name"> + <cstring>textLabel4</cstring> + </property> + <property name="text"> + <string>Company:</string> + </property> + </widget> + <widget class="KLineEdit" row="6" column="1"> + <property name="name"> + <cstring>leTelephoneWork</cstring> + </property> + </widget> + <widget class="KLineEdit" row="0" column="1"> + <property name="name"> + <cstring>leInitial</cstring> + </property> + </widget> + <widget class="QLabel" row="11" column="0"> + <property name="name"> + <cstring>textLabel12</cstring> + </property> + <property name="text"> + <string>Country:</string> + </property> + </widget> + <widget class="QLabel" row="1" column="0"> + <property name="name"> + <cstring>textLabel2</cstring> + </property> + <property name="text"> + <string>Title:</string> + </property> + </widget> + <widget class="QLabel" row="2" column="0"> + <property name="name"> + <cstring>textLabel3</cstring> + </property> + <property name="text"> + <string>Position:</string> + </property> + </widget> + <widget class="QLabel" row="8" column="0"> + <property name="name"> + <cstring>textLabel9</cstring> + </property> + <property name="text"> + <string>Street:</string> + </property> + </widget> + <widget class="QLabel" row="5" column="0"> + <property name="name"> + <cstring>textLabel6</cstring> + </property> + <property name="text"> + <string>Telephone (home):</string> + </property> + </widget> + <widget class="KLineEdit" row="11" column="1"> + <property name="name"> + <cstring>leCountry</cstring> + </property> + </widget> + <widget class="KLineEdit" row="10" column="1"> + <property name="name"> + <cstring>leCity</cstring> + </property> + </widget> + </grid> + </widget> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout4</cstring> + </property> + <hbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="KPushButton"> + <property name="name"> + <cstring>pbLoadKABC</cstring> + </property> + <property name="text"> + <string>&Load From Address Book</string> + </property> + </widget> + <widget class="KPushButton"> + <property name="name"> + <cstring>pbDelete</cstring> + </property> + <property name="text"> + <string>Delete Personal Data</string> + </property> + <property name="accel"> + <string></string> + </property> + </widget> + </hbox> + </widget> + <spacer> + <property name="name"> + <cstring>spacer1</cstring> + </property> + <property name="orientation"> + <enum>Vertical</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>20</width> + <height>16</height> + </size> + </property> + </spacer> + </vbox> +</widget> +<tabstops> + <tabstop>leFullName</tabstop> + <tabstop>leInitial</tabstop> + <tabstop>leAuthorTitle</tabstop> + <tabstop>leAuthorPosition</tabstop> + <tabstop>leCompany</tabstop> + <tabstop>leEmail</tabstop> + <tabstop>leTelephoneHome</tabstop> + <tabstop>leTelephoneWork</tabstop> + <tabstop>leFax</tabstop> + <tabstop>leStreet</tabstop> + <tabstop>lePostalCode</tabstop> + <tabstop>leCity</tabstop> + <tabstop>leCountry</tabstop> + <tabstop>pbLoadKABC</tabstop> +</tabstops> +<layoutdefaults spacing="6" margin="11"/> +<layoutfunctions spacing="KDialog::spacingHint" margin="KDialog::marginHint"/> +<includehints> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>kpushbutton.h</includehint> + <includehint>kpushbutton.h</includehint> +</includehints> +</UI> diff --git a/lib/kofficecore/koDocumentInfoUserMetadataWidget.ui b/lib/kofficecore/koDocumentInfoUserMetadataWidget.ui new file mode 100644 index 00000000..fd91dd5e --- /dev/null +++ b/lib/kofficecore/koDocumentInfoUserMetadataWidget.ui @@ -0,0 +1,233 @@ +<!DOCTYPE UI><UI version="3.3" stdsetdef="1"> +<class>KoDocumentInfoUserMetadataWidget</class> +<widget class="QWidget"> + <property name="name"> + <cstring>KoDocumentInfoUserMetadataWidget</cstring> + </property> + <property name="geometry"> + <rect> + <x>0</x> + <y>0</y> + <width>558</width> + <height>484</height> + </rect> + </property> + <grid> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <property name="margin"> + <number>0</number> + </property> + <widget class="KListView" row="0" column="0"> + <property name="name"> + <cstring>metaListView</cstring> + </property> + </widget> + <widget class="QLayoutWidget" row="0" column="1"> + <property name="name"> + <cstring>layout5</cstring> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="KPushButton"> + <property name="name"> + <cstring>kPushButton2</cstring> + </property> + <property name="minimumSize"> + <size> + <width>130</width> + <height>0</height> + </size> + </property> + <property name="text"> + <string>Add...</string> + </property> + </widget> + <widget class="KPushButton"> + <property name="name"> + <cstring>kPushButton4</cstring> + </property> + <property name="text"> + <string>Delete</string> + </property> + </widget> + <spacer> + <property name="name"> + <cstring>spacer5</cstring> + </property> + <property name="orientation"> + <enum>Vertical</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>20</width> + <height>101</height> + </size> + </property> + </spacer> + </vbox> + </widget> + <widget class="QButtonGroup" row="1" column="0" rowspan="1" colspan="2"> + <property name="name"> + <cstring>buttonGroup1</cstring> + </property> + <property name="title"> + <string>Type</string> + </property> + <grid> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="QRadioButton" row="1" column="0"> + <property name="name"> + <cstring>rbNumber</cstring> + </property> + <property name="text"> + <string>Number:</string> + </property> + <property name="checked"> + <bool>false</bool> + </property> + </widget> + <widget class="KDoubleSpinBox" row="1" column="1"> + <property name="name"> + <cstring>wNumber</cstring> + </property> + <property name="enabled"> + <bool>false</bool> + </property> + </widget> + <widget class="QRadioButton" row="0" column="0"> + <property name="name"> + <cstring>rString</cstring> + </property> + <property name="text"> + <string>String:</string> + </property> + <property name="checked"> + <bool>true</bool> + </property> + </widget> + <widget class="KLineEdit" row="0" column="1"> + <property name="name"> + <cstring>wString</cstring> + </property> + <property name="enabled"> + <bool>true</bool> + </property> + </widget> + <widget class="QRadioButton" row="2" column="0"> + <property name="name"> + <cstring>rbBoolean</cstring> + </property> + <property name="text"> + <string>Boolean:</string> + </property> + </widget> + <widget class="KComboBox" row="2" column="1"> + <item> + <property name="text"> + <string>true</string> + </property> + </item> + <item> + <property name="text"> + <string>false</string> + </property> + </item> + <property name="name"> + <cstring>wBoolean</cstring> + </property> + <property name="enabled"> + <bool>false</bool> + </property> + </widget> + <widget class="QRadioButton" row="3" column="0"> + <property name="name"> + <cstring>rbTime</cstring> + </property> + <property name="text"> + <string>Time:</string> + </property> + </widget> + <widget class="KTimeWidget" row="3" column="1"> + <property name="name"> + <cstring>wTime</cstring> + </property> + <property name="enabled"> + <bool>false</bool> + </property> + </widget> + <widget class="QRadioButton" row="4" column="0"> + <property name="name"> + <cstring>rbDate</cstring> + </property> + <property name="text"> + <string>Date:</string> + </property> + </widget> + <widget class="KDateTimeWidget" row="4" column="1"> + <property name="name"> + <cstring>wDate</cstring> + </property> + <property name="enabled"> + <bool>false</bool> + </property> + </widget> + </grid> + </widget> + </grid> +</widget> +<connections> + <connection> + <sender>rbNumber</sender> + <signal>toggled(bool)</signal> + <receiver>wNumber</receiver> + <slot>setEnabled(bool)</slot> + </connection> + <connection> + <sender>rbDate</sender> + <signal>toggled(bool)</signal> + <receiver>wDate</receiver> + <slot>setEnabled(bool)</slot> + </connection> + <connection> + <sender>rbTime</sender> + <signal>toggled(bool)</signal> + <receiver>wTime</receiver> + <slot>setEnabled(bool)</slot> + </connection> + <connection> + <sender>rbBoolean</sender> + <signal>toggled(bool)</signal> + <receiver>wBoolean</receiver> + <slot>setEnabled(bool)</slot> + </connection> + <connection> + <sender>rString</sender> + <signal>toggled(bool)</signal> + <receiver>wString</receiver> + <slot>setEnabled(bool)</slot> + </connection> +</connections> +<layoutdefaults spacing="6" margin="11"/> +<layoutfunctions spacing="KDialog::spacingHint" margin="KDialog::marginHint"/> +<includehints> + <includehint>klistview.h</includehint> + <includehint>kpushbutton.h</includehint> + <includehint>kpushbutton.h</includehint> + <includehint>knuminput.h</includehint> + <includehint>klineedit.h</includehint> + <includehint>kcombobox.h</includehint> + <includehint>ktimewidget.h</includehint> + <includehint>kdatetimewidget.h</includehint> + <includehint>kdatewidget.h</includehint> + <includehint>ktimewidget.h</includehint> +</includehints> +</UI> diff --git a/lib/kofficecore/koOpenPaneBase.ui b/lib/kofficecore/koOpenPaneBase.ui new file mode 100644 index 00000000..64a083a5 --- /dev/null +++ b/lib/kofficecore/koOpenPaneBase.ui @@ -0,0 +1,144 @@ +<!DOCTYPE UI><UI version="3.3" stdsetdef="1"> +<class>KoOpenPaneBase</class> +<widget class="QWidget"> + <property name="name"> + <cstring>KoOpenPaneBase</cstring> + </property> + <property name="geometry"> + <rect> + <x>0</x> + <y>0</y> + <width>534</width> + <height>482</height> + </rect> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="QSplitter"> + <property name="name"> + <cstring>m_splitter</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout5</cstring> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <property name="margin"> + <number>0</number> + </property> + <widget class="KListView"> + <column> + <property name="text"> + <string></string> + </property> + <property name="clickable"> + <bool>true</bool> + </property> + <property name="resizable"> + <bool>false</bool> + </property> + </column> + <property name="name"> + <cstring>m_sectionList</cstring> + </property> + <property name="enabled"> + <bool>true</bool> + </property> + <property name="allColumnsShowFocus"> + <bool>true</bool> + </property> + <property name="resizeMode"> + <enum>LastColumn</enum> + </property> + </widget> + <widget class="KPushButton"> + <property name="name"> + <cstring>m_openExistingButton</cstring> + </property> + <property name="text"> + <string></string> + </property> + </widget> + </vbox> + </widget> + <widget class="QLayoutWidget"> + <property name="name"> + <cstring>layout4</cstring> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <property name="margin"> + <number>0</number> + </property> + <widget class="QLabel"> + <property name="name"> + <cstring>m_headerLabel</cstring> + </property> + <property name="sizePolicy"> + <sizepolicy> + <hsizetype>5</hsizetype> + <vsizetype>4</vsizetype> + <horstretch>0</horstretch> + <verstretch>0</verstretch> + </sizepolicy> + </property> + <property name="font"> + <font> + <bold>1</bold> + </font> + </property> + <property name="text"> + <string>No Header</string> + </property> + </widget> + <widget class="Line"> + <property name="name"> + <cstring>line2</cstring> + </property> + <property name="frameShape"> + <enum>HLine</enum> + </property> + <property name="frameShadow"> + <enum>Sunken</enum> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + </widget> + <widget class="QWidgetStack"> + <property name="name"> + <cstring>m_widgetStack</cstring> + </property> + <widget class="QWidget"> + <property name="name"> + <cstring>WStackPage</cstring> + </property> + <attribute name="id"> + <number>0</number> + </attribute> + </widget> + </widget> + </vbox> + </widget> + </widget> + </vbox> +</widget> +<customwidgets> +</customwidgets> +<layoutdefaults spacing="6" margin="11"/> +<layoutfunctions spacing="KDialog::spacingHint" margin="KDialog::marginHint"/> +<includehints> + <includehint>klistview.h</includehint> + <includehint>kpushbutton.h</includehint> +</includehints> +</UI> diff --git a/lib/kofficecore/kodocinfopropspage.desktop b/lib/kofficecore/kodocinfopropspage.desktop new file mode 100644 index 00000000..8ef8c958 --- /dev/null +++ b/lib/kofficecore/kodocinfopropspage.desktop @@ -0,0 +1,68 @@ +[Desktop Entry] +Type=Service +Name=KOffice Document Info Properties Page +Name[af]=Koffice Dokument Inligting Eienskappe Bladsy +Name[ar]=صفحة الخصائص المعلوماتيّة الخاصة بمُستند KOffice +Name[az]=KOffice Sənəd Xüsusiyyətləri Səhifəsi +Name[bg]=Информация за документ на KOffice +Name[br]=Pajenn Titouriñ Perzhioù Teul KOffice +Name[bs]=Stranica sa osobinama KOffice dokumenta +Name[ca]=Pàgina d'informació de propietats de documents KOffice +Name[cs]=Stránka s informacemi o dokumentu KOffice +Name[cy]=Tudalen Priodweddau Gwybodaeth Dogfen KWord +Name[da]=Egenskabsside for KOffice-dokumentinformation +Name[de]=KOffice Dokumentinfo-Eigenschaftenseite +Name[el]=Σελίδα ιδιοτήτων των πληροφοριών εγγράφου του KOffice +Name[eo]=KOficeja Dokumentinformo-eco-paĝo +Name[es]=Página de propiedades de información de documentos de KOffice +Name[et]=KOffice'i dokumendiinfo omaduste lehekülg +Name[eu]=KOffice-en dokumentuen informazio-propietateen orria +Name[fa]=صفحۀ ویژگیهای اطلاعات سند KOffice +Name[fr]=Page des propriétés d'un document KOffice +Name[fy]=KOffice dokumint ynfo eigenskippen side +Name[gl]=Páxina de Información das Propiedades do Documento de KOffice +Name[he]=דף מאפיינים לגבי מידע מסמך של KOffice +Name[hi]=के-ऑफ़िस दस्तावेज़ जानकारी गुण पृष्ठ +Name[hr]=Stranica s podacima o svojstvima KOffice dokumenta +Name[hu]=KOffice dokumentumjellemzők tulajdonságlap +Name[id]=Halaman Informasi Properties KOffice +Name[is]=Eiginleikar KOffice skjals +Name[it]=Pagina proprietà informazioni documento KOffice +Name[ja]=KOffice ドキュメント情報プロパティページ +Name[km]=ទំព័រលក្ខណៈសម្បត្តិព័ត៌មានឯកសារ KOffice +Name[lo]=ຫນ້າຂໍ້ມູນຄຸນສົມບັດຂອງເອກະສານ ໂປຣແກມຊຸດສໍານັກງານ K +Name[lt]=KOffice Dokumento informacijos savybių puslapis +Name[lv]=KOffice dokumenta informācijas īpašību lapa +Name[mk]=Информации и параметри на KOffice документ +Name[ms]=Halaman Ciri Maklumat Dokumen KOffice +Name[mt]=Paġna ta' Informazzjoni dwar Dokumenti KOffice +Name[nb]=KOffice dokumentinformasjon +Name[nds]=Dokmentinformatschonen-Egenschappensiet för KOffice +Name[ne]=केडीई कार्यालय कागजात सूचना गुणहरूको पृष्ठ +Name[nl]=KOffice Document Info Eigenschappen Pagina +Name[nn]=KOffice Dokumentinformasjon +Name[pl]=Strona właściwości dokumentu KOffice +Name[pt]=Página de Informações sobre um Documento do KOffice +Name[pt_BR]=Página de Propriedades de Informações do Documento do KOffice +Name[ro]=Pagină proprietăţi document KOffice +Name[ru]=Страница сведений о документе KOffice +Name[se]=KOffice-dokumeantadieđut +Name[sk]=Popis vlastností dokumentu KOffice +Name[sl]=Stran z lastnostmi dokumenta za KOffice +Name[sr]=KOffice-ова страна са информацијама о документу +Name[sr@Latn]=KOffice-ova strana sa informacijama o dokumentu +Name[sv]=Koffice-sida med dokumentinformation +Name[tg]=KOffice Ҳуҷҷати Ахбороти Саҳифаи Ҳосиятҳо +Name[th]=หน้าข้อมูลคุณสมบัติของเอกสารโปรแกรมชุดสำนักงาน K +Name[tr]=KOffice Belge Özellikler Sayfası +Name[uk]=Сторінка властивостей документу KOffice +Name[uz]=KOffice hujjati haqida maʼlumot +Name[uz@cyrillic]=KOffice ҳужжати ҳақида маълумот +Name[ven]=Siatari la mafhungo a manwalwa a ofisi ya K +Name[wa]=Pådje des prôpietés des informåcions do documint KOffice +Name[xh]=Iphepha Lezinto zobumnini Zexwebhu lwe KOffice +Name[zh_CN]=KOffice 文档基本信息和属性 +Name[zh_TW]=KOffice 文件資訊和屬性 +Name[zu]=KOffice iphepha ledokhumenti Iphepha Eliqukethe Ulwazi +X-KDE-Library=kodocinfopropspage +ServiceTypes=KPropsDlg/Plugin,application/x-kspread,application/x-kword,application/x-kpresenter,application/x-kchart diff --git a/lib/kofficecore/koffice_export.h b/lib/kofficecore/koffice_export.h new file mode 100644 index 00000000..37a9e8f7 --- /dev/null +++ b/lib/kofficecore/koffice_export.h @@ -0,0 +1,210 @@ +/* + This file is part of kofficecore + Copyright (c) 2005 KOffice Team + + 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. +*/ + + + +#ifndef _KOFFICE_EXPORT_H +#define _KOFFICE_EXPORT_H + +#include <kdeversion.h> + +#ifdef Q_WS_WIN + +/* workaround for KDElibs < 3.2 on !win32 */ +#ifndef KDE_EXPORT +# define KDE_EXPORT +#endif + +#ifndef KOFFICECORE_EXPORT +# ifdef MAKE_KOFFICECORE_LIB +# define KOFFICECORE_EXPORT KDE_EXPORT +# elif KDE_MAKE_LIB +# define KOFFICECORE_EXPORT KDE_IMPORT +# else +# define KOFFICECORE_EXPORT +# endif +#endif + +#ifndef KOFFICEUI_EXPORT +# ifdef MAKE_KOFFICEUI_LIB +# define KOFFICEUI_EXPORT KDE_EXPORT +# elif KDE_MAKE_LIB +# define KOFFICEUI_EXPORT KDE_IMPORT +# else +# define KOFFICEUI_EXPORT +# endif +#endif + +#ifdef MAKE_KOSTORE_LIB +# define KOSTORE_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KOSTORE_EXPORT KDE_IMPORT +#else +# define KOSTORE_EXPORT +#endif + +#ifdef MAKE_KOPALETTE_LIB +# define KOPALETTE_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KOPALETTE_EXPORT KDE_IMPORT +#else +# define KOPALETTE_EXPORT +#endif + +#ifdef MAKE_KOWMF_LIB +# define KOWMF_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KOWMF_EXPORT KDE_IMPORT +#else +# define KOWMF_EXPORT +#endif + +#ifdef MAKE_KWMF_LIB +# define KWMF_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KWMF_EXPORT KDE_IMPORT +#else +# define KWMF_EXPORT +#endif + +#ifdef MAKE_KOTEXT_LIB +# define KOTEXT_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KOTEXT_EXPORT KDE_IMPORT +#else +# define KOTEXT_EXPORT +#endif + +#ifdef MAKE_KOFORMULA_LIB +# define KOFORMULA_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KOFORMULA_EXPORT KDE_IMPORT +#else +# define KOFORMULA_EXPORT +#endif + +#ifdef MAKE_KOPAINTER_LIB +# define KOPAINTER_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KOPAINTER_EXPORT KDE_IMPORT +#else +# define KOPAINTER_EXPORT +#endif + +#ifdef MAKE_KWORD_LIB +# define KWORD_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KWORD_EXPORT KDE_IMPORT +#else +# define KWORD_EXPORT +#endif + +#ifdef MAKE_KWMAILMERGE_LIB +# define KWMAILMERGE_EXPORT KDE_EXPORT +#elif KDE_MAKE_LIB +# define KWMAILMERGE_EXPORT KDE_IMPORT +#else +# define KWMAILMERGE_EXPORT +#endif + +#ifndef KOPROPERTY_EXPORT +# ifdef MAKE_KOPROPERTY_LIB +# define KOPROPERTY_EXPORT KDE_EXPORT +# elif KDE_MAKE_LIB +# define KOPROPERTY_EXPORT KDE_IMPORT +# else +# define KOPROPERTY_EXPORT +# endif +#endif + +#define KPRESENTER_EXPORT KDE_EXPORT +#define KCHART_EXPORT KDE_EXPORT +#define KDCHART_EXPORT KDE_EXPORT +#define KARBONCOMMON_EXPORT KDE_EXPORT +#define KARBONBASE_EXPORT KDE_EXPORT +#define KARBONCOMMAND_EXPORT KDE_EXPORT +#define KSPREAD_EXPORT KDE_EXPORT +#define KOSHELL_EXPORT KDE_EXPORT +#define KPLATO_EXPORT KDE_EXPORT +#define KPLATOCHART_EXPORT KDE_EXPORT +#define KUGAR_EXPORT KDE_EXPORT +#define KUGARDESIGNER_EXPORT KDE_EXPORT +#define KOFFICETOOLS_EXPORT KDE_EXPORT +#define KOFFICEFILTER_EXPORT KDE_EXPORT +#define KOCHARTINTERFACE_EXPORT KDE_EXPORT +#define KIVIOPLUGINS_EXPORT KDE_EXPORT +#define KIVIO_EXPORT KDE_EXPORT +#define KRITA_EXPORT KDE_EXPORT +#define KRITAUI_EXPORT KDE_EXPORT +#define KRITACORE_EXPORT KDE_EXPORT +#define KRITATOOL_EXPORT KDE_EXPORT +#define KRITAPAINT_EXPORT KDE_EXPORT +#define EXAMPLE_EXPORT KDE_EXPORT +#else // not windows + +#if KDE_VERSION >= KDE_MAKE_VERSION(3,3,90) +#define KOFFICE_EXPORT KDE_EXPORT +#else +#define KOFFICE_EXPORT +#endif + +/* kdemacros is OK, we can use gcc visibility macros */ +#define KOFFICECORE_EXPORT KOFFICE_EXPORT +#define KOFFICEUI_EXPORT KOFFICE_EXPORT +#define KOPALETTE_EXPORT KOFFICE_EXPORT +#define KOTEXT_EXPORT KOFFICE_EXPORT +#define KOFORMULA_EXPORT KOFFICE_EXPORT +#define KOSTORE_EXPORT KOFFICE_EXPORT +#define KOWMF_EXPORT KOFFICE_EXPORT +#define KOSCRIPT_EXPORT KOFFICE_EXPORT +#define KOPAINTER_EXPORT KOFFICE_EXPORT +#define KSPREAD_EXPORT KOFFICE_EXPORT +#define KFORMULA_EXPORT KOFFICE_EXPORT +#define KWORD_EXPORT KOFFICE_EXPORT +#define KWORD_MAILMERGE_EXPORT KOFFICE_EXPORT +#define KPRESENTER_EXPORT KOFFICE_EXPORT +#define KCHART_EXPORT KOFFICE_EXPORT +#define KDCHART_EXPORT KOFFICE_EXPORT +#define KARBONCOMMON_EXPORT KOFFICE_EXPORT +#define KARBONBASE_EXPORT KOFFICE_EXPORT +#define KARBONCOMMAND_EXPORT KOFFICE_EXPORT +#define KOSHELL_EXPORT KOFFICE_EXPORT +#define KPLATO_EXPORT KOFFICE_EXPORT +#define KPLATOCHART_EXPORT KOFFICE_EXPORT +#define KUGAR_EXPORT KOFFICE_EXPORT +#define KUGARDESIGNER_EXPORT KOFFICE_EXPORT +#define KOFFICETOOLS_EXPORT KOFFICE_EXPORT +#define KOFFICEFILTER_EXPORT KOFFICE_EXPORT +#define KOCHARTINTERFACE_EXPORT KOFFICE_EXPORT +#define KIVIOPLUGINS_EXPORT KOFFICE_EXPORT +#define KIVIO_EXPORT KOFFICE_EXPORT +#define KRITA_EXPORT KOFFICE_EXPORT +#define KRITAUI_EXPORT KOFFICE_EXPORT +#define KRITACORE_EXPORT KOFFICE_EXPORT +#define KRITATOOL_EXPORT KOFFICE_EXPORT +#define KRITAPAINT_EXPORT KOFFICE_EXPORT +#ifndef KOPROPERTY_EXPORT +# define KOPROPERTY_EXPORT KOFFICE_EXPORT +#endif +#define EXAMPLE_EXPORT KOFFICE_EXPORT + +#endif /* not windows */ + +#endif /* _KOFFICE_EXPORT_H */ diff --git a/lib/kofficecore/koffice_shell.rc b/lib/kofficecore/koffice_shell.rc new file mode 100644 index 00000000..a619390b --- /dev/null +++ b/lib/kofficecore/koffice_shell.rc @@ -0,0 +1,58 @@ +<!DOCTYPE kpartgui SYSTEM "kpartgui.dtd"> +<kpartgui name="KOffice" version="17"> +<MenuBar> + <Menu name="file" noMerge="1"><text>&File</text> + <Action name="file_new" /> + <Action name="file_open" /> + <Action name="file_open_recent" /> + <Separator /> + <Action name="file_save" /> + <Action name="file_save_as" /> + <Action name="file_reload_file" /> +<!-- <Action name="file_versions_file"/> --> + <Separator /> + <Action name="file_import_file" /> + <Action name="file_export_file" /> + <Action name="file_send_file" /> + <Separator /> + <Merge /> + <Separator /> + <Action name="file_print" /> + <Action name="file_print_preview" /> + <Separator /> + <Action name="file_documentinfo" /> + <Separator /> + <Action name="file_close" /> + <Action name="file_quit" /> + </Menu> + <Merge /> + <Menu name="settings" noMerge="1"><text>&Settings</text> + <DefineGroup name="settings_top"/> + <Merge/> +<!-- <ActionList name="toolbarlist"/> --> + <Separator/> + <DefineGroup name="settings_show"/> + <Separator/> + <Action name="options_configure_keybinding"/> + <Action name="options_configure_toolbars"/> + <DefineGroup name="settings_configure"/> + </Menu> + <Menu name="help" noMerge="1"><text>&Help</text> + <Merge /> + <Action name="help_contents"/> + <Action name="help_whats_this"/> + <Separator/> + <Action name="help_report_bug"/> + <Separator/> + <Action name="help_about_app"/> + <Action name="help_about_kde"/> + </Menu> +</MenuBar> +<ToolBar name="mainToolBar" fullWidth="false" noMerge="1"><Text>File</Text> + <Action name="file_new" /> + <Action name="file_open" /> + <Action name="file_save" /> + <Action name="file_print" /> + <Action name="file_print_preview" /> +</ToolBar> +</kpartgui> diff --git a/lib/kofficecore/kofficeversion.cc b/lib/kofficecore/kofficeversion.cc new file mode 100644 index 00000000..262664fc --- /dev/null +++ b/lib/kofficecore/kofficeversion.cc @@ -0,0 +1,46 @@ +/* This file is part of the KOffice libraries + Copyright (c) 2003 David Faure <faure@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 "kofficeversion.h" + +unsigned int KOffice::version() +{ + return KOFFICE_VERSION; +} + +unsigned int KOffice::versionMajor() +{ + return KOFFICE_VERSION_MAJOR; +} + +unsigned int KOffice::versionMinor() +{ + return KOFFICE_VERSION_MINOR; +} + +unsigned int KOffice::versionRelease() +{ + return KOFFICE_VERSION_RELEASE; +} + +const char *KOffice::versionString() +{ + return KOFFICE_VERSION_STRING; +} + diff --git a/lib/kofficecore/kofficeversion.h b/lib/kofficecore/kofficeversion.h new file mode 100644 index 00000000..89022330 --- /dev/null +++ b/lib/kofficecore/kofficeversion.h @@ -0,0 +1,86 @@ +/* This file is part of the KOffice libraries + Copyright (c) 2003 David Faure <faure@kde.org> + Copyright (c) 2003 Lukas Tinkl <lukas@kde.org> + Copyright (c) 2004 Nicolas Goutte <goutte@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. +*/ + +#ifndef _KOFFICE_VERSION_H_ +#define _KOFFICE_VERSION_H_ + +// Remmber to synchronize the version number with the file(s): +// - koffice/configure.in.in +// +// NOT synchronized are: +// - karbon/karbon_aboutdata.h +// + +#define KOFFICE_VERSION_STRING "1.6.3" +#define KOFFICE_VERSION_MAJOR 1 +#define KOFFICE_VERSION_MINOR 6 +#define KOFFICE_VERSION_RELEASE 3 +#define KOFFICE_MAKE_VERSION( a,b,c ) (((a) << 16) | ((b) << 8) | (c)) + +#define KOFFICE_VERSION \ + KOFFICE_MAKE_VERSION(KOFFICE_VERSION_MAJOR,KOFFICE_VERSION_MINOR,KOFFICE_VERSION_RELEASE) + +#define KOFFICE_IS_VERSION(a,b,c) ( KOFFICE_VERSION >= KOFFICE_MAKE_VERSION(a,b,c) ) + +/** + * Namespace for general KOFFICE functions. + */ +namespace KOffice +{ + /** + * Returns the encoded number of KOffice's version, see the KOFFICE_VERSION macro. + * In contrary to that macro this function returns the number of the actually + * installed KOffice version, not the number of the KOffice version that was + * installed when the program was compiled. + * @return the version number, encoded in a single uint + * @since 1.3 + */ + unsigned int version(); + /** + * Returns the major number of KOffice's version, e.g. + * 1 for KOffice 1.2.3. + * @return the major version number + * @since 1.3 + */ + unsigned int versionMajor(); + /** + * Returns the minor number of KOffice's version, e.g. + * 2 for KOffice 1.2.3. + * @return the minor version number + * @since 1.3 + */ + unsigned int versionMinor(); + /** + * Returns the release of KOffice's version, e.g. + * 3 for KOffice 1.2.3. + * @return the release number + * @since 1.3 + */ + unsigned int versionRelease(); + /** + * Returns the KOffice version as string, e.g. "1.2.3". + * @return the KOffice version. You can keep the string forever + * @since 1.3 + */ + const char *versionString(); +} + +#endif // _KOFFICE_VERSION_H_ diff --git a/lib/kofficecore/priorityqueue.h b/lib/kofficecore/priorityqueue.h new file mode 100644 index 00000000..68019143 --- /dev/null +++ b/lib/kofficecore/priorityqueue.h @@ -0,0 +1,216 @@ +/* This file is part of the KOffice libraries + Copyright (C) 2001 Werner Trobin <trobin@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 version 2 as published by the Free Software Foundation. + + 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. +*/ + +#ifndef priority_queue_h +#define priority_queue_h + +#include <vector> +#include <qstring.h> +#include <qasciidict.h> +#include <kdebug.h> + +// Better put all those internal classes in some namespace to avoid clashes +// as the names are quite generic +namespace KOffice { + + /** + * This PriorityQueue class is implemented as "upside down" heap, i.e. the item + * with the \b smallest key is at the root of the heap. + * If you feel like using that class your template parameter has to have a public + * method "unsigned int key() const" which returns - surprise - the key. The + * supplied key must not be "negative." We use UINT_MAX as value to represent + * "infinity" for the shortest path algorithm. + * As this is a very specialized class we also demand a "void setIndex(int i)" + * method in order to tell nodes where they are located inside the heap. This is + * a very ugly approach, but it's the only way I see if you want to avoid a O(n) + * search for the item where you decreased the key :} + * Just to make it even worse we also use a "int index() const" method + * to fetch the index... well, you most likely would need one anyway ;) + * Note: This class is pointer based (like QPtr*) - if you create PriorityQueue<X> + * we actually operate on X* ! We don't care about deleting your pointers at all. + * We don't copy them, we don't create new ones,... - you own them, you have + * to delete them :) + * + * In case you change a key value make sure to call keyDecreased and pass the + * item you touched. This is running in O(log n) according to Cormen... + * + * All the ideas are stol^H^H^H^Hborrowed from "Introduction to Algorithms", + * Cormen et al + * + * @author Werner Trobin <trobin@kde.org> + */ + template<class T> class PriorityQueue { + + public: + PriorityQueue() {} + PriorityQueue( const PriorityQueue<T>& rhs ) : m_vector( rhs.m_vector ) {} + PriorityQueue( const QAsciiDict<T>& items ); + ~PriorityQueue() {} + + PriorityQueue<T> &operator=( const PriorityQueue<T>& rhs ) { m_vector = rhs.m_vector; return *this; } + bool operator==( const PriorityQueue<T>& rhs ) { return m_vector == rhs.m_vector; } + + unsigned int count() const { return m_vector.size(); } + bool isEmpty() const { return m_vector.empty(); } + + void insert( T* item ); + + /** + * Call this method after decreasing the key of the ith item. The heap + * properties will no longer be valid if you either forget to call that + * method, or if you \b increase the key. + */ + void keyDecreased( T* item ); + + T* extractMinimum(); + + /** + * For debugging + */ + void dump() const; + + private: + // Note: We have to use a 1-based index here, and we get/return 0-based ones + int parent( int i ) { return ( ( i + 1 ) >> 1 ) - 1; } + int left( int i ) { return ( ( i + 1 ) << 1 ) - 1; } + int right( int i ) { return ( i + 1 ) << 1; } + + void heapify( int i ); + // item is !=0 for sure + void bubbleUp( T* item, int i ); + // Builds the heap in the vector in O(n) + void buildHeap(); + + std::vector<T*> m_vector; + }; + + template<class T> + PriorityQueue<T>::PriorityQueue( const QAsciiDict<T>& items ) : m_vector( items.count() ) + { + // First put all items into the vector + QAsciiDictIterator<T> it( items ); + for ( int i = 0; it.current(); ++it, ++i ) { + it.current()->setIndex( i ); + m_vector[ i ] = it.current(); + } + // Then build a heap in that vector + buildHeap(); + } + + template<class T> + void PriorityQueue<T>::insert( T* item ) + { + if ( !item ) + return; + int i = static_cast<int>( m_vector.size() ); + m_vector.push_back( 0 ); // extend the vector by one item. i == index to the last item + bubbleUp( item, i ); + } + + template<class T> + void PriorityQueue<T>::keyDecreased( T* item ) + { + if ( !item ) + return; + bubbleUp( item, item->index() ); + } + + template<class T> + T* PriorityQueue<T>::extractMinimum() + { + if ( m_vector.size() < 1 ) + return 0; + T *min = m_vector[ 0 ]; + m_vector[ 0 ] = m_vector[ m_vector.size() - 1 ]; + // update the index + m_vector[ 0 ]->setIndex( 0 ); + m_vector.pop_back(); + heapify( 0 ); + return min; + } + + template<class T> + void PriorityQueue<T>::dump() const + { + kdDebug( 30500 ) << "++++++++++ PriorityQueue::dump ++++++++++" << endl; + QString out; + int size = static_cast<int>( m_vector.size() ); + for ( int i = 0; i < size; ++i ) { + if ( m_vector[ i ]->index() != i ) + out += " ERROR: index out of sync. Should be " + QString::number( i ) + ", is " + + QString::number( m_vector[ i ]->index() ) + ". "; + out += QString::number( m_vector[ i ]->key() ); + out += ", "; + } + if ( out.isEmpty() ) + out = "(empty)"; + kdDebug( 30500 ) << out << endl; + kdDebug( 30500 ) << "++++++++++ PriorityQueue::dump (done) ++++++++++" << endl; + } + + template<class T> + void PriorityQueue<T>::heapify( int i ) + { + int l = left( i ), r = right( i ), size = static_cast<int>( m_vector.size() ); + int smallest; + + if ( l < size && m_vector[ l ]->key() < m_vector[ i ]->key() ) + smallest = l; + else + smallest = i; + if ( r < size && m_vector[ r ]->key() < m_vector[ smallest ]->key() ) + smallest = r; + + if ( smallest != i ) { + T* tmp = m_vector[ i ]; + m_vector[ i ] = m_vector[ smallest ]; + // update indices + m_vector[ i ]->setIndex( i ); + tmp->setIndex( smallest ); + m_vector[ smallest ] = tmp; + heapify( smallest ); + } + } + + template<class T> + void PriorityQueue<T>::bubbleUp( T* item, int i ) + { + int p = parent( i ); + while ( i > 0 && m_vector[ p ]->key() > item->key() ) { + // update the index first + m_vector[ p ]->setIndex( i ); + // then move it there + m_vector[ i ] = m_vector[ p ]; + i = p; + p = parent( i ); + } + item->setIndex( i ); + m_vector[ i ] = item; + } + + template<class T> + void PriorityQueue<T>::buildHeap() + { + // from size() / 2 down to 1 + for ( int i = ( m_vector.size() >> 1 ) - 1; i >= 0; --i ) + heapify( i ); + } + +} // namespace KOffice + +#endif // priority_queue_h diff --git a/lib/kofficecore/tests/Makefile.am b/lib/kofficecore/tests/Makefile.am new file mode 100644 index 00000000..3cc6827a --- /dev/null +++ b/lib/kofficecore/tests/Makefile.am @@ -0,0 +1,40 @@ +####### General stuff + +INCLUDES= -I$(srcdir)/.. $(KSTORE_INCLUDES) $(all_includes) + +####### Files + +check_PROGRAMS = priorityqueue_test filterchain_test filter_graph kogenstylestest \ + kodomtest kooasissettingstest kooasisstoretest kopointtest korecttest + +TESTS = kogenstylestest kodomtest kooasissettingstest kooasisstoretest kopointtest korecttest + +priorityqueue_test_SOURCES = priorityqueue_test.cpp +priorityqueue_test_LDADD = ../libkofficecore.la + +filterchain_test_SOURCES = filterchain_test.cpp +filterchain_test_LDADD = ../libkofficecore.la + +filter_graph_SOURCES = filter_graph.cpp +filter_graph_LDADD = ../libkofficecore.la + +kogenstylestest_SOURCES = kogenstylestest.cpp +kogenstylestest_LDADD = ../libkofficecore.la + +kodomtest_SOURCES = kodomtest.cpp +kodomtest_LDADD = ../libkofficecore.la + +kooasissettingstest_SOURCES = kooasissettingstest.cpp +kooasissettingstest_LDADD = ../libkofficecore.la + +kooasisstoretest_SOURCES = kooasisstoretest.cpp +kooasisstoretest_LDADD = ../libkofficecore.la + +kopointtest_SOURCES = kopointtest.cpp +kopointtest_LDADD = ../libkofficecore.la + +korecttest_SOURCES = korecttest.cpp +korecttest_LDADD = ../libkofficecore.la + +dot: + dot -Tpng -o graph.png graph.dot diff --git a/lib/kofficecore/tests/filter_graph.cpp b/lib/kofficecore/tests/filter_graph.cpp new file mode 100644 index 00000000..22323355 --- /dev/null +++ b/lib/kofficecore/tests/filter_graph.cpp @@ -0,0 +1,115 @@ +/* This file is part of the KDE project + Copyright (C) 2002 Werner Trobin <trobin@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 <qfile.h> +#include <KoQueryTrader.h> +#include <KoFilterManager.h> +#include <kinstance.h> +#include <kdebug.h> + +int main( int /*argc*/, char ** /*argv*/ ) +{ + KInstance instance( "filter_graph" ); // we need an instance when using the trader + + QCString output = "digraph filters {\n"; + + // The following code is shamelessly copied over from KOffice::Graph::buildGraph + // It wasn't feasible to do some serious changes in the lib for that tiny bit + // of duplicated code in a test file. + + QValueList<QString> vertices; // to keep track of already inserted values, not performance critical + + // Make sure that all available parts are added to the graph + QValueList<KoDocumentEntry> parts( KoDocumentEntry::query() ); + QValueList<KoDocumentEntry>::ConstIterator partIt( parts.begin() ); + QValueList<KoDocumentEntry>::ConstIterator partEnd( parts.end() ); + + while ( partIt != partEnd ) { + //kdDebug() << ( *partIt ).service()->desktopEntryName() << endl; + QStringList nativeMimeTypes = ( *partIt ).service()->property( "X-KDE-ExtraNativeMimeTypes" ).toStringList(); + nativeMimeTypes += ( *partIt ).service()->property( "X-KDE-NativeMimeType" ).toString(); + QStringList::ConstIterator it = nativeMimeTypes.begin(); + QStringList::ConstIterator end = nativeMimeTypes.end(); + for ( ; it != end; ++it ) { + QString key = *it; + //kdDebug() << " " << key << endl; + if ( !key.isEmpty() ) { + output += " \""; + output += key.latin1(); + output += "\" [shape=box, style=filled, fillcolor=lightblue];\n"; + if ( vertices.find( key ) == vertices.end() ) + vertices.append( key ); + } + } + ++partIt; + } + + QValueList<KoFilterEntry::Ptr> filters( KoFilterEntry::query() ); // no constraint here - we want *all* :) + QValueList<KoFilterEntry::Ptr>::ConstIterator it = filters.begin(); + QValueList<KoFilterEntry::Ptr>::ConstIterator end = filters.end(); + + for ( ; it != end; ++it ) { + kdDebug() << "import " << ( *it )->import << " export " << ( *it )->export_ << endl; + // First add the "starting points" + QStringList::ConstIterator importIt = ( *it )->import.begin(); + QStringList::ConstIterator importEnd = ( *it )->import.end(); + for ( ; importIt != importEnd; ++importIt ) { + // already there? + if ( vertices.find( *importIt ) == vertices.end() ) { + vertices.append( *importIt ); + output += " \""; + output += ( *importIt ).latin1(); + output += "\";\n"; + } + } + + QStringList::ConstIterator exportIt = ( *it )->export_.begin(); + QStringList::ConstIterator exportEnd = ( *it )->export_.end(); + + for ( ; exportIt != exportEnd; ++exportIt ) { + // First make sure the export vertex is in place + if ( vertices.find( *exportIt ) == vertices.end() ) { + output += " \""; + output += ( *exportIt ).latin1(); + output += "\";\n"; + vertices.append( *exportIt ); + } + // Then create the appropriate edges + importIt = ( *it )->import.begin(); + for ( ; importIt != importEnd; ++importIt ) { + output += " \""; + output += ( *importIt ).latin1(); + output += "\" -> \""; + output += ( *exportIt ).latin1(); + if ( KoFilterManager::filterAvailable( *it ) ) + output += "\";\n"; + else + output += "\" [style=dotted];\n"; + } + } + } + + output += "}\n"; + + QFile f( "graph.dot" ); + if ( f.open( IO_WriteOnly ) ) + f.writeBlock( output.data(), output.size() - 1 ); + f.close(); + return 0; +} diff --git a/lib/kofficecore/tests/filterchain_test.cpp b/lib/kofficecore/tests/filterchain_test.cpp new file mode 100644 index 00000000..23f620a1 --- /dev/null +++ b/lib/kofficecore/tests/filterchain_test.cpp @@ -0,0 +1,107 @@ +/* This file is part of the KDE project + Copyright (C) 2001 Werner Trobin <trobin@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 <KoFilterChain.h> +#include <KoFilterManager.h> +#include <kinstance.h> +#include <kdebug.h> + +int main( int /*argc*/, char ** /*argv*/ ) +{ + KInstance instance( "filterchain_test" ); // we need an instance when using the trader + KOffice::Graph g( "application/x-kspread" ); + g.dump(); + g.setSourceMimeType( "application/x-kword" ); + g.dump(); + + KoFilterManager *manager = new KoFilterManager( 0 ); + kdDebug() << "Trying to build some filter chains..." << endl; + QCString mimeType( "foo/bar" ); + KoFilterChain::Ptr chain = g.chain( manager, mimeType ); + if ( !chain ) + kdDebug() << "Chain for 'foo/bar' is not available, OK" << endl; + else { + kdError() << "Chain for 'foo/bar' is available!" << endl; + chain->dump(); + } + + mimeType = "application/x-krita"; + chain = g.chain( manager, mimeType ); + if ( !chain ) + kdDebug() << "Chain for 'application/x-krita' is not available, OK" << endl; + else { + kdError() << "Chain 'application/x-krita' is available!" << endl; + chain->dump(); + } + + mimeType = "text/x-csv"; + chain = g.chain( manager, mimeType ); + if ( !chain ) + kdError() << "Chain for 'text/x-csv' is not available!" << endl; + else { + kdDebug() << "Chain for 'text/x-csv' is available, OK" << endl; + chain->dump(); + } + + // Try to find the closest KOffice part + mimeType = ""; + chain = g.chain( manager, mimeType ); + if ( !chain ) + kdDebug() << "It was already a KOffice part, OK" << endl; + else + kdError() << "We really got a chain? ugh :}" << endl; + + g.setSourceMimeType( "text/x-csv" ); + mimeType = ""; + chain = g.chain( manager, mimeType ); + if ( !chain ) + kdError() << "Hmm... why didn't we find a chain?" << endl; + else { + kdDebug() << "Chain for 'text/x-csv' -> closest part is available (" + << mimeType << "), OK" << endl; + chain->dump(); + } + + kdDebug() << "Checking mimeFilter() for Import:" << endl; + QStringList list = KoFilterManager::mimeFilter( "application/x-kword", KoFilterManager::Import ); + QStringList::ConstIterator it = list.begin(); + QStringList::ConstIterator end = list.end(); + for ( ; it != end; ++it ) + kdDebug() << " " << *it << endl; + kdDebug() << " " << list.count() << " entries." << endl; + + kdDebug() << "Checking mimeFilter() for Export:" << endl; + list = KoFilterManager::mimeFilter( "application/x-kword", KoFilterManager::Export ); + it = list.begin(); + end = list.end(); + for ( ; it != end; ++it ) + kdDebug() << " " << *it << endl; + kdDebug() << " " << list.count() << " entries." << endl; + + kdDebug() << "Checking KoShell's mimeFilter():" << endl; + list = KoFilterManager::mimeFilter(); + it = list.begin(); + end = list.end(); + for ( ; it != end; ++it ) + kdDebug() << " " << *it << endl; + kdDebug() << " " << list.count() << " entries." << endl; + + delete manager; + return 0; +} diff --git a/lib/kofficecore/tests/kodomtest.cpp b/lib/kofficecore/tests/kodomtest.cpp new file mode 100644 index 00000000..c8e3c539 --- /dev/null +++ b/lib/kofficecore/tests/kodomtest.cpp @@ -0,0 +1,129 @@ +/* This file is part of the KDE project + Copyright (C) 2004 David Faure <faure@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 "KoDom.h" +#include "KoXmlNS.h" + +#include <qapplication.h> +#include <assert.h> + +//static void debugElemNS( const QDomElement& elem ) +//{ +// qDebug( "nodeName=%s tagName=%s localName=%s prefix=%s namespaceURI=%s", elem.nodeName().latin1(), elem.tagName().latin1(), elem.localName().latin1(), elem.prefix().latin1(), elem.namespaceURI().latin1() ); +//} + +void testQDom( const QDomDocument& doc ) +{ + QDomElement docElem = doc.documentElement(); + //debugElemNS( docElem ); + assert( docElem.nodeName() == "o:document-content" ); + assert( docElem.tagName() == "document-content" ); + assert( docElem.localName() == "document-content" ); + assert( docElem.prefix() == "o" ); + assert( docElem.namespaceURI() == KoXmlNS::office ); + + // WARNING, elementsByTagNameNS is recursive!!! + QDomNodeList docElemChildren = docElem.elementsByTagNameNS( KoXmlNS::office, "body" ); + assert( docElemChildren.length() == 1 ); + + QDomElement elem = docElemChildren.item( 0 ).toElement(); + //debugElemNS( elem ); + assert( elem.tagName() == "body" ); + assert( elem.localName() == "body" ); + assert( elem.prefix() == "o" ); + assert( elem.namespaceURI() == KoXmlNS::office ); + + QDomNode n = elem.firstChild(); + for ( ; !n.isNull() ; n = n.nextSibling() ) { + if ( !n.isElement() ) + continue; + QDomElement e = n.toElement(); + //debugElemNS( e ); + assert( e.tagName() == "p" ); + assert( e.localName() == "p" ); + assert( e.prefix().isEmpty() ); + assert( e.namespaceURI() == KoXmlNS::text ); + } + + qDebug("testQDom... ok"); +} + +void testKoDom( const QDomDocument& doc ) +{ + QDomElement docElem = KoDom::namedItemNS( doc, KoXmlNS::office, "document-content" ); + assert( !docElem.isNull() ); + assert( docElem.localName() == "document-content" ); + assert( docElem.namespaceURI() == KoXmlNS::office ); + + QDomElement body = KoDom::namedItemNS( docElem, KoXmlNS::office, "body" ); + assert( !body.isNull() ); + assert( body.localName() == "body" ); + assert( body.namespaceURI() == KoXmlNS::office ); + + QDomElement p = KoDom::namedItemNS( body, KoXmlNS::text, "p" ); + assert( !p.isNull() ); + assert( p.localName() == "p" ); + assert( p.namespaceURI() == KoXmlNS::text ); + + const QDomElement officeStyle = KoDom::namedItemNS( docElem, KoXmlNS::office, "styles" ); + assert( !officeStyle.isNull() ); + + // Look for a non-existing element + QDomElement notexist = KoDom::namedItemNS( body, KoXmlNS::text, "notexist" ); + assert( notexist.isNull() ); + + int count = 0; + QDomElement elem; + forEachElement( elem, body ) { + assert( elem.localName() == "p" ); + assert( elem.namespaceURI() == KoXmlNS::text ); + ++count; + } + assert( count == 2 ); + + // Attributes + // ### Qt bug: it doesn't work if using style-name instead of text:style-name in the XML + const QString styleName = p.attributeNS( KoXmlNS::text, "style-name", QString::null ); + qDebug( "%s", styleName.latin1() ); + assert( styleName == "L1" ); + + qDebug("testKoDom... ok"); +} + +int main( int argc, char** argv ) { + QApplication app( argc, argv, QApplication::Tty ); + + const QCString xml = QCString( "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n" + "<o:document-content xmlns:o=\"" ) + + KoXmlNS::office + + "\" xmlns=\"" + KoXmlNS::text + + "\" xmlns:text=\"" + KoXmlNS::text + + "\">\n" + "<o:body><p text:style-name=\"L1\">foobar</p><p>2nd</p></o:body><o:styles></o:styles>\n" + "</o:document-content>\n"; + QDomDocument doc; + //QXmlSimpleReader reader; + //reader.setFeature( "http://xml.org/sax/features/namespaces", TRUE ); + //reader.setFeature( "http://xml.org/sax/features/namespace-prefixes", FALSE ); + bool ok = doc.setContent( xml, true /* namespace processing */ ); + assert( ok ); + + testQDom(doc); + testKoDom(doc); +} diff --git a/lib/kofficecore/tests/kogenstylestest.cpp b/lib/kofficecore/tests/kogenstylestest.cpp new file mode 100644 index 00000000..ddfac1f6 --- /dev/null +++ b/lib/kofficecore/tests/kogenstylestest.cpp @@ -0,0 +1,270 @@ +/* This file is part of the KDE project + Copyright (C) 2004-2006 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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 <KoGenStyles.h> +#include <KoXmlWriter.h> +#include "../../store/tests/xmlwritertest.h" +#include <kdebug.h> +#include <assert.h> + +int testLookup() +{ + kdDebug() << k_funcinfo << endl; + KoGenStyles coll; + + QMap<QString, QString> map1; + map1.insert( "map1key", "map1value" ); + QMap<QString, QString> map2; + map2.insert( "map2key1", "map2value1" ); + map2.insert( "map2key2", "map2value2" ); + + KoGenStyle first( KoGenStyle::STYLE_AUTO, "paragraph" ); + first.addAttribute( "style:master-page-name", "Standard" ); + first.addProperty( "style:page-number", "0" ); + first.addProperty( "style:foobar", "2", KoGenStyle::TextType ); + first.addStyleMap( map1 ); + first.addStyleMap( map2 ); + + QString firstName = coll.lookup( first ); + kdDebug() << "The first style got assigned the name " << firstName << endl; + assert( firstName == "A1" ); // it's fine if it's something else, but the koxmlwriter tests require a known name + assert( first.type() == KoGenStyle::STYLE_AUTO ); + + KoGenStyle second( KoGenStyle::STYLE_AUTO, "paragraph" ); + second.addAttribute( "style:master-page-name", "Standard" ); + second.addProperty( "style:page-number", "0" ); + second.addProperty( "style:foobar", "2", KoGenStyle::TextType ); + second.addStyleMap( map1 ); + second.addStyleMap( map2 ); + + QString secondName = coll.lookup( second ); + kdDebug() << "The second style got assigned the name " << secondName << endl; + + assert( firstName == secondName ); // check that sharing works + assert( first == second ); // check that operator== works :) + + const KoGenStyle* s = coll.style( firstName ); // check lookup of existing style + assert( s ); + assert( *s == first ); + s = coll.style( "foobarblah" ); // check lookup of non-existing style + assert( !s ); + + KoGenStyle third( KoGenStyle::STYLE_AUTO, "paragraph", secondName ); // inherited style + third.addProperty( "style:margin-left", "1.249cm" ); + third.addProperty( "style:page-number", "0" ); // same as parent + third.addProperty( "style:foobar", "3", KoGenStyle::TextType ); // different from parent + assert( third.parentName() == secondName ); + + QString thirdName = coll.lookup( third, "P" ); + kdDebug() << "The third style got assigned the name " << thirdName << endl; + assert( thirdName == "P1" ); + + KoGenStyle user( KoGenStyle::STYLE_USER, "paragraph" ); // differs from third since it doesn't inherit second, and has a different type + user.addProperty( "style:margin-left", "1.249cm" ); + + QString userStyleName = coll.lookup( user, "User", KoGenStyles::DontForceNumbering ); + kdDebug() << "The user style got assigned the name " << userStyleName << endl; + assert( userStyleName == "User" ); + + KoGenStyle sameAsParent( KoGenStyle::STYLE_AUTO, "paragraph", secondName ); // inherited style + sameAsParent.addAttribute( "style:master-page-name", "Standard" ); + sameAsParent.addProperty( "style:page-number", "0" ); + sameAsParent.addProperty( "style:foobar", "2", KoGenStyle::TextType ); + sameAsParent.addStyleMap( map1 ); + sameAsParent.addStyleMap( map2 ); + QString sapName = coll.lookup( sameAsParent, "foobar" ); + kdDebug() << "The 'same as parent' style got assigned the name " << sapName << endl; + + assert( sapName == secondName ); + assert( coll.styles().count() == 3 ); + + // OK, now add a style marked as for styles.xml; it looks like the above style, but + // since it's marked for styles.xml it shouldn't be shared with it. + KoGenStyle headerStyle( KoGenStyle::STYLE_AUTO, "paragraph" ); + headerStyle.addAttribute( "style:master-page-name", "Standard" ); + headerStyle.addProperty( "style:page-number", "0" ); + headerStyle.addProperty( "style:foobar", "2", KoGenStyle::TextType ); + headerStyle.addStyleMap( map1 ); + headerStyle.addStyleMap( map2 ); + headerStyle.setAutoStyleInStylesDotXml( true ); + QString headerStyleName = coll.lookup( headerStyle, "foobar" ); + + assert( coll.styles().count() == 4 ); + assert( coll.styles( KoGenStyle::STYLE_AUTO ).count() == 2 ); + assert( coll.styles( KoGenStyle::STYLE_USER ).count() == 1 ); + + QValueList<KoGenStyles::NamedStyle> stylesXmlStyles = coll.styles( KoGenStyle::STYLE_AUTO, true ); + assert( stylesXmlStyles.count() == 1 ); + KoGenStyles::NamedStyle firstStyle = stylesXmlStyles.first(); + assert( firstStyle.name == headerStyleName ); + + TEST_BEGIN( 0, 0 ); + first.writeStyle( &writer, coll, "style:style", firstName, "style:paragraph-properties" ); + TEST_END( "XML for first/second style", "<r>\n <style:style style:name=\"A1\" style:family=\"paragraph\" style:master-page-name=\"Standard\">\n <style:paragraph-properties style:page-number=\"0\"/>\n <style:text-properties style:foobar=\"2\"/>\n <style:map map1key=\"map1value\"/>\n <style:map map2key1=\"map2value1\" map2key2=\"map2value2\"/>\n </style:style>\n</r>\n" ); + + TEST_BEGIN( 0, 0 ); + third.writeStyle( &writer, coll, "style:style", thirdName, "style:paragraph-properties" ); + TEST_END( "XML for third style", "<r>\n <style:style style:name=\"P1\" style:parent-style-name=\"A1\" style:family=\"paragraph\">\n <style:paragraph-properties style:margin-left=\"1.249cm\"/>\n <style:text-properties style:foobar=\"3\"/>\n </style:style>\n</r>\n" ); + + coll.markStyleForStylesXml( firstName ); + { + QValueList<KoGenStyles::NamedStyle> stylesXmlStyles = coll.styles( KoGenStyle::STYLE_AUTO, true ); + assert( stylesXmlStyles.count() == 2 ); + QValueList<KoGenStyles::NamedStyle> contentXmlStyles = coll.styles( KoGenStyle::STYLE_AUTO, false ); + assert( contentXmlStyles.count() == 1 ); + } + + return 0; +} + +int testDefaultStyle() +{ + kdDebug() << k_funcinfo << endl; + /* Create a default style, + * and then an auto style with exactly the same attributes + * -> the lookup gives the default style. + * + * Also checks how the default style gets written out to XML. + */ + KoGenStyles coll; + + KoGenStyle defaultStyle( KoGenStyle::STYLE_AUTO, "paragraph" ); + defaultStyle.addAttribute( "style:master-page-name", "Standard" ); + defaultStyle.addProperty( "myfont", "isBold" ); + defaultStyle.setDefaultStyle( true ); + QString defaultStyleName = coll.lookup( defaultStyle ); + assert( !defaultStyleName.isEmpty() ); // whatever, but not empty + assert( defaultStyle.type() == KoGenStyle::STYLE_AUTO ); + assert( defaultStyle.isDefaultStyle() ); + + KoGenStyle anotherStyle( KoGenStyle::STYLE_AUTO, "paragraph" ); + anotherStyle.addAttribute( "style:master-page-name", "Standard" ); + anotherStyle.addProperty( "myfont", "isBold" ); + QString anotherStyleName = coll.lookup( anotherStyle ); + assert( anotherStyleName == defaultStyleName ); + + assert( coll.styles().count() == 1 ); + + TEST_BEGIN( 0, 0 ); + defaultStyle.writeStyle( &writer, coll, "style:default-style", defaultStyleName, "style:paragraph-properties" ); + TEST_END( "XML for default style", "<r>\n <style:default-style style:family=\"paragraph\" style:master-page-name=\"Standard\">\n <style:paragraph-properties myfont=\"isBold\"/>\n </style:default-style>\n</r>\n" ); + + // The kspread case: not writing out all properties, only if they differ + // from the default style. + // KoGenStyles doesn't fetch info from the parent style when testing + // for equality, so KSpread uses isEmpty() to check for equality-to-parent. + KoGenStyle dataStyle( KoGenStyle::STYLE_AUTO, "paragraph", defaultStyleName ); + assert( dataStyle.isEmpty() ); + // and then it doesn't look up the auto style, but rather uses the parent style directly. + + return 0; +} + +int testUserStyles() +{ + kdDebug() << k_funcinfo << endl; + /* Two user styles with exactly the same attributes+properties will not get merged, since + * they don't have exactly the same attributes after all: the display-name obviously differs :) + */ + KoGenStyles coll; + + KoGenStyle user1( KoGenStyle::STYLE_USER, "paragraph" ); + user1.addAttribute( "style:display-name", "User 1" ); + user1.addProperty( "myfont", "isBold" ); + + QString user1StyleName = coll.lookup( user1, "User1", KoGenStyles::DontForceNumbering ); + kdDebug() << "The user style got assigned the name " << user1StyleName << endl; + assert( user1StyleName == "User1" ); + + KoGenStyle user2( KoGenStyle::STYLE_USER, "paragraph" ); + user2.addAttribute( "style:display-name", "User 2" ); + user2.addProperty( "myfont", "isBold" ); + + QString user2StyleName = coll.lookup( user2, "User2", KoGenStyles::DontForceNumbering ); + kdDebug() << "The user style got assigned the name " << user2StyleName << endl; + assert( user2StyleName == "User2" ); + + // And now, what if the data uses that style? + // This is like sameAsParent in the other test, but this time the + // parent is a STYLE_USER... + KoGenStyle dataStyle( KoGenStyle::STYLE_AUTO, "paragraph", user2StyleName ); + dataStyle.addProperty( "myfont", "isBold" ); + + QString dataStyleName = coll.lookup( dataStyle, "DataStyle" ); + kdDebug() << "The auto style got assigned the name " << dataStyleName << endl; + assert( dataStyleName == "User2" ); // it found the parent as equal + + // Let's do the opposite test, just to make sure + KoGenStyle dataStyle2( KoGenStyle::STYLE_AUTO, "paragraph", user2StyleName ); + dataStyle2.addProperty( "myfont", "isNotBold" ); + + QString dataStyle2Name = coll.lookup( dataStyle2, "DataStyle" ); + kdDebug() << "The different auto style got assigned the name " << dataStyle2Name << endl; + assert( dataStyle2Name == "DataStyle1" ); + + assert( coll.styles().count() == 3 ); + + TEST_BEGIN( 0, 0 ); + user1.writeStyle( &writer, coll, "style:style", user1StyleName, "style:paragraph-properties" ); + TEST_END( "XML for user style 1", "<r>\n <style:style style:name=\"User1\" style:display-name=\"User 1\" style:family=\"paragraph\">\n <style:paragraph-properties myfont=\"isBold\"/>\n </style:style>\n</r>\n" ); + + TEST_BEGIN( 0, 0 ); + user2.writeStyle( &writer, coll, "style:style", user2StyleName, "style:paragraph-properties" ); + TEST_END( "XML for user style 2", "<r>\n <style:style style:name=\"User2\" style:display-name=\"User 2\" style:family=\"paragraph\">\n <style:paragraph-properties myfont=\"isBold\"/>\n </style:style>\n</r>\n" ); + + return 0; +} + +int testStylesDotXml() +{ + kdDebug() << k_funcinfo << endl; + KoGenStyles coll; + + // Check that an autostyle-in-style.xml and an autostyle-in-content.xml + // don't get the same name. It confuses KoGenStyle's named-based maps. + KoGenStyle headerStyle( KoGenStyle::STYLE_AUTO, "paragraph" ); + headerStyle.addAttribute( "style:master-page-name", "Standard" ); + headerStyle.addProperty( "style:page-number", "0" ); + headerStyle.setAutoStyleInStylesDotXml( true ); + QString headerStyleName = coll.lookup( headerStyle, "P" ); + assert( headerStyleName == "P1" ); + + //coll.dump(); + + KoGenStyle first( KoGenStyle::STYLE_AUTO, "paragraph" ); + first.addAttribute( "style:master-page-name", "Standard" ); + QString firstName = coll.lookup( first, "P" ); + kdDebug() << "The auto style got assigned the name " << firstName << endl; + assert( firstName == "P2" ); // anything but not P1. + return 0; +} + +int main( int, char** ) { + fprintf( stderr, "OK\n" ); + + if ( testLookup() ) + return 1; + if ( testDefaultStyle() ) + return 1; + if ( testUserStyles() ) + return 1; + if ( testStylesDotXml() ) + return 1; + + return 0; +} diff --git a/lib/kofficecore/tests/kooasissettingstest.cpp b/lib/kofficecore/tests/kooasissettingstest.cpp new file mode 100644 index 00000000..ef29536d --- /dev/null +++ b/lib/kofficecore/tests/kooasissettingstest.cpp @@ -0,0 +1,106 @@ +/* This file is part of the KDE project + Copyright (C) 2004 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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 <KoOasisSettings.h> +#include <KoDom.h> +#include <kdebug.h> +#include <assert.h> + +void testSelectItemSet( KoOasisSettings& settings ) +{ + KoOasisSettings::Items items = settings.itemSet( "notexist" ); + assert( items.isNull() ); + items = settings.itemSet( "view-settings" ); + assert( !items.isNull() ); + kdDebug() << "testSelectItemSet OK" << endl; +} + +void testParseConfigItemString( KoOasisSettings& settings ) +{ + KoOasisSettings::Items viewSettings = settings.itemSet( "view-settings" ); + const QString unit = viewSettings.parseConfigItemString( "unit" ); + qDebug( "%s", unit.latin1() ); + assert( unit == "mm" ); + kdDebug() << "testParseConfigItemString OK" << endl; +} + +void testIndexedMap( KoOasisSettings& settings ) +{ + KoOasisSettings::Items viewSettings = settings.itemSet( "view-settings" ); + assert( !viewSettings.isNull() ); + KoOasisSettings::IndexedMap viewMap = viewSettings.indexedMap( "Views" ); + assert( !viewMap.isNull() ); + KoOasisSettings::Items firstView = viewMap.entry( 0 ); + assert( !firstView.isNull() ); + const short zoomFactor = firstView.parseConfigItemShort( "ZoomFactor" ); + assert( zoomFactor == 100 ); + KoOasisSettings::Items secondView = viewMap.entry( 1 ); + assert( secondView.isNull() ); + kdDebug() << "testIndexedMap OK" << endl; +} + +void testNamedMap( KoOasisSettings& settings ) +{ + KoOasisSettings::Items viewSettings = settings.itemSet( "view-settings" ); + assert( !viewSettings.isNull() ); + KoOasisSettings::NamedMap viewMap = viewSettings.namedMap( "NamedMap" ); + assert( !viewMap.isNull() ); + KoOasisSettings::Items foo = viewMap.entry( "foo" ); + assert( !foo.isNull() ); + const int zoomFactor = foo.parseConfigItemShort( "ZoomFactor" ); + assert( zoomFactor == 100 ); + KoOasisSettings::Items secondView = viewMap.entry( "foobar" ); + assert( secondView.isNull() ); + kdDebug() << "testNamedMap OK" << endl; +} + +int main( int, char** ) { + + const QCString xml = "\ +<?xml version=\"1.0\" encoding=\"UTF-8\"?> \ +<office:document-settings xmlns:office=\"urn:oasis:names:tc:opendocument:xmlns:office:1.0\" xmlns:config=\"urn:oasis:names:tc:opendocument:xmlns:config:1.0\"> \ + <office:settings> \ + <config:config-item-set config:name=\"view-settings\"> \ + <config:config-item config:name=\"unit\" config:type=\"string\">mm</config:config-item> \ + <config:config-item-map-indexed config:name=\"Views\"> \ + <config:config-item-map-entry> \ + <config:config-item config:name=\"ZoomFactor\" config:type=\"short\">100</config:config-item> \ + </config:config-item-map-entry> \ + </config:config-item-map-indexed> \ + <config:config-item-map-named config:name=\"NamedMap\"> \ + <config:config-item-map-entry config:name=\"foo\"> \ + <config:config-item config:name=\"ZoomFactor\" config:type=\"int\">100</config:config-item> \ + </config:config-item-map-entry> \ + </config:config-item-map-named> \ + </config:config-item-set> \ + </office:settings> \ +</office:document-settings> \ +"; + + QDomDocument doc; + bool ok = doc.setContent( xml, true /* namespace processing */ ); + assert( ok ); + + KoOasisSettings settings( doc ); + + testSelectItemSet( settings ); + testParseConfigItemString( settings ); + testIndexedMap( settings ); + testNamedMap( settings ); + return 0; +} diff --git a/lib/kofficecore/tests/kooasisstoretest.cpp b/lib/kofficecore/tests/kooasisstoretest.cpp new file mode 100644 index 00000000..6750904b --- /dev/null +++ b/lib/kofficecore/tests/kooasisstoretest.cpp @@ -0,0 +1,58 @@ +/* This file is part of the KDE project + Copyright (C) 2005 David Faure <faure@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 version 2 as published by the Free Software Foundation. + + 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 <KoOasisStore.h> +#include <KoDom.h> +#include <kdebug.h> +#include <assert.h> + +void testMimeForPath( QDomDocument& doc ) +{ + QString mime = KoOasisStore::mimeForPath( doc, "Object 1" ); + kdDebug() << k_funcinfo << mime << endl; + assert( !mime.isNull() ); + assert( !mime.isEmpty() ); + assert( mime == "application/vnd.oasis.opendocument.text" ); + kdDebug() << "testMimeForPath OK" << endl; +} + +int main( int, char** ) { + + const QCString xml = "\ +<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\ +<manifest:manifest xmlns:manifest=\"urn:oasis:names:tc:opendocument:xmlns:manifest:1.0\">\n\ + <manifest:file-entry manifest:media-type=\"application/vnd.oasis.opendocument.text\" manifest:full-path=\"/\"/>\n\ + <manifest:file-entry manifest:media-type=\"text/xml\" manifest:full-path=\"content.xml\"/>\n\ + <manifest:file-entry manifest:media-type=\"application/vnd.oasis.opendocument.text\" manifest:full-path=\"Object 1\"/>\n\ +</manifest:manifest> \ +"; + + QDomDocument doc; + QString errorMsg; + int errorLine, errorColumn; + bool ok = doc.setContent( xml, true /* namespace processing */, &errorMsg, &errorLine, &errorColumn ); + if ( !ok ) { + kdError() << "Parsing error! Aborting!" << endl + << " In line: " << errorLine << ", column: " << errorColumn << endl + << " Error message: " << errorMsg << endl; + return 1; + } + + testMimeForPath( doc ); + return 0; +} diff --git a/lib/kofficecore/tests/kopointtest.cpp b/lib/kofficecore/tests/kopointtest.cpp new file mode 100644 index 00000000..25286a5d --- /dev/null +++ b/lib/kofficecore/tests/kopointtest.cpp @@ -0,0 +1,89 @@ +#include <KoPoint.h> +#include <kdebug.h> +#include <kglobal.h> +#include <stdlib.h> +#include <math.h> + +bool check( QString txt, bool res, bool expected ) +{ + if ( res == expected ) { + kdDebug() << txt << " : checking '" << res << "' against expected value '" << expected << "'... " << "ok" << endl; + } else { + kdDebug() << txt << " : checking '" << res << "' against expected value '" << expected << "'... " << "KO !" << endl; + exit( 1 ); + } + return true; +} + +bool check( QString txt, double res, double expected ) +{ + if ( kAbs(res - expected) < 0.000001 ) { + kdDebug() << txt << " : checking '" << res << "' against expected value '" << expected << "'... " << "ok" << endl; + } else { + kdDebug() << txt << " : checking '" << res << "' against expected value '" << expected << "'... " << "KO !" << endl; + exit( 1 ); + } + return true; +} + +int main() +{ + KoPoint p0; + check( "KoPoint() is null", p0.isNull(), true ); + KoPoint p1( 10.1, 20.2 ); + check( "KoPoint(...) is not null", p1.isNull(), false ); + check( "KoPoint::x()", p1.x(), 10.1 ); + check( "KoPoint::y()", p1.y(), 20.2 ); + p1.setX( 2.1 ); + p1.setY( 3.2 ); + check( "KoPoint::setX()", p1.x(), 2.1 ); + check( "KoPoint::setY()", p1.y(), 3.2 ); + + KoPoint p2( QPoint( -30, -40 ) ); + check( "KoPoint(const QPoint&)", p2.x(), -30.0 ); + check( "KoPoint(const QPoint&)", p2.y(), -40.0 ); + p2.rx() = 1.5; + p2.ry() = 2.5; + check( "KoPoint::rx()", p2.x(), 1.5 ); + check( "KoPoint::ry()", p2.y(), 2.5 ); + p2.setCoords( -1.6, -1.7 ); + check( "KoPoint::setCoords(double, double)", p2.x(), -1.6 ); + check( "KoPoint::setCoords(double, double)", p2.y(), -1.7 ); + + check( "KoPoint::operator==(const KoPoint&)", p1 == p1, true ); + check( "KoPoint::operator!=(const KoPoint&)", p1 != p2, true ); + + KoPoint p4( 10.2, 20.2 ); + KoPoint p5( 30.4, 40.4 ); + p4 += p5; + check( "KoPoint::operator+=(const KoPoint&)", p4.x(), 40.6 ); + check( "KoPoint::operator+=(const KoPoint&)", p4.y(), 60.6 ); + p4 -= p5; + check( "KoPoint::operator-=(const KoPoint&)", p4.x(), 10.2 ); + check( "KoPoint::operator-=(const KoPoint&)", p4.y(), 20.2 ); + p4 *= 2.0; + check( "KoPoint::operator*=(double)", p4.x(), 20.4 ); + check( "KoPoint::operator*=(double)", p4.y(), 40.4 ); + p4 = p5; + check( "KoPoint::operator=(const KoPoint&)", p4.x(), 30.4 ); + check( "KoPoint::operator=(const KoPoint&)", p4.y(), 40.4 ); + + // ### global operator+, operator- and operator* + // ### transform + + KoPoint p6( 1.0, 2.0 ); + check( "KoPoint::isNear()", p6.isNear( p6, 0 ), true ); + check( "KoPoint::isNear()", p6.isNear( KoPoint( 2, 10 ), 1 ), false ); + check( "KoPoint::isNear()", p6.isNear( KoPoint( 2, 1 ), 1 ), true ); + + KoPoint p7( 10, 10 ); + check( "KoPoint::getAngle()", + fmod( KoPoint::getAngle( p7, p7 * -1 ) + 10*360.0, 360.0 ), 45.0 ); + // ### ??? check( "KoPoint::getAngle()", + // fmod( KoPoint::getAngle( -1 * p7, p7 ) + 10*360.0, 360.0 ), 45.0 ); + // can we define a behaviour ? + // ### check( "KoPoint::getAngle()", KoPoint::getAngle( p7, p7 ), ??? ); + + kdDebug() << endl << "Test OK !" << endl; +} + diff --git a/lib/kofficecore/tests/korecttest.cpp b/lib/kofficecore/tests/korecttest.cpp new file mode 100644 index 00000000..9c2cb8b0 --- /dev/null +++ b/lib/kofficecore/tests/korecttest.cpp @@ -0,0 +1,53 @@ +#include <KoRect.h> +#include <stdio.h> +#include <kapplication.h> +#include <stdlib.h> +#include <kdebug.h> +#include <kglobal.h> + +bool check(QString txt, bool res, bool expected) +{ + if (res == expected) { + kdDebug() << txt << " : checking '" << res << "' against expected value '" << expected << "'... " << "ok" << endl; + } + else { + kdDebug() << txt << " : checking '" << res << "' against expected value '" << expected << "'... " << "KO !" << endl; + exit(1); + } + return true; +} + +bool check(QString txt, QString a, QString b) +{ + if (a.isEmpty()) + a = QString::null; + if (b.isEmpty()) + b = QString::null; + if (a == b) { + kdDebug() << txt << " : checking '" << a << "' against expected value '" << b << "'... " << "ok" << endl; + } + else { + kdDebug() << txt << " : checking '" << a << "' against expected value '" << b << "'... " << "KO !" << endl; + exit(1); + } + return true; +} + +int main(int argc, char *argv[]) +{ + KApplication app(argc,argv,"korecttest",false,false); + + KoRect emptyRect; + check( "KoRect() is null", emptyRect.isNull(), true ); + check( "KoRect() is empty", emptyRect.isEmpty(), true ); + KoRect rect( 1, 15, 250, 156.14 ); + check( "KoRect(...) is not null", rect.isNull(), false ); + check( "KoRect(...) is not empty", rect.isEmpty(), false ); + KoRect unionRect = rect | emptyRect; + check( "Union is not null", unionRect.isNull(), false ); + check( "Union is not empty", unionRect.isEmpty(), false ); + kdDebug() << unionRect << endl; + + printf("\nTest OK !\n"); +} + diff --git a/lib/kofficecore/tests/koxmlreadertest.cpp b/lib/kofficecore/tests/koxmlreadertest.cpp new file mode 100644 index 00000000..213fe0db --- /dev/null +++ b/lib/kofficecore/tests/koxmlreadertest.cpp @@ -0,0 +1,2074 @@ +/* This file is part of the KDE project + Copyright (C) 2005 Ariya Hidayat <ariya@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, +*/ + +// xmlreadertest.cpp - test KoXml classes +// Ariya Hidayat, November 2005 + +#include <qstring.h> +#include <qcstring.h> +#include <qbuffer.h> +#include <qtextstream.h> +#include <qdatetime.h> +#include <qfile.h> + +#include "KoXmlReader.h" +#include <qxml.h> + +#define CHECK(x,y) check(__FILE__,__LINE__,#x,x,y) + +static int testCount = 0; +static int testFailed = 0; + +template<typename T> +void check( const char *file, int line, const char* msg, +const T& result, const T& expected ) +{ + testCount++; + if( result != expected ) + { + testFailed++; + QString message; + QTextStream ts( &message, IO_WriteOnly ); + ts << msg; + ts << " Result:"; + ts << result; + ts << ", "; + ts << "Expected:"; + ts << expected; + printf( "%s [%d]: %s\n", file, line, message.latin1() ); + } +} + +void check( const char *file, int line, const char* msg, bool result, +bool expected ) +{ + testCount++; + if( result != expected ) + { + testFailed++; + QString message; + QTextStream ts( &message, IO_WriteOnly ); + ts << msg; + ts << " Result: "; + if( result ) ts << "True"; else ts << "False"; + ts << ", "; + ts << "Expected: "; + if( expected ) ts << "True"; else ts << "False"; + printf( "%s [%d]: %s\n", file, line, message.latin1() ); + } +} + +void testNode() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<earth>"; + xmlstream << "<continents>"; + xmlstream << "<asia/>"; + xmlstream << "<africa/>"; + xmlstream << "<europe/>"; + xmlstream << "<america/>"; + xmlstream << "<australia/>"; + xmlstream << "<antartic/>"; + xmlstream << "</continents>"; + xmlstream << "<oceans>"; + xmlstream << "<pacific/>"; + xmlstream << "<atlantic/>"; + xmlstream << "</oceans>"; + xmlstream << "</earth>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + // null node + KoXmlNode node1; + CHECK( node1.nodeName(), QString::null ); + CHECK( node1.isNull(), true ); + CHECK( node1.isElement(), false ); + CHECK( node1.isElement(), false ); + CHECK( node1.isDocument(), false ); + CHECK( node1.ownerDocument().isNull(), false ); + CHECK( node1.parentNode().isNull(), true ); + CHECK( node1.hasChildNodes(), false ); + CHECK( node1.firstChild().isNull(), true ); + CHECK( node1.lastChild().isNull(), true ); + CHECK( node1.previousSibling().isNull(), true ); + CHECK( node1.nextSibling().isNull(), true ); + + // compare with another null node + KoXmlNode node2; + CHECK( node2.isNull(), true ); + CHECK( node1==node2, true ); + CHECK( node1!=node2, false ); + + // a node which is a document + KoXmlNode node3 = doc; + CHECK( node3.nodeName(), QString("#document") ); + CHECK( node3.isNull(), false ); + CHECK( node3.isElement(), false ); + CHECK( node3.isText(), false ); + CHECK( node3.isDocument(), true ); + CHECK( node3.ownerDocument().isNull(), false ); + CHECK( node3.ownerDocument()==doc, true ); + CHECK( node3.toDocument()==doc, true ); + + // convert to document and the compare + KoXmlDocument doc2 = node3.toDocument(); + CHECK( doc2.nodeName(), QString("#document") ); + CHECK( doc2.isNull(), false ); + CHECK( doc2.isDocument(), true ); + CHECK( node3==doc2, true ); + + // a document is of course can't be converted to element + KoXmlElement invalidElement = node3.toElement(); + CHECK( invalidElement.nodeName(), QString::null ); + CHECK( invalidElement.isNull(), true ); + CHECK( invalidElement.isElement(), true ); + CHECK( invalidElement.isText(), false ); + CHECK( invalidElement.isDocument(), false ); + + // clear() makes it a null node again + node3.clear(); + CHECK( node3.isNull(), true ); + CHECK( node3.nodeName(), QString::null ); + CHECK( node3.isElement(), false ); + CHECK( node3.isText(), false ); + CHECK( node3.isDocument(), false ); + CHECK( node3.ownerDocument().isNull(), false ); + CHECK( node1==node3, true ); + CHECK( node1!=node3, false ); + + // a node which is an element for <earth> + KoXmlNode node4 = doc.firstChild(); + CHECK( node4.isNull(), false ); + CHECK( node4.isElement(), true ); + CHECK( node4.isText(), false ); + CHECK( node4.isDocument(), false ); + CHECK( node4.hasChildNodes(), true ); + CHECK( node4.ownerDocument()==doc, true ); + CHECK( node4.toElement()==doc.firstChild().toElement(), true ); + + // clear() makes it a null node again + node4.clear(); + CHECK( node4.isNull(), true ); + CHECK( node4.isElement(), false ); + CHECK( node4.isText(), false ); + CHECK( node4.isDocument(), false ); + CHECK( node4==node1, true ); + CHECK( node4!=node1, false ); + + // a node which is an element for <continents> + KoXmlNode node5 = doc.firstChild().firstChild(); + CHECK( node5.nodeName(), QString("continents") ); + CHECK( node5.isNull(), false ); + CHECK( node5.isElement(), true ); + CHECK( node5.isText(), false ); + CHECK( node5.isDocument(), false ); + CHECK( node5.hasChildNodes(), true ); + CHECK( node5.ownerDocument()==doc, true ); + + // convert to element and the compare + KoXmlElement continentsElement = node5.toElement(); + CHECK( node5==continentsElement, true ); + CHECK( continentsElement.isNull(), false ); + CHECK( continentsElement.isElement(), true ); + CHECK( continentsElement.isText(), false ); + CHECK( continentsElement.hasChildNodes(), true ); + CHECK( continentsElement.ownerDocument()==doc, true ); + + // and it doesn't make sense to convert that node to document + // (instead a brand new document is created, i.e. not a null node) + KoXmlDocument invalidDoc = node5.toDocument(); + CHECK( invalidDoc.isNull(), false ); + CHECK( invalidDoc.isElement(), false ); + CHECK( invalidDoc.isText(), false ); + CHECK( invalidDoc.isDocument(), true ); + + // node for <europe> using namedItem() function + KoXmlNode europeNode = continentsElement.namedItem( QString("europe") ); + CHECK( europeNode.nodeName(), QString("europe") ); + CHECK( europeNode.isNull(), false ); + CHECK( europeNode.isElement(), true ); + CHECK( europeNode.isText(), false ); + CHECK( europeNode.hasChildNodes(), false ); + CHECK( europeNode.ownerDocument()==doc, true ); + + // search non-existing node + KoXmlNode fooNode = continentsElement.namedItem( QString("foobar") ); + CHECK( fooNode.isNull(), true ); + CHECK( fooNode.isElement(), false ); + CHECK( fooNode.isText(), false ); + CHECK( fooNode.isCDATASection(), false ); +} + +void testElement() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<html>"; + xmlstream << "<body bgcolor=\"#000\">"; + xmlstream << "<p>"; + xmlstream << "Hello, world!"; + xmlstream << "</p>"; + xmlstream << "</body>"; + xmlstream << "</html>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + // element for <html> + KoXmlElement rootElement; + rootElement = doc.documentElement(); + CHECK( rootElement.nodeName(), QString("html") ); + CHECK( rootElement.isNull(), false ); + CHECK( rootElement.isElement(), true ); + CHECK( rootElement.isDocument(), false ); + CHECK( rootElement.ownerDocument().isNull(), false ); + CHECK( rootElement.ownerDocument()==doc, true ); + CHECK( rootElement.parentNode().isNull(), false ); + CHECK( rootElement.parentNode().toDocument()==doc, true ); + CHECK( rootElement.hasChildNodes(), true ); + CHECK( rootElement.tagName(), QString("html") ); + CHECK( rootElement.prefix().isNull(), true ); + + // element for <body> + KoXmlElement bodyElement; + bodyElement = rootElement.firstChild().toElement(); + CHECK( bodyElement.nodeName(), QString("body") ); + CHECK( bodyElement.isNull(), false ); + CHECK( bodyElement.isElement(), true ); + CHECK( bodyElement.isDocument(), false ); + CHECK( bodyElement.ownerDocument().isNull(), false ); + CHECK( bodyElement.ownerDocument()==doc, true ); + CHECK( bodyElement.parentNode().isNull(), false ); + CHECK( bodyElement.parentNode()==rootElement, true ); + CHECK( bodyElement.hasChildNodes(), true ); + CHECK( bodyElement.tagName(), QString("body") ); + CHECK( bodyElement.prefix().isNull(), true ); + CHECK( bodyElement.hasAttribute("bgcolor"), true ); + CHECK( bodyElement.attribute("bgcolor"), QString("#000") ); + + // a shared copy of <body>, will still have access to attribute bgcolor + KoXmlElement body2Element; + body2Element = bodyElement; + CHECK( body2Element.nodeName(), QString("body") ); + CHECK( body2Element.isNull(), false ); + CHECK( body2Element.isElement(), true ); + CHECK( body2Element.isDocument(), false ); + CHECK( body2Element.ownerDocument().isNull(), false ); + CHECK( body2Element.ownerDocument()==doc, true ); + CHECK( body2Element==bodyElement, true ); + CHECK( body2Element!=bodyElement, false ); + CHECK( body2Element.hasChildNodes(), true ); + CHECK( body2Element.tagName(), QString("body") ); + CHECK( body2Element.prefix().isNull(), true ); + CHECK( body2Element.hasAttribute("bgcolor"), true ); + CHECK( body2Element.attribute("bgcolor"), QString("#000") ); + + // empty element, by default constructor + KoXmlElement testElement; + CHECK( testElement.nodeName(), QString::null ); + CHECK( testElement.isNull(), true ); + CHECK( testElement.isElement(), true ); + CHECK( testElement.isDocument(), false ); + CHECK( testElement.ownerDocument().isNull(), false ); + CHECK( testElement.ownerDocument()!=doc, true ); + CHECK( testElement==rootElement, false ); + CHECK( testElement!=rootElement, true ); + CHECK( testElement.parentNode().isNull(), true ); + CHECK( testElement.hasChildNodes(), false ); + + // check assignment operator + testElement = rootElement; + CHECK( testElement.nodeName(), QString("html") ); + CHECK( testElement.isNull(), false ); + CHECK( testElement.isElement(), true ); + CHECK( testElement.isDocument(), false ); + CHECK( testElement==rootElement, true ); + CHECK( testElement!=rootElement, false ); + CHECK( testElement.parentNode().isNull(), false ); + CHECK( testElement.parentNode().toDocument()==doc, true ); + CHECK( testElement.tagName(), QString("html") ); + CHECK( testElement.prefix().isNull(), true ); + + // assigned from another empty element + testElement = KoXmlElement(); + CHECK( testElement.isNull(), true ); + CHECK( testElement!=rootElement, true ); + + // assigned from <body> + testElement = bodyElement; + CHECK( testElement.isNull(), false ); + CHECK( testElement.isElement(), true ); + CHECK( testElement.isDocument(), false ); + CHECK( testElement.ownerDocument().isNull(), false ); + CHECK( testElement.ownerDocument()==doc, true ); + CHECK( testElement==bodyElement, true ); + CHECK( testElement.parentNode().isNull(), false ); + CHECK( testElement.tagName(), QString("body") ); + CHECK( testElement.prefix().isNull(), true ); + CHECK( testElement.hasChildNodes(), true ); + + // copy constructor + KoXmlElement dummyElement( rootElement ); + CHECK( dummyElement.isNull(), false ); + CHECK( dummyElement.isElement(), true ); + CHECK( dummyElement.isDocument(), false ); + CHECK( dummyElement.ownerDocument().isNull(), false ); + CHECK( dummyElement.ownerDocument()==doc, true ); + CHECK( dummyElement==rootElement, true ); + CHECK( dummyElement.parentNode().isNull(), false ); + CHECK( dummyElement.hasChildNodes(), true ); + CHECK( dummyElement.tagName(), QString("html") ); + CHECK( dummyElement.prefix().isNull(), true ); + + // clear() turns element to null node + dummyElement.clear(); + CHECK( dummyElement.isNull(), true ); + CHECK( dummyElement.isElement(), true ); + CHECK( dummyElement.isDocument(), false ); + CHECK( dummyElement.ownerDocument().isNull(), false ); + CHECK( dummyElement.ownerDocument()==doc, false ); + CHECK( dummyElement.hasChildNodes(), false ); + CHECK( dummyElement==rootElement, false ); + CHECK( dummyElement!=rootElement, true ); + + // check for plain null node converted to element + KoXmlNode dummyNode; + dummyElement = dummyNode.toElement(); + CHECK( dummyElement.isNull(), true ); + CHECK( dummyElement.isElement(), true ); + CHECK( dummyElement.isDocument(), false ); + CHECK( dummyElement.ownerDocument().isNull(), false ); + CHECK( dummyElement.hasChildNodes(), false ); + CHECK( dummyElement.ownerDocument()==doc, false ); +} + +void testAttributes() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<p>"; + xmlstream << "<img src=\"foo.png\" width=\"300\" height=\"150\"/>"; + xmlstream << "</p>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + KoXmlElement rootElement; + rootElement = doc.documentElement(); + CHECK( rootElement.isNull(), false ); + CHECK( rootElement.isElement(), true ); + CHECK( rootElement.parentNode().isNull(), false ); + CHECK( rootElement.parentNode().toDocument()==doc, true ); + CHECK( rootElement.tagName(), QString("p") ); + CHECK( rootElement.prefix().isNull(), true ); + + KoXmlElement imgElement; + imgElement = rootElement.firstChild().toElement(); + CHECK( imgElement.isNull(), false ); + CHECK( imgElement.isElement(), true ); + CHECK( imgElement.tagName(), QString("img") ); + CHECK( imgElement.prefix().isNull(), true ); + CHECK( imgElement.hasAttribute("src"), true ); + CHECK( imgElement.hasAttribute("width"), true ); + CHECK( imgElement.hasAttribute("height"), true ); + CHECK( imgElement.hasAttribute("non-exist"), false ); + CHECK( imgElement.hasAttribute("SRC"), false ); + CHECK( imgElement.attribute("src"), QString("foo.png") ); + CHECK( imgElement.attribute("width"), QString("300") ); + CHECK( imgElement.attribute("width").toInt(), 300 ); + CHECK( imgElement.attribute("height"), QString("150") ); + CHECK( imgElement.attribute("height").toInt(), 150 ); + CHECK( imgElement.attribute("border").isEmpty(), true ); + CHECK( imgElement.attribute("border","0").toInt(), 0 ); + CHECK( imgElement.attribute("border","-1").toInt(), -1 ); +} + +void testText() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<p>"; + xmlstream << "Hello "; + xmlstream << "<b>world</b>"; + xmlstream << "</p>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + // element for <p> + KoXmlElement parElement; + parElement = doc.documentElement(); + CHECK( parElement.isNull(), false ); + CHECK( parElement.isElement(), true ); + CHECK( parElement.isText(), false ); + CHECK( parElement.isDocument(), false ); + CHECK( parElement.ownerDocument().isNull(), false ); + CHECK( parElement.ownerDocument()==doc, true ); + CHECK( parElement.parentNode().isNull(), false ); + CHECK( parElement.parentNode().toDocument()==doc, true ); + CHECK( parElement.hasChildNodes(), true ); + CHECK( parElement.tagName(), QString("p") ); + CHECK( parElement.prefix().isNull(), true ); + CHECK( parElement.text(), QString("Hello world") ); + + // node for "Hello" + KoXmlNode helloNode; + helloNode = parElement.firstChild(); + CHECK( helloNode.nodeName(), QString("#text") ); + CHECK( helloNode.isNull(), false ); + CHECK( helloNode.isElement(), false ); + CHECK( helloNode.isText(), true ); + CHECK( helloNode.isDocument(), false ); + + // "Hello" text + KoXmlText helloText; + helloText = helloNode.toText(); + CHECK( helloText.nodeName(), QString("#text") ); + CHECK( helloText.isNull(), false ); + CHECK( helloText.isElement(), false ); + CHECK( helloText.isText(), true ); + CHECK( helloText.isDocument(), false ); + CHECK( helloText.data(), QString("Hello ") ); + + // shared copy of the text + KoXmlText hello2Text; + hello2Text = helloText; + CHECK( hello2Text.isNull(), false ); + CHECK( hello2Text.isElement(), false ); + CHECK( hello2Text.isText(), true ); + CHECK( hello2Text.isDocument(), false ); + CHECK( hello2Text.data(), QString("Hello ") ); + + // element for <b> + KoXmlElement boldElement; + boldElement = helloNode.nextSibling().toElement(); + CHECK( boldElement.isNull(), false ); + CHECK( boldElement.isElement(), true ); + CHECK( boldElement.isText(), false ); + CHECK( boldElement.isDocument(), false ); + CHECK( boldElement.ownerDocument().isNull(), false ); + CHECK( boldElement.ownerDocument()==doc, true ); + CHECK( boldElement.parentNode().isNull(), false ); + CHECK( boldElement.hasChildNodes(), true ); + CHECK( boldElement.tagName(), QString("b") ); + CHECK( boldElement.prefix().isNull(), true ); + + // "world" text + KoXmlText worldText; + worldText = boldElement.firstChild().toText(); + CHECK( worldText.isNull(), false ); + CHECK( worldText.isElement(), false ); + CHECK( worldText.isText(), true ); + CHECK( worldText.isDocument(), false ); + CHECK( worldText.data(), QString("world") ); +} + +void testCDATA() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<p>"; + xmlstream << "Hello "; + xmlstream << "<![CDATA[world]]>"; + xmlstream << "</p>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + // element for <p> + KoXmlElement parElement; + parElement = doc.documentElement(); + CHECK( parElement.isNull(), false ); + CHECK( parElement.isElement(), true ); + CHECK( parElement.isText(), false ); + CHECK( parElement.isDocument(), false ); + CHECK( parElement.ownerDocument().isNull(), false ); + CHECK( parElement.ownerDocument()==doc, true ); + CHECK( parElement.parentNode().isNull(), false ); + CHECK( parElement.parentNode().toDocument()==doc, true ); + CHECK( parElement.hasChildNodes(), true ); + CHECK( parElement.tagName(), QString("p") ); + CHECK( parElement.prefix().isNull(), true ); + CHECK( parElement.text(), QString("Hello world") ); + + // node for "Hello" + KoXmlNode helloNode; + helloNode = parElement.firstChild(); + CHECK( helloNode.isNull(), false ); + CHECK( helloNode.isElement(), false ); + CHECK( helloNode.isText(), true ); + CHECK( helloNode.isDocument(), false ); + + // "Hello" text + KoXmlText helloText; + helloText = helloNode.toText(); + CHECK( helloText.isNull(), false ); + CHECK( helloText.isElement(), false ); + CHECK( helloText.isText(), true ); + CHECK( helloText.isDocument(), false ); + CHECK( helloText.data(), QString("Hello ") ); + + // node for CDATA "world!" + // Note: isText() is also true for CDATA + KoXmlNode worldNode; + worldNode = helloNode.nextSibling(); + CHECK( worldNode.nodeName(), QString("#cdata-section") ); + CHECK( worldNode.isNull(), false ); + CHECK( worldNode.isElement(), false ); + CHECK( worldNode.isText(), true ); + CHECK( worldNode.isCDATASection(), true ); + CHECK( worldNode.isDocument(), false ); + + // CDATA section for "world!" + // Note: isText() is also true for CDATA + KoXmlCDATASection worldCDATA; + worldCDATA = worldNode.toCDATASection(); + CHECK( worldCDATA.nodeName(), QString("#cdata-section") ); + CHECK( worldCDATA.isNull(), false ); + CHECK( worldCDATA.isElement(), false ); + CHECK( worldCDATA.isText(), true ); + CHECK( worldCDATA.isCDATASection(), true ); + CHECK( worldCDATA.isDocument(), false ); + CHECK( worldCDATA.data(), QString("world") ); +} + +void testDocument() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<koffice>"; + xmlstream << " <kword/>\n"; + xmlstream << " <kpresenter/>\n"; + xmlstream << " <krita/>\n"; + xmlstream << "</koffice>"; + + KoXmlDocument doc; + + // empty document + CHECK( doc.nodeName(), QString("#document") ); + CHECK( doc.isNull(), false ); + CHECK( doc.isElement(), false ); + CHECK( doc.isDocument(), true ); + CHECK( doc.parentNode().isNull(), true ); + CHECK( doc.firstChild().isNull(), true ); + CHECK( doc.lastChild().isNull(), true ); + CHECK( doc.previousSibling().isNull(), true ); + CHECK( doc.nextSibling().isNull(), true ); + + // now give something as the content + CHECK( doc.setContent(&xmldevice,&errorMsg,&errorLine,&errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + // this document has something already + CHECK( doc.nodeName(), QString("#document") ); + CHECK( doc.isNull(), false ); + CHECK( doc.isElement(), false ); + CHECK( doc.isDocument(), true ); + CHECK( doc.parentNode().isNull(), true ); + CHECK( doc.firstChild().isNull(), false ); + CHECK( doc.lastChild().isNull(), false ); + CHECK( doc.previousSibling().isNull(), true ); + CHECK( doc.nextSibling().isNull(), true ); + + // make sure its children are fine + KoXmlElement rootElement; + rootElement = doc.firstChild().toElement(); + CHECK( rootElement.isNull(), false ); + CHECK( rootElement.isElement(), true ); + CHECK( rootElement.isDocument(), false ); + CHECK( rootElement.parentNode().isNull(), false ); + CHECK( rootElement.parentNode().toDocument()==doc, true ); + rootElement = doc.lastChild().toElement(); + CHECK( rootElement.isNull(), false ); + CHECK( rootElement.isElement(), true ); + CHECK( rootElement.isDocument(), false ); + CHECK( rootElement.parentNode().isNull(), false ); + CHECK( rootElement.parentNode().toDocument()==doc, true ); + + // clear() converts it into null node + doc.clear(); + CHECK( doc.nodeName(), QString::null ); + CHECK( doc.isNull(), true ); + CHECK( doc.isElement(), false ); + CHECK( doc.isDocument(), true ); + CHECK( doc.parentNode().isNull(), true ); + CHECK( doc.firstChild().isNull(), true ); + CHECK( doc.lastChild().isNull(), true ); + CHECK( doc.previousSibling().isNull(), true ); + CHECK( doc.nextSibling().isNull(), true ); + + // assigned from another empty document + doc = KoXmlDocument(); + CHECK( doc.nodeName(), QString("#document") ); + CHECK( doc.isNull(), false ); + CHECK( doc.isElement(), false ); + CHECK( doc.isDocument(), true ); + CHECK( doc.parentNode().isNull(), true ); +} + +void testNamespace() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + + // taken from example in Qt documentation (xml.html) + xmlstream << "<document xmlns:book = \"http://trolltech.com/fnord/book/\""; + xmlstream << " xmlns = \"http://trolltech.com/fnord/\" >"; + xmlstream << "<book>"; + xmlstream << " <book:title>Practical XML</book:title>"; + xmlstream << " <book:author xmlns:fnord = \"http://trolltech.com/fnord/\""; + xmlstream << " title=\"Ms\""; + xmlstream << " fnord:title=\"Goddess\""; + xmlstream << " name=\"Eris Kallisti\"/>"; + xmlstream << " <chapter>"; + xmlstream << " <title>A Namespace Called fnord</title>"; + xmlstream << " </chapter>"; + xmlstream << "</book>"; + xmlstream << "</document>"; + + KoXmlDocument doc; + KoXmlElement rootElement; + KoXmlElement bookElement; + KoXmlElement bookTitleElement; + KoXmlElement bookAuthorElement; + + // ------------- first without any namespace processing ----------- + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + rootElement = doc.documentElement(); + CHECK( rootElement.isNull(), false ); + CHECK( rootElement.isElement(), true ); + CHECK( rootElement.tagName(), QString("document") ); + CHECK( rootElement.prefix().isNull(), true ); + + bookElement = rootElement.firstChild().toElement(); + CHECK( bookElement.isNull(), false ); + CHECK( bookElement.isElement(), true ); + CHECK( bookElement.tagName(), QString("book") ); + CHECK( bookElement.prefix().isNull(), true ); + CHECK( bookElement.localName(), QString::null ); + + bookTitleElement = bookElement.firstChild().toElement(); + CHECK( bookTitleElement.isNull(), false ); + CHECK( bookTitleElement.isElement(), true ); + CHECK( bookTitleElement.tagName(), QString("book:title") ); + CHECK( bookTitleElement.prefix().isNull(), true ); + CHECK( bookTitleElement.localName(), QString::null ); + + bookAuthorElement = bookTitleElement.nextSibling().toElement(); + CHECK( bookAuthorElement.isNull(), false ); + CHECK( bookAuthorElement.isElement(), true ); + CHECK( bookAuthorElement.tagName(), QString("book:author") ); + CHECK( bookAuthorElement.prefix().isNull(), true ); + CHECK( bookAuthorElement.attribute("title"), QString("Ms") ); + CHECK( bookAuthorElement.attribute("fnord:title"), QString("Goddess") ); + CHECK( bookAuthorElement.attribute("name"), QString("Eris Kallisti") ); + + // ------------- now with namespace processing ----------- + xmldevice.at(0); // just to rewind + + CHECK( doc.setContent( &xmldevice, true, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + char* defaultNS = "http://trolltech.com/fnord/"; + char* bookNS = "http://trolltech.com/fnord/book/"; + char* fnordNS = "http://trolltech.com/fnord/"; + + // <document> + rootElement = doc.documentElement(); + CHECK( rootElement.isNull(), false ); + CHECK( rootElement.isElement(), true ); + CHECK( rootElement.tagName(), QString("document") ); + CHECK( rootElement.prefix().isEmpty(), true ); + CHECK( rootElement.namespaceURI(), QString( defaultNS ) ); + CHECK( rootElement.localName(), QString("document") ); + + // <book> + bookElement = rootElement.firstChild().toElement(); + CHECK( bookElement.isNull(), false ); + CHECK( bookElement.isElement(), true ); + CHECK( bookElement.tagName(), QString("book") ); + CHECK( bookElement.prefix().isEmpty(), true ); + CHECK( bookElement.namespaceURI(), QString( defaultNS ) ); + CHECK( bookElement.localName(), QString("book") ); + + // <book:title> + bookTitleElement = bookElement.firstChild().toElement(); + CHECK( bookTitleElement.isNull(), false ); + CHECK( bookTitleElement.isElement(), true ); + CHECK( bookTitleElement.tagName(), QString("title") ); + CHECK( bookTitleElement.prefix(), QString("book") ); + CHECK( bookTitleElement.namespaceURI(), QString(bookNS) ); + CHECK( bookTitleElement.localName(), QString("title") ); + + // another way, find it using namedItemNS() + KoXmlElement book2TitleElement; + book2TitleElement = KoXml::namedItemNS( rootElement.firstChild(), bookNS, "title" ); + //book2TitleElement = bookElement.namedItemNS( bookNS, "title" ).toElement(); + CHECK( book2TitleElement==bookTitleElement, true ); + CHECK( book2TitleElement.isNull(), false ); + CHECK( book2TitleElement.isElement(), true ); + CHECK( book2TitleElement.tagName(), QString("title") ); + + // <book:author> + bookAuthorElement = bookTitleElement.nextSibling().toElement(); + CHECK( bookAuthorElement.isNull(), false ); + CHECK( bookAuthorElement.isElement(), true ); + CHECK( bookAuthorElement.tagName(), QString("author") ); + CHECK( bookAuthorElement.prefix(), QString("book") ); + CHECK( bookAuthorElement.namespaceURI(), QString(bookNS) ); + CHECK( bookAuthorElement.localName(), QString("author") ); + + // another way, find it using namedItemNS() + KoXmlElement book2AuthorElement; + book2AuthorElement = KoXml::namedItemNS( bookElement, bookNS, "author" ); + //book2AuthorElement = bookElement.namedItemNS( bookNS, "author" ).toElement(); + CHECK( book2AuthorElement==bookAuthorElement, true ); + CHECK( book2AuthorElement.isNull(), false ); + CHECK( book2AuthorElement.isElement(), true ); + CHECK( book2AuthorElement.tagName(), QString("author") ); + + // attributes in <book:author> + // Note: with namespace processing, attribute's prefix is taken out and + // hence "fnord:title" will simply override "title" + // and searching attribute with prefix will give no result + CHECK( bookAuthorElement.hasAttribute("title"), true ); + CHECK( bookAuthorElement.hasAttribute("fnord:title"), false ); + CHECK( bookAuthorElement.hasAttribute("name"), true ); + CHECK( bookAuthorElement.attribute("title"), QString("Goddess") ); + CHECK( bookAuthorElement.attribute("fnord:title").isEmpty(), true ); + CHECK( bookAuthorElement.attribute("name"), QString("Eris Kallisti") ); + + // attributes in <book:author>, with NS family of functions + // those without prefix are not accessible at all, because they do not belong + // to any namespace at all. + // Note: default namespace does not apply to attribute names! + CHECK( bookAuthorElement.hasAttributeNS(defaultNS,"title"), true ); + CHECK( bookAuthorElement.hasAttributeNS(bookNS,"title"), false ); + CHECK( bookAuthorElement.hasAttributeNS(fnordNS,"title"), true ); + + CHECK( bookAuthorElement.attributeNS(defaultNS,"title",""), QString("Goddess") ); + CHECK( bookAuthorElement.attributeNS(bookNS,"title",""), QString("") ); + CHECK( bookAuthorElement.attributeNS(fnordNS,"title",""), QString("Goddess") ); + + CHECK( bookAuthorElement.hasAttributeNS(defaultNS,"fnord:title"), false ); + CHECK( bookAuthorElement.hasAttributeNS(bookNS,"fnord:title"), false ); + CHECK( bookAuthorElement.hasAttributeNS(fnordNS,"fnord:title"), false ); + + CHECK( bookAuthorElement.hasAttributeNS(defaultNS,"name"), false ); + CHECK( bookAuthorElement.hasAttributeNS(bookNS,"name"), false ); + CHECK( bookAuthorElement.hasAttributeNS(fnordNS,"name"), false ); + + CHECK( bookAuthorElement.attributeNS(defaultNS,"name",QString::null).isEmpty(), true ); + CHECK( bookAuthorElement.attributeNS(bookNS,"name",QString::null).isEmpty(), true ); + CHECK( bookAuthorElement.attributeNS(fnordNS,"name",QString::null).isEmpty(), true ); +} + +void testUnload() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<earth>"; + xmlstream << "<continents>"; + xmlstream << "<asia/>"; + xmlstream << "<africa/>"; + xmlstream << "<europe/>"; + xmlstream << "<america/>"; + xmlstream << "<australia/>"; + xmlstream << "<antartic/>"; + xmlstream << "</continents>"; + xmlstream << "<oceans>"; + xmlstream << "<pacific/>"; + xmlstream << "<atlantic/>"; + xmlstream << "</oceans>"; + xmlstream << "</earth>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + KoXmlElement earthElement; + earthElement = doc.documentElement().toElement(); + CHECK( earthElement.isNull(), false ); + CHECK( earthElement.isElement(), true ); + CHECK( earthElement.parentNode().isNull(), false ); + CHECK( earthElement.hasChildNodes(), true ); + CHECK( earthElement.tagName(), QString("earth") ); + CHECK( earthElement.prefix().isNull(), true ); + + // this ensures that all child nodes of <earth> are loaded + earthElement.firstChild(); + + // explicitly unload all child nodes of <earth> + KoXml::unload( earthElement ); + + // we should get the correct first child + KoXmlElement continentsElement = earthElement.firstChild().toElement(); + CHECK( continentsElement.nodeName(), QString("continents") ); + CHECK( continentsElement.isNull(), false ); + CHECK( continentsElement.isElement(), true ); + CHECK( continentsElement.isText(), false ); + CHECK( continentsElement.isDocument(), false ); + CHECK( continentsElement.hasChildNodes(), true ); + + // let us unload everything again + KoXml::unload( earthElement ); + + // we should get the correct last child + KoXmlElement oceansElement = earthElement.lastChild().toElement(); + CHECK( oceansElement.nodeName(), QString("oceans") ); + CHECK( oceansElement.isNull(), false ); + CHECK( oceansElement.isElement(), true ); + CHECK( oceansElement.isText(), false ); + CHECK( oceansElement.isDocument(), false ); + CHECK( oceansElement.hasChildNodes(), true ); +} + +void testSimpleXML() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<solarsystem>"; + xmlstream << " <mercurius/>\n"; + xmlstream << " <venus/>\n"; + xmlstream << " <earth>\n"; + xmlstream << " <moon/>\n"; + xmlstream << " </earth>\n"; + xmlstream << " <mars/>\n"; + xmlstream << " <jupiter/>\n"; + xmlstream << "</solarsystem>"; + + KoXmlDocument doc; + CHECK( doc.setContent(&xmldevice,&errorMsg,&errorLine,&errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + // <solarsystem> + KoXmlElement rootElement; + rootElement = doc.documentElement(); + CHECK( rootElement.isNull(), false ); + CHECK( rootElement.isElement(), true ); + CHECK( rootElement.parentNode().isNull(), false ); + CHECK( rootElement.hasChildNodes(), true ); + CHECK( rootElement.tagName(), QString("solarsystem") ); + CHECK( rootElement.prefix().isNull(), true ); + + // node <mercurius> + KoXmlNode firstPlanetNode; + firstPlanetNode = rootElement.firstChild(); + CHECK( firstPlanetNode.isNull(), false ); + CHECK( firstPlanetNode.isElement(), true ); + CHECK( firstPlanetNode.nextSibling().isNull(), false ); + CHECK( firstPlanetNode.previousSibling().isNull(), true ); + CHECK( firstPlanetNode.parentNode().isNull(), false ); + CHECK( firstPlanetNode.parentNode()==rootElement, true ); + CHECK( firstPlanetNode.parentNode()!=rootElement, false ); + CHECK( firstPlanetNode.hasChildNodes(), false ); + CHECK( firstPlanetNode.firstChild().isNull(), true ); + CHECK( firstPlanetNode.lastChild().isNull(), true ); + + // element <mercurius> + KoXmlElement firstPlanetElement; + firstPlanetElement = firstPlanetNode.toElement(); + CHECK( firstPlanetElement.isNull(), false ); + CHECK( firstPlanetElement.isElement(), true ); + CHECK( firstPlanetElement.parentNode().isNull(), false ); + CHECK( firstPlanetElement.parentNode()==rootElement, true ); + CHECK( firstPlanetElement.hasChildNodes(), false ); + CHECK( firstPlanetElement.firstChild().isNull(), true ); + CHECK( firstPlanetElement.lastChild().isNull(), true ); + CHECK( firstPlanetElement.tagName(), QString("mercurius") ); + CHECK( firstPlanetElement.prefix().isNull(), true ); + + // node <venus> + KoXmlNode secondPlanetNode; + secondPlanetNode = firstPlanetNode.nextSibling(); + CHECK( secondPlanetNode.isNull(), false ); + CHECK( secondPlanetNode.isElement(), true ); + CHECK( secondPlanetNode.nextSibling().isNull(), false ); + CHECK( secondPlanetNode.previousSibling().isNull(), false ); + CHECK( secondPlanetNode.previousSibling()==firstPlanetNode, true ); + CHECK( secondPlanetNode.previousSibling()==firstPlanetElement, true ); + CHECK( secondPlanetNode.parentNode().isNull(), false ); + CHECK( secondPlanetNode.parentNode()==rootElement, true ); + CHECK( secondPlanetNode.parentNode()!=rootElement, false ); + CHECK( secondPlanetNode.hasChildNodes(), false ); + CHECK( secondPlanetNode.firstChild().isNull(), true ); + CHECK( secondPlanetNode.lastChild().isNull(), true ); + + // element <venus> + KoXmlElement secondPlanetElement; + secondPlanetElement = secondPlanetNode.toElement(); + CHECK( secondPlanetElement.isNull(), false ); + CHECK( secondPlanetElement.isElement(), true ); + CHECK( secondPlanetElement.nextSibling().isNull(), false ); + CHECK( secondPlanetElement.previousSibling().isNull(), false ); + CHECK( secondPlanetElement.previousSibling()==firstPlanetNode, true ); + CHECK( secondPlanetElement.previousSibling()==firstPlanetElement, true ); + CHECK( secondPlanetElement.parentNode().isNull(), false ); + CHECK( secondPlanetElement.parentNode()==rootElement, true ); + CHECK( secondPlanetElement.parentNode()!=rootElement, false ); + CHECK( secondPlanetElement.hasChildNodes(), false ); + CHECK( secondPlanetElement.firstChild().isNull(), true ); + CHECK( secondPlanetElement.lastChild().isNull(), true ); + CHECK( secondPlanetElement.tagName(), QString("venus") ); + CHECK( secondPlanetElement.prefix().isNull(), true ); +} + +void testRootError() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + // multiple root nodes are not valid ! + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<earth></earth><moon></moon>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), false ); + CHECK( errorMsg.isEmpty(), false ); + CHECK( errorMsg, QString("unexpected character") ); + CHECK( errorLine, 1 ); + CHECK( errorColumn, 17 ); +} + +void testMismatchedTag() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + xmlstream << "<earth></e>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, &errorMsg, &errorLine, &errorColumn ), false ); + CHECK( errorMsg.isEmpty(), false ); + CHECK( errorMsg, QString("tag mismatch") ); + CHECK( errorLine, 1 ); + CHECK( errorColumn, 11 ); +} + +void testSimpleOpenDocumentText() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + + // content.xml from a simple OpenDocument text + // it has only paragraph "Hello, world!" + // automatic styles, declarations and unnecessary namespaces are omitted. + xmlstream << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"; + xmlstream << "<office:document-content "; + xmlstream << " xmlns:office=\"urn:oasis:names:tc:opendocument:xmlns:office:1.0\""; + xmlstream << " xmlns:style=\"urn:oasis:names:tc:opendocument:xmlns:style:1.0\""; + xmlstream << " xmlns:text=\"urn:oasis:names:tc:opendocument:xmlns:text:1.0\" "; + xmlstream << " office:version=\"1.0\">"; + xmlstream << " <office:automatic-styles/>"; + xmlstream << " <office:body>"; + xmlstream << " <office:text>"; + xmlstream << " <text:p text:style-name=\"Standard\">Hello, world!</text:p>"; + xmlstream << " </office:text>"; + xmlstream << " </office:body>"; + xmlstream << "</office:document-content>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, true, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + char* officeNS = "urn:oasis:names:tc:opendocument:xmlns:office:1.0"; + char* textNS = "urn:oasis:names:tc:opendocument:xmlns:text:1.0"; + + // <office:document-content> + KoXmlElement contentElement; + contentElement = doc.documentElement(); + CHECK( contentElement.isNull(), false ); + CHECK( contentElement.isElement(), true ); + CHECK( contentElement.parentNode().isNull(), false ); + CHECK( contentElement.parentNode().toDocument()==doc, true ); + CHECK( contentElement.firstChild().isNull(), false ); + CHECK( contentElement.lastChild().isNull(), false ); + CHECK( contentElement.previousSibling().isNull(), false ); + CHECK( contentElement.nextSibling().isNull(), true ); + CHECK( contentElement.localName(), QString("document-content") ); + CHECK( contentElement.hasAttributeNS(officeNS,"version"), true ); + CHECK( contentElement.attributeNS(officeNS,"version",""), QString("1.0") ); + + // <office:automatic-styles> + KoXmlElement stylesElement; + stylesElement = KoXml::namedItemNS( contentElement, officeNS, "automatic-styles" ); + CHECK( stylesElement.isNull(), false ); + CHECK( stylesElement.isElement(), true ); + CHECK( stylesElement.parentNode().isNull(), false ); + CHECK( stylesElement.parentNode()==contentElement, true ); + CHECK( stylesElement.firstChild().isNull(), true ); + CHECK( stylesElement.lastChild().isNull(), true ); + CHECK( stylesElement.previousSibling().isNull(), true ); + CHECK( stylesElement.nextSibling().isNull(), false ); + CHECK( stylesElement.localName(), QString("automatic-styles") ); + + // also same <office:automatic-styles>, but without namedItemNS + KoXmlNode styles2Element; + styles2Element = contentElement.firstChild().toElement(); + CHECK( styles2Element.isNull(), false ); + CHECK( styles2Element.isElement(), true ); + CHECK( styles2Element.parentNode().isNull(), false ); + CHECK( styles2Element.parentNode()==contentElement, true ); + CHECK( styles2Element.firstChild().isNull(), true ); + CHECK( styles2Element.lastChild().isNull(), true ); + CHECK( styles2Element.previousSibling().isNull(), true ); + CHECK( styles2Element.nextSibling().isNull(), false ); + CHECK( styles2Element.localName(), QString("automatic-styles") ); + + // <office:body> + KoXmlElement bodyElement; + bodyElement = KoXml::namedItemNS( contentElement, officeNS, "body" ); + CHECK( bodyElement.isNull(), false ); + CHECK( bodyElement.isElement(), true ); + CHECK( bodyElement.parentNode().isNull(), false ); + CHECK( bodyElement.parentNode()==contentElement, true ); + CHECK( bodyElement.firstChild().isNull(), false ); + CHECK( bodyElement.lastChild().isNull(), false ); + CHECK( bodyElement.previousSibling().isNull(), false ); + CHECK( bodyElement.nextSibling().isNull(), true ); + CHECK( bodyElement.localName(), QString("body") ); + + // also same <office:body>, but without namedItemNS + KoXmlElement body2Element; + body2Element = stylesElement.nextSibling().toElement(); + CHECK( body2Element.isNull(), false ); + CHECK( body2Element.isElement(), true ); + CHECK( body2Element.parentNode().isNull(), false ); + CHECK( body2Element.parentNode()==contentElement, true ); + CHECK( body2Element.firstChild().isNull(), false ); + CHECK( body2Element.lastChild().isNull(), false ); + CHECK( body2Element.previousSibling().isNull(), false ); + CHECK( body2Element.nextSibling().isNull(), true ); + CHECK( body2Element.localName(), QString("body") ); + + // <office:text> + KoXmlElement textElement; + textElement = KoXml::namedItemNS( bodyElement, officeNS, "text" ); + CHECK( textElement.isNull(), false ); + CHECK( textElement.isElement(), true ); + CHECK( textElement.parentNode().isNull(), false ); + CHECK( textElement.parentNode()==bodyElement, true ); + CHECK( textElement.firstChild().isNull(), false ); + CHECK( textElement.lastChild().isNull(), false ); + CHECK( textElement.previousSibling().isNull(), true ); + CHECK( textElement.nextSibling().isNull(), true ); + CHECK( textElement.localName(), QString("text") ); + + // the same <office:text>, but without namedItemNS + KoXmlElement text2Element; + text2Element = bodyElement.firstChild().toElement(); + CHECK( text2Element.isNull(), false ); + CHECK( text2Element.isElement(), true ); + CHECK( text2Element.parentNode().isNull(), false ); + CHECK( text2Element.parentNode()==bodyElement, true ); + CHECK( text2Element.firstChild().isNull(), false ); + CHECK( text2Element.lastChild().isNull(), false ); + CHECK( text2Element.previousSibling().isNull(), true ); + CHECK( text2Element.nextSibling().isNull(), true ); + CHECK( text2Element.localName(), QString("text") ); + + // <text:p> + KoXmlElement parElement; + parElement = textElement.firstChild().toElement(); + CHECK( parElement.isNull(), false ); + CHECK( parElement.isElement(), true ); + CHECK( parElement.parentNode().isNull(), false ); + CHECK( parElement.parentNode()==textElement, true ); + CHECK( parElement.firstChild().isNull(), false ); + CHECK( parElement.lastChild().isNull(), false ); + CHECK( parElement.previousSibling().isNull(), true ); + CHECK( parElement.nextSibling().isNull(), true ); + CHECK( parElement.tagName(), QString("p") ); + CHECK( parElement.text(), QString("Hello, world!") ); + CHECK( parElement.attributeNS( QString(textNS),"style-name",""), QString("Standard") ); +} + +void testSimpleOpenDocumentSpreadsheet() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + + // content.xml from a simple OpenDocument spreadsheet + // the document has three worksheets, the last two are empty. + // on the first sheet, cell A1 contains the text "Hello, world". + // automatic styles, font declarations and unnecessary namespaces are omitted. + + xmlstream << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"; + xmlstream << "<office:document-content "; + xmlstream << "xmlns:office=\"urn:oasis:names:tc:opendocument:xmlns:office:1.0\""; + xmlstream << "xmlns:text=\"urn:oasis:names:tc:opendocument:xmlns:text:1.0\" "; + xmlstream << "xmlns:table=\"urn:oasis:names:tc:opendocument:xmlns:table:1.0\">"; + xmlstream << "<office:body>"; + xmlstream << "<office:spreadsheet>"; + xmlstream << "<table:table table:name=\"Sheet1\" table:style-name=\"ta1\" table:print=\"false\">"; + xmlstream << "<table:table-column table:style-name=\"co1\" table:default-cell-style-name=\"Default\"/>"; + xmlstream << "<table:table-row table:style-name=\"ro1\">"; + xmlstream << "<table:table-cell office:value-type=\"string\">"; + xmlstream << "<text:p>Hello, world</text:p>"; + xmlstream << "</table:table-cell>"; + xmlstream << "</table:table-row>"; + xmlstream << "</table:table>"; + xmlstream << "<table:table table:name=\"Sheet2\" table:style-name=\"ta1\" table:print=\"false\">"; + xmlstream << "<table:table-column table:style-name=\"co1\" table:default-cell-style-name=\"Default\"/>"; + xmlstream << "<table:table-row table:style-name=\"ro1\">"; + xmlstream << "<table:table-cell/>"; + xmlstream << "</table:table-row>"; + xmlstream << "</table:table>"; + xmlstream << "<table:table table:name=\"Sheet3\" table:style-name=\"ta1\" table:print=\"false\">"; + xmlstream << "<table:table-column table:style-name=\"co1\" table:default-cell-style-name=\"Default\"/>"; + xmlstream << "<table:table-row table:style-name=\"ro1\">"; + xmlstream << "<table:table-cell/>"; + xmlstream << "</table:table-row>"; + xmlstream << "</table:table>"; + xmlstream << "</office:spreadsheet>"; + xmlstream << "</office:body>"; + xmlstream << "</office:document-content>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, true, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + QString officeNS = "urn:oasis:names:tc:opendocument:xmlns:office:1.0"; + QString tableNS = "urn:oasis:names:tc:opendocument:xmlns:table:1.0"; + QString textNS = "urn:oasis:names:tc:opendocument:xmlns:text:1.0"; + + // <office:document-content> + KoXmlElement contentElement; + contentElement = doc.documentElement(); + CHECK( contentElement.isNull(), false ); + CHECK( contentElement.isElement(), true ); + CHECK( contentElement.parentNode().isNull(), false ); + CHECK( contentElement.parentNode().toDocument()==doc, true ); + CHECK( contentElement.firstChild().isNull(), false ); + CHECK( contentElement.lastChild().isNull(), false ); + CHECK( contentElement.previousSibling().isNull(), false ); + CHECK( contentElement.nextSibling().isNull(), true ); + CHECK( contentElement.localName(), QString("document-content") ); + + // <office:body> + KoXmlElement bodyElement; + bodyElement = contentElement.firstChild().toElement(); + CHECK( bodyElement.isNull(), false ); + CHECK( bodyElement.isElement(), true ); + CHECK( bodyElement.parentNode().isNull(), false ); + CHECK( bodyElement.parentNode()==contentElement, true ); + CHECK( bodyElement.firstChild().isNull(), false ); + CHECK( bodyElement.lastChild().isNull(), false ); + CHECK( bodyElement.previousSibling().isNull(), true ); + CHECK( bodyElement.nextSibling().isNull(), true ); + CHECK( bodyElement.localName(), QString("body") ); + + // <office:spreadsheet> + KoXmlElement spreadsheetElement; + spreadsheetElement = bodyElement.firstChild().toElement(); + CHECK( spreadsheetElement.isNull(), false ); + CHECK( spreadsheetElement.isElement(), true ); + CHECK( spreadsheetElement.parentNode().isNull(), false ); + CHECK( spreadsheetElement.parentNode()==bodyElement, true ); + CHECK( spreadsheetElement.firstChild().isNull(), false ); + CHECK( spreadsheetElement.lastChild().isNull(), false ); + CHECK( spreadsheetElement.previousSibling().isNull(), true ); + CHECK( spreadsheetElement.nextSibling().isNull(), true ); + CHECK( spreadsheetElement.localName(), QString("spreadsheet") ); + + // <table:table> for Sheet1 + KoXmlElement sheet1Element; + sheet1Element = spreadsheetElement.firstChild().toElement(); + CHECK( sheet1Element.isNull(), false ); + CHECK( sheet1Element.isElement(), true ); + CHECK( sheet1Element.parentNode().isNull(), false ); + CHECK( sheet1Element.parentNode()==spreadsheetElement, true ); + CHECK( sheet1Element.firstChild().isNull(), false ); + CHECK( sheet1Element.lastChild().isNull(), false ); + CHECK( sheet1Element.previousSibling().isNull(), true ); + CHECK( sheet1Element.nextSibling().isNull(), false ); + CHECK( sheet1Element.tagName(), QString("table") ); + CHECK( sheet1Element.hasAttributeNS(tableNS,"name"), true ); + CHECK( sheet1Element.attributeNS(tableNS,"name",""), QString("Sheet1") ); + CHECK( sheet1Element.attributeNS(tableNS,"style-name",""), QString("ta1") ); + CHECK( sheet1Element.attributeNS(tableNS,"print",""), QString("false") ); + + KoXml::load( sheet1Element, 100 ); + + // <table:table-column> + KoXmlElement columnElement; + columnElement = sheet1Element.firstChild().toElement(); + CHECK( columnElement.isNull(), false ); + CHECK( columnElement.isElement(), true ); + CHECK( columnElement.parentNode().isNull(), false ); + CHECK( columnElement.parentNode()==sheet1Element, true ); + CHECK( columnElement.firstChild().isNull(), true ); + CHECK( columnElement.lastChild().isNull(), true ); + CHECK( columnElement.previousSibling().isNull(), true ); + CHECK( columnElement.nextSibling().isNull(), false ); + CHECK( columnElement.tagName(), QString("table-column") ); + CHECK( columnElement.attributeNS(tableNS,"style-name",""), QString("co1") ); + CHECK( columnElement.attributeNS(tableNS,"default-cell-style-name",""), QString("Default") ); + + // <table:table-row> + KoXmlElement rowElement; + rowElement = columnElement.nextSibling().toElement(); + CHECK( rowElement.isNull(), false ); + CHECK( rowElement.isElement(), true ); + CHECK( rowElement.parentNode().isNull(), false ); + CHECK( rowElement.parentNode()==sheet1Element, true ); + CHECK( rowElement.firstChild().isNull(), false ); + CHECK( rowElement.lastChild().isNull(), false ); + CHECK( rowElement.previousSibling().isNull(), false ); + CHECK( rowElement.nextSibling().isNull(), true ); + CHECK( rowElement.tagName(), QString("table-row") ); + CHECK( rowElement.attributeNS(tableNS,"style-name",""), QString("ro1") ); + + // <table:table-cell> + KoXmlElement cellElement; + cellElement = rowElement.firstChild().toElement(); + CHECK( cellElement.isNull(), false ); + CHECK( cellElement.isElement(), true ); + CHECK( cellElement.parentNode().isNull(), false ); + CHECK( cellElement.parentNode()==rowElement, true ); + CHECK( cellElement.firstChild().isNull(), false ); + CHECK( cellElement.lastChild().isNull(), false ); + CHECK( cellElement.previousSibling().isNull(), true ); + CHECK( cellElement.nextSibling().isNull(), true ); + CHECK( cellElement.tagName(), QString("table-cell") ); + CHECK( cellElement.attributeNS(officeNS,"value-type",""), QString("string") ); + + // <text:p> + KoXmlElement parElement; + parElement = cellElement.firstChild().toElement(); + CHECK( parElement.isNull(), false ); + CHECK( parElement.isElement(), true ); + CHECK( parElement.parentNode().isNull(), false ); + CHECK( parElement.parentNode()==cellElement, true ); + CHECK( parElement.firstChild().isNull(), false ); + CHECK( parElement.lastChild().isNull(), false ); + CHECK( parElement.previousSibling().isNull(), true ); + CHECK( parElement.nextSibling().isNull(), true ); + CHECK( parElement.tagName(), QString("p") ); + CHECK( parElement.text(), QString("Hello, world") ); + + // <table:table> for Sheet2 + KoXmlElement sheet2Element; + sheet2Element = sheet1Element.nextSibling().toElement(); + CHECK( sheet2Element.isNull(), false ); + CHECK( sheet2Element.isElement(), true ); + CHECK( sheet2Element.parentNode().isNull(), false ); + CHECK( sheet2Element.parentNode()==spreadsheetElement, true ); + CHECK( sheet2Element.firstChild().isNull(), false ); + CHECK( sheet2Element.lastChild().isNull(), false ); + CHECK( sheet2Element.previousSibling().isNull(), false ); + CHECK( sheet2Element.nextSibling().isNull(), false ); + CHECK( sheet2Element.tagName(), QString("table") ); + + // </table:table> for Sheet3 + KoXmlElement sheet3Element; + sheet3Element = sheet2Element.nextSibling().toElement(); + CHECK( sheet3Element.isNull(), false ); + CHECK( sheet3Element.isElement(), true ); + CHECK( sheet3Element.parentNode().isNull(), false ); + CHECK( sheet3Element.parentNode()==spreadsheetElement, true ); + CHECK( sheet3Element.firstChild().isNull(), false ); + CHECK( sheet3Element.lastChild().isNull(), false ); + CHECK( sheet3Element.previousSibling().isNull(), false ); + CHECK( sheet3Element.nextSibling().isNull(), true ); + CHECK( sheet3Element.tagName(), QString("table") ); +} + +void testSimpleOpenDocumentPresentation() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + + // content.xml from a simple OpenDocument presentation + // styles, declarations and unnecessary namespaces are omitted + // the first page is "Title" and has two text boxes + // the second page is + + xmlstream << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"; + xmlstream << "<office:document-content "; + xmlstream << " xmlns:office=\"urn:oasis:names:tc:opendocument:xmlns:office:1.0\" "; + xmlstream << " xmlns:text=\"urn:oasis:names:tc:opendocument:xmlns:text:1.0\" "; + xmlstream << " xmlns:draw=\"urn:oasis:names:tc:opendocument:xmlns:drawing:1.0\" "; + xmlstream << " xmlns:presentation=\"urn:oasis:names:tc:opendocument:xmlns:presentation:1.0\" "; + xmlstream << " xmlns:svg=\"urn:oasis:names:tc:opendocument:xmlns:svg-compatible:1.0\" "; + xmlstream << " office:version=\"1.0\">"; + xmlstream << " <office:scripts/>"; + xmlstream << " <office:automatic-styles/>"; + xmlstream << " <office:body>"; + xmlstream << " <office:presentation>"; + xmlstream << " <draw:page draw:name=\"Title\" draw:style-name=\"dp1\" "; + xmlstream << " draw:master-page-name=\"lyt-cool\" "; + xmlstream << " presentation:presentation-page-layout-name=\"AL1T0\">"; + xmlstream << " <draw:frame presentation:style-name=\"pr1\" "; + xmlstream << " draw:text-style-name=\"P2\" draw:layer=\"layout\" "; + xmlstream << " svg:width=\"23.912cm\" svg:height=\"3.508cm\" "; + xmlstream << " svg:x=\"2.058cm\" svg:y=\"1.543cm\" "; + xmlstream << " presentation:class=\"title\" "; + xmlstream << " presentation:user-transformed=\"true\">"; + xmlstream << " <draw:text-box>"; + xmlstream << " <text:p text:style-name=\"P1\">Foobar</text:p>"; + xmlstream << " </draw:text-box>"; + xmlstream << " </draw:frame>"; + xmlstream << " <draw:frame presentation:style-name=\"pr2\" "; + xmlstream << " draw:text-style-name=\"P3\" draw:layer=\"layout\""; + xmlstream << " svg:width=\"23.912cm\" svg:height=\"13.231cm\""; + xmlstream << " svg:x=\"2.058cm\" svg:y=\"5.838cm\" "; + xmlstream << " presentation:class=\"subtitle\">"; + xmlstream << " <draw:text-box>"; + xmlstream << " <text:p text:style-name=\"P3\">Foo</text:p>"; + xmlstream << " </draw:text-box>"; + xmlstream << " </draw:frame>"; + xmlstream << " <presentation:notes draw:style-name=\"dp2\">"; + xmlstream << " <draw:page-thumbnail draw:style-name=\"gr1\" draw:layer=\"layout\" svg:width=\"13.706cm\" svg:height=\"10.28cm\" svg:x=\"3.647cm\" svg:y=\"2.853cm\" draw:page-number=\"1\" presentation:class=\"page\"/>"; + xmlstream << " <draw:frame presentation:style-name=\"pr3\" draw:text-style-name=\"P1\" draw:layer=\"layout\" svg:width=\"14.518cm\" svg:height=\"11.411cm\" svg:x=\"3.249cm\" svg:y=\"14.13cm\" presentation:class=\"notes\" presentation:placeholder=\"true\">"; + xmlstream << " <draw:text-box/>"; + xmlstream << " </draw:frame>"; + xmlstream << " </presentation:notes>"; + xmlstream << " </draw:page>"; + xmlstream << " <presentation:settings presentation:stay-on-top=\"true\"/>"; + xmlstream << " </office:presentation>"; + xmlstream << " </office:body>"; + xmlstream << "</office:document-content>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, true, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + char* officeNS = "urn:oasis:names:tc:opendocument:xmlns:office:1.0"; + char* drawNS = "urn:oasis:names:tc:opendocument:xmlns:drawing:1.0"; + char* textNS = "urn:oasis:names:tc:opendocument:xmlns:text:1.0"; + char* presentationNS = "urn:oasis:names:tc:opendocument:xmlns:presentation:1.0"; + char* svgNS = "urn:oasis:names:tc:opendocument:xmlns:svg-compatible:1.0"; + + // <office:document-content> + KoXmlElement contentElement; + contentElement = doc.documentElement(); + CHECK( contentElement.isNull(), false ); + + CHECK( contentElement.isElement(), true ); + CHECK( contentElement.parentNode().isNull(), false ); + CHECK( contentElement.parentNode().toDocument()==doc, true ); + CHECK( contentElement.firstChild().isNull(), false ); + CHECK( contentElement.lastChild().isNull(), false ); + CHECK( contentElement.previousSibling().isNull(), false ); + + CHECK( contentElement.nextSibling().isNull(), true ); + CHECK( contentElement.localName(), QString("document-content") ); + CHECK( contentElement.hasAttributeNS(officeNS,"version"), true ); + CHECK( contentElement.attributeNS(officeNS,"version",""), QString("1.0") ); + + // <office:scripts> + KoXmlElement scriptsElement; + scriptsElement = KoXml::namedItemNS( contentElement, officeNS, "scripts" ); + CHECK( scriptsElement.isNull(), false ); + CHECK( scriptsElement.isElement(), true ); + CHECK( scriptsElement.parentNode().isNull(), false ); + CHECK( scriptsElement.parentNode()==contentElement, true ); + CHECK( scriptsElement.firstChild().isNull(), true ); + CHECK( scriptsElement.lastChild().isNull(), true ); + CHECK( scriptsElement.previousSibling().isNull(), true ); + CHECK( scriptsElement.nextSibling().isNull(), false ); + CHECK( scriptsElement.localName(), QString("scripts") ); + + // <office:automatic-styles> + KoXmlElement stylesElement; + stylesElement = KoXml::namedItemNS( contentElement, officeNS, "automatic-styles" ); + CHECK( stylesElement.isNull(), false ); + CHECK( stylesElement.isElement(), true ); + CHECK( stylesElement.parentNode().isNull(), false ); + CHECK( stylesElement.parentNode()==contentElement, true ); + CHECK( stylesElement.firstChild().isNull(), true ); + CHECK( stylesElement.lastChild().isNull(), true ); + CHECK( stylesElement.previousSibling().isNull(), false ); + CHECK( stylesElement.nextSibling().isNull(), false ); + CHECK( stylesElement.localName(), QString("automatic-styles") ); + + // also same <office:automatic-styles>, but without namedItemNS + KoXmlNode styles2Element; + styles2Element = scriptsElement.nextSibling().toElement(); + CHECK( styles2Element.isNull(), false ); + CHECK( styles2Element.isElement(), true ); + CHECK( styles2Element.parentNode().isNull(), false ); + CHECK( styles2Element.parentNode()==contentElement, true ); + CHECK( styles2Element.firstChild().isNull(), true ); + CHECK( styles2Element.lastChild().isNull(), true ); + CHECK( styles2Element.previousSibling().isNull(), false ); + CHECK( styles2Element.nextSibling().isNull(), false ); + CHECK( styles2Element.localName(), QString("automatic-styles") ); + + // <office:body> + KoXmlElement bodyElement; + bodyElement = KoXml::namedItemNS( contentElement, officeNS, "body" ); + CHECK( bodyElement.isNull(), false ); + CHECK( bodyElement.isElement(), true ); + CHECK( bodyElement.parentNode().isNull(), false ); + CHECK( bodyElement.parentNode()==contentElement, true ); + CHECK( bodyElement.firstChild().isNull(), false ); + CHECK( bodyElement.lastChild().isNull(), false ); + CHECK( bodyElement.previousSibling().isNull(), false ); + CHECK( bodyElement.nextSibling().isNull(), true ); + CHECK( bodyElement.localName(), QString("body") ); + + // also same <office:body>, but without namedItemNS + KoXmlElement body2Element; + body2Element = stylesElement.nextSibling().toElement(); + CHECK( body2Element.isNull(), false ); + CHECK( body2Element.isElement(), true ); + CHECK( body2Element.parentNode().isNull(), false ); + CHECK( body2Element.parentNode()==contentElement, true ); + CHECK( body2Element.firstChild().isNull(), false ); + CHECK( body2Element.lastChild().isNull(), false ); + CHECK( body2Element.previousSibling().isNull(), false ); + CHECK( body2Element.nextSibling().isNull(), true ); + CHECK( body2Element.localName(), QString("body") ); + + // <office:presentation> + KoXmlElement presentationElement; + presentationElement = KoXml::namedItemNS( bodyElement, officeNS, "presentation" ); + CHECK( presentationElement.isNull(), false ); + CHECK( presentationElement.isElement(), true ); + CHECK( presentationElement.parentNode().isNull(), false ); + CHECK( presentationElement.parentNode()==bodyElement, true ); + CHECK( presentationElement.firstChild().isNull(), false ); + CHECK( presentationElement.lastChild().isNull(), false ); + CHECK( presentationElement.previousSibling().isNull(), true ); + CHECK( presentationElement.nextSibling().isNull(), true ); + CHECK( presentationElement.localName(), QString("presentation") ); + + // the same <office:presentation>, but without namedItemNS + KoXmlElement presentation2Element; + presentation2Element = bodyElement.firstChild().toElement(); + CHECK( presentation2Element.isNull(), false ); + CHECK( presentation2Element.isElement(), true ); + CHECK( presentation2Element.parentNode().isNull(), false ); + CHECK( presentation2Element.parentNode()==bodyElement, true ); + CHECK( presentation2Element.firstChild().isNull(), false ); + CHECK( presentation2Element.lastChild().isNull(), false ); + CHECK( presentation2Element.previousSibling().isNull(), true ); + CHECK( presentation2Element.nextSibling().isNull(), true ); + CHECK( presentation2Element.localName(), QString("presentation") ); + + // <draw:page> for "Title" + KoXmlElement titlePageElement; + titlePageElement = presentationElement.firstChild().toElement(); + CHECK( titlePageElement.isNull(), false ); + CHECK( titlePageElement.isElement(), true ); + CHECK( titlePageElement.parentNode().isNull(), false ); + CHECK( titlePageElement.parentNode()==presentationElement, true ); + CHECK( titlePageElement.firstChild().isNull(), false ); + CHECK( titlePageElement.lastChild().isNull(), false ); + CHECK( titlePageElement.previousSibling().isNull(), true ); + CHECK( titlePageElement.nextSibling().isNull(), false ); + CHECK( titlePageElement.localName(), QString("page") ); + CHECK( titlePageElement.attributeNS(drawNS,"name",""), QString("Title") ); + CHECK( titlePageElement.attributeNS(drawNS,"style-name",""), QString("dp1") ); + CHECK( titlePageElement.attributeNS(drawNS,"master-page-name",""), QString("lyt-cool") ); + CHECK( titlePageElement.attributeNS(presentationNS, + "presentation-page-layout-name",""), QString("AL1T0") ); + + // <draw:frame> for the title frame + KoXmlElement titleFrameElement; + titleFrameElement = titlePageElement.firstChild().toElement(); + CHECK( titleFrameElement.isNull(), false ); + CHECK( titleFrameElement.isElement(), true ); + CHECK( titleFrameElement.parentNode().isNull(), false ); + CHECK( titleFrameElement.parentNode()==titlePageElement, true ); + CHECK( titleFrameElement.firstChild().isNull(), false ); + CHECK( titleFrameElement.lastChild().isNull(), false ); + CHECK( titleFrameElement.previousSibling().isNull(), true ); + CHECK( titleFrameElement.nextSibling().isNull(), false ); + CHECK( titleFrameElement.localName(), QString("frame") ); + CHECK( titleFrameElement.attributeNS(presentationNS,"style-name",""), QString("pr1") ); + CHECK( titleFrameElement.attributeNS(presentationNS,"class",""), QString("title") ); + CHECK( titleFrameElement.attributeNS(presentationNS,"user-transformed",""), QString("true") ); + CHECK( titleFrameElement.attributeNS(drawNS,"text-style-name",""), QString("P2") ); + CHECK( titleFrameElement.attributeNS(drawNS,"layer",""), QString("layout") ); + CHECK( titleFrameElement.attributeNS(svgNS,"width",""), QString("23.912cm") ); + CHECK( titleFrameElement.attributeNS(svgNS,"height",""), QString("3.508cm") ); + CHECK( titleFrameElement.attributeNS(svgNS,"x",""), QString("2.058cm") ); + CHECK( titleFrameElement.attributeNS(svgNS,"y",""), QString("1.543cm") ); + + // <draw:text-box> of the title frame + KoXmlElement titleBoxElement; + titleBoxElement = titleFrameElement.firstChild().toElement(); + CHECK( titleBoxElement.isNull(), false ); + CHECK( titleBoxElement.isElement(), true ); + CHECK( titleBoxElement.parentNode().isNull(), false ); + CHECK( titleBoxElement.parentNode()==titleFrameElement, true ); + CHECK( titleBoxElement.firstChild().isNull(), false ); + CHECK( titleBoxElement.lastChild().isNull(), false ); + CHECK( titleBoxElement.previousSibling().isNull(), true ); + CHECK( titleBoxElement.nextSibling().isNull(), true ); + CHECK( titleBoxElement.localName(), QString("text-box") ); + + // <text:p> for the title text-box + KoXmlElement titleParElement; + titleParElement = titleBoxElement.firstChild().toElement(); + CHECK( titleParElement.isNull(), false ); + CHECK( titleParElement.isElement(), true ); + CHECK( titleParElement.parentNode().isNull(), false ); + CHECK( titleParElement.parentNode()==titleBoxElement, true ); + CHECK( titleParElement.firstChild().isNull(), false ); + CHECK( titleParElement.lastChild().isNull(), false ); + CHECK( titleParElement.previousSibling().isNull(), true ); + CHECK( titleParElement.nextSibling().isNull(), true ); + CHECK( titleParElement.localName(), QString("p") ); + CHECK( titleParElement.attributeNS(textNS,"style-name",""), QString("P1") ); + CHECK( titleParElement.text(), QString("Foobar") ); + + // <draw:frame> for the subtitle frame + KoXmlElement subtitleFrameElement; + subtitleFrameElement = titleFrameElement.nextSibling().toElement(); + CHECK( subtitleFrameElement.isNull(), false ); + CHECK( subtitleFrameElement.isElement(), true ); + CHECK( subtitleFrameElement.parentNode().isNull(), false ); + CHECK( subtitleFrameElement.parentNode()==titlePageElement, true ); + CHECK( subtitleFrameElement.firstChild().isNull(), false ); + CHECK( subtitleFrameElement.lastChild().isNull(), false ); + CHECK( subtitleFrameElement.previousSibling().isNull(), false ); + CHECK( subtitleFrameElement.nextSibling().isNull(), false ); + CHECK( subtitleFrameElement.localName(), QString("frame") ); + CHECK( subtitleFrameElement.attributeNS(presentationNS,"style-name",""), QString("pr2") ); + CHECK( subtitleFrameElement.attributeNS(presentationNS,"class",""), QString("subtitle") ); + CHECK( subtitleFrameElement.hasAttributeNS(presentationNS,"user-transformed"), false ); + CHECK( subtitleFrameElement.attributeNS(drawNS,"text-style-name",""), QString("P3") ); + CHECK( subtitleFrameElement.attributeNS(drawNS,"layer",""), QString("layout") ); + CHECK( subtitleFrameElement.attributeNS(svgNS,"width",""), QString("23.912cm") ); + CHECK( subtitleFrameElement.attributeNS(svgNS,"height",""), QString("13.231cm") ); + CHECK( subtitleFrameElement.attributeNS(svgNS,"x",""), QString("2.058cm") ); + CHECK( subtitleFrameElement.attributeNS(svgNS,"y",""), QString("5.838cm") ); + + // <draw:text-box> of the subtitle frame + KoXmlElement subtitleBoxElement; + subtitleBoxElement = subtitleFrameElement.firstChild().toElement(); + CHECK( subtitleBoxElement.isNull(), false ); + CHECK( subtitleBoxElement.isElement(), true ); + CHECK( subtitleBoxElement.parentNode().isNull(), false ); + CHECK( subtitleBoxElement.parentNode()==subtitleFrameElement, true ); + CHECK( subtitleBoxElement.firstChild().isNull(), false ); + CHECK( subtitleBoxElement.lastChild().isNull(), false ); + CHECK( subtitleBoxElement.previousSibling().isNull(), true ); + CHECK( subtitleBoxElement.nextSibling().isNull(), true ); + CHECK( subtitleBoxElement.localName(), QString("text-box") ); + + // <text:p> for the subtitle text-box + KoXmlElement subtitleParElement; + subtitleParElement = subtitleBoxElement.firstChild().toElement(); + CHECK( subtitleParElement.isNull(), false ); + CHECK( subtitleParElement.isElement(), true ); + CHECK( subtitleParElement.parentNode().isNull(), false ); + CHECK( subtitleParElement.parentNode()==subtitleBoxElement, true ); + CHECK( subtitleParElement.firstChild().isNull(), false ); + CHECK( subtitleParElement.lastChild().isNull(), false ); + CHECK( subtitleParElement.previousSibling().isNull(), true ); + CHECK( subtitleParElement.nextSibling().isNull(), true ); + CHECK( subtitleParElement.localName(), QString("p") ); + CHECK( subtitleParElement.attributeNS(textNS,"style-name",""), QString("P3") ); + CHECK( subtitleParElement.text(), QString("Foo") ); +} + +void testSimpleOpenDocumentFormula() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + + // content.xml from a simple OpenDocument formula + // this is essentially MathML + xmlstream << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"; + xmlstream << "<!DOCTYPE math:math PUBLIC \"-//OpenOffice.org//DTD Modified W3C MathML 1.01//EN\" \"math.dtd\">"; + xmlstream << "<math:math xmlns:math=\"http://www.w3.org/1998/Math/MathML\">"; + xmlstream << " <math:semantics>"; + xmlstream << " <math:mrow>"; + xmlstream << " <math:mi>E</math:mi>"; + xmlstream << " <math:mo math:stretchy=\"false\">=</math:mo>"; + xmlstream << " <math:msup>"; + xmlstream << " <math:mi math:fontstyle=\"italic\">mc</math:mi>"; + xmlstream << " <math:mn>2</math:mn>"; + xmlstream << " </math:msup>"; + xmlstream << " </math:mrow>"; + xmlstream << " <math:annotation math:encoding=\"StarMath 5.0\">E = mc^2 </math:annotation>"; + xmlstream << " </math:semantics>"; + xmlstream << "</math:math>"; + + KoXmlDocument doc; + CHECK( doc.setContent( &xmldevice, true, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + char* mathNS = "http://www.w3.org/1998/Math/MathML"; + + // <math:math> + KoXmlElement mathElement; + mathElement = doc.documentElement(); + CHECK( mathElement.isNull(), false ); + CHECK( mathElement.isElement(), true ); + CHECK( mathElement.parentNode().isNull(), false ); + CHECK( mathElement.parentNode().toDocument()==doc, true ); + CHECK( mathElement.firstChild().isNull(), false ); + CHECK( mathElement.lastChild().isNull(), false ); + CHECK( mathElement.previousSibling().isNull(), false ); + CHECK( mathElement.nextSibling().isNull(), true ); + CHECK( mathElement.localName(), QString("math") ); + + // <math:semantics> + KoXmlElement semanticsElement; + semanticsElement = KoXml::namedItemNS( mathElement, mathNS, "semantics" ); + CHECK( semanticsElement.isNull(), false ); + CHECK( semanticsElement.isElement(), true ); + CHECK( semanticsElement.parentNode().isNull(), false ); + CHECK( semanticsElement.parentNode().toElement()==mathElement, true ); + CHECK( semanticsElement.firstChild().isNull(), false ); + CHECK( semanticsElement.lastChild().isNull(), false ); + CHECK( semanticsElement.previousSibling().isNull(), true ); + CHECK( semanticsElement.nextSibling().isNull(), true ); + CHECK( semanticsElement.localName(), QString("semantics") ); + + // the same <math:semantics> but without namedItemNS + KoXmlElement semantics2Element; + semantics2Element = mathElement.firstChild().toElement(); + CHECK( semantics2Element.isNull(), false ); + CHECK( semantics2Element.isElement(), true ); + CHECK( semantics2Element.parentNode().isNull(), false ); + CHECK( semantics2Element.parentNode().toElement()==mathElement, true ); + CHECK( semantics2Element.firstChild().isNull(), false ); + CHECK( semantics2Element.lastChild().isNull(), false ); + CHECK( semantics2Element.previousSibling().isNull(), true ); + CHECK( semantics2Element.nextSibling().isNull(), true ); + CHECK( semantics2Element.localName(), QString("semantics") ); + + // <math:mrow> + KoXmlElement mrowElement; + mrowElement = semanticsElement.firstChild().toElement(); + CHECK( mrowElement.isNull(), false ); + CHECK( mrowElement.isElement(), true ); + CHECK( mrowElement.parentNode().isNull(), false ); + CHECK( mrowElement.parentNode().toElement()==semanticsElement, true ); + CHECK( mrowElement.firstChild().isNull(), false ); + CHECK( mrowElement.lastChild().isNull(), false ); + CHECK( mrowElement.previousSibling().isNull(), true ); + CHECK( mrowElement.nextSibling().isNull(), false ); + CHECK( mrowElement.localName(), QString("mrow") ); + + // <math:mi> for "E" + KoXmlElement miElement; + miElement = mrowElement.firstChild().toElement(); + CHECK( miElement.isNull(), false ); + CHECK( miElement.isElement(), true ); + CHECK( miElement.parentNode().isNull(), false ); + CHECK( miElement.parentNode().toElement()==mrowElement, true ); + CHECK( miElement.firstChild().isNull(), false ); + CHECK( miElement.lastChild().isNull(), false ); + CHECK( miElement.previousSibling().isNull(), true ); + CHECK( miElement.nextSibling().isNull(), false ); + CHECK( miElement.localName(), QString("mi") ); + + // <math:mo> for "=" + KoXmlElement moElement; + moElement = miElement.nextSibling().toElement(); + CHECK( moElement.isNull(), false ); + CHECK( moElement.isElement(), true ); + CHECK( moElement.parentNode().isNull(), false ); + CHECK( moElement.parentNode().toElement()==mrowElement, true ); + CHECK( moElement.firstChild().isNull(), false ); + CHECK( moElement.lastChild().isNull(), false ); + CHECK( moElement.previousSibling().isNull(), false ); + CHECK( moElement.nextSibling().isNull(), false ); + CHECK( moElement.localName(), QString("mo") ); + CHECK( moElement.attributeNS(mathNS,"stretchy",""), QString("false") ); + + // <math:msup> for "mc" and superscripted "2" + KoXmlElement msupElement; + msupElement = moElement.nextSibling().toElement(); + CHECK( msupElement.isNull(), false ); + CHECK( msupElement.isElement(), true ); + CHECK( msupElement.parentNode().isNull(), false ); + CHECK( msupElement.parentNode().toElement()==mrowElement, true ); + CHECK( msupElement.firstChild().isNull(), false ); + CHECK( msupElement.lastChild().isNull(), false ); + CHECK( msupElement.previousSibling().isNull(), false ); + CHECK( msupElement.nextSibling().isNull(), true ); + CHECK( msupElement.localName(), QString("msup") ); + + // <math:mi> inside the <math:msup> for "mc" + KoXmlElement mcElement; + mcElement = msupElement.firstChild().toElement(); + CHECK( mcElement.isNull(), false ); + CHECK( mcElement.isElement(), true ); + CHECK( mcElement.parentNode().isNull(), false ); + CHECK( mcElement.parentNode().toElement()==msupElement, true ); + CHECK( mcElement.firstChild().isNull(), false ); + CHECK( mcElement.lastChild().isNull(), false ); + CHECK( mcElement.previousSibling().isNull(), true ); + CHECK( mcElement.nextSibling().isNull(), false ); + CHECK( mcElement.localName(), QString("mi") ); + CHECK( mcElement.text(), QString("mc") ); + CHECK( mcElement.attributeNS(mathNS,"fontstyle",""), QString("italic") ); + + // <math:mn> inside the <math:msup> for "2" (superscript) + KoXmlElement mnElement; + mnElement = mcElement.nextSibling().toElement(); + CHECK( mnElement.isNull(), false ); + CHECK( mnElement.isElement(), true ); + CHECK( mnElement.parentNode().isNull(), false ); + CHECK( mnElement.parentNode().toElement()==msupElement, true ); + CHECK( mnElement.firstChild().isNull(), false ); + CHECK( mnElement.lastChild().isNull(), false ); + CHECK( mnElement.previousSibling().isNull(), false ); + CHECK( mnElement.nextSibling().isNull(), true ); + CHECK( mnElement.localName(), QString("mn") ); + CHECK( mnElement.text(), QString("2") ); + + // <math:annotation> + KoXmlElement annotationElement; + annotationElement = semanticsElement.lastChild().toElement(); + CHECK( annotationElement.isNull(), false ); + CHECK( annotationElement.isElement(), true ); + CHECK( annotationElement.parentNode().isNull(), false ); + CHECK( annotationElement.parentNode().toElement()==semanticsElement, true ); + CHECK( annotationElement.firstChild().isNull(), false ); + CHECK( annotationElement.lastChild().isNull(), false ); + CHECK( annotationElement.previousSibling().isNull(), false ); + CHECK( annotationElement.nextSibling().isNull(), true ); + CHECK( annotationElement.localName(), QString("annotation") ); + CHECK( annotationElement.text(), QString("E = mc^2 ") ); + CHECK( annotationElement.attributeNS(mathNS,"encoding",""), QString("StarMath 5.0") ); +} + +void testLargeOpenDocumentSpreadsheet() +{ + QString errorMsg; + int errorLine = 0; + int errorColumn = 0; + + int sheetCount = 4; + int rowCount = 200; + int colCount = 200*2; + + QByteArray xmlbuf; + QBuffer xmldevice( xmlbuf ); + QTextStream xmlstream( xmlbuf, IO_WriteOnly ); + + // content.xml + xmlstream << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"; + xmlstream << "<office:document-content "; + xmlstream << "xmlns:office=\"urn:oasis:names:tc:opendocument:xmlns:office:1.0\" "; + xmlstream << "xmlns:text=\"urn:oasis:names:tc:opendocument:xmlns:text:1.0\" "; + xmlstream << "xmlns:table=\"urn:oasis:names:tc:opendocument:xmlns:table:1.0\" >"; + xmlstream << "<office:body>"; + xmlstream << "<office:spreadsheet>"; + for( int i = 0; i < sheetCount; i++ ) + { + QString sheetName = QString("Sheet%1").arg(i+1); + xmlstream << "<table:table table:name=\"" << sheetName; + xmlstream << "\" table:print=\"false\">"; + for( int j = 0; j < rowCount; j++ ) + { + xmlstream << "<table:table-row>"; + for( int k = 0; k < colCount; k++ ) + { + xmlstream << "<table:table-cell office:value-type=\"string\">"; + xmlstream << "<text:p>Hello, world</text:p>"; + xmlstream << "</table:table-cell>"; + } + xmlstream << "</table:table-row>"; + } + xmlstream << "</table:table>"; + } + xmlstream << "</office:spreadsheet>"; + xmlstream << "</office:body>"; + xmlstream << "</office:document-content>"; + + printf("Raw XML size: %d KB\n", xmlbuf.size()/1024 ); + + QTime timer; + timer.start(); + +#if 1 + // just to test parsing speed with plain dumb handler + QXmlSimpleReader* reader = new QXmlSimpleReader; + reader->setFeature( "http://xml.org/sax/features/namespaces", true ); + QXmlDefaultHandler handler; + reader->setContentHandler( &handler ); + reader->setErrorHandler( &handler ); + reader->setLexicalHandler( &handler ); + reader->setDeclHandler( &handler ); + reader->setDTDHandler( &handler ); + QXmlInputSource xmlSource; + xmlSource.setData( QString::fromUtf8( xmlbuf.data(), xmlbuf.size() ) ); + timer.start(); + reader->parse( &xmlSource ); + printf("Large spreadsheet: QXmlDefaultHandler parsing time is %d ms\n", timer.elapsed() ); + delete reader; +#endif + + timer.start(); + KoXmlDocument doc; + + // uncomment to see the performance if on-demand/lazy loading is not used + // will definitely eat more memory +#ifndef KOXML_USE_QDOM + doc.setFastLoading( true ); +#endif + + CHECK( doc.setContent( &xmldevice, true, &errorMsg, &errorLine, &errorColumn ), true ); + CHECK( errorMsg.isEmpty(), true ); + CHECK( errorLine, 0 ); + CHECK( errorColumn, 0 ); + + printf("Large spreadsheet: parsing time is %d ms\n", timer.elapsed() ); + + // release memory taken by the XML document content + xmlbuf.resize( 0 ); + + // namespaces that will be used + QString officeNS = "urn:oasis:names:tc:opendocument:xmlns:office:1.0"; + QString tableNS = "urn:oasis:names:tc:opendocument:xmlns:table:1.0"; + QString textNS = "urn:oasis:names:tc:opendocument:xmlns:text:1.0"; + + // <office:document-content> + KoXmlElement contentElement; + contentElement = doc.documentElement(); + CHECK( contentElement.isNull(), false ); + CHECK( contentElement.isElement(), true ); + CHECK( contentElement.localName(), QString("document-content") ); + + // <office:body> + KoXmlElement bodyElement; + bodyElement = contentElement.firstChild().toElement(); + CHECK( bodyElement.isNull(), false ); + CHECK( bodyElement.isElement(), true ); + CHECK( bodyElement.localName(), QString("body") ); + + // <office:spreadsheet> + KoXmlElement spreadsheetElement; + spreadsheetElement = bodyElement.firstChild().toElement(); + CHECK( spreadsheetElement.isNull(), false ); + CHECK( spreadsheetElement.isElement(), true ); + CHECK( spreadsheetElement.localName(), QString("spreadsheet") ); + + // now we visit every sheet, every row, every cell + timer.start(); + KoXmlElement tableElement; + tableElement = spreadsheetElement.firstChild().toElement(); + for( int table=0; table < sheetCount; table++ ) + { + QString tableName = QString("Sheet%1").arg(table+1); + CHECK( tableElement.isNull(), false ); + CHECK( tableElement.isElement(), true ); + CHECK( tableElement.localName(), QString("table") ); + CHECK( tableElement.hasAttributeNS(tableNS,"name"), true ); + CHECK( tableElement.attributeNS(tableNS,"name",""), tableName ); + CHECK( tableElement.attributeNS(tableNS,"print",""), QString("false") ); + + // load everything for this table + KoXml::load( tableElement, 99 ); + + CHECK( tableElement.parentNode().isNull(), false ); + CHECK( tableElement.parentNode()==spreadsheetElement, true ); + CHECK( tableElement.firstChild().isNull(), false ); + CHECK( tableElement.lastChild().isNull(), false ); + + KoXmlElement rowElement; + rowElement = tableElement.firstChild().toElement(); + for( int row=0; row < rowCount; row++ ) + { + CHECK( rowElement.isNull(), false ); + CHECK( rowElement.isElement(), true ); + CHECK( rowElement.localName(), QString("table-row") ); + CHECK( rowElement.parentNode().isNull(), false ); + CHECK( rowElement.parentNode()==tableElement, true ); + CHECK( rowElement.firstChild().isNull(), false ); + CHECK( rowElement.lastChild().isNull(), false ); + + KoXmlElement cellElement; + cellElement = rowElement.firstChild().toElement(); + for( int col=0; col < colCount; col++ ) + { + CHECK( cellElement.isNull(), false ); + CHECK( cellElement.isElement(), true ); + CHECK( cellElement.localName(), QString("table-cell") ); + CHECK( cellElement.text(), QString("Hello, world") ); + CHECK( cellElement.hasAttributeNS(officeNS,"value-type"), true ); + CHECK( cellElement.attributeNS(officeNS,"value-type",""), QString("string" ) ); + CHECK( cellElement.parentNode().isNull(), false ); + CHECK( cellElement.parentNode()==rowElement, true ); + CHECK( cellElement.firstChild().isNull(), false ); + CHECK( cellElement.lastChild().isNull(), false ); + cellElement = cellElement.nextSibling().toElement(); + } + + //KoXml::unload( rowElement ); + rowElement = rowElement.nextSibling().toElement(); + } + + KoXml::unload( tableElement ); + tableElement = tableElement.nextSibling().toElement(); + } + + printf("Large spreadsheet: iterating time is %d ms\n", timer.elapsed() ); +} + +int main( int argc, char** argv ) +{ + Q_UNUSED( argc ); + Q_UNUSED( argv ); + + testNode(); + testElement(); + testAttributes(); + testText(); + testCDATA(); + testDocument(); + testNamespace(); + + testUnload(); + + testSimpleXML(); + testRootError(); + testMismatchedTag(); + + testSimpleOpenDocumentText(); + testSimpleOpenDocumentSpreadsheet(); + testSimpleOpenDocumentPresentation(); + testSimpleOpenDocumentFormula(); + +// testLargeOpenDocumentSpreadsheet(); + +#ifdef KOXML_USE_QDOM + printf("Using QDom: "); +#else + printf("Using KoXml: "); +#endif + printf("%d tests, %d failed\n", testCount, testFailed ); + + return testFailed; +} diff --git a/lib/kofficecore/tests/priorityqueue_test.cpp b/lib/kofficecore/tests/priorityqueue_test.cpp new file mode 100644 index 00000000..d2c7eafd --- /dev/null +++ b/lib/kofficecore/tests/priorityqueue_test.cpp @@ -0,0 +1,87 @@ +/* This file is part of the KDE project + Copyright (C) 2001 Werner Trobin <trobin@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 <priorityqueue.h> +#include <kdebug.h> +#include <qptrlist.h> +#include <qasciidict.h> +#include <stdlib.h> +#include <time.h> + +struct Node { + Node( unsigned int key ) : m_key( key ), m_index( 0 ) {} + + unsigned int key() const { return m_key; } + void setKey( unsigned int key ) { m_key = key; } + + int index() const { return m_index; } + void setIndex( int i ) { m_index = i; } +private: + unsigned int m_key; + int m_index; +}; + +static const char* const keys[] = { "one", "two", "three", "four", "five", + "six", "seven", "eight", "nine", "ten", + "eleven", "twelve", 0 }; + +int main( int /*argc*/, char **/*argv*/ ) +{ + QPtrList<Node> list; + list.setAutoDelete( true ); + QAsciiDict<Node> dict; + + KOffice::PriorityQueue<Node> queue; + + srand( time( 0 ) ); + for ( int i = 0; i < 12; ++i ) { + Node *n = new Node( rand() % 20 ); + list.append( n ); + queue.insert( n ); + // Check whether the AsciiDict CTOR is okay + Node *n2 = new Node( *n ); + dict.insert( keys[ i ], n2 ); + } + + kdDebug() << "##### Queue 1: " << endl; + queue.dump(); + + kdDebug() << "##### Queue 2: " << endl; + KOffice::PriorityQueue<Node> queue2( dict ); + queue2.dump(); + + Node *n = list.at( 6 ); + kdDebug() << "##### Decreasing node: " << n->key() << " at " << n->index() << endl; + n->setKey( 2 ); + queue.keyDecreased( n ); + queue.dump(); + + n = list.at( 2 ); + kdDebug() << "##### Decreasing node: " << n->key() << " at " << n->index() << endl; + n->setKey( 0 ); + queue.keyDecreased( n ); + queue.dump(); + + n = queue.extractMinimum(); + while ( n ) { + queue.dump(); + n = queue.extractMinimum(); + } + return 0; +} |