+ support of merging projects on App level
This commit is contained in:
parent
2608d0aaf2
commit
c02590c666
|
@ -77,7 +77,8 @@ SET(Document_CPP_SRCS
|
|||
Plane.cpp
|
||||
Transactions.cpp
|
||||
VRMLObject.cpp
|
||||
MaterialObject.cpp
|
||||
MaterialObject.cpp
|
||||
MergeDocuments.cpp
|
||||
)
|
||||
|
||||
SET(Document_HPP_SRCS
|
||||
|
@ -99,7 +100,8 @@ SET(Document_HPP_SRCS
|
|||
Plane.h
|
||||
Transactions.h
|
||||
VRMLObject.h
|
||||
MaterialObject.h
|
||||
MaterialObject.h
|
||||
MergeDocuments.h
|
||||
)
|
||||
SET(Document_SRCS
|
||||
${Document_CPP_SRCS}
|
||||
|
|
|
@ -33,6 +33,11 @@
|
|||
<UserDocu>Restore the document from disk</UserDocu>
|
||||
</Documentation>
|
||||
</Methode>
|
||||
<Methode Name="mergeProject">
|
||||
<Documentation>
|
||||
<UserDocu>Merges this document with another project file</UserDocu>
|
||||
</Documentation>
|
||||
</Methode>
|
||||
<Methode Name="exportGraphviz">
|
||||
<Documentation>
|
||||
<UserDocu>Export the dependencies of the objects as graph</UserDocu>
|
||||
|
|
|
@ -31,6 +31,7 @@
|
|||
#include <Base/FileInfo.h>
|
||||
#include "DocumentObject.h"
|
||||
#include "DocumentObjectPy.h"
|
||||
#include "MergeDocuments.h"
|
||||
|
||||
// inclusion of the generated files (generated By DocumentPy.xml)
|
||||
#include "DocumentPy.h"
|
||||
|
@ -135,6 +136,22 @@ PyObject* DocumentPy::restore(PyObject * args)
|
|||
Py_Return;
|
||||
}
|
||||
|
||||
PyObject* DocumentPy::mergeProject(PyObject * args)
|
||||
{
|
||||
char* filename;
|
||||
if (!PyArg_ParseTuple(args, "s", &filename)) // convert args: Python->C
|
||||
return NULL; // NULL triggers exception
|
||||
|
||||
PY_TRY {
|
||||
Base::FileInfo fi(filename);
|
||||
Base::ifstream str(fi, std::ios::in | std::ios::binary);
|
||||
App::Document* doc = getDocumentPtr();
|
||||
MergeDocuments md(doc);
|
||||
md.importObjects(str);
|
||||
Py_Return;
|
||||
} PY_CATCH;
|
||||
}
|
||||
|
||||
PyObject* DocumentPy::exportGraphviz(PyObject * args)
|
||||
{
|
||||
char* fn=0;
|
||||
|
|
161
src/App/MergeDocuments.cpp
Normal file
161
src/App/MergeDocuments.cpp
Normal file
|
@ -0,0 +1,161 @@
|
|||
/***************************************************************************
|
||||
* 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>
|
||||
|
||||
using namespace App;
|
||||
|
||||
namespace App {
|
||||
|
||||
class XMLMergeReader : public Base::XMLReader
|
||||
{
|
||||
public:
|
||||
XMLMergeReader(std::map<std::string, std::string>& name, const char* FileName, std::istream& str)
|
||||
: Base::XMLReader(FileName, str), nameMap(name)
|
||||
{}
|
||||
|
||||
void addName(const char* s1, const char* s2)
|
||||
{
|
||||
nameMap[s1] = s2;
|
||||
}
|
||||
const char* getName(const char* name) const
|
||||
{
|
||||
std::map<std::string, std::string>::const_iterator it = nameMap.find(name);
|
||||
if (it != nameMap.end())
|
||||
return it->second.c_str();
|
||||
else
|
||||
return name;
|
||||
}
|
||||
bool doNameMapping() const
|
||||
{
|
||||
return true;
|
||||
}
|
||||
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 == "LinkSub" || (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:
|
||||
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));
|
||||
}
|
||||
|
||||
MergeDocuments::~MergeDocuments()
|
||||
{
|
||||
connectExport.disconnect();
|
||||
connectImport.disconnect();
|
||||
}
|
||||
|
||||
unsigned int MergeDocuments::getMemSize (void) const
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
std::vector<App::DocumentObject*>
|
||||
MergeDocuments::importObjects(std::istream& input)
|
||||
{
|
||||
this->nameMap.clear();
|
||||
this->stream = new zipios::ZipInputStream(input);
|
||||
XMLMergeReader reader(this->nameMap,"<memory>", *stream);
|
||||
std::vector<App::DocumentObject*> objs = appdoc->importObjects(reader);
|
||||
|
||||
delete this->stream;
|
||||
this->stream = 0;
|
||||
|
||||
return objs;
|
||||
}
|
||||
|
||||
void MergeDocuments::importObject(const std::vector<App::DocumentObject*>& o, Base::XMLReader & r)
|
||||
{
|
||||
objects = o;
|
||||
Restore(r);
|
||||
r.readFiles(*this->stream);
|
||||
}
|
||||
|
||||
void MergeDocuments::exportObject(const std::vector<App::DocumentObject*>& o, Base::Writer & w)
|
||||
{
|
||||
objects = o;
|
||||
Save(w);
|
||||
}
|
||||
|
||||
void MergeDocuments::Save (Base::Writer & w) const
|
||||
{
|
||||
// Save view provider stuff
|
||||
}
|
||||
|
||||
void MergeDocuments::Restore(Base::XMLReader &r)
|
||||
{
|
||||
// Restore view provider stuff
|
||||
}
|
||||
|
||||
void MergeDocuments::SaveDocFile (Base::Writer & w) const
|
||||
{
|
||||
// Save view provider stuff
|
||||
}
|
||||
|
||||
void MergeDocuments::RestoreDocFile(Base::Reader & r)
|
||||
{
|
||||
// Restore view provider stuff
|
||||
}
|
63
src/App/MergeDocuments.h
Normal file
63
src/App/MergeDocuments.h
Normal file
|
@ -0,0 +1,63 @@
|
|||
/***************************************************************************
|
||||
* 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 *
|
||||
* *
|
||||
***************************************************************************/
|
||||
|
||||
|
||||
#ifndef APP_MERGEDOCUMENTS_H
|
||||
#define APP_MERGEDOCUMENTS_H
|
||||
|
||||
#include <boost/signals.hpp>
|
||||
#include <Base/Persistence.h>
|
||||
|
||||
namespace zipios {
|
||||
class ZipInputStream;
|
||||
}
|
||||
|
||||
namespace App {
|
||||
class Document;
|
||||
class DocumentObject;
|
||||
class AppExport MergeDocuments : public Base::Persistence
|
||||
{
|
||||
public:
|
||||
MergeDocuments(App::Document* doc);
|
||||
~MergeDocuments();
|
||||
unsigned int getMemSize (void) const;
|
||||
std::vector<App::DocumentObject*> importObjects(std::istream&);
|
||||
void importObject(const std::vector<App::DocumentObject*>& o, Base::XMLReader & r);
|
||||
void exportObject(const std::vector<App::DocumentObject*>& o, Base::Writer & w);
|
||||
void Save (Base::Writer & w) const;
|
||||
void Restore(Base::XMLReader &r);
|
||||
void SaveDocFile (Base::Writer & w) const;
|
||||
void RestoreDocFile(Base::Reader & r);
|
||||
|
||||
private:
|
||||
zipios::ZipInputStream* stream;
|
||||
App::Document* appdoc;
|
||||
std::vector<App::DocumentObject*> objects;
|
||||
std::map<std::string, std::string> nameMap;
|
||||
typedef boost::signals::connection Connection;
|
||||
Connection connectExport;
|
||||
Connection connectImport;
|
||||
};
|
||||
|
||||
} // namespace App
|
||||
|
||||
#endif // APP_MERGEDOCUMENTS_H
|
|
@ -853,7 +853,8 @@ void Document::exportObjects(const std::vector<App::DocumentObject*>& obj, Base:
|
|||
writer.Stream() << "</Document>" << std::endl;
|
||||
}
|
||||
|
||||
void Document::importObjects(const std::vector<App::DocumentObject*>& obj, Base::Reader& reader)
|
||||
void Document::importObjects(const std::vector<App::DocumentObject*>& obj, Base::Reader& reader,
|
||||
const std::map<std::string, std::string>& nameMapping)
|
||||
{
|
||||
// We must create an XML parser to read from the input stream
|
||||
Base::XMLReader xmlReader("GuiDocument.xml", reader);
|
||||
|
@ -874,15 +875,12 @@ void Document::importObjects(const std::vector<App::DocumentObject*>& obj, Base:
|
|||
// thus we try to match by type. This should work because the order of
|
||||
// objects should not have changed
|
||||
xmlReader.readElement("ViewProvider");
|
||||
std::string type = xmlReader.getAttribute("type");
|
||||
ViewProvider* pObj = getViewProvider(*it);
|
||||
while (pObj && type != pObj->getTypeId().getName()) {
|
||||
if (it != obj.end()) {
|
||||
++it;
|
||||
pObj = getViewProvider(*it);
|
||||
}
|
||||
}
|
||||
if (pObj && type == pObj->getTypeId().getName())
|
||||
std::string name = xmlReader.getAttribute("name");
|
||||
std::map<std::string, std::string>::const_iterator jt = nameMapping.find(name);
|
||||
if (jt != nameMapping.end())
|
||||
name = jt->second;
|
||||
Gui::ViewProvider* pObj = this->getViewProviderByName(name.c_str());
|
||||
if (pObj)
|
||||
pObj->Restore(xmlReader);
|
||||
xmlReader.readEndElement("ViewProvider");
|
||||
if (it == obj.end())
|
||||
|
@ -892,6 +890,10 @@ void Document::importObjects(const std::vector<App::DocumentObject*>& obj, Base:
|
|||
}
|
||||
|
||||
xmlReader.readEndElement("Document");
|
||||
|
||||
// In the file GuiDocument.xml new data files might be added
|
||||
if (!xmlReader.getFilenames().empty())
|
||||
xmlReader.readFiles(static_cast<zipios::ZipInputStream&>(reader.getStream()));
|
||||
}
|
||||
|
||||
void Document::addRootObjectsToGroup(const std::vector<App::DocumentObject*>& obj, App::DocumentObjectGroup* grp)
|
||||
|
|
|
@ -124,7 +124,8 @@ public:
|
|||
/// This method is used to restore large amounts of data from a binary file.
|
||||
virtual void RestoreDocFile(Base::Reader &reader);
|
||||
void exportObjects(const std::vector<App::DocumentObject*>&, Base::Writer&);
|
||||
void importObjects(const std::vector<App::DocumentObject*>&, Base::Reader&);
|
||||
void importObjects(const std::vector<App::DocumentObject*>&, Base::Reader&,
|
||||
const std::map<std::string, std::string>& nameMapping);
|
||||
/// Add all root objects of the given array to a group
|
||||
void addRootObjectsToGroup(const std::vector<App::DocumentObject*>&, App::DocumentObjectGroup*);
|
||||
//@}
|
||||
|
|
|
@ -164,44 +164,7 @@ void MergeDocuments::SaveDocFile (Base::Writer & w) const
|
|||
document->exportObjects(objects, w);
|
||||
}
|
||||
|
||||
void MergeDocuments::RestoreDocFile(Base::Reader & reader)
|
||||
void MergeDocuments::RestoreDocFile(Base::Reader & r)
|
||||
{
|
||||
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");
|
||||
|
||||
// In the file GuiDocument.xml new data files might be added
|
||||
if (!xmlReader.getFilenames().empty())
|
||||
xmlReader.readFiles(static_cast<zipios::ZipInputStream&>(reader.getStream()));
|
||||
document->importObjects(objects, r, nameMap);
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue
Block a user