You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
koffice/kivio/kiviopart/kivio_doc.cpp

1074 lines
28 KiB

/*
* Kivio - Visual Modelling and Flowcharting
* Copyright (C) 2000-2001 theKompany.com & Dave Marotti
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include <tqdom.h>
#include <tqtextstream.h>
#include <tqbuffer.h>
#include <tqtabwidget.h>
#include <tqpaintdevicemetrics.h>
#include "kivio_doc.h"
#include "kivio_page.h"
#include "kivio_map.h"
#include "kivio_view.h"
#include "kivio_factory.h"
#include "export_page_dialog.h"
#include "kivio_common.h"
#include "kivio_group_stencil.h"
#include "kivio_icon_view.h"
#include "kivio_layer.h"
#include "kivio_painter.h"
#include "kivio_screen_painter.h"
#include "kivio_stencil.h"
#include "kivio_stencil_spawner_set.h"
#include "kivioglobal.h"
#include "kivio_config.h"
#include "polylineconnectorspawner.h"
#include "kivio_canvas.h"
#include "stencilbarbutton.h"
#include <unistd.h>
#include <kmessagebox.h>
#include <klocale.h>
#include <kprinter.h>
#include <kdebug.h>
#include <kurl.h>
#include <kapplication.h>
#include <assert.h>
#include <tqdatetime.h>
#include <sys/types.h>
#include <dirent.h>
#include <sys/stat.h>
#include <pwd.h>
#include <tqdir.h>
#include <tqfileinfo.h>
#include <tqfile.h>
#include <tqpixmap.h>
#include <tqpainter.h>
#include <tqpen.h>
#include <tqfont.h>
#include <tqvaluelist.h>
#include <kstandarddirs.h>
#include <kpopupmenu.h>
#include <kmenubar.h>
#include <kiconloader.h>
#include <khelpmenu.h>
#include <kconfig.h>
#include <KoTemplateChooseDia.h>
#include <KoFilterManager.h>
#include <KoStoreDevice.h>
#include "KIvioDocIface.h"
#include <kcommand.h>
#include <KoZoomHandler.h>
#include <KoApplication.h>
#include <kglobal.h>
#include <KoCommandHistory.h>
#include <KoXmlWriter.h>
#include <KoGenStyles.h>
#include <KoOasisSettings.h>
#include <KoDom.h>
#include <KoXmlNS.h>
//using namespace std;
/*****************************************************************************
*
* KivioDoc
*
*****************************************************************************/
TQPtrList<KivioDoc>* KivioDoc::s_docs = 0;
int KivioDoc::s_docId = 0;
KivioDoc::KivioDoc( TQWidget *parentWidget, const char* widgetName, TQObject* tqparent, const char* name, bool singleViewMode )
: KoDocument( parentWidget, widgetName, tqparent, name, singleViewMode )
{
dcop = 0;
if (!s_docs)
s_docs = new TQPtrList<KivioDoc>;
s_docs->append(this);
m_pLstSpawnerSets = new TQPtrList<KivioStencilSpawnerSet>;
m_pLstSpawnerSets->setAutoDelete(true);
m_loadTimer = 0;
m_currentFile = 0;
setInstance( KivioFactory::global(), false );
setTemplateType("kivio_template");
if ( !name )
{
TQString tmp( "Document%1" ); //lukas: FIXME
tmp = tmp.tqarg( s_docId++ );
setName( tmp.latin1() );
}
m_iPageId = 1;
m_bLoading = false;
m_pMap = new KivioMap( this, "Map" );
// Load autoLoadStencils in internal StencilSpawnerSet
m_pInternalSet = new KivioStencilSpawnerSet("Kivio_Internal");
m_pInternalSet->setId("Kivio - Internal - Do Not Touch");
TQStringList list = instance()->dirs()->findAllResources("data",instance()->instanceName()+"/autoloadStencils/*",true,false);
TQStringList::ConstIterator pIt = list.begin();
TQStringList::ConstIterator pEnd = list.end();
for (; pIt != pEnd; ++pIt )
{
m_pInternalSet->loadFile(*pIt);
}
// Add the polyline connector spawner to the internal stencil set.
addInternalStencilSpawner(new Kivio::PolyLineConnectorSpawner(internalSpawnerSet()));
initConfig();
//laurent: Why don't use menu history for undo/redo command ? sync with other koffice application
m_commandHistory = new KoCommandHistory( actionCollection(), /*false*/true ) ;
connect( m_commandHistory, TQT_SIGNAL( documentRestored() ), this, TQT_SLOT( slotDocumentRestored() ) );
connect( m_commandHistory, TQT_SIGNAL( commandExecuted() ), this, TQT_SLOT( slotCommandExecuted() ) );
if ( name )
dcopObject();
}
DCOPObject* KivioDoc::dcopObject()
{
if ( !dcop ) {
dcop = new KIvioDocIface( this );
}
return dcop;
}
TQPtrList<KivioDoc>& KivioDoc::documents()
{
if ( s_docs == 0 ) {
s_docs = new TQPtrList<KivioDoc>;
}
return *s_docs;
}
bool KivioDoc::initDoc(InitDocFlags flags, TQWidget* parentWidget)
{
KivioPage *t = createPage();
m_pMap->addPage( t );
m_docOpened = false; // Used to for a hack that make kivio not crash if you cancel startup dialog.
if(flags == KoDocument::InitDocEmpty) {
setEmpty();
m_docOpened = true; // Used to for a hack that make kivio not crash if you cancel startup dialog.
return true;
}
TQString f;
KoTemplateChooseDia::ReturnType ret;
KoTemplateChooseDia::DialogType dlgtype;
if(flags != KoDocument::InitDocFileNew) {
dlgtype = KoTemplateChooseDia::Everything;
initConfig();
} else {
dlgtype = KoTemplateChooseDia::OnlyTemplates;
}
ret = KoTemplateChooseDia::choose( KivioFactory::global(), f,
dlgtype, "kivio_template", parentWidget );
if( ret == KoTemplateChooseDia::File ) {
KURL url(f);
bool ok = openURL(url);
m_docOpened = ok; // Used to for a hack that make kivio not crash if you cancel startup dialog.
return ok;
} else if ( ret == KoTemplateChooseDia::Template ) {
TQFileInfo fileInfo( f );
TQString fileName( fileInfo.dirPath(true) + "/" + fileInfo.baseName() + ".kft" );
resetURL();
bool ok = loadNativeFormat( fileName );
if ( !ok )
showLoadingErrorDialog();
setEmpty();
m_docOpened = ok; // Used to for a hack that make kivio not crash if you cancel startup dialog.
return ok;
} else if ( ret == KoTemplateChooseDia::Empty ) {
setEmpty();
m_docOpened = true; // Used to for a hack that make kivio not crash if you cancel startup dialog.
return true;
} else {
return false;
}
}
void KivioDoc::openExistingFile( const TQString& file )
{
KivioPage* t = createPage();
m_pMap->addPage(t);
KoDocument::openExistingFile(file);
}
void KivioDoc::openTemplate( const TQString& file )
{
KivioPage* t = createPage();
m_pMap->addPage(t);
KoDocument::openTemplate(file);
}
void KivioDoc::initEmpty()
{
KivioPage* t = createPage();
m_pMap->addPage(t);
KoDocument::initEmpty();
}
KoView* KivioDoc::createViewInstance( TQWidget* tqparent, const char* name )
{
if (!name) {
name = "View";
}
return new KivioView( tqparent, name, this );
}
TQDomDocument KivioDoc::saveXML()
{
TQDomDocument doc( "kiviodoc" );
doc.appendChild( doc.createProcessingInstruction( "xml", "version=\"1.0\" encoding=\"UTF-8\"" ) );
TQDomElement kivio = doc.createElement( "kiviosheet" );
kivio.setAttribute( "editor", "Kivio" );
kivio.setAttribute( "mime", MIME_TYPE );
kivio.setAttribute( "units", unitName() );
gridData.save(kivio,"grid");
TQDomElement viewItemsElement = doc.createElement("ViewItems");
kivio.appendChild(viewItemsElement);
doc.appendChild(kivio);
// Save the list of stencils spawners we have loaded.
// We save these as the following:
//
// <KivioStencilSpawnerSet id="whatever the id is">
// <KivioSMLStencilSpawner id="whatever the id is"/>
// <KivioSMLStencilSpawner id="whatever the id is"/>
// </KivioStencilSpawnerSet>
// ....
//
// This is so we can load them back in, and check that we actually
// have all these spawners on disk.
KivioStencilSpawnerSet *pSet = m_pLstSpawnerSets->first();
while( pSet )
{
if(checkStencilsForSpawnerSet(pSet)) {
kivio.appendChild(pSet->saveXML(doc));
}
pSet = m_pLstSpawnerSets->next();
}
TQDomElement e = m_pMap->save(doc);
kivio.appendChild(e);
setModified(false);
return doc;
}
bool KivioDoc::saveOasis(KoStore* store, KoXmlWriter* manifestWriter)
{
KoStoreDevice storeDev(store);
KoGenStyles styles;
KoGenStyle pageLayout = Kivio::Config::defaultPageLayout().saveOasis();
TQString layoutName = styles.lookup(pageLayout, "PL");
KoGenStyle masterPage(KoGenStyle::STYLE_MASTER);
masterPage.addAttribute("style:page-tqlayout-name", layoutName);
styles.lookup(masterPage, "Standard", false);
if(!store->open("content.xml")) {
return false;
}
KoXmlWriter* docWriter = createOasisXmlWriter(&storeDev, "office:document-content");
docWriter->startElement("office:body");
docWriter->startElement("office:drawing");
m_pMap->saveOasis(store, docWriter, &styles); // Save contents
docWriter->endElement(); // office:drawing
docWriter->endElement(); // office:body
docWriter->endElement(); // Root element
docWriter->endDocument();
delete docWriter;
if(!store->close()) {
return false;
}
manifestWriter->addManifestEntry("content.xml", "text/xml");
if(!store->open("styles.xml")) {
return false;
}
KoXmlWriter* styleWriter = createOasisXmlWriter(&storeDev, "office:document-styles");
styleWriter->startElement("office:automatic-styles");
TQValueList<KoGenStyles::NamedStyle> styleList = styles.styles(KoGenStyle::STYLE_PAGELAYOUT);
TQValueList<KoGenStyles::NamedStyle>::const_iterator it = styleList.begin();
for ( ; it != styleList.end(); ++it) {
(*it).style->writeStyle(styleWriter, styles, "style:page-tqlayout", (*it).name, "style:page-tqlayout-properties");
}
styleList = styles.styles(Kivio::STYLE_PAGE);
it = styleList.begin();
for ( ; it != styleList.end(); ++it) {
(*it).style->writeStyle(styleWriter, styles, "style:style", (*it).name, "style:properties");
}
styleWriter->endElement(); // office:automatic-styles
styleList = styles.styles(KoGenStyle::STYLE_MASTER);
it = styleList.begin();
styleWriter->startElement("office:master-styles");
for ( ; it != styleList.end(); ++it) {
(*it).style->writeStyle(styleWriter, styles, "style:master-page", (*it).name, "");
}
styleWriter->endElement(); // office:master-styles
styleWriter->endElement(); // Root element
styleWriter->endDocument();
delete styleWriter;
if(!store->close()) {
return false;
}
manifestWriter->addManifestEntry("styles.xml", "text/xml");
if(!store->open("settings.xml")) {
return false;
}
KoXmlWriter* settingsWriter = createOasisXmlWriter(&storeDev, "office:document-settings");
settingsWriter->startElement("office:settings");
settingsWriter->startElement("config:config-item-set");
settingsWriter->addAttribute("config:name", "view-settings");
KoUnit::saveOasis( settingsWriter, unit() );
saveOasisSettings( *settingsWriter );
settingsWriter->endElement(); // config:config-item-set
settingsWriter->endElement(); // office:settings
settingsWriter->endDocument();
delete settingsWriter;
if(!store->close()) {
return false;
}
manifestWriter->addManifestEntry("settings.xml", "text/xml");
setModified(false);
return true;
}
void KivioDoc::saveOasisSettings( KoXmlWriter &/*settingsWriter*/ )
{
//todo
}
bool KivioDoc::loadOasis( const TQDomDocument& doc, KoOasisStyles& oasisStyles, const TQDomDocument& settings, KoStore* )
{
kdDebug(43000) << "Start loading OASIS document..." << endl;
m_bLoading = true;
TQDomElement contents = doc.documentElement();
TQDomElement body(KoDom::namedItemNS( contents, KoXmlNS::office, "body"));
if(body.isNull()) {
kdDebug(43000) << "No office:body found!" << endl;
setErrorMessage(i18n("Invalid OASIS document. No office:body tag found."));
m_bLoading = false;
return false;
}
body = KoDom::namedItemNS( body, KoXmlNS::office, "drawing");
if(body.isNull()) {
kdDebug(43000) << "No office:drawing found!" << endl;
setErrorMessage(i18n("Invalid OASIS document. No office:drawing tag found."));
m_bLoading = false;
return false;
}
TQDomNode node = body.firstChild();
TQString localName;
m_pMap->clear();
// TODO: port to forEachElement
while(!node.isNull()) {
localName = node.localName();
if(localName == "page" /* && and namespace is KoXmlNS::draw*/) {
KivioPage* p = createPage();
addPage(p);
if(!p->loadOasis(node.toElement(), oasisStyles)) {
m_bLoading = false;
return false;
}
}
node = node.nextSibling();
}
loadOasisSettings( settings );
emit loadingFinished();
return true;
}
void KivioDoc::loadOasisSettings( const TQDomDocument&settingsDoc )
{
if ( settingsDoc.isNull() )
return ; //not a error some file doesn't have settings.xml
KoOasisSettings settings( settingsDoc );
KoOasisSettings::Items viewSettings = settings.itemSet( "view-settings" );
if ( !viewSettings.isNull() )
{
setUnit(KoUnit::unit(viewSettings.parseConfigItemString("unit")));
//todo add other config here.
}
}
bool KivioDoc::loadXML( TQIODevice *, const TQDomDocument& doc )
{
m_bLoading = true;
if ( doc.doctype().name() != "kiviodoc" ) {
m_bLoading = false;
return false;
}
TQDomElement kivio = doc.documentElement();
if ( kivio.attribute( "mime" ) != "application/x-kivio" &&
kivio.attribute( "mime" ) != "application/vnd.kde.kivio" )
{
kdDebug(43000) << "KivioDoc::loadXML() - Invalid mime type" << endl;
m_bLoading = false;
return false;
}
TQDomNode node = kivio.firstChild();
while( !node.isNull() )
{
TQString name = node.nodeName();
if( name == "KivioMap" )
{
if( !m_pMap->loadXML( node.toElement() ) )
{
m_bLoading = false;
return false;
}
}
else if( name == "KivioStencilSpawnerSet" )
{
TQString id = XmlReadString( node.toElement(), "id", "" );
bool hidden = (XmlReadString(node.toElement(), "hidden", "false") == "true");
if( id.isEmpty() )
{
kdDebug(43000) << "KivioDoc::loadXML() - Bad KivioStencilSpawnerSet found, it contains no id!" << endl;
}
else
{
loadStencilSpawnerSet(id, hidden);
}
}
else if( name == "Options" )
{
// Not used anymore use users default page instead from Settings to create new pages...
}
else
{
kdDebug(43000) << "KivioDoc::loadXML() - Unknown node " << name << endl;
}
node = node.nextSibling();
}
TQString us = kivio.attribute("units", Kivio::Config::unit());
bool isInt = false;
int u = us.toInt(&isInt);
if(!isInt) {
setUnit(KoUnit::unit(us));
} else {
setUnit(Kivio::convToKoUnit(u));
}
if(kivio.hasAttribute("gridIsShow")) {
gridData.load(kivio,"grid");
}
emit loadingFinished();
return true;
}
bool KivioDoc::loadStencilSpawnerSet(const TQString &id, bool hidden)
{
KStandardDirs *dirs = KGlobal::dirs();
TQStringList dirList = dirs->findDirs("data", "kivio/stencils");
TQString rootDir;
// Iterate through all data directories
for( TQStringList::Iterator it = dirList.begin(); it != dirList.end(); ++it )
{
rootDir = (*it);
// Within each data directory, iterate through all directories looking
// for a filename (dir) that matches the parameter
TQDir d(rootDir);
d.setFilter( TQDir::Dirs );
d.setSorting( TQDir::Name );
const TQFileInfoList *list = d.entryInfoList();
TQFileInfoListIterator listIT( *list );
TQFileInfo *fi;
// Loop through the outer directories (like BasicFlowcharting)
while( (fi=listIT.current()) )
{
if( fi->fileName() != "." &&
fi->fileName() != ".." )
{
TQDir innerD(fi->absFilePath() );
innerD.setFilter( TQDir::Dirs );
innerD.setSorting( TQDir::Name );
const TQFileInfoList *innerList = innerD.entryInfoList();
TQFileInfoListIterator innerIT( *innerList );
TQFileInfo *innerFI;
// Loop through the inner directories (like FlowChartingShapes1)
while( (innerFI = innerIT.current()) )
{
if( innerFI->fileName() != ".." &&
innerFI->fileName() != "." )
{
// Compare the descriptions
TQString foundId;
// TODO: use ID system here for loading
foundId = KivioStencilSpawnerSet::readId(innerFI->absFilePath());
if( foundId == id)
{
// Load the spawner set with rootDir + "/" + fi.fileName()
addSpawnerSetDuringLoad(innerFI->absFilePath(), hidden);
return true;
}
}
++innerIT;
}
}
++listIT;
}
}
return false;
}
bool KivioDoc::completeLoading( KoStore* )
{
m_bLoading = false;
m_pMap->update();
setModified( false );
return true;
}
KivioPage* KivioDoc::createPage()
{
TQString s( i18n("Page%1") );
s = s.tqarg( m_iPageId++ );
KivioPage* t = new KivioPage(m_pMap, s);
t->setPageName(s,true);
return t;
}
void KivioDoc::addPage( KivioPage* page )
{
m_pMap->addPage(page);
setModified(true);
emit sig_addPage(page);
}
void KivioDoc::insertPage( KivioPage * page )
{
TQPtrListIterator<KoView> it( views() );
for (; it.current(); ++it )
((KivioView*)it.current())->insertPage( page );
}
void KivioDoc::takePage( KivioPage * page )
{
TQPtrListIterator<KoView> it( views() );
for (; it.current(); ++it )
((KivioView*)it.current())->removePage( page );
}
void KivioDoc::paintContent( TQPainter& painter, const TQRect& rect, bool transparent, double /*zoomX*/, double /*zoomY*/ )
{
KivioPage* page = m_pMap->firstPage();
if ( !page )
return;
KoZoomHandler zoom;
zoom.setZoomAndResolution(100, KoGlobal::dpiX(),
KoGlobal::dpiY());
KoRect r = page->getRectForAllStencils();
float zw = (float) rect.width() / (float)zoom.zoomItX(r.width());
float zh = (float) rect.height() / (float)zoom.zoomItY(r.height());
float z = TQMIN(zw, zh);
//kdDebug(43000) << "paintContent: w = " << rect.width() << " h = " << rect.height() << endl;
zoom.setZoomAndResolution(tqRound(z * 100), KoGlobal::dpiX(),
KoGlobal::dpiY());
KivioScreenPainter ksp(&painter);
ksp.painter()->translate( - zoom.zoomItX(r.x()), - zoom.zoomItY(r.y()) );
paintContent(ksp,rect,transparent,page, TQPoint(zoom.zoomItX(r.x()), zoom.zoomItY(r.y())), &zoom, false, false);
ksp.setPainter(0L); // Important! Don't delete the TQPainter!!!
}
void KivioDoc::paintContent( KivioPainter& painter, const TQRect& rect, bool transparent, KivioPage* page, TQPoint p0, KoZoomHandler* zoom, bool drawConnectorTargets, bool drawSelection )
{
if ( isLoading() )
return;
page->paintContent(painter,rect,transparent,p0,zoom, drawConnectorTargets, drawSelection);
}
void KivioDoc::printContent( KPrinter &prn )
{
KivioScreenPainter p;
TQValueList<int> pages = prn.pageList();
KivioPage *pPage;
// ### HACK: disable zooming-when-printing if embedded parts are used.
// No koffice app supports zooming in paintContent currently.
// Disable in ALL cases now
bool doZoom = false;
int dpiX = doZoom ? 300 : KoGlobal::dpiX();
int dpiY = doZoom ? 300 : KoGlobal::dpiY();
p.start(&prn);
TQPaintDeviceMetrics metrics( &prn );
p.painter()->scale( (double)metrics.logicalDpiX() / (double)dpiX,
(double)metrics.logicalDpiY() / (double)dpiY );
TQValueList<int>::iterator it;
for(it = pages.begin(); it != pages.end(); ++it) {
pPage = m_pMap->pageList().at((*it)-1);
pPage->printContent(p, dpiX, dpiY);
if( it != (--pages.end()) ) {
prn.newPage();
}
}
p.stop();
}
// TODO: Fix for id system
KivioStencilSpawnerSet* KivioDoc::findSpawnerSet(const TQString& dirName, const TQString& id)
{
KivioStencilSpawnerSet *pSet = m_pLstSpawnerSets->first();
while(pSet)
{
if( pSet->dir() == dirName || pSet->id() == id )
{
return pSet;
}
pSet = m_pLstSpawnerSets->next();
}
return 0;
}
void KivioDoc::addSpawnerSet( const TQString &dirName )
{
TQString id = KivioStencilSpawnerSet::readId( dirName );
KivioStencilSpawnerSet* set = findSpawnerSet(dirName, id);
if(set)
{
// Unhide the set if it's only hidden
if(set->hidden()) {
set->setHidden(false);
emit sig_addSpawnerSet(set);
}
return;
}
set = new KivioStencilSpawnerSet();
if(!set->loadDir(dirName))
{
kdDebug(43000) << "KivioDoc::addSpawnerSet() - Error loading dir set" << endl;
delete set;
return;
}
// Queue set for loading stencils
m_stencilSetLoadQueue.append(set);
if(!m_loadTimer) {
m_loadTimer = new TQTimer(this);
connect(m_loadTimer, TQT_SIGNAL(timeout()), this, TQT_SLOT(loadStencil()));
}
if(!m_loadTimer->isActive()) {
emit initProgress();
m_loadTimer->start(0, false);
}
}
void KivioDoc::addSpawnerSetDuringLoad(const TQString& dirName, bool hidden)
{
KivioStencilSpawnerSet *set;
set = new KivioStencilSpawnerSet();
set->setHidden(hidden);
if( set->loadDir(dirName)==false )
{
kdDebug(43000) << "KivioDoc::addSpawnerSetDuringLoad() - Error loading dir set" << endl;
delete set;
return;
}
TQStringList::iterator it;
TQStringList files = set->files();
for(it = files.begin(); it != files.end(); ++it) {
TQString fileName = set->dir() + "/" + (*it);
set->loadFile(fileName);
}
m_pLstSpawnerSets->append(set);
}
KivioDoc::~KivioDoc()
{
if(m_docOpened) {
saveConfig();
}
// ***MUST*** Delete the pages first because they may
// contain plugins which will be unloaded soon. The stencils which are
// spawned by plugins NEED the plugins still loaded when their destructor
// is called or the program will slit it's throat.
delete m_pMap;
delete dcop;
delete m_commandHistory;
delete m_pLstSpawnerSets;
m_pLstSpawnerSets = 0;
s_docs->removeRef(this);
}
void KivioDoc::saveConfig()
{
Kivio::Config::self()->writeConfig();
}
void KivioDoc::initConfig()
{
setUnit( KoUnit::unit(Kivio::Config::unit()) );
m_font = Kivio::Config::font();
m_pageLayout = Kivio::Config::defaultPageLayout();
}
bool KivioDoc::removeSpawnerSet( KivioStencilSpawnerSet *pSet )
{
return m_pLstSpawnerSets->removeRef( pSet );
}
/**
* Iterates through all spawner objects in the stencil set checking if
* they exist in any of the pages.
*/
void KivioDoc::slotDeleteStencilSet( DragBarButton *pBtn, TQWidget *w, KivioStackBar *pBar )
{
KivioIconView* pIconView = static_cast<KivioIconView*>(w);
KivioStencilSpawnerSet* pSet = pIconView->spawnerSet();
// Only actually remove the set if it isn't used...
if(!checkStencilsForSpawnerSet(pSet)) {
removeSpawnerSet(pSet);
} else {
pSet->setHidden(true);
}
// And emit the signal to kill the set (page & button)
emit sig_deleteStencilSet(pBtn, w, pBar);
}
/**
* Checks if any stencils in the document use this spawner
*/
bool KivioDoc::checkStencilsForSpawnerSet(KivioStencilSpawnerSet* spawnerSet)
{
KivioPage *pPage = m_pMap->firstPage();
KivioLayer *pLayer;
KivioStencil *pStencil;
// Iterate across all the pages
while(pPage) {
pLayer = pPage->layers()->first();
while(pLayer) {
pStencil = pLayer->stencilList()->first();
while(pStencil) {
// If this is a group stencil, then we must check all child stencils
if(pStencil->groupList() && pStencil->groupList()->count() > 0) {
if(checkGroupForSpawnerSet(pStencil, spawnerSet)) {
return true;
}
} else if(pStencil->spawner()->set() == spawnerSet) {
return true;
}
pStencil = pLayer->stencilList()->next();
}
pLayer = pPage->layers()->next();
}
pPage = m_pMap->nextPage();
}
return false;
}
bool KivioDoc::checkGroupForSpawnerSet(KivioStencil* pGroup, KivioStencilSpawnerSet* spawnerSet)
{
KivioStencil *pStencil = pGroup->groupList()->first();
while(pStencil) {
if(pStencil->groupList() && pStencil->groupList()->count() > 0) {
if(checkGroupForSpawnerSet(pStencil, spawnerSet))
return true;
} else if(pStencil->spawner()->set() == spawnerSet) {
return true;
}
pStencil = pGroup->groupList()->next();
}
return false;
}
void KivioDoc::slotSelectionChanged()
{
emit sig_selectionChanged();
}
KivioStencilSpawner* KivioDoc::findStencilSpawner( const TQString& setId, const TQString& stencilId )
{
KivioStencilSpawnerSet *pSpawnerSet = m_pLstSpawnerSets->first();
while( pSpawnerSet )
{
if( pSpawnerSet->id() == setId && pSpawnerSet->tqfind(stencilId) )
{
return pSpawnerSet->tqfind(stencilId);
// return pSpawnerSet->tqfind(name)
}
pSpawnerSet = m_pLstSpawnerSets->next();
}
if( m_pInternalSet->id() == setId && m_pInternalSet->tqfind(stencilId) )
{
return m_pInternalSet->tqfind(stencilId);
}
return NULL;
}
KivioStencilSpawner* KivioDoc::findInternalStencilSpawner( const TQString& stencilId )
{
return m_pInternalSet->tqfind(stencilId);
}
void KivioDoc::addInternalStencilSpawner(KivioStencilSpawner* spawner)
{
m_pInternalSet->addSpawner(spawner);
}
void KivioDoc::updateView(KivioPage* page)
{
emit sig_updateView(page);
}
void KivioDoc::updateButton()
{
TQPtrListIterator<KoView> it( views() );
for (; it.current(); ++it )
((KivioView*)it.current())->updateButton();
}
void KivioDoc::resetLayerPanel()
{
TQPtrListIterator<KoView> it( views() );
for (; it.current(); ++it )
((KivioView*)it.current())->resetLayerPanel();
}
void KivioDoc::addCommand( KCommand * cmd )
{
kdDebug(43000) << "KivioDoc::addCommand " << cmd->name() << endl;
m_commandHistory->addCommand( cmd, false );
setModified( true );
}
int KivioDoc::undoRedoLimit() const
{
return m_commandHistory->undoLimit();
}
void KivioDoc::setUndoRedoLimit(int val)
{
m_commandHistory->setUndoLimit(val);
m_commandHistory->setRedoLimit(val);
}
void KivioDoc::slotDocumentRestored()
{
setModified( false );
}
void KivioDoc::slotCommandExecuted()
{
setModified( true );
}
void KivioDoc::updateProtectPanelCheckBox()
{
TQPtrListIterator<KoView> it( views() );
for (; it.current(); ++it )
((KivioView*)it.current())->updateProtectPanelCheckBox();
}
void KivioDoc::loadStencil()
{
KivioStencilSpawnerSet* set = m_stencilSetLoadQueue.first();
TQString fileName = set->dir() + "/" + set->files()[m_currentFile];
set->loadFile(fileName);
m_currentFile++;
emit progress(tqRound(((float)m_currentFile / (float)set->files().count()) * 100.0));
if(m_currentFile >= set->files().count()) {
m_pLstSpawnerSets->append(set);
if(!m_bLoading) {
setModified(true);
emit sig_addSpawnerSet(set);
}
m_currentFile = 0;
set = 0;
m_stencilSetLoadQueue.pop_front();
if(m_stencilSetLoadQueue.isEmpty()) {
m_loadTimer->stop();
emit endProgress();
} else {
emit initProgress();
}
}
}
void KivioDoc::updateGuideLines(KoView* sender)
{
TQValueList<double> hGuideLines;
TQValueList<double> vGuideLines;
KivioView* view = static_cast<KivioView*>(sender);
view->canvasWidget()->guideLines().getGuideLines(hGuideLines, vGuideLines);
view->activePage()->setGuideLines(hGuideLines, vGuideLines);
TQPtrListIterator<KoView> it(views());
KivioView* itView = 0;
for (; it.current(); ++it ) {
itView = static_cast<KivioView*>(it.current());
if(it.current() != sender && (itView->activePage() == view->activePage())) {
itView->canvasWidget()->guideLines().setGuideLines(hGuideLines, vGuideLines);
}
}
}
void KivioDoc::updateGuideLines(KivioPage* page)
{
TQValueList<double> hGuideLines = page->horizontalGuideLines();
TQValueList<double> vGuideLines = page->verticalGuideLines();
TQPtrListIterator<KoView> it(views());
KivioView* itView = 0;
for (; it.current(); ++it ) {
itView = static_cast<KivioView*>(it.current());
if(itView->activePage() == page) {
itView->canvasWidget()->guideLines().setGuideLines(hGuideLines, vGuideLines);
}
}
}
#include "kivio_doc.moc"