/*************************************************************************** * Copyright (c) 2009 Werner Mayer * * * * 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_ #endif #include "VRMLObject.h" #include "Document.h" #include "DocumentObjectPy.h" #include #include #include #include using namespace App; PROPERTY_SOURCE(App::VRMLObject, App::GeoFeature) VRMLObject::VRMLObject() : index(0) { ADD_PROPERTY_TYPE(VrmlFile,(0),"",Prop_None,"Included file with the VRML definition"); ADD_PROPERTY_TYPE(Urls,(""),"",static_cast(Prop_ReadOnly|Prop_Output|Prop_Transient), "Resource files loaded by the VRML file"); ADD_PROPERTY_TYPE(Resources,(""),"",static_cast(Prop_ReadOnly|Prop_Output), "Resource files loaded by the VRML file"); Urls.setSize(0); Resources.setSize(0); } VRMLObject::~VRMLObject() { } short VRMLObject::mustExecute(void) const { return 0; } void VRMLObject::onChanged(const App::Property* prop) { if (prop == &VrmlFile) { std::string orig = VrmlFile.getOriginalFileName(); if (!orig.empty()) { // store the path name of the VRML file Base::FileInfo fi(orig); this->vrmlPath = fi.dirPath(); } } else if (prop == &Urls) { // save the relative paths to the resource files in the project file Resources.setSize(Urls.getSize()); const std::vector& urls = Urls.getValues(); int index=0; for (std::vector::const_iterator it = urls.begin(); it != urls.end(); ++it, ++index) { std::string output = getRelativePath(this->vrmlPath, *it); Resources.set1Value(index, output); } } GeoFeature::onChanged(prop); } PyObject *VRMLObject::getPyObject() { if (PythonObject.is(Py::_None())){ // ref counter is set to 1 PythonObject = Py::Object(new DocumentObjectPy(this),true); } return Py::new_reference_to(PythonObject); } std::string VRMLObject::getRelativePath(const std::string& prefix, const std::string& resource) const { std::string str; std::string intname = this->getNameInDocument(); if (!prefix.empty()) { if (resource.substr(0, prefix.size()) == prefix) { std::string suffix = resource.substr(prefix.size()); str = intname + suffix; } } if (str.empty()) { Base::FileInfo fi(resource); str = intname + "/" + fi.fileName(); } return str; } std::string VRMLObject::fixRelativePath(const std::string& name, const std::string& resource) const { // the part before the first '/' must match with object's internal name std::string::size_type pos = resource.find('/'); if (pos != std::string::npos) { std::string prefix = resource.substr(0, pos); std::string suffix = resource.substr(pos); if (prefix != name) { return name + suffix; } } return resource; } void VRMLObject::makeDirectories(const std::string& path, const std::string& subdir) { std::string::size_type pos = subdir.find('/'); while (pos != std::string::npos) { std::string sub = subdir.substr(0, pos); std::string dir = path + "/" + sub; Base::FileInfo fi(dir); if (!fi.createDirectory()) break; pos = subdir.find('/', pos+1); } } void VRMLObject::Save (Base::Writer &writer) const { App::GeoFeature::Save(writer); // save also the inline files if there const std::vector& urls = Resources.getValues(); for (std::vector::const_iterator it = urls.begin(); it != urls.end(); ++it) { writer.addFile(it->c_str(), this); } this->index = 0; } void VRMLObject::Restore(Base::XMLReader &reader) { App::GeoFeature::Restore(reader); Urls.setSize(Resources.getSize()); // restore also the inline files if there const std::vector& urls = Resources.getValues(); for(std::vector::const_iterator it = urls.begin(); it != urls.end(); ++it) { reader.addFile(it->c_str(), this); } this->index = 0; } void VRMLObject::SaveDocFile (Base::Writer &writer) const { // store the inline files of the VRML file if (this->index < Urls.getSize()) { std::string url = Urls[this->index]; Base::FileInfo fi(url); // it can happen that the transient directory has changed after // saving the 'URLs' in RestoreDocFile() and then we have to // try again with the new transient directory. if (!fi.exists()) { std::string path = getDocument()->TransientDir.getValue(); url = Resources[this->index]; url = path + "/" + url; fi.setFile(url); } this->index++; Base::ifstream file(fi, std::ios::in | std::ios::binary); if (file) { writer.Stream() << file.rdbuf(); } } } void VRMLObject::RestoreDocFile(Base::Reader &reader) { if (this->index < Resources.getSize()) { std::string path = getDocument()->TransientDir.getValue(); std::string url = Resources[this->index]; std::string intname = this->getNameInDocument(); url = fixRelativePath(intname, url); Resources.set1Value(this->index, url); makeDirectories(path, url); url = path + "/" + url; Base::FileInfo fi(url); Urls.set1Value(this->index, url); this->index++; Base::ofstream file(fi, std::ios::out | std::ios::binary); if (file) { reader >> file.rdbuf(); file.close(); } // after restoring all inline files reload the VRML file if (this->index == Urls.getSize()) { VrmlFile.touch(); Base::FileInfo fi(VrmlFile.getValue()); this->vrmlPath = fi.dirPath(); } } }