deba@1409: /* -*- C++ -*- deba@1409: * src/lemon/lemon_writer.h - Part of LEMON, a generic C++ optimization library deba@1409: * deba@1409: * Copyright (C) 2005 Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport deba@1409: * (Egervary Research Group on Combinatorial Optimization, EGRES). deba@1409: * deba@1409: * Permission to use, modify and distribute this software is granted deba@1409: * provided that this copyright notice appears in all copies. For deba@1409: * precise terms see the accompanying LICENSE file. deba@1409: * deba@1409: * This software is provided "AS IS" with no warranty of any kind, deba@1409: * express or implied, and with no claim as to its suitability for any deba@1409: * purpose. deba@1409: * deba@1409: */ deba@1409: deba@1409: ///\ingroup io_group deba@1409: ///\file deba@1409: ///\brief Lemon Format writer. deba@1409: deba@1409: #ifndef LEMON_LEMON_WRITER_H deba@1409: #define LEMON_LEMON_WRITER_H deba@1409: deba@1409: #include deba@1409: #include deba@1409: #include deba@1409: #include deba@1409: #include deba@1409: #include deba@1409: #include deba@1409: deba@1409: #include deba@1409: #include deba@1409: #include deba@1409: deba@1409: deba@1409: namespace lemon { deba@1409: deba@1409: /// \ingroup io_group deba@1409: /// \brief Lemon Format writer class. deba@1409: /// deba@1409: /// The Lemon Format contains several sections. We do not want to deba@1409: /// determine what sections are in a lemon file we give only a framework deba@1409: /// to write a section oriented format. deba@1409: /// deba@1409: /// In the Lemon Format each section starts with a line contains a \c \@ deba@1409: /// character on the first not white space position. This line is the deba@1409: /// header line of the section. Each next lines belong to this section deba@1409: /// while it does not starts with \c \@ character. This line can start a deba@1409: /// new section or if it can close the file with the \c \@end line. deba@1409: /// The file format ignore the empty lines and it may contain comments deba@1409: /// started with a \c # character to the end of the line. deba@1409: /// deba@1409: /// The framework provides an abstract LemonWriter::SectionWriter class deba@1409: /// what defines the interface of a SectionWriter. The SectionWriter deba@1409: /// has the \c header() member function what gives back the header of the deba@1409: /// section. After that it will be called the \c write() member which deba@1409: /// should write the content of the section. deba@1409: /// deba@1409: /// \relates GraphWriter deba@1409: /// \relates NodeSetWriter deba@1409: /// \relates EdgeSetWriter deba@1409: /// \relates NodesWriter deba@1409: /// \relates EdgesWriter deba@1409: /// \relates AttributeWriter deba@1409: class LemonWriter { deba@1409: public: deba@1409: deba@1409: /// \brief Abstract base class for writing a section. deba@1409: /// deba@1409: /// This class has an \c header() member function what gives back deba@1409: /// the header line of the section. The \c write() member should deba@1409: /// write the content of the section to the stream. deba@1409: class SectionWriter { deba@1409: friend class LemonWriter; deba@1409: protected: deba@1409: /// \brief Constructor for SectionWriter. deba@1409: /// deba@1409: /// Constructor for SectionWriter. It attach this writer to deba@1409: /// the given LemonWriter. deba@1409: SectionWriter(LemonWriter& writer) { deba@1409: writer.attach(*this); deba@1409: } deba@1409: deba@1409: /// \brief The header of section. deba@1409: /// deba@1409: /// It gives back the header of the section. deba@1409: virtual std::string header() = 0; deba@1409: deba@1409: /// \brief Writer function of the section. deba@1409: /// deba@1409: /// Write the content of the section. deba@1409: virtual void write(std::ostream& os) = 0; deba@1409: }; deba@1409: deba@1409: /// \brief Constructor for LemonWriter. deba@1409: /// deba@1409: /// Constructor for LemonWriter which writes to the given stream. deba@1409: LemonWriter(std::ostream& _os) deba@1409: : os(&_os), own_os(false) {} deba@1409: deba@1409: /// \brief Constructor for LemonWriter. deba@1409: /// deba@1409: /// Constructor for LemonWriter which writes to the given file. deba@1409: LemonWriter(const std::string& filename) deba@1409: : os(0), own_os(true) { deba@1409: os = new std::ofstream(filename.c_str()); deba@1409: } deba@1409: deba@1409: /// \brief Desctructor for LemonWriter. deba@1409: /// deba@1409: /// Desctructor for LemonWriter. deba@1409: ~LemonWriter() { deba@1409: if (own_os) { deba@1409: delete os; deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: LemonWriter(const LemonWriter&); deba@1409: void operator=(const LemonWriter&); deba@1409: deba@1409: void attach(SectionWriter& writer) { deba@1409: writers.push_back(&writer); deba@1409: } deba@1409: deba@1409: public: deba@1409: deba@1409: /// \brief Executes the LemonWriter. deba@1409: /// deba@1409: /// It executes the LemonWriter. deba@1409: void run() { deba@1409: SectionWriters::iterator it; deba@1409: for (it = writers.begin(); it != writers.end(); ++it) { deba@1409: *os << (*it)->header() << std::endl; deba@1409: (*it)->write(*os); deba@1409: } deba@1409: *os << "@end" << std::endl; deba@1409: } deba@1409: deba@1409: deba@1409: private: deba@1409: deba@1409: std::ostream* os; deba@1409: bool own_os; deba@1409: deba@1409: typedef std::vector SectionWriters; deba@1409: SectionWriters writers; deba@1409: deba@1409: }; deba@1409: deba@1409: /// \brief Helper class for implementing the common SectionWriters. deba@1409: /// deba@1409: /// Helper class for implementing the common SectionWriters. deba@1409: class CommonSectionWriterBase : public LemonWriter::SectionWriter { deba@1409: typedef LemonWriter::SectionWriter Parent; deba@1409: protected: deba@1409: deba@1409: /// \brief Constructor for CommonSectionWriterBase. deba@1409: /// deba@1409: /// Constructor for CommonSectionWriterBase. It attach this writer to deba@1409: /// the given LemonWriter. deba@1409: CommonSectionWriterBase(LemonWriter& _writer) deba@1409: : Parent(_writer) {} deba@1409: deba@1409: // Writers deba@1409: deba@1409: template deba@1409: class WriterBase { deba@1409: public: deba@1409: typedef _Item Item; deba@1409: deba@1409: virtual ~WriterBase() {} deba@1409: deba@1409: virtual void write(std::ostream& os, const Item& item) = 0; deba@1409: }; deba@1409: deba@1409: deba@1409: template deba@1409: class MapWriter : public WriterBase<_Item> { deba@1409: public: deba@1409: typedef _Map Map; deba@1409: typedef _Writer Writer; deba@1409: typedef typename Writer::Value Value; deba@1409: typedef _Item Item; deba@1409: deba@1409: const Map& map; deba@1409: Writer writer; deba@1409: deba@1409: MapWriter(const Map& _map, const Writer& _writer) deba@1409: : map(_map), writer(_writer) {} deba@1409: deba@1409: virtual ~MapWriter() {} deba@1409: deba@1409: virtual void write(std::ostream& os, const Item& item) { deba@1409: Value value = map[item]; deba@1409: writer.write(os, value); deba@1409: } deba@1409: deba@1409: }; deba@1409: deba@1409: deba@1409: class ValueWriterBase { deba@1409: public: deba@1409: virtual void write(std::ostream&) = 0; deba@1409: }; deba@1409: deba@1409: template deba@1409: class ValueWriter : public ValueWriterBase { deba@1409: public: deba@1409: typedef _Value Value; deba@1409: typedef _Writer Writer; deba@1409: deba@1409: ValueWriter(const Value& _value, const Writer& _writer) deba@1409: : value(_value), writer(_writer) {} deba@1409: deba@1409: virtual void write(std::ostream& os) { deba@1411: writer.write(os, value); deba@1409: } deba@1409: private: deba@1409: const Value& value; deba@1409: Writer writer; deba@1409: }; deba@1409: deba@1409: deba@1409: template deba@1409: class IdWriterBase { deba@1409: public: deba@1409: typedef _Item Item; deba@1409: virtual void write(std::ostream&, const Item&) const = 0; deba@1409: }; deba@1409: deba@1409: template deba@1409: class IdWriter : public IdWriterBase<_Item> { deba@1409: public: deba@1409: typedef _Item Item; deba@1409: typedef _BoxedIdWriter BoxedIdWriter; deba@1409: deba@1409: const BoxedIdWriter& idWriter; deba@1409: deba@1409: IdWriter(const BoxedIdWriter& _idWriter) deba@1409: : idWriter(_idWriter) {} deba@1409: deba@1409: virtual void write(std::ostream& os, const Item& item) const { deba@1409: return idWriter.writeId(os, item); deba@1409: } deba@1409: }; deba@1409: }; deba@1409: deba@1409: /// \ingroup io_group deba@1409: /// \brief SectionWriter for writing a graph's nodeset. deba@1409: /// deba@1409: /// The lemon format can store multiple graph nodesets with several maps. deba@1409: /// The nodeset section's header line is \c \@nodeset \c nodeset_id, but the deba@1409: /// \c nodeset_id may be empty. deba@1409: /// deba@1409: /// The first line of the section contains the names of the maps separated deba@1409: /// with white spaces. Each next lines describes a node in the nodeset, and deba@1409: /// contains the mapped values for each map. deba@1409: /// deba@1409: /// If the nodeset contains an \c "id" named map then it will be regarded deba@1409: /// as id map. This map should contain only unique values and when the deba@1409: /// \c writeId() member will be called with a node it will write it's id. deba@1409: /// Otherwise if the \c _forceIdMap constructor parameter is true then deba@1409: /// the id map will be the id in the graph. deba@1409: /// deba@1409: /// \relates LemonWriter deba@1409: template deba@1409: class NodeSetWriter : public CommonSectionWriterBase { deba@1409: typedef CommonSectionWriterBase Parent; deba@1409: public: deba@1409: deba@1409: typedef _Graph Graph; deba@1409: typedef _Traits Traits; deba@1409: typedef typename Graph::Node Item; deba@1409: deba@1409: /// \brief Constructor. deba@1409: /// deba@1409: /// Constructor for NodeSetWriter. It creates the NodeSetWriter and deba@1409: /// attach it into the given LemonWriter. If the \c _forceIdMap deba@1409: /// parameter is true then the writer will write own id map when deba@1409: /// the user does not give "id" named map. deba@1409: NodeSetWriter(LemonWriter& _writer, const Graph& _graph, deba@1409: const std::string& _id = std::string(), deba@1409: bool _forceIdMap = true) deba@1409: : Parent(_writer), idMap(0), forceIdMap(_forceIdMap), deba@1409: graph(_graph), id(_id) {} deba@1409: deba@1409: /// \brief Destructor. deba@1409: /// deba@1409: /// Destructor for NodeSetWriter. deba@1409: virtual ~NodeSetWriter() { deba@1409: typename MapWriters::iterator it; deba@1409: for (it = writers.begin(); it != writers.end(); ++it) { deba@1409: delete it->second; deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: NodeSetWriter(const NodeSetWriter&); deba@1409: void operator=(const NodeSetWriter&); deba@1409: deba@1409: public: deba@1409: deba@1409: /// \brief Add a new node map writer command for the writer. deba@1409: /// deba@1409: /// Add a new node map writer command for the writer. deba@1409: template deba@1409: NodeSetWriter& writeMap(std::string name, const Map& map) { deba@1409: return writeMap, Map>(name, map); deba@1409: } deba@1409: deba@1409: /// \brief Add a new node map writer command for the writer. deba@1409: /// deba@1409: /// Add a new node map writer command for the writer. deba@1409: template deba@1409: NodeSetWriter& writeMap(std::string name, const Map& map, deba@1409: const Writer& writer = Writer()) { deba@1409: writers.push_back( deba@1409: make_pair(name, new MapWriter(map, writer))); deba@1409: return *this; deba@1409: } deba@1409: deba@1409: protected: deba@1409: deba@1409: /// \brief The header of the section. deba@1409: /// deba@1409: /// It gives back the header of the section. deba@1409: virtual std::string header() { deba@1409: return "@nodeset " + id; deba@1409: } deba@1409: deba@1409: /// \brief Writer function of the section. deba@1409: /// deba@1409: /// Write the content of the section. deba@1409: virtual void write(std::ostream& os) { deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: if (writers[i].first == "id") { deba@1409: idMap = writers[i].second; deba@1409: forceIdMap = false; deba@1409: break; deba@1409: } deba@1409: } deba@1409: if (forceIdMap) { deba@1409: os << "id\t"; deba@1409: } deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: os << writers[i].first << '\t'; deba@1409: } deba@1409: os << std::endl; deba@1409: for (typename Graph::NodeIt it(graph); it != INVALID; ++it) { deba@1409: if (forceIdMap) { deba@1409: os << graph.id(it) << '\t'; deba@1409: } deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: writers[i].second->write(os, it); deba@1409: os << '\t'; deba@1409: } deba@1409: os << std::endl; deba@1409: } deba@1409: } deba@1409: deba@1409: public: deba@1409: deba@1409: /// \brief Returns true if the nodeset can write the ids of the nodes. deba@1409: /// deba@1409: /// Returns true if the nodeset can write the ids of the nodes. deba@1409: /// It is possible only if an "id" named map was written or the deba@1409: /// \c _forceIdMap constructor parameter was true. deba@1409: bool isIdWriter() const { deba@1409: return idMap != 0 || forceIdMap; deba@1409: } deba@1409: deba@1409: /// \brief Write the id of the given node. deba@1409: /// deba@1409: /// It writes the id of the given node. If there was written an "id" deba@1409: /// named map then it will write the map value belongs to the node. deba@1409: /// Otherwise if the \c forceId parameter was true it will write deba@1409: /// its id in the graph. deba@1409: void writeId(std::ostream& os, const Item& item) const { deba@1409: if (forceIdMap) { deba@1409: os << graph.id(item); deba@1409: } else { deba@1409: idMap->write(os, item); deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: deba@1409: typedef std::vector*> > MapWriters; deba@1409: MapWriters writers; deba@1409: deba@1409: WriterBase* idMap; deba@1409: bool forceIdMap; deba@1409: deba@1409: const Graph& graph; deba@1409: std::string id; deba@1409: deba@1409: }; deba@1409: deba@1409: /// \ingroup io_group deba@1409: /// \brief SectionWriter for writing a graph's edgeset. deba@1409: /// deba@1409: /// The lemon format can store multiple graph edgesets with several maps. deba@1409: /// The edgeset section's header line is \c \@edgeset \c edgeset_id, but the deba@1409: /// \c edgeset_id may be empty. deba@1409: /// deba@1409: /// The first line of the section contains the names of the maps separated deba@1409: /// with white spaces. Each next lines describes a edge in the edgeset. The deba@1409: /// line contains the source and the target nodes' id and the mapped deba@1409: /// values for each map. deba@1409: /// deba@1409: /// If the edgeset contains an \c "id" named map then it will be regarded deba@1409: /// as id map. This map should contain only unique values and when the deba@1409: /// \c writeId() member will be called with a edge it will write it's id. deba@1409: /// Otherwise if the \c _forceIdMap constructor parameter is true then deba@1409: /// the id map will be the id in the graph. deba@1409: /// deba@1409: /// The edgeset writer needs a node id writer to identify which nodes deba@1409: /// have to be connected. If a NodeSetWriter can write the nodes' id, deba@1409: /// it will be able to use with this class. deba@1409: /// deba@1409: /// \relates LemonWriter deba@1409: template deba@1409: class EdgeSetWriter : public CommonSectionWriterBase { deba@1409: typedef CommonSectionWriterBase Parent; deba@1409: public: deba@1409: deba@1409: typedef _Graph Graph; deba@1409: typedef _Traits Traits; deba@1409: typedef typename Graph::Edge Item; deba@1409: deba@1409: /// \brief Constructor. deba@1409: /// deba@1409: /// Constructor for EdgeSetWriter. It creates the EdgeSetWriter and deba@1409: /// attach it into the given LemonWriter. It will write node ids by deba@1409: /// the \c _nodeIdWriter. If the \c _forceIdMap parameter is true deba@1409: /// then the writer will write own id map when the user does not give deba@1409: /// "id" named map. deba@1409: template deba@1409: EdgeSetWriter(LemonWriter& _writer, const Graph& _graph, deba@1409: const NodeIdWriter& _nodeIdWriter, deba@1409: const std::string& _id = std::string(), deba@1409: bool _forceIdMap = true) deba@1409: : Parent(_writer), idMap(0), forceIdMap(_forceIdMap), deba@1409: graph(_graph), id(_id), deba@1409: nodeIdWriter(new IdWriter deba@1409: (_nodeIdWriter)) {} deba@1409: deba@1409: /// \brief Destructor. deba@1409: /// deba@1409: /// Destructor for EdgeSetWriter. deba@1409: virtual ~EdgeSetWriter() { deba@1409: typename MapWriters::iterator it; deba@1409: for (it = writers.begin(); it != writers.end(); ++it) { deba@1409: delete it->second; deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: EdgeSetWriter(const EdgeSetWriter&); deba@1409: void operator=(const EdgeSetWriter&); deba@1409: deba@1409: public: deba@1409: deba@1409: /// \brief Add a new node map writer command for the writer. deba@1409: /// deba@1409: /// Add a new node map writer command for the writer. deba@1409: template deba@1409: EdgeSetWriter& writeMap(std::string name, const Map& map) { deba@1409: return writeMap, Map>(name, map); deba@1409: } deba@1409: deba@1409: /// \brief Add a new node map writer command for the writer. deba@1409: /// deba@1409: /// Add a new node map writer command for the writer. deba@1409: template deba@1409: EdgeSetWriter& writeMap(std::string name, const Map& map, deba@1409: const Writer& writer = Writer()) { deba@1409: writers.push_back( deba@1409: make_pair(name, new MapWriter(map, writer))); deba@1409: return *this; deba@1409: } deba@1409: deba@1409: protected: deba@1409: deba@1409: /// \brief The header of the section. deba@1409: /// deba@1409: /// It gives back the header of the section. deba@1409: virtual std::string header() { deba@1409: return "@edgeset " + id; deba@1409: } deba@1409: deba@1409: /// \brief Writer function of the section. deba@1409: /// deba@1409: /// Write the content of the section. deba@1409: virtual void write(std::ostream& os) { deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: if (writers[i].first == "id") { deba@1409: idMap = writers[i].second; deba@1409: forceIdMap = false; deba@1409: break; deba@1409: } deba@1409: } deba@1409: os << "\t\t"; deba@1409: if (forceIdMap) { deba@1409: os << "id\t"; deba@1409: } deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: os << writers[i].first << '\t'; deba@1409: } deba@1409: os << std::endl; deba@1409: for (typename Graph::EdgeIt it(graph); it != INVALID; ++it) { deba@1409: nodeIdWriter->write(os, graph.source(it)); deba@1409: os << '\t'; deba@1409: nodeIdWriter->write(os, graph.target(it)); deba@1409: os << '\t'; deba@1409: if (forceIdMap) { deba@1409: os << graph.id(it) << '\t'; deba@1409: } deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: writers[i].second->write(os, it); deba@1409: os << '\t'; deba@1409: } deba@1409: os << std::endl; deba@1409: } deba@1409: } deba@1409: deba@1409: public: deba@1409: deba@1409: /// \brief Returns true if the edgeset can write the ids of the edges. deba@1409: /// deba@1409: /// Returns true if the edgeset can write the ids of the edges. deba@1409: /// It is possible only if an "id" named map was written or the deba@1409: /// \c _forceIdMap constructor parameter was true. deba@1409: bool isIdWriter() const { deba@1409: return forceIdMap || idMap != 0; deba@1409: } deba@1409: deba@1409: /// \brief Write the id of the given edge. deba@1409: /// deba@1409: /// It writes the id of the given edge. If there was written an "id" deba@1409: /// named map then it will write the map value belongs to the edge. deba@1409: /// Otherwise if the \c forceId parameter was true it will write deba@1409: /// its id in the graph. deba@1409: void writeId(std::ostream& os, const Item& item) const { deba@1409: if (forceIdMap) { deba@1409: os << graph.id(item); deba@1409: } else { deba@1409: idMap->write(os, item); deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: deba@1409: typedef std::vector*> > MapWriters; deba@1409: MapWriters writers; deba@1409: deba@1409: WriterBase* idMap; deba@1409: bool forceIdMap; deba@1409: deba@1409: const Graph& graph; deba@1409: std::string id; deba@1409: deba@1409: std::auto_ptr > nodeIdWriter; deba@1409: }; deba@1409: deba@1409: /// \ingroup io_group deba@1409: /// \brief SectionWriter for writing labeled nodes. deba@1409: /// deba@1409: /// The nodes section's header line is \c \@nodes \c nodes_id, but the deba@1409: /// \c nodes_id may be empty. deba@1409: /// deba@1409: /// Each line in the section contains the label of the node and deba@1409: /// then the node id. deba@1409: /// deba@1409: /// \relates LemonWriter deba@1409: template deba@1409: class NodeWriter : public CommonSectionWriterBase { deba@1409: typedef CommonSectionWriterBase Parent; deba@1409: typedef _Graph Graph; deba@1409: typedef typename Graph::Node Item; deba@1409: public: deba@1409: deba@1409: /// \brief Constructor. deba@1409: /// deba@1409: /// Constructor for NodeWriter. It creates the NodeWriter and deba@1409: /// attach it into the given LemonWriter. The given \c _IdWriter deba@1409: /// will write the nodes' id what can be a nodeset writer. deba@1409: template deba@1409: NodeWriter(LemonWriter& _writer, const _IdWriter& _idWriter, deba@1409: const std::string& _id = std::string()) deba@1409: : Parent(_writer), id(_id), deba@1409: idWriter(new IdWriter(_idWriter)) {} deba@1409: deba@1409: /// \brief Destructor. deba@1409: /// deba@1409: /// Destructor for NodeWriter. deba@1409: virtual ~NodeWriter() {} deba@1409: deba@1409: private: deba@1409: NodeWriter(const NodeWriter&); deba@1409: void operator=(const NodeWriter&); deba@1409: deba@1409: public: deba@1409: deba@1409: /// \brief Add a node writer command for the NodeWriter. deba@1409: /// deba@1409: /// Add a node writer command for the NodeWriter. deba@1409: void writeNode(const std::string& name, const Item& item) { deba@1409: writers.push_back(make_pair(name, &item)); deba@1409: } deba@1409: deba@1409: protected: deba@1409: deba@1409: /// \brief Header checking function. deba@1409: /// deba@1409: /// It gives back true when the header line start with \c @nodes, deba@1409: /// and the header line's id and the writer's id are the same. deba@1409: virtual std::string header() { deba@1409: return "@nodes " + id; deba@1409: } deba@1409: deba@1409: /// \brief Writer function of the section. deba@1409: /// deba@1409: /// Write the content of the section. deba@1409: virtual void write(std::ostream& os) { deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: os << writers[i].first << ' '; deba@1409: idWriter->write(os, *(writers[i].second)); deba@1409: os << std::endl; deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: deba@1409: std::string id; deba@1409: deba@1409: typedef std::vector > ItemWriters; deba@1409: ItemWriters writers; deba@1409: std::auto_ptr > idWriter; deba@1409: }; deba@1409: deba@1409: /// \ingroup io_group deba@1409: /// \brief SectionWriter for writeing labeled edges. deba@1409: /// deba@1409: /// The edges section's header line is \c \@edges \c edges_id, but the deba@1409: /// \c edges_id may be empty. deba@1409: /// deba@1409: /// Each line in the section contains the label of the edge and deba@1409: /// then the edge id. deba@1409: /// deba@1409: /// \relates LemonWriter deba@1409: template deba@1409: class EdgeWriter : public CommonSectionWriterBase { deba@1409: typedef CommonSectionWriterBase Parent; deba@1409: typedef _Graph Graph; deba@1409: typedef typename Graph::Edge Item; deba@1409: public: deba@1409: deba@1409: /// \brief Constructor. deba@1409: /// deba@1409: /// Constructor for EdgeWriter. It creates the EdgeWriter and deba@1409: /// attach it into the given LemonWriter. The given \c _IdWriter deba@1409: /// will write the edges' id what can be a edgeset writer. deba@1409: template deba@1409: EdgeWriter(LemonWriter& _writer, const _IdWriter& _idWriter, deba@1409: const std::string& _id = std::string()) deba@1409: : Parent(_writer), id(_id), deba@1409: idWriter(new IdWriter(_idWriter)) {} deba@1409: deba@1409: /// \brief Destructor. deba@1409: /// deba@1409: /// Destructor for EdgeWriter. deba@1409: virtual ~EdgeWriter() {} deba@1409: private: deba@1409: EdgeWriter(const EdgeWriter&); deba@1409: void operator=(const EdgeWriter&); deba@1409: deba@1409: public: deba@1409: deba@1409: /// \brief Add an edge writer command for the NodeWriter. deba@1409: /// deba@1409: /// Add an edge writer command for the NodeWriter. deba@1409: void writeEdge(const std::string& name, const Item& item) { deba@1409: writers.push_back(make_pair(name, &item)); deba@1409: } deba@1409: deba@1409: protected: deba@1409: deba@1409: /// \brief Header checking function. deba@1409: /// deba@1409: /// It gives back true when the header line start with \c @nodes, deba@1409: /// and the header line's id and the writer's id are the same. deba@1409: virtual std::string header() { deba@1409: return "@edges " + id; deba@1409: } deba@1409: deba@1409: /// \brief Writer function of the section. deba@1409: /// deba@1409: /// Write the content of the section. deba@1409: virtual void write(std::ostream& os) { deba@1409: for (int i = 0; i < (int)writers.size(); ++i) { deba@1409: os << writers[i].first << ' '; deba@1409: idWriter->write(os, *(writers[i].second)); deba@1409: os << std::endl; deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: deba@1409: std::string id; deba@1409: deba@1409: typedef std::vector > ItemWriters; deba@1409: ItemWriters writers; deba@1409: deba@1409: std::auto_ptr > idWriter; deba@1409: }; deba@1409: deba@1409: /// \ingroup io_group deba@1409: /// \brief SectionWriter for attributes. deba@1409: /// deba@1409: /// The lemon format can store multiple attribute set. Each set has deba@1409: /// the header line \c \@attributes \c attributeset_id, but the deba@1409: /// attributeset_id may be empty. deba@1409: /// deba@1409: /// The attributeset section contains several lines. Each of them starts deba@1409: /// with the name of attribute and then the value. deba@1409: /// deba@1409: /// \relates LemonWriter deba@1409: template deba@1409: class AttributeWriter : public CommonSectionWriterBase { deba@1409: typedef CommonSectionWriterBase Parent; deba@1409: typedef _Traits Traits; deba@1409: public: deba@1409: /// \brief Constructor. deba@1409: /// deba@1409: /// Constructor for AttributeWriter. It creates the AttributeWriter and deba@1409: /// attach it into the given LemonWriter. deba@1409: AttributeWriter(LemonWriter& _writer, deba@1409: const std::string& _id = std::string()) deba@1409: : Parent(_writer), id(_id) {} deba@1409: deba@1409: /// \brief Destructor. deba@1409: /// deba@1409: /// Destructor for AttributeWriter. deba@1409: virtual ~AttributeWriter() { deba@1409: typename Writers::iterator it; deba@1409: for (it = writers.begin(); it != writers.end(); ++it) { deba@1409: delete it->second; deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: AttributeWriter(const AttributeWriter&); deba@1409: void operator=(AttributeWriter&); deba@1409: deba@1409: public: deba@1409: /// \brief Add an attribute writer command for the writer. deba@1409: /// deba@1409: /// Add an attribute writer command for the writer. deba@1409: template deba@1409: AttributeWriter& writeAttribute(const std::string& id, deba@1409: const Value& value) { deba@1409: return deba@1409: writeAttribute >(id, value); deba@1409: } deba@1409: deba@1409: /// \brief Add an attribute writer command for the writer. deba@1409: /// deba@1409: /// Add an attribute writer command for the writer. deba@1409: template deba@1409: AttributeWriter& writeAttribute(const std::string& name, deba@1409: const Value& value, deba@1409: const Writer& writer = Writer()) { deba@1409: writers.push_back(make_pair(name, new ValueWriter deba@1409: (value, writer))); deba@1409: return *this; deba@1409: } deba@1409: deba@1409: protected: deba@1409: deba@1409: /// \brief The header of section. deba@1409: /// deba@1409: /// It gives back the header of the section. deba@1409: std::string header() { deba@1409: return "@attributes " + id; deba@1409: } deba@1409: deba@1409: /// \brief Writer function of the section. deba@1409: /// deba@1409: /// Write the content of the section. deba@1409: void write(std::ostream& os) { deba@1409: typename Writers::iterator it; deba@1409: for (it = writers.begin(); it != writers.end(); ++it) { deba@1409: os << it->first << ' '; deba@1409: it->second->write(os); deba@1409: os << std::endl; deba@1409: } deba@1409: } deba@1409: deba@1409: private: deba@1409: std::string id; deba@1409: deba@1409: typedef std::vector > Writers; deba@1409: Writers writers; deba@1409: }; deba@1409: deba@1409: deba@1409: } deba@1409: #endif