This repository has been archived by the owner on Jan 25, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 5
/
geo.php
101 lines (81 loc) · 2.62 KB
/
geo.php
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
<?php
/**
* Kirby GEO Plugin
*
* @author Bastian Allgeier <[email protected]>
*/
use Kirby\Geo;
/**
* Autoloader for all Kirby GEO Classes
*/
load([
'kirby\\geo' => __DIR__ . DS . 'lib' . DS . 'geo.php',
'kirby\\geo\\point' => __DIR__ . DS . 'lib' . DS . 'geo' . DS . 'point.php'
]);
/**
* Creates a class alias for the GEO class, to make it more usable
*/
class_alias('Kirby\\Geo', 'Geo');
/**
* Adds a new radius filter to all collections
*/
collection::$filters['radius'] = function($collection, $field, $options) {
$origin = geo::point(a::get($options, 'lat'), a::get($options, 'lng'));
$radius = intval(a::get($options, 'radius'));
$unit = a::get($options, 'unit', 'km') === 'km' ? 'km' : 'mi';
if(!$origin) {
throw new Exception('Invalid geo point for radius filter. You must specify valid lat and lng values');
}
if($radius === 0) {
throw new Exception('Invalid radius value for radius filter. You must specify a valid integer value');
}
foreach($collection->data as $key => $item) {
$value = collection::extractValue($item, $field);
// skip invalid points
if(!is_string($value) and !is_a($value, 'Field')) {
unset($collection->$key);
continue;
}
try {
$point = geo::point((string)$value);
} catch(Exception $e) {
unset($collection->$key);
continue;
}
$distance = geo::distance($origin, $point, $unit);
if($distance > $radius) {
unset($collection->$key);
}
}
return $collection;
};
/**
* Adds a new field method "coordinates",
* which can be used to convert a field with
* comma separated lat and long values to a Kirby Geo Point
*/
field::$methods['coordinates'] = function($field) {
return geo::point($field->value);
};
/**
* Adds a new field method "distance",
* which can be used to calculate the distance between a
* field with comma separated lat and long values and a
* valid Kirby Geo Point
*/
field::$methods['distance'] = function($field, $point, $unit = 'km') {
if(!is_a($point, 'Kirby\\Geo\\Point')) {
throw new Exception('You must pass a valid Geo Point object to measure the distance');
}
return geo::distance($field->coordinates(), $point, $unit);
};
/**
* Same as distance, but will return a human readable version
* of the distance instead of a long float
*/
field::$methods['niceDistance'] = function($field, $point, $unit = 'km') {
if(!is_a($point, 'Kirby\\Geo\\Point')) {
throw new Exception('You must pass a valid Geo Point object to measure the distance');
}
return geo::niceDistance($field->coordinates(), $point, $unit);
};