namespace CGAL { /*! \mainpage User Manual \anchor Chapter_IO_Streams \anchor Chapter_operator_for_IO_streams \cgalAutoToc \authors Andreas Fabri, Geert-Jan Giezeman, Lutz Kettner, and Maxime Gimeno \section IOstreamIntro Introduction An algorithm is the application of a serie of steps to convert some input data into output data. As such, it is necessary to have a way to input and output external data into \cgal data structures. This is achieved using input/output (I/O) streams of data, which enables reading and writing to and from files, the console, or other custom structures. \subsection StreamSupportOutline Outline - \ref IOstreamIO : the most essential data structures of \cgal, its kernel objects, all provide adapted input and output operators. In addition, \cgal provides tools to enhance the existing stream system of the STL to easily read and write classes external to \cgal. - \ref IOstreamSupportedFormats : a number of traditional data structures such as point sets or polygon meshes have well-established standards specifying a particular data format to facilitate data exchanges. Formats supported for \cgal data structures are detailed in this section; the page \ref IOStreamSupportedFileFormats offers the reversed viewpoint (file format to \cgal data structure). Finally, an alternative to standard file formats is serialization of data, see Section \ref IOstreamOther. \section IOstreamPrecision Stream Precision When manipulating floating point-based number types, it is important to always keep in mind that the stream of the precision should be chosen appropriately, lest potentially significant loss of data might incur. For example, consider the following extreme case: \code{.cpp} double v = 184, w = 182; std::cout << std::setprecision(2) << v << " " << w << std::endl; \endcode This snipplet will produce the output `1.8e+02 1.8e+02` despite `v` and `w` being different. As the default precision of output streams of the \stl is `6`, this can be a real source of bugs and wrong outputs. The stream precision of \stl streams can be modified using std::setprecision. File I/O functions also offer an optional named parameter to set the output stream precision. \section IOstreamIO Reading and Writing Data With Streams All classes in the \cgal kernel provide input and output operators for I/O streams. Classes external to \cgal are also supported, by means of `oformat()` (Section \ref seciofornoncgaltypes "I/O for Non CGAL Types"). The basic task of such an operator is to produce a representation of an object that can be written as a sequence of characters on devices as a console, a file, or a pipe. In \cgal we distinguish between a raw \ascii, a raw binary, and a pretty printing format. \code{.cpp} enum Mode {ASCII = 0, BINARY, PRETTY}; \endcode In \ascii mode, objects are written as a set of numbers, e.g. the coordinates of a point or the coefficients of a line, in a machine independent format. In `BINARY` mode, data are written in a binary format, e.g. a double is represented as a sequence of four byte. The format depends on the machine. The mode `PRETTY` serves mainly for debugging as the type of the geometric object is written, as well as the data defining the object. For example for a point at the origin with Cartesian double coordinates, the output would be `PointC2(0.0, 0.0)`. At the moment \cgal does not provide input operations for pretty printed data. By default a stream is in \ascii mode. \cgal provides the following functions to modify the mode of an I/O stream. \code{.cpp} IO::Mode set_mode(std::ios& s, IO::Mode m); IO::Mode IO::set_ascii_mode(std::ios& s); IO::Mode IO::set_binary_mode(std::ios& s); IO::Mode IO::set_pretty_mode(std::ios& s); \endcode The following functions enable testing whether a stream is in a certain mode: \code{.cpp} IO::Mode get_mode(std::ios& s); bool is_ascii(std::ios& s); bool is_binary(std::ios& s); bool is_pretty(std::ios& s); \endcode \subsection IOstreamInput Input Operator \cgal defines input operators for classes that are derived from the class `istream`. This allows to read from istreams as `std::cin`, as well as from `std::istringstream` and `std::ifstream`. The input operator is defined for all classes in the \cgal `Kernel`. Let `is` be an input stream. \code{.cpp} // Extracts object `c` from the stream `is`. Returns `is`. istream& operator>>(istream& is, Class c); \endcode \code{.cpp} #include #include #include #include typedef CGAL::Point_2< CGAL::Cartesian > Point; typedef CGAL::Segment_2< CGAL::Cartesian > Segment; int main() { Point p, q; Segment s; CGAL::IO::set_ascii_mode(std::cin); std::cin >> p >> q; std::ifstream f("data.txt"); CGAL::IO::set_binary_mode(f); f >> s >> p; return 1; } \endcode \subsection IOstreamOutput Output Operator \cgal defines output operators for classes that are derived from the class `ostream`. This allows to write to ostreams as `std::cout` or `std::cerr`, as well as to `std::ostringstream` and `std::ofstream`. The output operator is defined for all classes in the \cgal `Kernel` and for the class `Color` (see Section \ref IOstreamColors) as well. Let `os` be an output stream. \code{.cpp} // Inserts object `c` in the stream `os`. Returns `os`. ostream& operator<<(ostream& os, Class c); \endcode \code{.cpp} #include #include #include #include typedef CGAL::Point_2< CGAL::Cartesian > Point; typedef CGAL::Segment_2< CGAL::Cartesian > Segment; int main() { Point p(0,1), q(2,2); Segment s(p,q); CGAL::IO::set_pretty_mode(std::cout); std::cout << p << std::endl << q << std::endl; std::ofstream f("data.txt"); CGAL::IO::set_binary_mode(f); f << s << p ; return 1; } \endcode \subsection seciofornoncgaltypes IO for Non-CGAL Types \subsubsection IOstreamUsingOutputFormatting Using Output Formatting To ensure that non-\cgal types are formatted correctly (i.e., respecting \link PkgStreamSupportEnumRef `IO::Mode` \endlink), `oformat()` can be used. For types with a `Output_rep` specialization, the respective output routine of `Output_rep` will be called by `oformat()`. Otherwise, the stream output operator will be called. \code{.cpp} std::cout << CGAL::IO::oformat( myobject ); \endcode Optionally, you can provide a second template parameter `F` as a formatting tag: \code{.cpp} std::cout << CGAL::IO::oformat( myobject, My_formatting_tag() ); \endcode For a list of formatting tags supported by the type `T`, please refer to the documentation of the respective type. \subsubsection IOstreamCustomizingOutputFormatting Customizing Output Formatting In some situations, you want to control the output formatting for a type `T`. For external types (third party libraries etc.), there might be problems if their stream output operator does not respect \link PkgStreamSupportEnumRef `IO::Mode` \endlink. The purpose of `Output_rep` is to provide a way to control output formatting that works independently of the object's stream output operator. Instead of putting `T` directly into an output stream, `T` is wrapped into an output representation `Output_rep`. For convenience, a function `oformat()` exists, which constructs an instance of `Output_rep`. If you do not specialize `Output_rep` for `T`, `T`'s stream output operator is called from within `Output_rep`, by default. If you want another behaviour for your type `T`, you have to provide a specialization for that type. Furthermore, you can provide specializations with a second template parameter (a formatting tag). The second template parameter defaults to `Null_tag` and means default behaviour. For example, specializing `Output_rep` for `CORE::BigRat` (without a formatting tag parameter) could look like this: \code{.cpp} template class Output_rep< ::CORE::BigRat, F> { const ::CORE::BigRat& t; public: Output_rep( const ::CORE::BigRat& tt) : t(tt) {} std::ostream& operator()( std::ostream& out) const { switch (get_mode(out)) { case IO::PRETTY:{ if(CGAL_CORE_DENOMINATOR(t) == ::CORE::BigRat(1)) return out <rgb-value of the color. There are a 11 predefined `Color` functions available: `black()`, `white()`, `gray()`, `red()`, `green()`, `deep_blue()`, `blue()`, `purple()`, `violet()`, `orange()`, and `yellow()`. \subsection IOstreamStream Stream Support Three classes are provided by \cgal as adaptors to input and output stream iterators. The class `Istream_iterator` is an input iterator adaptor and is particularly useful for classes that are similar but not compatible to `std::istream`. Similarly, the class `Ostream_iterator` is an output iterator adaptor. The class `Verbose_ostream` can be used as an output stream. The stream output operator `<<` is defined for any type. The class stores in an internal state a stream and whether the output is active or not. If the state is active, the stream output operator `<<` uses the internal stream to output its argument. If the state is inactive, nothing happens. \section IOstreamSupportedFormats Importing and Exporting Data using Standard File Formats Specific standards have been created to facilite the exchange of data for traditional data structures such as point sets or polygon meshes. \cgal aims to provide a uniform and consistent approach for Input/Output functions, providing for a given file format with shorter name `XXX` an input function `read_XXX(input, data_structure)`, and an output function called `write_XXX(output, data_structure)`. Whenever possible, a generic I/O function which can deduce the file format from the extension is also provided for convenience. In the following subsections, file formats supported in \cgal are listed along with compatible \cgal data structures. A reverse viewpoint, listing the data structures which can be used for a specific file format is available on the page: \ref IOStreamSupportedFileFormats. Note that these categories will grow as the on-going effort to uniformize I/O in \cgal progresses. \subsection IOstreamPointSetIO Point Set IO A set of points - possibly with normals - is the basic input of many algorithms. The following table shows which file formats can be read from and written for point ranges.
Generic \ref IOStreamOFF "OFF" \ref IOStreamXYZ "XYZ" \ref IOStreamPLY "PLY" \ref IOStreamLAS "LAS"
Input `read_points()` \link PkgPointSetProcessing3IOOff `read_OFF()` \endlink \link PkgPointSetProcessing3IOXyz `read_XYZ()` \endlink \link PkgPointSetProcessing3IOPly `read_PLY()` \endlink \link PkgPointSetProcessing3IOLas `read_LAS()` \endlink
Output `write_points()` \link PkgPointSetProcessing3IOOff `write_OFF()` \endlink \link PkgPointSetProcessing3IOXyz `write_XYZ()` \endlink \link PkgPointSetProcessing3IOPly `write_PLY()` \endlink \link PkgPointSetProcessing3IOLas `write_LAS()` \endlink
All of these functions (with the exception of the LAS format) can read and write either points alone or points with normals (depending on whether the `normal_map` named parameter is used by the user or not). Note that the %PLY format handles both \ascii and binary formats. In addition, %PLY and %LAS are extensible formats that can embed additional properties. These can also be read by \cgal (see Section \ref Point_set_processing_3Properties_io). The class `CGAL::Point_set_3` is the data structure used in \cgal to represent point sets. It is a vector-based data structure that contains a default property (named point) for the coordinates of the points, and is able to work with dynamic properties. The file formats supported for `CGAL::Point_set_3` are detailed in the table below.
Generic \ref IOStreamOFF "OFF" \ref IOStreamXYZ "XYZ" \ref IOStreamPLY "PLY" \ref IOStreamLAS "LAS"
Input `read_point_set()` \link PkgPointSet3IOOFF `read_OFF()` \endlink \link PkgPointSet3IOXYZ `read_XYZ()` \endlink \link PkgPointSet3IOPLY `read_PLY()` \endlink \link PkgPointSet3IOLAS `read_LAS()` \endlink
Output `write_point_set()` \link PkgPointSet3IOOFF `write_OFF()` \endlink \link PkgPointSet3IOXYZ `write_XYZ()` \endlink \link PkgPointSet3IOPLY `write_PLY()` \endlink \link PkgPointSet3IOLAS `write_LAS()` \endlink
\subsection IOstreamPolygonSoupIO Polygon Soup IO A polygon soup is a set of polygons with no global combinatorial information, stored in a two containers: one storing the points, and the other one storing their indices per face (i.e a vector of 3 integers represent a triangle face).
Generic \ref IOStreamOFF "OFF" \ref IOStreamOBJ "OBJ" \ref IOStreamSTL "STL" \ref IOStreamPLY "PLY" \ref IOStreamVTK "VTP" \ref IOStreamGocad "GOCAD" \ref IOStreamWKT "WKT" \ref IOStream3MF "3MF"
Input `read_polygon_soup()` \link PkgStreamSupportIoFuncsOFF `read_OFF()` \endlink \link PkgStreamSupportIoFuncsOBJ `read_OBJ()` \endlink \link PkgStreamSupportIoFuncsSTL `read_STL()` \endlink \link PkgStreamSupportIoFuncsPLY `read_PLY()` \endlink \link PkgStreamSupportIoFuncsVTP `read_VTP()` \endlink \link PkgStreamSupportIoFuncsGOCAD `read_GOCAD()` \endlink \link PkgStreamSupportIoFuncsWKT `read_WKT()` \endlink \link PkgStreamSupportIoFuncs3MF `read_3MF()` \endlink
Output `write_polygon_soup()` \link PkgStreamSupportIoFuncsOFF `write_OFF()` \endlink \link PkgStreamSupportIoFuncsOBJ `write_OBJ()` \endlink \link PkgStreamSupportIoFuncsSTL `write_STL()` \endlink \link PkgStreamSupportIoFuncsPLY `write_PLY()` \endlink \link PkgStreamSupportIoFuncsVTP `write_VTP()` \endlink \link PkgStreamSupportIoFuncsGOCAD `write_GOCAD()` \endlink \link PkgStreamSupportIoFuncsWKT `write_WKT()` \endlink \link PkgStreamSupportIoFuncs3MF `write_3MF()` \endlink
\subsection IOstreamPolygonMeshIO Polygon Mesh IO A \a polygon \a mesh is a consistent and orientable surface mesh, that can have one or more boundaries. This refines the concept of `FaceGraph` with some additional restrictions; a complete definition can be found \ref PMPDef "here". The table above only lists the functions that work with any polygon mesh.
Generic \ref IOStreamOFF "OFF" \ref IOStreamSTL "STL" \ref IOStreamVTK "VTP" \ref IOStreamOBJ "OBJ" \ref IOStreamGocad "GOCAD" \ref IOStreamWRL "WRL"
Input `read_polygon_mesh()` `read_OFF()` `read_STL()` `read_VTP()` `read_OBJ()` `read_GOCAD()`
-
Output `write_polygon_mesh()` `write_OFF()` `write_STL()` `write_VTP()` `write_OBJ()` `write_GOCAD()` `write_WRL()`
Some particular polygon mesh data structures such as `CGAL::Surface_mesh` have specific overloads of these functions, enabling reading and writing of dynamic information for some file format. See the reference manual of each data structure for more information. The functions above require the input to represent a 2-manifold surface (possibly with boundaries). If this is not the case, the package \ref PkgPolygonMeshProcessing offers the function \link PMP_IO_grp `CGAL::Polygon_mesh_processing::IO::read_polygon_mesh()` \endlink which can perform some combinatorial repairing to ensure the input data is a 2-manifold. \subsection IOstream2DGeo Simple 2D Geometry IO `WKT` stands for Well-Known Text and it is a text markup language for representing vector geometry objects on a geographical map. See the wikipedia page for details. CGAL supports a subset of WKT types: point, multipoint, linestring, multilinestring, polygon and multipolygon. Free functions are provided for reading and writing several \cgal types using these WKT types, namely: - `CGAL::Point_2` - `CGAL::Point_3` - `CGAL::Polygon_with_holes_2` - random access range of the above types. You can find more information about WKT \ref PkgStreamSupportRef "here". The following table summarizes the most useful functions. Note that the name deviates from the standard \cgal I/O functions.
Generic Points Polylines Polygons
Input read_WKT() read_multi_point_WKT() read_multi_linestring_WKT() read_multi_polygon_WKT()
Output
-
write_multi_point_WKT() write_multi_linestring_WKT() write_multi_polygon_WKT()
\subsection IOstreamOtherIO Other Data Structures Other data structures of \cgal have their own I/O functions, see their respective user and reference manuals. \subsection IOstreamOther Reading Unsupported Formats If \cgal does not support the file format of your data, the `boost::property_tree` might come in handy if your data has a simple-enough structure. The following small example shows how to parse an XML file, which might look as follows: \code{.xml} \endcode \cgalExample{Stream_support/read_xml.cpp} */ } /* namespace CGAL */