-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathrect.rs
116 lines (95 loc) · 2.67 KB
/
rect.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
use std::io::Write;
use geo_traits::{CoordTrait, LineStringTrait, PolygonTrait, RectTrait};
use crate::error::WKBResult;
use crate::writer::{polygon_wkb_size, write_polygon};
use crate::Endianness;
struct Coord2D {
x: f64,
y: f64,
}
impl CoordTrait for Coord2D {
type T = f64;
fn x(&self) -> Self::T {
self.x
}
fn y(&self) -> Self::T {
self.y
}
fn dim(&self) -> geo_traits::Dimensions {
geo_traits::Dimensions::Xy
}
fn nth_or_panic(&self, n: usize) -> Self::T {
match n {
0 => self.x,
1 => self.y,
_ => panic!(),
}
}
}
/// A wrapper around an impl RectTrait to provide LineStringTrait and PolygonTrait
struct RectWrapper<'a, G: RectTrait<T = f64>>(&'a G);
impl<'a, G: RectTrait<T = f64>> LineStringTrait for &'a RectWrapper<'a, G> {
type T = f64;
type CoordType<'b> = Coord2D where G: 'b, Self: 'b;
fn dim(&self) -> geo_traits::Dimensions {
self.0.dim()
}
fn num_coords(&self) -> usize {
5
}
unsafe fn coord_unchecked(&self, i: usize) -> Self::CoordType<'_> {
let min_coord = self.0.min();
let max_coord = self.0.max();
match i {
0 => Coord2D {
x: min_coord.x(),
y: min_coord.y(),
},
1 => Coord2D {
x: min_coord.x(),
y: max_coord.y(),
},
2 => Coord2D {
x: max_coord.x(),
y: max_coord.y(),
},
3 => Coord2D {
x: max_coord.x(),
y: min_coord.y(),
},
4 => Coord2D {
x: min_coord.x(),
y: min_coord.y(),
},
_ => unreachable!(),
}
}
}
impl<'a, G: RectTrait<T = f64>> PolygonTrait for RectWrapper<'a, G> {
type T = f64;
type RingType<'b> = &'b RectWrapper<'b, G> where G: 'b, Self: 'b;
fn dim(&self) -> geo_traits::Dimensions {
self.0.dim()
}
fn exterior(&self) -> Option<Self::RingType<'_>> {
Some(self)
}
fn num_interiors(&self) -> usize {
0
}
unsafe fn interior_unchecked(&self, _i: usize) -> Self::RingType<'_> {
unreachable!()
}
}
/// The number of bytes this Rect will take up when encoded as WKB
pub fn rect_wkb_size(geom: &impl RectTrait<T = f64>) -> usize {
polygon_wkb_size(&RectWrapper(geom))
}
/// Write a Rect geometry to a Writer encoded as WKB
pub fn write_rect(
writer: &mut impl Write,
geom: &impl RectTrait<T = f64>,
endianness: Endianness,
) -> WKBResult<()> {
write_polygon(writer, &RectWrapper(geom), endianness)
}