1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015 |
- #ifndef BOOST_GRAPH_GRID_GRAPH_HPP
- #define BOOST_GRAPH_GRID_GRAPH_HPP
- #include <cmath>
- #include <functional>
- #include <numeric>
- #include <boost/array.hpp>
- #include <boost/bind.hpp>
- #include <boost/limits.hpp>
- #include <boost/graph/graph_traits.hpp>
- #include <boost/graph/properties.hpp>
- #include <boost/iterator/counting_iterator.hpp>
- #include <boost/iterator/transform_iterator.hpp>
- #include <boost/property_map/property_map.hpp>
- #define BOOST_GRID_GRAPH_TEMPLATE_PARAMS \
- std::size_t DimensionsT, typename VertexIndexT, \
- typename EdgeIndexT
- #define BOOST_GRID_GRAPH_TYPE \
- grid_graph<DimensionsT, VertexIndexT, EdgeIndexT>
- #define BOOST_GRID_GRAPH_TRAITS_T \
- typename graph_traits<BOOST_GRID_GRAPH_TYPE >
- namespace boost {
-
- template<BOOST_GRID_GRAPH_TEMPLATE_PARAMS>
- class grid_graph;
-
-
-
- template <typename Graph,
- typename Descriptor,
- typename Index>
- struct grid_graph_index_map {
- public:
- typedef Index value_type;
- typedef Index reference_type;
- typedef reference_type reference;
- typedef Descriptor key_type;
- typedef readable_property_map_tag category;
- grid_graph_index_map() { }
- grid_graph_index_map(const Graph& graph) :
- m_graph(&graph) { }
- value_type operator[](key_type key) const {
- return (m_graph->index_of(key));
- }
- friend inline Index
- get(const grid_graph_index_map<Graph, Descriptor, Index>& index_map,
- const typename grid_graph_index_map<Graph, Descriptor, Index>::key_type& key)
- {
- return (index_map[key]);
- }
- protected:
- const Graph* m_graph;
- };
- template<BOOST_GRID_GRAPH_TEMPLATE_PARAMS>
- struct property_map<BOOST_GRID_GRAPH_TYPE, vertex_index_t> {
- typedef grid_graph_index_map<BOOST_GRID_GRAPH_TYPE,
- BOOST_GRID_GRAPH_TRAITS_T::vertex_descriptor,
- BOOST_GRID_GRAPH_TRAITS_T::vertices_size_type> type;
- typedef type const_type;
- };
- template<BOOST_GRID_GRAPH_TEMPLATE_PARAMS>
- struct property_map<BOOST_GRID_GRAPH_TYPE, edge_index_t> {
- typedef grid_graph_index_map<BOOST_GRID_GRAPH_TYPE,
- BOOST_GRID_GRAPH_TRAITS_T::edge_descriptor,
- BOOST_GRID_GRAPH_TRAITS_T::edges_size_type> type;
- typedef type const_type;
- };
-
-
-
- template <typename Descriptor>
- struct grid_graph_reverse_edge_map {
- public:
- typedef Descriptor value_type;
- typedef Descriptor reference_type;
- typedef reference_type reference;
- typedef Descriptor key_type;
- typedef readable_property_map_tag category;
- grid_graph_reverse_edge_map() { }
- value_type operator[](const key_type& key) const {
- return (value_type(key.second, key.first));
- }
- friend inline Descriptor
- get(const grid_graph_reverse_edge_map<Descriptor>& rev_map,
- const typename grid_graph_reverse_edge_map<Descriptor>::key_type& key)
- {
- return (rev_map[key]);
- }
- };
- template<BOOST_GRID_GRAPH_TEMPLATE_PARAMS>
- struct property_map<BOOST_GRID_GRAPH_TYPE, edge_reverse_t> {
- typedef grid_graph_reverse_edge_map<BOOST_GRID_GRAPH_TRAITS_T::edge_descriptor> type;
- typedef type const_type;
- };
-
-
-
- namespace detail {
-
- template <typename Graph>
- struct grid_graph_vertex_at {
- typedef typename graph_traits<Graph>::vertex_descriptor result_type;
- grid_graph_vertex_at() : m_graph(0) {}
- grid_graph_vertex_at(const Graph* graph) :
- m_graph(graph) { }
- result_type
- operator()
- (typename graph_traits<Graph>::vertices_size_type vertex_index) const {
- return (vertex(vertex_index, *m_graph));
- }
- private:
- const Graph* m_graph;
- };
-
- template <typename Graph>
- struct grid_graph_out_edge_at {
- private:
- typedef typename graph_traits<Graph>::vertex_descriptor vertex_descriptor;
- public:
- typedef typename graph_traits<Graph>::edge_descriptor result_type;
- grid_graph_out_edge_at() : m_vertex(), m_graph(0) {}
- grid_graph_out_edge_at(vertex_descriptor source_vertex,
- const Graph* graph) :
- m_vertex(source_vertex),
- m_graph(graph) { }
- result_type
- operator()
- (typename graph_traits<Graph>::degree_size_type out_edge_index) const {
- return (out_edge_at(m_vertex, out_edge_index, *m_graph));
- }
- private:
- vertex_descriptor m_vertex;
- const Graph* m_graph;
- };
-
- template <typename Graph>
- struct grid_graph_in_edge_at {
- private:
- typedef typename graph_traits<Graph>::vertex_descriptor vertex_descriptor;
- public:
- typedef typename graph_traits<Graph>::edge_descriptor result_type;
- grid_graph_in_edge_at() : m_vertex(), m_graph(0) {}
- grid_graph_in_edge_at(vertex_descriptor target_vertex,
- const Graph* graph) :
- m_vertex(target_vertex),
- m_graph(graph) { }
- result_type
- operator()
- (typename graph_traits<Graph>::degree_size_type in_edge_index) const {
- return (in_edge_at(m_vertex, in_edge_index, *m_graph));
- }
- private:
- vertex_descriptor m_vertex;
- const Graph* m_graph;
- };
-
- template <typename Graph>
- struct grid_graph_edge_at {
- typedef typename graph_traits<Graph>::edge_descriptor result_type;
- grid_graph_edge_at() : m_graph(0) {}
- grid_graph_edge_at(const Graph* graph) :
- m_graph(graph) { }
- result_type
- operator()
- (typename graph_traits<Graph>::edges_size_type edge_index) const {
- return (edge_at(edge_index, *m_graph));
- }
- private:
- const Graph* m_graph;
- };
-
- template <typename Graph>
- struct grid_graph_adjacent_vertex_at {
- public:
- typedef typename graph_traits<Graph>::vertex_descriptor result_type;
- grid_graph_adjacent_vertex_at(result_type source_vertex,
- const Graph* graph) :
- m_vertex(source_vertex),
- m_graph(graph) { }
- result_type
- operator()
- (typename graph_traits<Graph>::degree_size_type adjacent_index) const {
- return (target(out_edge_at(m_vertex, adjacent_index, *m_graph), *m_graph));
- }
- private:
- result_type m_vertex;
- const Graph* m_graph;
- };
- }
-
-
-
- template <std::size_t Dimensions,
- typename VertexIndex = std::size_t,
- typename EdgeIndex = VertexIndex>
- class grid_graph {
- private:
- typedef boost::array<bool, Dimensions> WrapDimensionArray;
- grid_graph() { };
- public:
- typedef grid_graph<Dimensions, VertexIndex, EdgeIndex> type;
-
- typedef VertexIndex vertices_size_type;
- typedef EdgeIndex edges_size_type;
- typedef EdgeIndex degree_size_type;
-
- typedef boost::array<VertexIndex, Dimensions> vertex_descriptor;
- typedef std::pair<vertex_descriptor, vertex_descriptor> edge_descriptor;
-
- typedef counting_iterator<vertices_size_type> vertex_index_iterator;
- typedef detail::grid_graph_vertex_at<type> vertex_function;
- typedef transform_iterator<vertex_function, vertex_index_iterator> vertex_iterator;
-
- typedef counting_iterator<edges_size_type> edge_index_iterator;
- typedef detail::grid_graph_edge_at<type> edge_function;
- typedef transform_iterator<edge_function, edge_index_iterator> edge_iterator;
-
- typedef counting_iterator<degree_size_type> degree_iterator;
- typedef detail::grid_graph_out_edge_at<type> out_edge_function;
- typedef transform_iterator<out_edge_function, degree_iterator> out_edge_iterator;
-
- typedef detail::grid_graph_in_edge_at<type> in_edge_function;
- typedef transform_iterator<in_edge_function, degree_iterator> in_edge_iterator;
-
- typedef detail::grid_graph_adjacent_vertex_at<type> adjacent_vertex_function;
- typedef transform_iterator<adjacent_vertex_function, degree_iterator> adjacency_iterator;
-
- typedef directed_tag directed_category;
- typedef disallow_parallel_edge_tag edge_parallel_category;
- struct traversal_category : virtual public incidence_graph_tag,
- virtual public adjacency_graph_tag,
- virtual public vertex_list_graph_tag,
- virtual public edge_list_graph_tag,
- virtual public bidirectional_graph_tag,
- virtual public adjacency_matrix_tag { };
- static inline vertex_descriptor null_vertex()
- {
- vertex_descriptor maxed_out_vertex;
- std::fill(maxed_out_vertex.begin(), maxed_out_vertex.end(),
- (std::numeric_limits<vertices_size_type>::max)());
- return (maxed_out_vertex);
- }
-
- grid_graph(vertex_descriptor dimension_lengths) :
- m_dimension_lengths(dimension_lengths) {
- std::fill(m_wrap_dimension.begin(),
- m_wrap_dimension.end(), false);
- precalculate();
- }
-
-
- grid_graph(vertex_descriptor dimension_lengths,
- bool wrap_all_dimensions) :
- m_dimension_lengths(dimension_lengths) {
-
- std::fill(m_wrap_dimension.begin(),
- m_wrap_dimension.end(),
- wrap_all_dimensions);
- precalculate();
- }
-
-
- grid_graph(vertex_descriptor dimension_lengths,
- WrapDimensionArray wrap_dimension) :
- m_dimension_lengths(dimension_lengths),
- m_wrap_dimension(wrap_dimension) {
- precalculate();
- }
-
- inline std::size_t dimensions() const {
- return (Dimensions);
- }
-
- inline vertices_size_type length(std::size_t dimension) const {
- return (m_dimension_lengths[dimension]);
- }
-
- inline bool wrapped(std::size_t dimension) const {
- return (m_wrap_dimension[dimension]);
- }
-
-
- vertex_descriptor next
- (vertex_descriptor vertex,
- std::size_t dimension_index,
- vertices_size_type distance = 1) const {
- vertices_size_type new_position =
- vertex[dimension_index] + distance;
- if (wrapped(dimension_index)) {
- new_position %= length(dimension_index);
- }
- else {
-
- new_position =
- (std::min)(new_position,
- vertices_size_type(length(dimension_index) - 1));
- }
- vertex[dimension_index] = new_position;
- return (vertex);
- }
-
-
- vertex_descriptor previous
- (vertex_descriptor vertex,
- std::size_t dimension_index,
- vertices_size_type distance = 1) const {
-
-
-
- vertex[dimension_index] =
- (distance > vertex[dimension_index]) ?
- (wrapped(dimension_index) ?
- (length(dimension_index) - (distance % length(dimension_index))) : 0) :
- vertex[dimension_index] - distance;
- return (vertex);
- }
- protected:
-
- inline vertices_size_type num_vertices() const {
- return (m_num_vertices);
- }
-
-
- inline edges_size_type num_edges() const {
- return (m_num_edges);
- }
-
- inline edges_size_type num_edges
- (std::size_t dimension_index) const {
- return (m_edge_count[dimension_index]);
- }
-
- vertices_size_type index_of(vertex_descriptor vertex) const {
- vertices_size_type vertex_index = 0;
- vertices_size_type index_multiplier = 1;
- for (std::size_t dimension_index = 0;
- dimension_index < Dimensions;
- ++dimension_index) {
- vertex_index += (vertex[dimension_index] * index_multiplier);
- index_multiplier *= length(dimension_index);
- }
- return (vertex_index);
- }
-
-
- vertex_descriptor vertex_at
- (vertices_size_type vertex_index) const {
-
- boost::array<vertices_size_type, Dimensions> vertex;
- vertices_size_type index_divider = 1;
- for (std::size_t dimension_index = 0;
- dimension_index < Dimensions;
- ++dimension_index) {
- vertex[dimension_index] = (vertex_index / index_divider) %
- length(dimension_index);
- index_divider *= length(dimension_index);
- }
- return (vertex);
- }
-
-
-
- edge_descriptor edge_at(edges_size_type edge_index) const {
-
- std::size_t dimension_index = 0;
- edges_size_type dimension_edges = num_edges(0);
- while (edge_index >= dimension_edges) {
- edge_index -= dimension_edges;
- ++dimension_index;
- dimension_edges = num_edges(dimension_index);
- }
- vertex_descriptor vertex_source, vertex_target;
- bool is_forward = ((edge_index / (num_edges(dimension_index) / 2)) == 0);
- if (wrapped(dimension_index)) {
- vertex_source = vertex_at(edge_index % num_vertices());
- vertex_target = is_forward ?
- next(vertex_source, dimension_index) :
- previous(vertex_source, dimension_index);
- }
- else {
-
-
-
-
-
-
- edges_size_type vertex_offset = edge_index % num_edges(dimension_index);
-
-
-
- vertices_size_type index_divider = 1;
- for (std::size_t dimension_index_iter = 0;
- dimension_index_iter < Dimensions;
- ++dimension_index_iter) {
- std::size_t dimension_length = (dimension_index_iter == dimension_index) ?
- length(dimension_index_iter) - 1 :
- length(dimension_index_iter);
- vertex_source[dimension_index_iter] = (vertex_offset / index_divider) %
- dimension_length;
- index_divider *= dimension_length;
- }
- if (is_forward) {
- vertex_target = next(vertex_source, dimension_index);
- }
- else {
-
-
- vertex_target = vertex_source;
- ++vertex_source[dimension_index];
- }
- }
-
- return (std::make_pair(vertex_source, vertex_target));
- }
-
-
- edges_size_type index_of(edge_descriptor edge) const {
- vertex_descriptor source_vertex = source(edge, *this);
- vertex_descriptor target_vertex = target(edge, *this);
- BOOST_ASSERT (source_vertex != target_vertex);
-
-
- std::size_t different_dimension_index = 0;
- while (source_vertex[different_dimension_index] ==
- target_vertex[different_dimension_index]) {
- ++different_dimension_index;
- }
- edges_size_type edge_index = 0;
-
-
-
- for (std::size_t dimension_index = 0;
- dimension_index < different_dimension_index;
- ++dimension_index) {
- edge_index += num_edges(dimension_index);
- }
-
- vertices_size_type source_position = source_vertex[different_dimension_index];
- vertices_size_type target_position = target_vertex[different_dimension_index];
-
- bool is_forward = true;
-
- if (wrapped(different_dimension_index)) {
-
-
-
-
- if (((target_position < source_position) &&
- ((source_position - target_position) == 1)) ||
- ((source_position < target_position) &&
- ((target_position - source_position) > 1))) {
- is_forward = false;
- }
- }
- else if (target_position < source_position) {
- is_forward = false;
- }
-
- if (!is_forward) {
- edge_index += (num_edges(different_dimension_index) / 2);
- }
-
- if (wrapped(different_dimension_index)) {
- edge_index += index_of(source_vertex);
- }
- else {
- vertices_size_type index_multiplier = 1;
- if (!is_forward) {
- --source_vertex[different_dimension_index];
- }
- for (std::size_t dimension_index = 0;
- dimension_index < Dimensions;
- ++dimension_index) {
- edge_index += (source_vertex[dimension_index] * index_multiplier);
- index_multiplier *= (dimension_index == different_dimension_index) ?
- length(dimension_index) - 1 :
- length(dimension_index);
- }
- }
- return (edge_index);
- }
-
- degree_size_type out_degree(vertex_descriptor vertex) const {
- degree_size_type out_edge_count = 0;
- for (std::size_t dimension_index = 0;
- dimension_index < Dimensions;
- ++dimension_index) {
-
-
-
- if ((vertex[dimension_index] == 0) ||
- (vertex[dimension_index] == (length(dimension_index) - 1))) {
- out_edge_count += (wrapped(dimension_index) ? 2 : 1);
- }
- else {
-
- out_edge_count += 2;
- }
- }
- return (out_edge_count);
- }
-
-
- edge_descriptor out_edge_at
- (vertex_descriptor vertex,
- degree_size_type out_edge_index) const {
- edges_size_type edges_left = out_edge_index + 1;
- std::size_t dimension_index = 0;
- bool is_forward = false;
-
-
- while (edges_left > 0) {
- if (!wrapped(dimension_index)) {
- if (!is_forward && (vertex[dimension_index] == 0)) {
- is_forward = true;
- continue;
- }
- else if (is_forward &&
- (vertex[dimension_index] == (length(dimension_index) - 1))) {
- is_forward = false;
- ++dimension_index;
- continue;
- }
- }
- --edges_left;
- if (edges_left > 0) {
- is_forward = !is_forward;
-
- if (!is_forward) {
- ++dimension_index;
- }
- }
- }
- return (std::make_pair(vertex, is_forward ?
- next(vertex, dimension_index) :
- previous(vertex, dimension_index)));
- }
-
- inline degree_size_type in_degree(vertex_descriptor vertex) const {
- return (out_degree(vertex));
- }
-
-
- edge_descriptor in_edge_at
- (vertex_descriptor vertex,
- edges_size_type in_edge_index) const {
- edge_descriptor out_edge = out_edge_at(vertex, in_edge_index);
- return (std::make_pair(target(out_edge, *this), source(out_edge, *this)));
- }
-
- void precalculate() {
- m_num_vertices =
- std::accumulate(m_dimension_lengths.begin(),
- m_dimension_lengths.end(),
- vertices_size_type(1),
- std::multiplies<vertices_size_type>());
-
- m_num_edges = 0;
- for (std::size_t dimension_index = 0;
- dimension_index < Dimensions;
- ++dimension_index) {
- if (wrapped(dimension_index)) {
- m_edge_count[dimension_index] = num_vertices() * 2;
- }
- else {
- m_edge_count[dimension_index] =
- (num_vertices() - (num_vertices() / length(dimension_index))) * 2;
- }
- m_num_edges += num_edges(dimension_index);
- }
- }
- const vertex_descriptor m_dimension_lengths;
- WrapDimensionArray m_wrap_dimension;
- vertices_size_type m_num_vertices;
- boost::array<edges_size_type, Dimensions> m_edge_count;
- edges_size_type m_num_edges;
- public:
-
-
-
- friend inline std::pair<typename type::vertex_iterator,
- typename type::vertex_iterator>
- vertices(const type& graph) {
- typedef typename type::vertex_iterator vertex_iterator;
- typedef typename type::vertex_function vertex_function;
- typedef typename type::vertex_index_iterator vertex_index_iterator;
- return (std::make_pair
- (vertex_iterator(vertex_index_iterator(0),
- vertex_function(&graph)),
- vertex_iterator(vertex_index_iterator(graph.num_vertices()),
- vertex_function(&graph))));
- }
- friend inline typename type::vertices_size_type
- num_vertices(const type& graph) {
- return (graph.num_vertices());
- }
- friend inline typename type::vertex_descriptor
- vertex(typename type::vertices_size_type vertex_index,
- const type& graph) {
- return (graph.vertex_at(vertex_index));
- }
-
-
-
- friend inline std::pair<typename type::out_edge_iterator,
- typename type::out_edge_iterator>
- out_edges(typename type::vertex_descriptor vertex,
- const type& graph) {
- typedef typename type::degree_iterator degree_iterator;
- typedef typename type::out_edge_function out_edge_function;
- typedef typename type::out_edge_iterator out_edge_iterator;
- return (std::make_pair
- (out_edge_iterator(degree_iterator(0),
- out_edge_function(vertex, &graph)),
- out_edge_iterator(degree_iterator(graph.out_degree(vertex)),
- out_edge_function(vertex, &graph))));
- }
- friend inline typename type::degree_size_type
- out_degree
- (typename type::vertex_descriptor vertex,
- const type& graph) {
- return (graph.out_degree(vertex));
- }
- friend inline typename type::edge_descriptor
- out_edge_at(typename type::vertex_descriptor vertex,
- typename type::degree_size_type out_edge_index,
- const type& graph) {
- return (graph.out_edge_at(vertex, out_edge_index));
- }
-
-
-
- friend typename std::pair<typename type::adjacency_iterator,
- typename type::adjacency_iterator>
- adjacent_vertices (typename type::vertex_descriptor vertex,
- const type& graph) {
- typedef typename type::degree_iterator degree_iterator;
- typedef typename type::adjacent_vertex_function adjacent_vertex_function;
- typedef typename type::adjacency_iterator adjacency_iterator;
- return (std::make_pair
- (adjacency_iterator(degree_iterator(0),
- adjacent_vertex_function(vertex, &graph)),
- adjacency_iterator(degree_iterator(graph.out_degree(vertex)),
- adjacent_vertex_function(vertex, &graph))));
- }
-
-
-
- friend inline typename type::edges_size_type
- num_edges(const type& graph) {
- return (graph.num_edges());
- }
- friend inline typename type::edge_descriptor
- edge_at(typename type::edges_size_type edge_index,
- const type& graph) {
- return (graph.edge_at(edge_index));
- }
- friend inline std::pair<typename type::edge_iterator,
- typename type::edge_iterator>
- edges(const type& graph) {
- typedef typename type::edge_index_iterator edge_index_iterator;
- typedef typename type::edge_function edge_function;
- typedef typename type::edge_iterator edge_iterator;
- return (std::make_pair
- (edge_iterator(edge_index_iterator(0),
- edge_function(&graph)),
- edge_iterator(edge_index_iterator(graph.num_edges()),
- edge_function(&graph))));
- }
-
-
-
- friend inline std::pair<typename type::in_edge_iterator,
- typename type::in_edge_iterator>
- in_edges(typename type::vertex_descriptor vertex,
- const type& graph) {
- typedef typename type::in_edge_function in_edge_function;
- typedef typename type::degree_iterator degree_iterator;
- typedef typename type::in_edge_iterator in_edge_iterator;
- return (std::make_pair
- (in_edge_iterator(degree_iterator(0),
- in_edge_function(vertex, &graph)),
- in_edge_iterator(degree_iterator(graph.in_degree(vertex)),
- in_edge_function(vertex, &graph))));
- }
- friend inline typename type::degree_size_type
- in_degree (typename type::vertex_descriptor vertex,
- const type& graph) {
- return (graph.in_degree(vertex));
- }
- friend inline typename type::degree_size_type
- degree (typename type::vertex_descriptor vertex,
- const type& graph) {
- return (graph.out_degree(vertex) * 2);
- }
- friend inline typename type::edge_descriptor
- in_edge_at(typename type::vertex_descriptor vertex,
- typename type::degree_size_type in_edge_index,
- const type& graph) {
- return (graph.in_edge_at(vertex, in_edge_index));
- }
-
-
-
- friend std::pair<typename type::edge_descriptor, bool>
- edge (typename type::vertex_descriptor source_vertex,
- typename type::vertex_descriptor destination_vertex,
- const type& graph) {
- std::pair<typename type::edge_descriptor, bool> edge_exists =
- std::make_pair(std::make_pair(source_vertex, destination_vertex), false);
- for (std::size_t dimension_index = 0;
- dimension_index < Dimensions;
- ++dimension_index) {
- typename type::vertices_size_type dim_difference = 0;
- typename type::vertices_size_type
- source_dim = source_vertex[dimension_index],
- dest_dim = destination_vertex[dimension_index];
- dim_difference = (source_dim > dest_dim) ?
- (source_dim - dest_dim) : (dest_dim - source_dim);
- if (dim_difference > 0) {
-
-
-
- if (edge_exists.second) {
- edge_exists.second = false;
- break;
- }
-
-
-
-
- if ((dim_difference == 1) ||
- (graph.wrapped(dimension_index) &&
- (((source_dim == 0) && (dest_dim == (graph.length(dimension_index) - 1))) ||
- ((dest_dim == 0) && (source_dim == (graph.length(dimension_index) - 1)))))) {
- edge_exists.second = true;
-
- }
- else {
-
- edge_exists.second = false;
- break;
- }
- }
- }
- return (edge_exists);
- }
-
-
-
- friend inline typename type::vertices_size_type
- get(vertex_index_t,
- const type& graph,
- typename type::vertex_descriptor vertex) {
- return (graph.index_of(vertex));
- }
- friend inline typename type::edges_size_type
- get(edge_index_t,
- const type& graph,
- typename type::edge_descriptor edge) {
- return (graph.index_of(edge));
- }
- friend inline grid_graph_index_map<
- type,
- typename type::vertex_descriptor,
- typename type::vertices_size_type>
- get(vertex_index_t, const type& graph) {
- return (grid_graph_index_map<
- type,
- typename type::vertex_descriptor,
- typename type::vertices_size_type>(graph));
- }
- friend inline grid_graph_index_map<
- type,
- typename type::edge_descriptor,
- typename type::edges_size_type>
- get(edge_index_t, const type& graph) {
- return (grid_graph_index_map<
- type,
- typename type::edge_descriptor,
- typename type::edges_size_type>(graph));
- }
- friend inline grid_graph_reverse_edge_map<
- typename type::edge_descriptor>
- get(edge_reverse_t, const type& graph) {
- return (grid_graph_reverse_edge_map<
- typename type::edge_descriptor>());
- }
- template<typename Graph,
- typename Descriptor,
- typename Index>
- friend struct grid_graph_index_map;
- template<typename Descriptor>
- friend struct grid_graph_reverse_edge_map;
- };
- }
- #undef BOOST_GRID_GRAPH_TYPE
- #undef BOOST_GRID_GRAPH_TEMPLATE_PARAMS
- #undef BOOST_GRID_GRAPH_TRAITS_T
- #endif
|