xref: /plugin/openlayersmap/StaticMap.php (revision 504fe8a3556bd0092986954bd0af2315d06c9420)
1628e43ccSMark Prins<?php
2628e43ccSMark Prins/*
37a37f5a9SMark Prins * Copyright (c) 2012-2016 Mark C. Prins <mprins@users.sf.net>
4628e43ccSMark Prins *
5f4b9bdacSMark Prins * In part based on staticMapLite 0.03 available at http://staticmaplite.svn.sourceforge.net/viewvc/staticmaplite/
6628e43ccSMark Prins *
7628e43ccSMark Prins * Copyright (c) 2009 Gerhard Koch <gerhard.koch AT ymail.com>
8628e43ccSMark Prins *
9628e43ccSMark Prins * Licensed under the Apache License, Version 2.0 (the "License");
10628e43ccSMark Prins * you may not use this file except in compliance with the License.
11628e43ccSMark Prins * You may obtain a copy of the License at
12628e43ccSMark Prins *
13628e43ccSMark Prins *     http://www.apache.org/licenses/LICENSE-2.0
14628e43ccSMark Prins *
15628e43ccSMark Prins * Unless required by applicable law or agreed to in writing, software
16628e43ccSMark Prins * distributed under the License is distributed on an "AS IS" BASIS,
17628e43ccSMark Prins * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18628e43ccSMark Prins * See the License for the specific language governing permissions and
19628e43ccSMark Prins * limitations under the License.
20628e43ccSMark Prins */
21f4b9bdacSMark Prinsinclude_once (realpath ( dirname ( __FILE__ ) ) . '/../geophp/geoPHP/geoPHP.inc');
22628e43ccSMark Prins/**
23ab8cbd2bSMark Prins *
24628e43ccSMark Prins * @author Mark C. Prins <mprins@users.sf.net>
25628e43ccSMark Prins * @author Gerhard Koch <gerhard.koch AT ymail.com>
26628e43ccSMark Prins *
27628e43ccSMark Prins */
28628e43ccSMark Prinsclass StaticMap {
29ab8cbd2bSMark Prins	// this should probably not be changed
30628e43ccSMark Prins	protected $tileSize = 256;
31628e43ccSMark Prins
32628e43ccSMark Prins	// the final output
33628e43ccSMark Prins	var $doc = '';
34628e43ccSMark Prins
35628e43ccSMark Prins	protected $tileInfo = array (
36628e43ccSMark Prins			// OSM sources
37628e43ccSMark Prins			'openstreetmap' => array (
38628e43ccSMark Prins					'txt' => '(c) OpenStreetMap CC-BY-SA',
39628e43ccSMark Prins					'logo' => 'osm_logo.png',
40ab8cbd2bSMark Prins					'url' => 'http://tile.openstreetmap.org/{Z}/{X}/{Y}.png'
41ab8cbd2bSMark Prins			),
42628e43ccSMark Prins			// OCM sources
43628e43ccSMark Prins			'cycle' => array (
44628e43ccSMark Prins					'txt' => 'OpenCycleMap tiles',
45628e43ccSMark Prins					'logo' => 'cycle_logo.png',
46ab8cbd2bSMark Prins					'url' => 'http://tile.opencyclemap.org/cycle/{Z}/{X}/{Y}.png'
47ab8cbd2bSMark Prins			),
48628e43ccSMark Prins			'transport' => array (
49628e43ccSMark Prins					'txt' => 'OpenCycleMap tiles',
50628e43ccSMark Prins					'logo' => 'cycle_logo.png',
51ab8cbd2bSMark Prins					'url' => 'http://tile2.opencyclemap.org/transport/{Z}/{X}/{Y}.png'
52ab8cbd2bSMark Prins			),
53628e43ccSMark Prins			'landscape' => array (
54628e43ccSMark Prins					'txt' => 'OpenCycleMap tiles',
55628e43ccSMark Prins					'logo' => 'cycle_logo.png',
56ab8cbd2bSMark Prins					'url' => 'http://tile3.opencyclemap.org/landscape/{Z}/{X}/{Y}.png'
57ab8cbd2bSMark Prins			),
58e72390dbSMark Prins			'toner-lite' => array (
59e72390dbSMark Prins					'txt' => 'OpenCycleMap tiles',
60e72390dbSMark Prins					'logo' => 'stamen.png',
61e72390dbSMark Prins					'url' => 'http://tile.stamen.com/toner-lite/{Z}/{X}/{Y}.png'
62e72390dbSMark Prins			),
63e72390dbSMark Prins			'terrain' => array (
64e72390dbSMark Prins					'txt' => 'OpenCycleMap tiles',
65e72390dbSMark Prins					'logo' => 'stamen.png',
66e72390dbSMark Prins					'url' => 'http://tile.stamen.com/terrain/{Z}/{X}/{Y}.png'
67ab8cbd2bSMark Prins			),
68628e43ccSMark Prins			// 'piste'=>array(
69628e43ccSMark Prins			// 'txt'=>'OpenPisteMap tiles',
70628e43ccSMark Prins			// 'logo'=>'piste_logo.png',
71628e43ccSMark Prins			// 'url'=>''),
72628e43ccSMark Prins			// 'sea'=>array(
73628e43ccSMark Prins			// 'txt'=>'OpenSeaMap tiles',
74628e43ccSMark Prins			// 'logo'=>'sea_logo.png',
75628e43ccSMark Prins			// 'url'=>''),
76e72390dbSMark Prins			// H&B sources
77e72390dbSMark Prins			'hikeandbike' => array (
78e72390dbSMark Prins					'txt' => 'Hike & Bike Map',
79e72390dbSMark Prins					'logo' => 'hnb_logo.png',
80*504fe8a3SMark Prins					//'url' => 'http://toolserver.org/tiles/hikebike/{Z}/{X}/{Y}.png'
81*504fe8a3SMark Prins					//moved to: https://www.toolserver.org/tiles/hikebike/12/2105/1388.png
82*504fe8a3SMark Prins					'url' => 'http://c.tiles.wmflabs.org/hikebike/{Z}/{X}/{Y}.png'
83ab8cbd2bSMark Prins			)
84628e43ccSMark Prins	);
85628e43ccSMark Prins	protected $tileDefaultSrc = 'openstreetmap';
86628e43ccSMark Prins
87628e43ccSMark Prins	// set up markers
88628e43ccSMark Prins	protected $markerPrototypes = array (
89628e43ccSMark Prins			// found at http://www.mapito.net/map-marker-icons.html
90628e43ccSMark Prins			// these are 17x19 px with a pointer at the bottom left
91ab8cbd2bSMark Prins			'lightblue' => array (
92ab8cbd2bSMark Prins					'regex' => '/^lightblue([0-9]+)$/',
93628e43ccSMark Prins					'extension' => '.png',
94628e43ccSMark Prins					'shadow' => false,
95628e43ccSMark Prins					'offsetImage' => '0,-19',
96628e43ccSMark Prins					'offsetShadow' => false
97628e43ccSMark Prins			),
98628e43ccSMark Prins			// openlayers std markers are 21x25px with shadow
99ab8cbd2bSMark Prins			'ol-marker' => array (
100ab8cbd2bSMark Prins					'regex' => '/^marker(|-blue|-gold|-green|-red)+$/',
101628e43ccSMark Prins					'extension' => '.png',
102628e43ccSMark Prins					'shadow' => 'marker_shadow.png',
103628e43ccSMark Prins					'offsetImage' => '-10,-25',
104628e43ccSMark Prins					'offsetShadow' => '-1,-13'
105628e43ccSMark Prins			),
106628e43ccSMark Prins			// these are 16x16 px
107ab8cbd2bSMark Prins			'ww_icon' => array (
108ab8cbd2bSMark Prins					'regex' => '/ww_\S+$/',
109628e43ccSMark Prins					'extension' => '.png',
110628e43ccSMark Prins					'shadow' => false,
111628e43ccSMark Prins					'offsetImage' => '-8,-8',
112628e43ccSMark Prins					'offsetShadow' => false
113628e43ccSMark Prins			),
114628e43ccSMark Prins			// assume these are 16x16 px
115ab8cbd2bSMark Prins			'rest' => array (
116ab8cbd2bSMark Prins					'regex' => '/^(?!lightblue([0-9]+)$)(?!(ww_\S+$))(?!marker(|-blue|-gold|-green|-red)+$)(.*)/',
117628e43ccSMark Prins					'extension' => '.png',
118628e43ccSMark Prins					'shadow' => 'marker_shadow.png',
119628e43ccSMark Prins					'offsetImage' => '-8,-8',
120628e43ccSMark Prins					'offsetShadow' => '-1,-1'
121628e43ccSMark Prins			)
122628e43ccSMark Prins	);
123628e43ccSMark Prins	protected $centerX, $centerY, $offsetX, $offsetY, $image;
1246914b920SMark Prins	protected $zoom, $lat, $lon, $width, $height, $markers, $maptype, $kmlFileName, $gpxFileName, $geojsonFileName, $autoZoomExtent;
125628e43ccSMark Prins	protected $tileCacheBaseDir, $mapCacheBaseDir, $mediaBaseDir;
126628e43ccSMark Prins	protected $useTileCache = true;
127628e43ccSMark Prins	protected $mapCacheID = '';
128628e43ccSMark Prins	protected $mapCacheFile = '';
129628e43ccSMark Prins	protected $mapCacheExtension = 'png';
130628e43ccSMark Prins
131628e43ccSMark Prins	/**
132f4b9bdacSMark Prins	 * Constructor.
133ab8cbd2bSMark Prins	 *
134ab8cbd2bSMark Prins	 * @param float $lat
135ab8cbd2bSMark Prins	 *        	Latitude (x) of center of map
136ab8cbd2bSMark Prins	 * @param float $lon
137ab8cbd2bSMark Prins	 *        	Longitude (y) of center of map
138ab8cbd2bSMark Prins	 * @param int $zoom
139ab8cbd2bSMark Prins	 *        	Zoomlevel
140ab8cbd2bSMark Prins	 * @param int $width
141ab8cbd2bSMark Prins	 *        	Width in pixels
142ab8cbd2bSMark Prins	 * @param int $height
143ab8cbd2bSMark Prins	 *        	Height in pixels
144ab8cbd2bSMark Prins	 * @param string $maptype
145ab8cbd2bSMark Prins	 *        	Name of the map
146ab8cbd2bSMark Prins	 * @param mixed $markers
147ab8cbd2bSMark Prins	 *        	array of markers
148ab8cbd2bSMark Prins	 * @param string $gpx
149ab8cbd2bSMark Prins	 *        	GPX filename
150ab8cbd2bSMark Prins	 * @param string $kml
151ab8cbd2bSMark Prins	 *        	KML filename
152ab8cbd2bSMark Prins	 * @param string $mediaDir
153ab8cbd2bSMark Prins	 *        	Directory to store/cache maps
154ab8cbd2bSMark Prins	 * @param string $tileCacheBaseDir
155ab8cbd2bSMark Prins	 *        	Directory to cache map tiles
156ab8cbd2bSMark Prins	 * @param boolean $autoZoomExtent
157ab8cbd2bSMark Prins	 *        	Wheter or not to override zoom/lat/lon and zoom to the extent of gpx/kml and markers
158628e43ccSMark Prins	 */
1596914b920SMark Prins	public function __construct($lat, $lon, $zoom, $width, $height, $maptype, $markers, $gpx, $kml, $geojson, $mediaDir, $tileCacheBaseDir, $autoZoomExtent = TRUE) {
160628e43ccSMark Prins		$this->zoom = $zoom;
161628e43ccSMark Prins		$this->lat = $lat;
162628e43ccSMark Prins		$this->lon = $lon;
163628e43ccSMark Prins		$this->width = $width;
164628e43ccSMark Prins		$this->height = $height;
165628e43ccSMark Prins		// validate + set maptype
166628e43ccSMark Prins		$this->maptype = $this->tileDefaultSrc;
167628e43ccSMark Prins		if (array_key_exists ( $maptype, $this->tileInfo )) {
168628e43ccSMark Prins			$this->maptype = $maptype;
169628e43ccSMark Prins		}
1702d11d700SMark Prins		$this->markers = $markers;
1712d11d700SMark Prins		$this->kmlFileName = $kml;
1722d11d700SMark Prins		$this->gpxFileName = $gpx;
1736914b920SMark Prins		$this->geojsonFileName = $geojson;
1742d11d700SMark Prins		$this->mediaBaseDir = $mediaDir;
175628e43ccSMark Prins		$this->tileCacheBaseDir = $tileCacheBaseDir . '/olmaptiles';
176628e43ccSMark Prins		$this->useTileCache = $this->tileCacheBaseDir !== '';
177628e43ccSMark Prins		$this->mapCacheBaseDir = $mediaDir . '/olmapmaps';
1782d11d700SMark Prins		$this->autoZoomExtent = $autoZoomExtent;
179628e43ccSMark Prins	}
180628e43ccSMark Prins
181628e43ccSMark Prins	/**
182628e43ccSMark Prins	 *
183628e43ccSMark Prins	 * @param number $long
184628e43ccSMark Prins	 * @param number $zoom
185f4b9bdacSMark Prins	 * @return number
186628e43ccSMark Prins	 */
187628e43ccSMark Prins	public function lonToTile($long, $zoom) {
188628e43ccSMark Prins		return (($long + 180) / 360) * pow ( 2, $zoom );
189628e43ccSMark Prins	}
190628e43ccSMark Prins	/**
191628e43ccSMark Prins	 *
192628e43ccSMark Prins	 * @param number $lat
193628e43ccSMark Prins	 * @param number $zoom
194628e43ccSMark Prins	 * @return number
195628e43ccSMark Prins	 */
196628e43ccSMark Prins	public function latToTile($lat, $zoom) {
1972d11d700SMark Prins		return (1 - log ( tan ( $lat * pi () / 180 ) + 1 / cos ( $lat * M_PI / 180 ) ) / M_PI) / 2 * pow ( 2, $zoom );
198628e43ccSMark Prins	}
199f4b9bdacSMark Prins
200628e43ccSMark Prins	/**
201628e43ccSMark Prins	 */
202628e43ccSMark Prins	public function initCoords() {
203628e43ccSMark Prins		$this->centerX = $this->lonToTile ( $this->lon, $this->zoom );
204628e43ccSMark Prins		$this->centerY = $this->latToTile ( $this->lat, $this->zoom );
205628e43ccSMark Prins		$this->offsetX = floor ( (floor ( $this->centerX ) - $this->centerX) * $this->tileSize );
206628e43ccSMark Prins		$this->offsetY = floor ( (floor ( $this->centerY ) - $this->centerY) * $this->tileSize );
207628e43ccSMark Prins	}
208628e43ccSMark Prins
209628e43ccSMark Prins	/**
210628e43ccSMark Prins	 * make basemap image.
211628e43ccSMark Prins	 */
212628e43ccSMark Prins	public function createBaseMap() {
213628e43ccSMark Prins		$this->image = imagecreatetruecolor ( $this->width, $this->height );
214628e43ccSMark Prins		$startX = floor ( $this->centerX - ($this->width / $this->tileSize) / 2 );
215628e43ccSMark Prins		$startY = floor ( $this->centerY - ($this->height / $this->tileSize) / 2 );
216628e43ccSMark Prins		$endX = ceil ( $this->centerX + ($this->width / $this->tileSize) / 2 );
217628e43ccSMark Prins		$endY = ceil ( $this->centerY + ($this->height / $this->tileSize) / 2 );
218628e43ccSMark Prins		$this->offsetX = - floor ( ($this->centerX - floor ( $this->centerX )) * $this->tileSize );
219628e43ccSMark Prins		$this->offsetY = - floor ( ($this->centerY - floor ( $this->centerY )) * $this->tileSize );
220628e43ccSMark Prins		$this->offsetX += floor ( $this->width / 2 );
221628e43ccSMark Prins		$this->offsetY += floor ( $this->height / 2 );
222628e43ccSMark Prins		$this->offsetX += floor ( $startX - floor ( $this->centerX ) ) * $this->tileSize;
223628e43ccSMark Prins		$this->offsetY += floor ( $startY - floor ( $this->centerY ) ) * $this->tileSize;
224628e43ccSMark Prins
225628e43ccSMark Prins		for($x = $startX; $x <= $endX; $x ++) {
226628e43ccSMark Prins			for($y = $startY; $y <= $endY; $y ++) {
227ab8cbd2bSMark Prins				$url = str_replace ( array (
228ab8cbd2bSMark Prins						'{Z}',
229ab8cbd2bSMark Prins						'{X}',
230ab8cbd2bSMark Prins						'{Y}'
231ab8cbd2bSMark Prins				), array (
232ab8cbd2bSMark Prins						$this->zoom,
233ab8cbd2bSMark Prins						$x,
234ab8cbd2bSMark Prins						$y
235ab8cbd2bSMark Prins				), $this->tileInfo [$this->maptype] ['url'] );
236628e43ccSMark Prins				$tileData = $this->fetchTile ( $url );
237628e43ccSMark Prins				if ($tileData) {
238628e43ccSMark Prins					$tileImage = imagecreatefromstring ( $tileData );
239628e43ccSMark Prins				} else {
240628e43ccSMark Prins					$tileImage = imagecreate ( $this->tileSize, $this->tileSize );
241628e43ccSMark Prins					$color = imagecolorallocate ( $tileImage, 255, 255, 255 );
242628e43ccSMark Prins					@imagestring ( $tileImage, 1, 127, 127, 'err', $color );
243628e43ccSMark Prins				}
244628e43ccSMark Prins				$destX = ($x - $startX) * $this->tileSize + $this->offsetX;
245628e43ccSMark Prins				$destY = ($y - $startY) * $this->tileSize + $this->offsetY;
246628e43ccSMark Prins				imagecopy ( $this->image, $tileImage, $destX, $destY, 0, 0, $this->tileSize, $this->tileSize );
247628e43ccSMark Prins			}
248628e43ccSMark Prins		}
249628e43ccSMark Prins	}
250628e43ccSMark Prins
251628e43ccSMark Prins	/**
252628e43ccSMark Prins	 * Place markers on the map and number them in the same order as they are listed in the html.
253628e43ccSMark Prins	 */
254628e43ccSMark Prins	public function placeMarkers() {
255628e43ccSMark Prins		$count = 0;
256628e43ccSMark Prins		$color = imagecolorallocate ( $this->image, 0, 0, 0 );
257628e43ccSMark Prins		$bgcolor = imagecolorallocate ( $this->image, 200, 200, 200 );
258628e43ccSMark Prins		$markerBaseDir = dirname ( __FILE__ ) . '/icons';
259628e43ccSMark Prins		// loop thru marker array
260628e43ccSMark Prins		foreach ( $this->markers as $marker ) {
261628e43ccSMark Prins			// set some local variables
262628e43ccSMark Prins			$markerLat = $marker ['lat'];
263628e43ccSMark Prins			$markerLon = $marker ['lon'];
264628e43ccSMark Prins			$markerType = $marker ['type'];
265628e43ccSMark Prins			// clear variables from previous loops
266628e43ccSMark Prins			$markerFilename = '';
267628e43ccSMark Prins			$markerShadow = '';
268628e43ccSMark Prins			$matches = false;
269628e43ccSMark Prins			// check for marker type, get settings from markerPrototypes
270628e43ccSMark Prins			if ($markerType) {
271628e43ccSMark Prins				foreach ( $this->markerPrototypes as $markerPrototype ) {
272628e43ccSMark Prins					if (preg_match ( $markerPrototype ['regex'], $markerType, $matches )) {
273628e43ccSMark Prins						$markerFilename = $matches [0] . $markerPrototype ['extension'];
274628e43ccSMark Prins						if ($markerPrototype ['offsetImage']) {
27582c0f6baSMark Prins							list ( $markerImageOffsetX, $markerImageOffsetY ) = explode ( ",", $markerPrototype ['offsetImage'] );
276628e43ccSMark Prins						}
277628e43ccSMark Prins						$markerShadow = $markerPrototype ['shadow'];
278628e43ccSMark Prins						if ($markerShadow) {
27982c0f6baSMark Prins							list ( $markerShadowOffsetX, $markerShadowOffsetY ) = explode ( ",", $markerPrototype ['offsetShadow'] );
280628e43ccSMark Prins						}
281628e43ccSMark Prins					}
282628e43ccSMark Prins				}
283628e43ccSMark Prins			}
284628e43ccSMark Prins			// create img resource
285628e43ccSMark Prins			if (file_exists ( $markerBaseDir . '/' . $markerFilename )) {
286628e43ccSMark Prins				$markerImg = imagecreatefrompng ( $markerBaseDir . '/' . $markerFilename );
287628e43ccSMark Prins			} else {
288628e43ccSMark Prins				$markerImg = imagecreatefrompng ( $markerBaseDir . '/marker.png' );
289628e43ccSMark Prins			}
290628e43ccSMark Prins			// check for shadow + create shadow recource
291628e43ccSMark Prins			if ($markerShadow && file_exists ( $markerBaseDir . '/' . $markerShadow )) {
292628e43ccSMark Prins				$markerShadowImg = imagecreatefrompng ( $markerBaseDir . '/' . $markerShadow );
293628e43ccSMark Prins			}
294628e43ccSMark Prins			// calc position
295628e43ccSMark Prins			$destX = floor ( ($this->width / 2) - $this->tileSize * ($this->centerX - $this->lonToTile ( $markerLon, $this->zoom )) );
296628e43ccSMark Prins			$destY = floor ( ($this->height / 2) - $this->tileSize * ($this->centerY - $this->latToTile ( $markerLat, $this->zoom )) );
297628e43ccSMark Prins			// copy shadow on basemap
298628e43ccSMark Prins			if ($markerShadow && $markerShadowImg) {
299ab8cbd2bSMark Prins				imagecopy ( $this->image, $markerShadowImg, $destX + intval ( $markerShadowOffsetX ), $destY + intval ( $markerShadowOffsetY ), 0, 0, imagesx ( $markerShadowImg ), imagesy ( $markerShadowImg ) );
300628e43ccSMark Prins			}
301628e43ccSMark Prins			// copy marker on basemap above shadow
302ab8cbd2bSMark Prins			imagecopy ( $this->image, $markerImg, $destX + intval ( $markerImageOffsetX ), $destY + intval ( $markerImageOffsetY ), 0, 0, imagesx ( $markerImg ), imagesy ( $markerImg ) );
303628e43ccSMark Prins			// add label
304628e43ccSMark Prins			imagestring ( $this->image, 3, $destX - imagesx ( $markerImg ) + 1, $destY + intval ( $markerImageOffsetY ) + 1, ++ $count, $bgcolor );
305628e43ccSMark Prins			imagestring ( $this->image, 3, $destX - imagesx ( $markerImg ), $destY + intval ( $markerImageOffsetY ), $count, $color );
306ab8cbd2bSMark Prins		}
307ab8cbd2bSMark Prins		;
308628e43ccSMark Prins	}
30957e65445SMark Prins
310628e43ccSMark Prins	/**
311628e43ccSMark Prins	 *
312628e43ccSMark Prins	 * @param string $url
313628e43ccSMark Prins	 * @return string
314628e43ccSMark Prins	 */
315628e43ccSMark Prins	public function tileUrlToFilename($url) {
316ab8cbd2bSMark Prins		return $this->tileCacheBaseDir . "/" . str_replace ( array (
317ab8cbd2bSMark Prins				'http://'
318ab8cbd2bSMark Prins		), '', $url );
319628e43ccSMark Prins	}
32057e65445SMark Prins
321628e43ccSMark Prins	/**
322628e43ccSMark Prins	 *
323628e43ccSMark Prins	 * @param string $url
324628e43ccSMark Prins	 */
325628e43ccSMark Prins	public function checkTileCache($url) {
326628e43ccSMark Prins		$filename = $this->tileUrlToFilename ( $url );
327628e43ccSMark Prins		if (file_exists ( $filename )) {
328628e43ccSMark Prins			return file_get_contents ( $filename );
329628e43ccSMark Prins		}
330628e43ccSMark Prins	}
331628e43ccSMark Prins	public function checkMapCache() {
332628e43ccSMark Prins		$this->mapCacheID = md5 ( $this->serializeParams () );
333628e43ccSMark Prins		$filename = $this->mapCacheIDToFilename ();
334ab8cbd2bSMark Prins		if (file_exists ( $filename ))
335ab8cbd2bSMark Prins			return true;
336628e43ccSMark Prins	}
337628e43ccSMark Prins	public function serializeParams() {
338ab8cbd2bSMark Prins		return join ( "&", array (
339ab8cbd2bSMark Prins				$this->zoom,
340ab8cbd2bSMark Prins				$this->lat,
341ab8cbd2bSMark Prins				$this->lon,
342ab8cbd2bSMark Prins				$this->width,
343ab8cbd2bSMark Prins				$this->height,
344ab8cbd2bSMark Prins				serialize ( $this->markers ),
345ab8cbd2bSMark Prins				$this->maptype,
346ab8cbd2bSMark Prins				$this->kmlFileName,
347ab8cbd2bSMark Prins				$this->gpxFileName,
348ab8cbd2bSMark Prins				$this->geojsonFileName
349ab8cbd2bSMark Prins		) );
350628e43ccSMark Prins	}
351628e43ccSMark Prins	public function mapCacheIDToFilename() {
352628e43ccSMark Prins		if (! $this->mapCacheFile) {
353628e43ccSMark Prins			$this->mapCacheFile = $this->mapCacheBaseDir . "/" . $this->maptype . "/" . $this->zoom . "/cache_" . substr ( $this->mapCacheID, 0, 2 ) . "/" . substr ( $this->mapCacheID, 2, 2 ) . "/" . substr ( $this->mapCacheID, 4 );
354628e43ccSMark Prins		}
355628e43ccSMark Prins		return $this->mapCacheFile . "." . $this->mapCacheExtension;
356628e43ccSMark Prins	}
35757e65445SMark Prins
3582d11d700SMark Prins	/**
3592d11d700SMark Prins	 * Recursively create the directory.
360ab8cbd2bSMark Prins	 *
361ab8cbd2bSMark Prins	 * @param string $pathname
362ab8cbd2bSMark Prins	 *        	The directory path.
363ab8cbd2bSMark Prins	 * @param int $mode
364ab8cbd2bSMark Prins	 *        	File access mode. For more information on modes, read the details on the chmod manpage.
3652d11d700SMark Prins	 */
366628e43ccSMark Prins	public function mkdir_recursive($pathname, $mode) {
367628e43ccSMark Prins		is_dir ( dirname ( $pathname ) ) || $this->mkdir_recursive ( dirname ( $pathname ), $mode );
368628e43ccSMark Prins		return is_dir ( $pathname ) || @mkdir ( $pathname, $mode );
369628e43ccSMark Prins	}
370628e43ccSMark Prins
3712d11d700SMark Prins	/**
3722d11d700SMark Prins	 * Write a tile into the cache.
373ab8cbd2bSMark Prins	 *
3742d11d700SMark Prins	 * @param string $url
3752d11d700SMark Prins	 * @param mixed $data
3762d11d700SMark Prins	 */
377628e43ccSMark Prins	public function writeTileToCache($url, $data) {
378628e43ccSMark Prins		$filename = $this->tileUrlToFilename ( $url );
379628e43ccSMark Prins		$this->mkdir_recursive ( dirname ( $filename ), 0777 );
380628e43ccSMark Prins		file_put_contents ( $filename, $data );
381628e43ccSMark Prins	}
38257e65445SMark Prins
3832d11d700SMark Prins	/**
3842d11d700SMark Prins	 * Fetch a tile and (if configured) store it in the cache.
385ab8cbd2bSMark Prins	 *
3862d11d700SMark Prins	 * @param string $url
3872d11d700SMark Prins	 */
388628e43ccSMark Prins	public function fetchTile($url) {
389ab8cbd2bSMark Prins		if ($this->useTileCache && ($cached = $this->checkTileCache ( $url )))
390ab8cbd2bSMark Prins			return $cached;
391e4f115f4SMark Prins
392e4f115f4SMark Prins		$_UA = 'Mozilla/4.0 (compatible; DokuWikiSpatial HTTP Client; ' . PHP_OS . ')';
393e4f115f4SMark Prins		if (function_exists ( "curl_init" )) {
394e4f115f4SMark Prins			// use cUrl
395628e43ccSMark Prins			$ch = curl_init ();
396628e43ccSMark Prins			curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, 1 );
397e4f115f4SMark Prins			curl_setopt ( $ch, CURLOPT_USERAGENT, $_UA );
398628e43ccSMark Prins			curl_setopt ( $ch, CURLOPT_CONNECTTIMEOUT, 10 );
399628e43ccSMark Prins			curl_setopt ( $ch, CURLOPT_URL, $url );
400628e43ccSMark Prins			$tile = curl_exec ( $ch );
401628e43ccSMark Prins			curl_close ( $ch );
402e4f115f4SMark Prins		} else {
403e4f115f4SMark Prins			// use file_get_contents
404e4f115f4SMark Prins			global $conf;
405e4f115f4SMark Prins			$opts = array (
406e4f115f4SMark Prins					'http' => array (
407e4f115f4SMark Prins							'method' => "GET",
408ab8cbd2bSMark Prins							'header' => "Accept-language: en\r\n" . "User-Agent: $_UA\r\n" . "accept: image/png\r\n",
409e4f115f4SMark Prins							'proxy' => "tcp://" . $conf ['proxy'] ['host'] . ":" . $conf ['proxy'] ['port'],
410ab8cbd2bSMark Prins							'request_fulluri' => true
411e4f115f4SMark Prins					)
412e4f115f4SMark Prins			);
413e4f115f4SMark Prins			$context = stream_context_create ( $opts );
414e4f115f4SMark Prins			$tile = file_get_contents ( $url, false, $context );
415e4f115f4SMark Prins		}
416628e43ccSMark Prins		if ($tile && $this->useTileCache) {
417628e43ccSMark Prins			$this->writeTileToCache ( $url, $tile );
418628e43ccSMark Prins		}
419628e43ccSMark Prins		return $tile;
420628e43ccSMark Prins	}
421628e43ccSMark Prins
422628e43ccSMark Prins	/**
423628e43ccSMark Prins	 * Draw gpx trace on the map.
424628e43ccSMark Prins	 */
425628e43ccSMark Prins	public function drawGPX() {
426c977deacSMark Prins		$col = imagecolorallocatealpha ( $this->image, 0, 0, 255, .4 * 127 );
4276c6bb022SMark Prins		$gpxgeom = geoPHP::load ( file_get_contents ( $this->gpxFileName ), 'gpx' );
428c977deacSMark Prins		$this->drawGeometry ( $gpxgeom, $col );
429628e43ccSMark Prins	}
430628e43ccSMark Prins
4316914b920SMark Prins	/**
4326914b920SMark Prins	 * Draw geojson on the map.
4336914b920SMark Prins	 */
4346914b920SMark Prins	public function drawGeojson() {
4356914b920SMark Prins		$col = imagecolorallocatealpha ( $this->image, 255, 0, 255, .4 * 127 );
4366914b920SMark Prins		$gpxgeom = geoPHP::load ( file_get_contents ( $this->geojsonFileName ), 'json' );
4376914b920SMark Prins		$this->drawGeometry ( $gpxgeom, $col );
4386914b920SMark Prins	}
43957e65445SMark Prins
440628e43ccSMark Prins	/**
441628e43ccSMark Prins	 * Draw kml trace on the map.
442628e43ccSMark Prins	 */
443628e43ccSMark Prins	public function drawKML() {
4442d11d700SMark Prins		// TODO get colour from kml node (not currently supported in geoPHP)
445c977deacSMark Prins		$col = imagecolorallocatealpha ( $this->image, 255, 0, 0, .4 * 127 );
446c977deacSMark Prins		$kmlgeom = geoPHP::load ( file_get_contents ( $this->kmlFileName ), 'kml' );
447c977deacSMark Prins		$this->drawGeometry ( $kmlgeom, $col );
448c977deacSMark Prins	}
44957e65445SMark Prins
450c977deacSMark Prins	/**
451c977deacSMark Prins	 * Draw geometry or geometry collection on the map.
452ab8cbd2bSMark Prins	 *
453c977deacSMark Prins	 * @param Geometry $geom
454ab8cbd2bSMark Prins	 * @param int $colour
455ab8cbd2bSMark Prins	 *        	drawing colour
456c977deacSMark Prins	 */
457c977deacSMark Prins	private function drawGeometry($geom, $colour) {
4586c6bb022SMark Prins		switch ($geom->geometryType ()) {
459c977deacSMark Prins			case 'GeometryCollection' :
460c977deacSMark Prins				// recursively draw part of the collection
461c977deacSMark Prins				for($i = 1; $i < $geom->numGeometries () + 1; $i ++) {
462c977deacSMark Prins					$_geom = $geom->geometryN ( $i );
463c977deacSMark Prins					$this->drawGeometry ( $_geom, $colour );
464c977deacSMark Prins				}
4656c6bb022SMark Prins				break;
466c977deacSMark Prins			case 'MultiPolygon' :
467c977deacSMark Prins				// TODO implement / do nothing
468c977deacSMark Prins				break;
469c977deacSMark Prins			case 'MultiLineString' :
470c977deacSMark Prins				// TODO implement / do nothing
471c977deacSMark Prins				break;
472c977deacSMark Prins			case 'MultiPoint' :
473c977deacSMark Prins				// TODO implement / do nothing
4746c6bb022SMark Prins				break;
4756c6bb022SMark Prins			case 'Polygon' :
476c977deacSMark Prins				$this->drawPolygon ( $geom, $colour );
477c977deacSMark Prins				break;
478c977deacSMark Prins			case 'LineString' :
479c977deacSMark Prins				$this->drawLineString ( $geom, $colour );
480c977deacSMark Prins				break;
481c977deacSMark Prins			case 'Point' :
482c977deacSMark Prins				$this->drawPoint ( $geom, $colour );
4836c6bb022SMark Prins				break;
4846c6bb022SMark Prins			default :
4852d11d700SMark Prins				// draw nothing
4866c6bb022SMark Prins				break;
4876c6bb022SMark Prins		}
4886c6bb022SMark Prins	}
489c977deacSMark Prins
490e61425c7SMark Prins	/**
491e61425c7SMark Prins	 * Draw a line on the map.
492ab8cbd2bSMark Prins	 *
493e61425c7SMark Prins	 * @param LineString $line
494ab8cbd2bSMark Prins	 * @param int $colour
495ab8cbd2bSMark Prins	 *        	drawing colour
496e61425c7SMark Prins	 */
497da6f229fSMark Prins	private function drawLineString($line, $colour) {
498cc74a83cSMark Prins		imagesetthickness ( $this->image, 2 );
499da6f229fSMark Prins		for($p = 1; $p < $line->numGeometries (); $p ++) {
500da6f229fSMark Prins			// get first pair of points
501da6f229fSMark Prins			$p1 = $line->geometryN ( $p );
502da6f229fSMark Prins			$p2 = $line->geometryN ( $p + 1 );
503da6f229fSMark Prins			// translate to paper space
504da6f229fSMark Prins			$x1 = floor ( ($this->width / 2) - $this->tileSize * ($this->centerX - $this->lonToTile ( $p1->x (), $this->zoom )) );
505da6f229fSMark Prins			$y1 = floor ( ($this->height / 2) - $this->tileSize * ($this->centerY - $this->latToTile ( $p1->y (), $this->zoom )) );
506da6f229fSMark Prins			$x2 = floor ( ($this->width / 2) - $this->tileSize * ($this->centerX - $this->lonToTile ( $p2->x (), $this->zoom )) );
507da6f229fSMark Prins			$y2 = floor ( ($this->height / 2) - $this->tileSize * ($this->centerY - $this->latToTile ( $p2->y (), $this->zoom )) );
508da6f229fSMark Prins			// draw to image
509da6f229fSMark Prins			imageline ( $this->image, $x1, $y1, $x2, $y2, $colour );
510da6f229fSMark Prins		}
511e61425c7SMark Prins		imagesetthickness ( $this->image, 1 );
512da6f229fSMark Prins	}
513c977deacSMark Prins
514e61425c7SMark Prins	/**
515e61425c7SMark Prins	 * Draw a point on the map.
516ab8cbd2bSMark Prins	 *
517e61425c7SMark Prins	 * @param Point $point
518ab8cbd2bSMark Prins	 * @param int $colour
519ab8cbd2bSMark Prins	 *        	drawing colour
520e61425c7SMark Prins	 */
521da6f229fSMark Prins	private function drawPoint($point, $colour) {
522c977deacSMark Prins		imagesetthickness ( $this->image, 2 );
523da6f229fSMark Prins		// translate to paper space
524da6f229fSMark Prins		$cx = floor ( ($this->width / 2) - $this->tileSize * ($this->centerX - $this->lonToTile ( $point->x (), $this->zoom )) );
525da6f229fSMark Prins		$cy = floor ( ($this->height / 2) - $this->tileSize * ($this->centerY - $this->latToTile ( $point->y (), $this->zoom )) );
526cc74a83cSMark Prins		$r = 5;
527da6f229fSMark Prins		// draw to image
528cc74a83cSMark Prins		// imageellipse($this->image, $cx, $cy,$r, $r, $colour);
529cc74a83cSMark Prins		imagefilledellipse ( $this->image, $cx, $cy, $r, $r, $colour );
530cc74a83cSMark Prins		// don't use imageellipse because the imagesetthickness function has
531c977deacSMark Prins		// no effect. So the better workaround is to use imagearc.
532cc74a83cSMark Prins		imagearc ( $this->image, $cx, $cy, $r, $r, 0, 359, $colour );
533e61425c7SMark Prins		imagesetthickness ( $this->image, 1 );
534da6f229fSMark Prins	}
535c977deacSMark Prins
536e61425c7SMark Prins	/**
537e61425c7SMark Prins	 * Draw a polygon on the map.
538ab8cbd2bSMark Prins	 *
539e61425c7SMark Prins	 * @param Polygon $polygon
540ab8cbd2bSMark Prins	 * @param int $colour
541ab8cbd2bSMark Prins	 *        	drawing colour
542e61425c7SMark Prins	 */
543e61425c7SMark Prins	private function drawPolygon($polygon, $colour) {
544c977deacSMark Prins		// TODO implementation of drawing holes,
545c977deacSMark Prins		// maybe draw the polygon to an in-memory image and use imagecopy, draw polygon in col., draw holes in bgcol?
546c977deacSMark Prins
547c977deacSMark Prins		// print_r('Polygon:<br />');
548c977deacSMark Prins		// print_r($polygon);
549c977deacSMark Prins		$extPoints = array ();
550c977deacSMark Prins		// extring is a linestring actually..
551c977deacSMark Prins		$extRing = $polygon->exteriorRing ();
552c977deacSMark Prins
553c977deacSMark Prins		for($i = 1; $i < $extRing->numGeometries (); $i ++) {
554c977deacSMark Prins			$p1 = $extRing->geometryN ( $i );
555c977deacSMark Prins			$x = floor ( ($this->width / 2) - $this->tileSize * ($this->centerX - $this->lonToTile ( $p1->x (), $this->zoom )) );
556c977deacSMark Prins			$y = floor ( ($this->height / 2) - $this->tileSize * ($this->centerY - $this->latToTile ( $p1->y (), $this->zoom )) );
557c977deacSMark Prins			$extPoints [] = $x;
558c977deacSMark Prins			$extPoints [] = $y;
559e61425c7SMark Prins		}
560c977deacSMark Prins		// print_r('points:('.($i-1).')<br />');
561c977deacSMark Prins		// print_r($extPoints);
562c977deacSMark Prins		// imagepolygon ($this->image, $extPoints, $i-1, $colour );
563c977deacSMark Prins		imagefilledpolygon ( $this->image, $extPoints, $i - 1, $colour );
564c977deacSMark Prins	}
565c977deacSMark Prins
566628e43ccSMark Prins	/**
567628e43ccSMark Prins	 * add copyright and origin notice and icons to the map.
568628e43ccSMark Prins	 */
569628e43ccSMark Prins	public function drawCopyright() {
570628e43ccSMark Prins		$logoBaseDir = dirname ( __FILE__ ) . '/' . 'logo/';
571628e43ccSMark Prins		$logoImg = imagecreatefrompng ( $logoBaseDir . $this->tileInfo ['openstreetmap'] ['logo'] );
572628e43ccSMark Prins		$textcolor = imagecolorallocate ( $this->image, 0, 0, 0 );
573628e43ccSMark Prins		$bgcolor = imagecolorallocate ( $this->image, 200, 200, 200 );
574628e43ccSMark Prins
5752d11d700SMark Prins		imagecopy ( $this->image, $logoImg, 0, imagesy ( $this->image ) - imagesy ( $logoImg ), 0, 0, imagesx ( $logoImg ), imagesy ( $logoImg ) );
576628e43ccSMark Prins		imagestring ( $this->image, 1, imagesx ( $logoImg ) + 2, imagesy ( $this->image ) - imagesy ( $logoImg ) + 1, $this->tileInfo ['openstreetmap'] ['txt'], $bgcolor );
577628e43ccSMark Prins		imagestring ( $this->image, 1, imagesx ( $logoImg ) + 1, imagesy ( $this->image ) - imagesy ( $logoImg ), $this->tileInfo ['openstreetmap'] ['txt'], $textcolor );
578628e43ccSMark Prins
579628e43ccSMark Prins		// additional tile source info, ie. who created/hosted the tiles
580628e43ccSMark Prins		if ($this->maptype != 'openstreetmap') {
581628e43ccSMark Prins			$iconImg = imagecreatefrompng ( $logoBaseDir . $this->tileInfo [$this->maptype] ['logo'] );
5822d11d700SMark Prins			imagecopy ( $this->image, $iconImg, imagesx ( $logoImg ) + 1, imagesy ( $this->image ) - imagesy ( $iconImg ), 0, 0, imagesx ( $iconImg ), imagesy ( $iconImg ) );
583628e43ccSMark Prins			imagestring ( $this->image, 1, imagesx ( $logoImg ) + imagesx ( $iconImg ) + 4, imagesy ( $this->image ) - ceil ( imagesy ( $logoImg ) / 2 ) + 1, $this->tileInfo [$this->maptype] ['txt'], $bgcolor );
584628e43ccSMark Prins			imagestring ( $this->image, 1, imagesx ( $logoImg ) + imagesx ( $iconImg ) + 3, imagesy ( $this->image ) - ceil ( imagesy ( $logoImg ) / 2 ), $this->tileInfo [$this->maptype] ['txt'], $textcolor );
585628e43ccSMark Prins		}
586628e43ccSMark Prins	}
587cc74a83cSMark Prins
588628e43ccSMark Prins	/**
589628e43ccSMark Prins	 * make the map.
590628e43ccSMark Prins	 */
591628e43ccSMark Prins	public function makeMap() {
592628e43ccSMark Prins		$this->initCoords ();
593628e43ccSMark Prins		$this->createBaseMap ();
594ab8cbd2bSMark Prins		if (! empty ( $this->markers ))
595ab8cbd2bSMark Prins			$this->placeMarkers ();
596ab8cbd2bSMark Prins		if (file_exists ( $this->kmlFileName ))
597ab8cbd2bSMark Prins			$this->drawKML ();
598ab8cbd2bSMark Prins		if (file_exists ( $this->gpxFileName ))
599ab8cbd2bSMark Prins			$this->drawGPX ();
600ab8cbd2bSMark Prins		if (file_exists ( $this->geojsonFileName ))
601ab8cbd2bSMark Prins			$this->drawGeojson ();
6026914b920SMark Prins
603628e43ccSMark Prins		$this->drawCopyright ();
604628e43ccSMark Prins	}
605cc74a83cSMark Prins
606628e43ccSMark Prins	/**
607e4f115f4SMark Prins	 * Calculate the lat/lon/zoom values to make sure that all of the markers and gpx/kml are on the map.
608ab8cbd2bSMark Prins	 *
609ab8cbd2bSMark Prins	 * @param float $paddingFactor
610ab8cbd2bSMark Prins	 *        	buffer constant to enlarge (>1.0) the zoom level
6112d11d700SMark Prins	 */
6122d11d700SMark Prins	private function autoZoom($paddingFactor = 1.0) {
6132d11d700SMark Prins		$geoms = array ();
614e4f115f4SMark Prins		$geoms [] = new Point ( $this->lon, $this->lat );
615e4f115f4SMark Prins		if (! empty ( $this->markers )) {
6162d11d700SMark Prins			foreach ( $this->markers as $marker ) {
6172d11d700SMark Prins				$geoms [] = new Point ( $marker ['lon'], $marker ['lat'] );
6182d11d700SMark Prins			}
6192d11d700SMark Prins		}
6202d11d700SMark Prins		if (file_exists ( $this->kmlFileName )) {
6212d11d700SMark Prins			$geoms [] = geoPHP::load ( file_get_contents ( $this->kmlFileName ), 'kml' );
6222d11d700SMark Prins		}
6232d11d700SMark Prins		if (file_exists ( $this->gpxFileName )) {
6242d11d700SMark Prins			$geoms [] = geoPHP::load ( file_get_contents ( $this->gpxFileName ), 'gpx' );
6252d11d700SMark Prins		}
6266914b920SMark Prins		if (file_exists ( $this->geojsonFileName )) {
6276914b920SMark Prins			$geoms [] = geoPHP::load ( file_get_contents ( $this->geojsonFileName ), 'geojson' );
6286914b920SMark Prins		}
6296914b920SMark Prins
630ab8cbd2bSMark Prins		if (count ( $geoms ) <= 1)
631ab8cbd2bSMark Prins			return;
63253bfe4a3SMark Prins
6332d11d700SMark Prins		$geom = new GeometryCollection ( $geoms );
6342d11d700SMark Prins		$centroid = $geom->centroid ();
6352d11d700SMark Prins		$bbox = $geom->getBBox ();
6362d11d700SMark Prins
6372d11d700SMark Prins		// determine vertical resolution, this depends on the distance from the equator
6382d11d700SMark Prins		// $vy00 = log(tan(M_PI*(0.25 + $centroid->getY()/360)));
6392d11d700SMark Prins		$vy0 = log ( tan ( M_PI * (0.25 + $bbox ['miny'] / 360) ) );
6402d11d700SMark Prins		$vy1 = log ( tan ( M_PI * (0.25 + $bbox ['maxy'] / 360) ) );
64153bfe4a3SMark Prins		$zoomFactorPowered = ($this->height / 2) / (40.7436654315252 * ($vy1 - $vy0));
6422d11d700SMark Prins		$resolutionVertical = 360 / ($zoomFactorPowered * $this->tileSize);
6432d11d700SMark Prins		// determine horizontal resolution
6442d11d700SMark Prins		$resolutionHorizontal = ($bbox ['maxx'] - $bbox ['minx']) / $this->width;
6452d11d700SMark Prins		$resolution = max ( $resolutionHorizontal, $resolutionVertical ) * $paddingFactor;
6462d11d700SMark Prins		$zoom = log ( 360 / ($resolution * $this->tileSize), 2 );
6472d11d700SMark Prins
6482d11d700SMark Prins		$this->zoom = floor ( $zoom );
6492d11d700SMark Prins		$this->lon = $centroid->getX ();
6502d11d700SMark Prins		$this->lat = $centroid->getY ();
6512d11d700SMark Prins	}
6522d11d700SMark Prins
6532d11d700SMark Prins	/**
654628e43ccSMark Prins	 * get the map, this may return a reference to a cached copy.
655ab8cbd2bSMark Prins	 *
656628e43ccSMark Prins	 * @return string url relative to media dir
657628e43ccSMark Prins	 */
658628e43ccSMark Prins	public function getMap() {
659ab8cbd2bSMark Prins		if ($this->autoZoomExtent)
660ab8cbd2bSMark Prins			$this->autoZoom ();
6612d11d700SMark Prins
662628e43ccSMark Prins			// use map cache, so check cache for map
663628e43ccSMark Prins		if (! $this->checkMapCache ()) {
664628e43ccSMark Prins			// map is not in cache, needs to be build
665628e43ccSMark Prins			$this->makeMap ();
666628e43ccSMark Prins			$this->mkdir_recursive ( dirname ( $this->mapCacheIDToFilename () ), 0777 );
667628e43ccSMark Prins			imagepng ( $this->image, $this->mapCacheIDToFilename (), 9 );
668628e43ccSMark Prins		}
669628e43ccSMark Prins		$this->doc = $this->mapCacheIDToFilename ();
670628e43ccSMark Prins		// make url relative to media dir
671628e43ccSMark Prins		return str_replace ( $this->mediaBaseDir, '', $this->doc );
672628e43ccSMark Prins	}
673628e43ccSMark Prins}
674