git-svn-id: https://free-cad.svn.sourceforge.net/svnroot/free-cad/trunk@5000 e8eeb9e2-ec13-0410-a4a9-efa5cf37419d
239 lines
9.0 KiB
C++
239 lines
9.0 KiB
C++
/***************************************************************************
|
|
* Copyright (c) 2010 Werner Mayer <wmayer[at]users.sourceforge.net> *
|
|
* *
|
|
* This file is part of the FreeCAD CAx development system. *
|
|
* *
|
|
* 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., 59 Temple Place, *
|
|
* Suite 330, Boston, MA 02111-1307, USA *
|
|
* *
|
|
***************************************************************************/
|
|
|
|
#include "PreCompiled.h"
|
|
#ifndef _PreComp_
|
|
# include <stack>
|
|
# include <boost/bind.hpp>
|
|
#endif
|
|
#include "MergeDocuments.h"
|
|
#include <Base/Console.h>
|
|
#include <Base/Reader.h>
|
|
#include <Base/Writer.h>
|
|
#include <App/Document.h>
|
|
#include <App/DocumentObject.h>
|
|
#include <Gui/Application.h>
|
|
#include <Gui/Document.h>
|
|
#include <Gui/ViewProvider.h>
|
|
|
|
using namespace Gui;
|
|
|
|
namespace Gui {
|
|
|
|
class XMLMergeReader : public Base::XMLReader
|
|
{
|
|
public:
|
|
XMLMergeReader(const std::map<std::string, std::string>& name, const char* FileName, std::istream& str)
|
|
: Base::XMLReader(FileName, str), nameMap(name)
|
|
{}
|
|
|
|
protected:
|
|
void startElement(const XMLCh* const uri, const XMLCh* const localname,
|
|
const XMLCh* const qname,
|
|
const XERCES_CPP_NAMESPACE_QUALIFIER Attributes& attrs)
|
|
{
|
|
Base::XMLReader::startElement(uri, localname, qname, attrs);
|
|
if (LocalName == "Property")
|
|
propertyStack.push(std::make_pair(AttrMap["name"],AttrMap["type"]));
|
|
|
|
if (!propertyStack.empty()) {
|
|
// replace the stored object name with the real one
|
|
if (LocalName == "Link" || (LocalName == "String" && propertyStack.top().first == "Label")) {
|
|
for (std::map<std::string, std::string>::iterator it = AttrMap.begin(); it != AttrMap.end(); ++it) {
|
|
std::map<std::string, std::string>::const_iterator jt = nameMap.find(it->second);
|
|
if (jt != nameMap.end())
|
|
it->second = jt->second;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void endElement(const XMLCh* const uri, const XMLCh *const localname, const XMLCh *const qname)
|
|
{
|
|
Base::XMLReader::endElement(uri, localname, qname);
|
|
if (LocalName == "Property")
|
|
propertyStack.pop();
|
|
}
|
|
|
|
private:
|
|
const std::map<std::string, std::string>& nameMap;
|
|
typedef std::pair<std::string, std::string> PropertyTag;
|
|
std::stack<PropertyTag> propertyStack;
|
|
};
|
|
}
|
|
|
|
MergeDocuments::MergeDocuments(App::Document* doc) : appdoc(doc)
|
|
{
|
|
connectExport = doc->signalExportObjects.connect
|
|
(boost::bind(&MergeDocuments::exportObject, this, _1, _2));
|
|
connectImport = doc->signalImportObjects.connect
|
|
(boost::bind(&MergeDocuments::importObject, this, _1, _2));
|
|
document = Gui::Application::Instance->getDocument(doc);
|
|
}
|
|
|
|
MergeDocuments::~MergeDocuments()
|
|
{
|
|
connectExport.disconnect();
|
|
connectImport.disconnect();
|
|
}
|
|
|
|
unsigned int MergeDocuments::getMemSize (void) const
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
std::vector<App::DocumentObject*>
|
|
MergeDocuments::importObjects(std::istream& input)
|
|
{
|
|
//std::map<std::string, std::string> nameMap;
|
|
std::vector<App::DocumentObject*> objs;
|
|
zipios::ZipInputStream zipstream(input);
|
|
XMLMergeReader reader(nameMap,"<memory>", zipstream);
|
|
|
|
int i,Cnt;
|
|
reader.readElement("Document");
|
|
long scheme = reader.getAttributeAsInteger("SchemaVersion");
|
|
reader.DocumentSchema = scheme;
|
|
|
|
// read the object types
|
|
nameMap.clear();
|
|
reader.readElement("Objects");
|
|
Cnt = reader.getAttributeAsInteger("Count");
|
|
for (i=0 ;i<Cnt ;i++) {
|
|
reader.readElement("Object");
|
|
std::string type = reader.getAttribute("type");
|
|
std::string name = reader.getAttribute("name");
|
|
std::string docn = name;
|
|
|
|
// remove number from end to avoid lengthy names
|
|
size_t lastpos = docn.length()-1;
|
|
while (docn[lastpos] >= 48 && docn[lastpos] <= 57)
|
|
lastpos--;
|
|
docn = docn.substr(0, lastpos+1);
|
|
|
|
try {
|
|
App::DocumentObject* o = appdoc->addObject(type.c_str(),docn.c_str());
|
|
objs.push_back(o);
|
|
// use this name for the later access because an object with
|
|
// the given name may already exist
|
|
nameMap[name] = o->getNameInDocument();
|
|
}
|
|
catch (Base::Exception&) {
|
|
Base::Console().Message("Cannot create object '%s'\n", name.c_str());
|
|
}
|
|
}
|
|
reader.readEndElement("Objects");
|
|
|
|
// read the features itself
|
|
reader.readElement("ObjectData");
|
|
Cnt = reader.getAttributeAsInteger("Count");
|
|
for (i=0 ;i<Cnt ;i++) {
|
|
reader.readElement("Object");
|
|
std::string name = nameMap[reader.getAttribute("name")];
|
|
App::DocumentObject* pObj = appdoc->getObject(name.c_str());
|
|
if (pObj) { // check if this feature has been registered
|
|
// pObj->StatusBits.set(4);
|
|
pObj->Restore(reader);
|
|
// pObj->StatusBits.reset(4);
|
|
}
|
|
reader.readEndElement("Object");
|
|
}
|
|
reader.readEndElement("ObjectData");
|
|
|
|
reader.readEndElement("Document");
|
|
appdoc->signalImportObjects(objs, reader);
|
|
reader.readFiles(zipstream);
|
|
// reset all touched
|
|
for (std::vector<App::DocumentObject*>::iterator it= objs.begin();it!=objs.end();++it)
|
|
(*it)->purgeTouched();
|
|
return objs;
|
|
}
|
|
|
|
void MergeDocuments::importObject(const std::vector<App::DocumentObject*>& o, Base::XMLReader & r)
|
|
{
|
|
objects = o;
|
|
for (std::vector<App::DocumentObject*>::iterator it = objects.begin(); it != objects.end(); ++it) {
|
|
Gui::ViewProvider* vp = document->getViewProvider(*it);
|
|
if (vp) vp->hide();
|
|
}
|
|
Restore(r);
|
|
}
|
|
|
|
void MergeDocuments::exportObject(const std::vector<App::DocumentObject*>& o, Base::Writer & w)
|
|
{
|
|
objects = o;
|
|
Save(w);
|
|
}
|
|
|
|
void MergeDocuments::Save (Base::Writer & w) const
|
|
{
|
|
w.addFile("GuiDocument.xml", this);
|
|
}
|
|
|
|
void MergeDocuments::Restore(Base::XMLReader &r)
|
|
{
|
|
r.addFile("GuiDocument.xml", this);
|
|
}
|
|
|
|
void MergeDocuments::SaveDocFile (Base::Writer & w) const
|
|
{
|
|
document->exportObjects(objects, w);
|
|
}
|
|
|
|
void MergeDocuments::RestoreDocFile(Base::Reader & reader)
|
|
{
|
|
std::vector<App::DocumentObject*> obj = objects;
|
|
// We must create an XML parser to read from the input stream
|
|
Base::XMLReader xmlReader("GuiDocument.xml", reader);
|
|
xmlReader.readElement("Document");
|
|
long scheme = xmlReader.getAttributeAsInteger("SchemaVersion");
|
|
|
|
// At this stage all the document objects and their associated view providers exist.
|
|
// Now we must restore the properties of the view providers only.
|
|
//
|
|
// SchemeVersion "1"
|
|
if (scheme == 1) {
|
|
// read the viewproviders itself
|
|
xmlReader.readElement("ViewProviderData");
|
|
int Cnt = xmlReader.getAttributeAsInteger("Count");
|
|
std::vector<App::DocumentObject*>::const_iterator it = obj.begin();
|
|
for (int i=0;i<Cnt&&it!=obj.end();++i,++it) {
|
|
// The stored name usually doesn't match with the current name anymore
|
|
// thus we try to match by type. This should work because the order of
|
|
// objects should not have changed
|
|
xmlReader.readElement("ViewProvider");
|
|
std::string name = xmlReader.getAttribute("name");
|
|
name = nameMap[name];
|
|
Gui::ViewProvider* pObj = document->getViewProviderByName(name.c_str());
|
|
//Gui::ViewProvider* pObj = document->getViewProvider(*it);
|
|
if (pObj)
|
|
pObj->Restore(xmlReader);
|
|
xmlReader.readEndElement("ViewProvider");
|
|
if (it == obj.end())
|
|
break;
|
|
}
|
|
xmlReader.readEndElement("ViewProviderData");
|
|
}
|
|
|
|
xmlReader.readEndElement("Document");
|
|
}
|