cgal/Stream_support/include/CGAL/IO/WKT.h

582 lines
15 KiB
C++

// Copyright (c) 2018-2020 GeometryFactory Sarl (France).
// All rights reserved.
//
// Licensees holding a valid commercial license may use this file in
// accordance with the commercial license agreement provided with the software.
//
// This file is part of CGAL (www.cgal.org)
//
// $URL$
// $Id$
// SPDX-License-Identifier: LGPL-3.0-or-later OR LicenseRef-Commercial
//
// Author(s) : Maxime Gimeno
#ifndef CGAL_IO_WKT_H
#define CGAL_IO_WKT_H
#if defined(DOXYGEN_RUNNING) || (!defined(BOOST_GCC) || BOOST_GCC >= 40500))
#include <CGAL/Point_2.h>
#include <CGAL/Point_3.h>
#include <CGAL/Polygon_2.h>
#include <CGAL/Polygon_with_holes_2.h>
#include <CGAL/IO/WKT/traits_point.h>
#include <CGAL/IO/WKT/traits_point_3.h>
#include <CGAL/IO/WKT/traits_linestring.h>
#include <CGAL/IO/WKT/traits_polygon.h>
#include <CGAL/IO/WKT/traits_multipoint.h>
#include <CGAL/IO/WKT/traits_multilinestring.h>
#include <CGAL/IO/WKT/traits_multipolygon.h>
#include <boost/geometry/io/wkt/read.hpp>
#include <boost/geometry/io/wkt/write.hpp>
#include <iostream>
#include <fstream>
#include <sstream>
#include <string>
namespace CGAL {
namespace internal {
template <typename K>
void pop_back_if_equal_to_front(CGAL::Polygon_2<K>& poly)
{
typename CGAL::Polygon_2<K>::iterator it = poly.end();
--it;
if((*poly.begin()) == *it)
poly.erase(it);
}
template <typename K>
void pop_back_if_equal_to_front(CGAL::Polygon_with_holes_2<K>& pwh)
{
pop_back_if_equal_to_front(pwh.outer_boundary());
for(auto i = pwh.holes_begin(); i!= pwh.holes_end(); ++i)
pop_back_if_equal_to_front(*i);
}
} // namespace internal
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief fills a `Point` from a WKT stream.
//!
//! The first line starting with POINT in the stream will be used.
//!
//! \tparam Point can be a `CGAL::Point_2` or `CGAL::Point_3`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::Point_2`
//! \see `CGAL::Point_3`
template<typename Point>
bool read_point_WKT(std::istream& in,
Point& point)
{
if(!in.good())
return false;
std::string line;
while(std::getline(in, line))
{
std::istringstream iss(line);
std::string type;
iss >> type;
if(type.substr(0, 5).compare("POINT") == 0)
{
try
{
boost::geometry::read_wkt(line, point);
}
catch(...)
{
std::cerr << "error." << std::endl;
return false;
}
break;
}
}
return !in.fail();
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief overwrites the content of a `MultiPoint` with the first line starting with MULTIPOINT in the stream.
//!
//! \tparam MultiPoint must be a model of `RandomAccessRange` of `CGAL::Point_2` or `CGAL::Point_3`,
//! and have:
//! - a function `push_back()` that takes the same point type,
//! - a function `clear()`,
//! - a function `resize()` that takes an `size_type`
//! - an `operator[]()` that takes a `size_type`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::Point_2`
//! \see `CGAL::Point_3`
template<typename MultiPoint>
bool read_multi_point_WKT(std::istream& in,
MultiPoint& mp)
{
if(!in.good())
return false;
internal::Geometry_container<MultiPoint, boost::geometry::multi_point_tag> gc(mp);
std::string line;
while(std::getline(in, line))
{
std::istringstream iss(line);
std::string type;
iss >> type;
if(type.substr(0, 10).compare("MULTIPOINT") == 0)
{
try{
boost::geometry::read_wkt(line, gc);
} catch(...){
std::cerr << "error." << std::endl;
return false;
}
break;
}
}
return !in.fail();
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief fills a `Linestring` from a WKT stream.
//!
//! The first line starting with LINESTRING in the stream will be used.
//!
//! \tparam Linestring must be a model of `RandomAccessRange` of `CGAL::Point_2`,
//! and have:
//! - a function `push_back()` that takes a `CGAL::Point_2`.
//! - a function `clear()`,
//! - a function `resize()` that takes an `size_type`
//! - an `operator[]()` that takes a `size_type`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::Point_2`
template<typename LineString>
bool read_linestring_WKT(std::istream& in,
LineString& polyline)
{
if(!in.good())
return false;
internal::Geometry_container<LineString, boost::geometry::linestring_tag> gc(polyline);
std::string line;
while(std::getline(in, line))
{
std::istringstream iss(line);
std::string type;
iss >> type;
if(type.substr(0, 10).compare("LINESTRING") == 0)
{
try{
boost::geometry::read_wkt(line, gc);
} catch(...){
std::cerr << "error." << std::endl;
return false;
}
break;
}
}
return !in.fail();
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief overwrites the content of a `MultiLineString` with the first line starting with MULTILINESTRING in the stream.
//!
//! \tparam MultiLineString must be a model of `RandomAccessRange` of `Linestring`,
//! and have:
//! - a function `push_back()` that takes a `Linestring`,
//! - a function `clear()`,
//! - a function `resize()` that takes an `size_type`
//! - an `operator[]()` that takes a `size_type`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::Point_2`
template<typename MultiLineString>
bool read_multi_linestring_WKT(std::istream& in,
MultiLineString& mls)
{
if(!in.good())
return false;
typedef typename MultiLineString::value_type PointRange;
typedef internal::Geometry_container<PointRange, boost::geometry::linestring_tag> LineString;
std::vector<LineString> pr_range;
internal::Geometry_container<std::vector<LineString>, boost::geometry::multi_linestring_tag> gc(pr_range);
std::string line;
while(std::getline(in, line))
{
std::istringstream iss(line);
std::string type;
iss >> type;
if(type.substr(0, 15).compare("MULTILINESTRING") == 0)
{
try
{
boost::geometry::read_wkt(line, gc);
}
catch(...)
{
std::cerr << "error." << std::endl;
return false;
}
break;
}
}
for(LineString& ls : gc)
mls.push_back(*ls.range);
return !in.fail();
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief fills `polygon` from a WKT stream.
//!
//! The first line starting with POLYGON in the stream will be used.
//!
//! \tparam Polygon is a `CGAL::General_polygon_with_holes_2`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::General_polygon_with_holes_2`
template<typename Polygon>
bool read_polygon_WKT(std::istream& in,
Polygon& polygon)
{
if(!in.good())
return false;
std::string line;
while(std::getline(in, line))
{
std::istringstream iss(line);
std::string type;
iss >> type;
if(type.substr(0, 7).compare("POLYGON") == 0)
{
try
{
boost::geometry::read_wkt(line, polygon);
}
catch( ...)
{
in.setstate(std::ios::failbit);
return false;
};
internal::pop_back_if_equal_to_front(polygon);
break;
}
}
return !in.fail();
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief overwrites the content of a `MultiPolygon` with the first line starting with MULTIPOLYGON in the stream.
//!
//! \tparam MultiPolygon must be a model of `RandomAccessRange` of `CGAL::General_polygon_with_holes_2`,
//! and have:
//! - a function `push_back()` that takes a `CGAL::General_polygon_with_holes_2`,
//! - a function `clear()`,
//! - a function `resize()` that takes an `size_type`
//! - an `operator[]()` that takes a `size_type`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::General_polygon_with_holes_2`
template<typename MultiPolygon>
bool read_multi_polygon_WKT(std::istream& in,
MultiPolygon& polygons)
{
if(!in.good())
return false;
internal::Geometry_container<MultiPolygon, boost::geometry::multi_polygon_tag> gc(polygons);
std::string line;
while(std::getline(in, line))
{
std::istringstream iss(line);
std::string type;
iss >> type;
if(type.substr(0, 12).compare("MULTIPOLYGON") == 0)
{
try
{
boost::geometry::read_wkt(line, gc);
}
catch( ...)
{
in.setstate(std::ios::failbit);
return false;
};
for(typename internal::Geometry_container<MultiPolygon, boost::geometry::multi_polygon_tag>::iterator it = gc.begin(); it != gc.end(); ++it)
internal::pop_back_if_equal_to_front(*it);
break;
}
}
return !in.fail();
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief writes `point` into a WKT stream.
//!
//! \tparam Point is a `CGAL::Point_2`
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::Point_2`
template<typename Point>
std::ostream& write_point_WKT(std::ostream& out,
const Point& point)
{
if(!out.good())
return out;
out << boost::geometry::wkt(point) << std::endl;
return out;
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief writes `poly` into a WKT stream.
//!
//! \tparam Polygon must be a `CGAL::General_polygon_with_holes_2`
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::General_polygon_with_holes_2`
template<typename Polygon>
std::ostream& write_polygon_WKT(std::ostream& out,
const Polygon& poly)
{
if(!out.good())
return out;
out << boost::geometry::wkt(poly) << std::endl;
return out;
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief writes the content of `ls` into a WKT stream.
//!
//! \tparam LineString must be a `RandomAccessRange` of `CGAL::Point_2`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//!\see `CGAL::Point_2`
template<typename LineString>
std::ostream& write_linestring_WKT(std::ostream& out,
LineString ls)
{
if(!out.good())
return out;
internal::Geometry_container<LineString, boost::geometry::linestring_tag> gc(ls);
out << boost::geometry::wkt(gc) << std::endl;
return out;
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief writes the content of `mp` into a WKT stream.
//!
//! \tparam MultiPoint must be a `RandomAccessRange` of `CGAL::Point_2`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//!\see `CGAL::Point_2`
template<typename MultiPoint>
std::ostream& write_multi_point_WKT(std::ostream& out,
MultiPoint& mp)
{
if(!out.good())
return out;
internal::Geometry_container<MultiPoint, boost::geometry::multi_point_tag> gc(mp);
out << boost::geometry::wkt(gc) << std::endl;
return out;
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief writes the content of `polygons` into a WKT stream.
//!
//! \tparam MultiPolygon must be a `RandomAccessRange` of `CGAL::General_polygon_with_holes_2`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//!\see `CGAL::General_polygon_with_holes_2`
template<typename MultiPolygon>
std::ostream& write_multi_polygon_WKT(std::ostream& out,
MultiPolygon& polygons)
{
if(!out.good())
return out;
internal::Geometry_container<MultiPolygon, boost::geometry::multi_polygon_tag> gc(polygons);
out << boost::geometry::wkt(gc) << std::endl;
return out;
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! \brief writes the content of `mls` into a WKT stream.
//!
//! \tparam MultiLineString must be a `RandomAccessRange` of `LineString`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::write_linestring_WKT()`
template<typename MultiLineString>
std::ostream& write_multi_linestring_WKT(std::ostream& out,
MultiLineString& mls)
{
if(!out.good())
return out;
typedef typename MultiLineString::value_type PointRange;
typedef internal::Geometry_container<PointRange, boost::geometry::linestring_tag> LineString;
std::vector<LineString> pr_range;
for(PointRange& pr : mls)
{
LineString ls(pr);
pr_range.push_back(ls);
}
internal::Geometry_container<std::vector<LineString>, boost::geometry::multi_linestring_tag> gc(pr_range);
out << boost::geometry::wkt(gc) << std::endl;
return out;
}
//! \ingroup PkgStreamSupportIoFuncsWKT
//!
//! reads the content of a WKT stream and fills `points`, `polylines` and `polygons`
//! with all the POINT, MULTIPOINT, LINESTRING, MULTILINESTRING, POLYGON and MULTIPOLYGON it finds in `input`.
//!
//! \tparam MultiPoint must be a model of `RandomAccessRange` of `CGAL::Point_2` or `CGAL::Point_3`.
//! \tparam MultiLineString must be a `RandomAccessRange` of `Linestring`.
//! \tparam MultiPolygon must be a model of `RandomAccessRange` of `CGAL::General_polygon_with_holes_2`.
//!
//! \attention Only Cartesian Kernels with double or float as `FT` are supported.
//!
//! \see `CGAL::read_linestring_WKT()`
template<typename MultiPoint,
typename MultiLineString,
typename MultiPolygon>
bool read_WKT(std::istream& is,
MultiPoint& points,
MultiLineString& polylines,
MultiPolygon& polygons)
{
if(!is.good())
return false;
do
{
typedef typename MultiPoint::value_type Point;
typedef typename MultiLineString::value_type LineString;
typedef typename MultiPolygon::value_type Polygon;
std::string line;
std::streampos input_pos = is.tellg();
std::getline(is, line);
std::istringstream iss(line);
std::string t;
std::string type="";
iss >> t;
for(std::size_t pos=0; pos < t.length(); ++pos)
{
char c = t[pos];
if(c == '(')
break;
type.push_back(c);
}
is.seekg(input_pos);
if(type == "POINT")
{
Point p;
CGAL::read_point_WKT(is, p);
points.push_back(p);
}
else if(type == "LINESTRING")
{
LineString l;
CGAL::read_linestring_WKT(is, l);
polylines.push_back(l);
}
else if(type == "POLYGON")
{
Polygon p;
CGAL::read_polygon_WKT(is, p);
if(!p.outer_boundary().is_empty())
polygons.push_back(p);
}
else if(type == "MULTIPOINT")
{
MultiPoint mp;
CGAL::read_multi_point_WKT(is, mp);
for(const Point& point : mp)
points.push_back(point);
}
else if(type == "MULTILINESTRING")
{
MultiLineString mls;
CGAL::read_multi_linestring_WKT(is, mls);
for(const LineString& ls : mls)
polylines.push_back(ls);
}
else if(type == "MULTIPOLYGON")
{
MultiPolygon mp;
CGAL::read_multi_polygon_WKT(is, mp);
for(const Polygon& poly : mp)
polygons.push_back(poly);
}
}
while(is.good() && !is.eof());
return !is.fail();
}
} // namespace CGAL
#endif // BOOST VERSION CHECKS
#endif // CGAL_IO_WKT_H