xref: /plugin/openlayersmap/syntax/olmap.php (revision a57aaf07a34c563b41d065a48840d44bf9e56b1e)
138a47c60SMark Prins<?php
238a47c60SMark Prins/*
338a47c60SMark Prins * Copyright (c) 2008-2011 Mark C. Prins <mc.prins@gmail.com>
438a47c60SMark Prins *
538a47c60SMark Prins * Permission to use, copy, modify, and distribute this software for any
638a47c60SMark Prins * purpose with or without fee is hereby granted, provided that the above
738a47c60SMark Prins * copyright notice and this permission notice appear in all copies.
838a47c60SMark Prins *
938a47c60SMark Prins * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
1038a47c60SMark Prins * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
1138a47c60SMark Prins * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
1238a47c60SMark Prins * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
1338a47c60SMark Prins * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
1438a47c60SMark Prins * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
1538a47c60SMark Prins * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1638a47c60SMark Prins */
1738a47c60SMark Prins
1838a47c60SMark Prins/**
1938a47c60SMark Prins * Plugin OL Maps: Allow Display of a OpenLayers Map in a wiki page.
2038a47c60SMark Prins *
2138a47c60SMark Prins * @author Mark Prins
2238a47c60SMark Prins */
2338a47c60SMark Prins
2438a47c60SMark Prinsif (!defined('DOKU_INC'))
2538a47c60SMark Prinsdefine('DOKU_INC', realpath(dirname(__FILE__) . '/../../') . '/');
2638a47c60SMark Prinsif (!defined('DOKU_PLUGIN'))
2738a47c60SMark Prinsdefine('DOKU_PLUGIN', DOKU_INC . 'lib/plugins/');
2838a47c60SMark Prinsrequire_once (DOKU_PLUGIN . 'syntax.php');
2938a47c60SMark Prins
3038a47c60SMark Prins/**
3138a47c60SMark Prins * All DokuWiki plugins to extend the parser/rendering mechanism
3238a47c60SMark Prins * need to inherit from this class
3338a47c60SMark Prins */
3438a47c60SMark Prinsclass syntax_plugin_openlayersmap_olmap extends DokuWiki_Syntax_Plugin {
3538a47c60SMark Prins	/** defaults of the recognized attributes of the olmap tag. */
3638a47c60SMark Prins	private $dflt = array (
3738a47c60SMark Prins		'id' => 'olmap',
3838a47c60SMark Prins		'width' => '550px',
3938a47c60SMark Prins		'height' => '450px',
4038a47c60SMark Prins		'lat' => 50.0,
4138a47c60SMark Prins		'lon' => 5.1,
4238a47c60SMark Prins		'zoom' => 12,
4338a47c60SMark Prins		'toolbar' => true,
4438a47c60SMark Prins		'statusbar' => true,
4538a47c60SMark Prins		'controls' => true,
4638a47c60SMark Prins		'poihoverstyle' => false,
4738a47c60SMark Prins		'baselyr'=>'OpenStreetMap',
4838a47c60SMark Prins	 	'gpxfile' => '',
4945c50327SMark Prins 		'kmlfile' => '',
5045c50327SMark Prins		'summary'=>''
5138a47c60SMark Prins		);
5238a47c60SMark Prins
5338a47c60SMark Prins		/**
5438a47c60SMark Prins		 * Return the type of syntax this plugin defines.
5538a47c60SMark Prins		 * @Override
5638a47c60SMark Prins		 */
5738a47c60SMark Prins		function getType() {
5838a47c60SMark Prins			return 'substition';
5938a47c60SMark Prins		}
6038a47c60SMark Prins
6138a47c60SMark Prins		/**
6238a47c60SMark Prins		 * Defines how this syntax is handled regarding paragraphs.
6338a47c60SMark Prins		 * @Override
6438a47c60SMark Prins		 */
6538a47c60SMark Prins		function getPType() {
6638a47c60SMark Prins			return 'block';
6738a47c60SMark Prins		}
6838a47c60SMark Prins
6938a47c60SMark Prins		/**
7038a47c60SMark Prins		 * Returns a number used to determine in which order modes are added.
7138a47c60SMark Prins		 * @Override
7238a47c60SMark Prins		 */
7338a47c60SMark Prins		function getSort() {
7438a47c60SMark Prins			return 901;
7538a47c60SMark Prins		}
7638a47c60SMark Prins
7738a47c60SMark Prins		/**
7838a47c60SMark Prins		 * This function is inherited from Doku_Parser_Mode.
7938a47c60SMark Prins		 * Here is the place to register the regular expressions needed
8038a47c60SMark Prins		 * to match your syntax.
8138a47c60SMark Prins		 * @Override
8238a47c60SMark Prins		 */
8338a47c60SMark Prins		function connectTo($mode) {
8438a47c60SMark Prins			$this->Lexer->addSpecialPattern('<olmap ?[^>\n]*>.*?</olmap>', $mode, 'plugin_openlayersmap_olmap');
8538a47c60SMark Prins		}
8638a47c60SMark Prins
8738a47c60SMark Prins		/**
8838a47c60SMark Prins		 * handle each olmap tag. prepare the matched syntax for use in the renderer.
8938a47c60SMark Prins		 * @Override
9038a47c60SMark Prins		 */
9138a47c60SMark Prins		function handle($match, $state, $pos, &$handler) {
9238a47c60SMark Prins			// break matched cdata into its components
9338a47c60SMark Prins			list ($str_params, $str_points) = explode('>', substr($match, 7, -9), 2);
94554f43b5SMark Prins			// get the lat/lon for adding them to the metadata (used by geotag)
95554f43b5SMark Prins			preg_match('(lat[:|=]\"\d*\.\d*\")',$match,$mainLat);
96554f43b5SMark Prins			preg_match('(lon[:|=]\"\d*\.\d*\")',$match,$mainLon);
97554f43b5SMark Prins			$mainLat=substr($mainLat[0],5,-1);
98554f43b5SMark Prins			$mainLon=substr($mainLon[0],5,-1);
9938a47c60SMark Prins
10038a47c60SMark Prins			$gmap = $this->_extract_params($str_params);
10138a47c60SMark Prins			$overlay = $this->_extract_points($str_points);
10238a47c60SMark Prins
10338a47c60SMark Prins			$mapid = $gmap['id'];
10438a47c60SMark Prins
10538a47c60SMark Prins			// determine width and height (inline styles) for the map image
10638a47c60SMark Prins			if ($gmap['width'] || $gmap['height']) {
10738a47c60SMark Prins				$style = $gmap['width'] ? 'width: ' . $gmap['width'] . ";" : "";
10838a47c60SMark Prins				$style .= $gmap['height'] ? 'height: ' . $gmap['height'] . ";" : "";
10938a47c60SMark Prins				$style = "style='$style'";
11038a47c60SMark Prins			} else {
11138a47c60SMark Prins				$style = '';
11238a47c60SMark Prins			}
11338a47c60SMark Prins
11438a47c60SMark Prins			// unset gmap values for width and height - they don't go into javascript
11538a47c60SMark Prins			unset ($gmap['width'], $gmap['height']);
11638a47c60SMark Prins
11738a47c60SMark Prins			// create a javascript parameter string for the map
11838a47c60SMark Prins			$param = '';
11938a47c60SMark Prins			foreach ($gmap as $key => $val) {
12038a47c60SMark Prins				$param .= is_numeric($val) ? "$key: $val, " : "$key: '" . hsc($val) . "', ";
12138a47c60SMark Prins			}
12238a47c60SMark Prins			if (!empty ($param)) {
12345c50327SMark Prins				$param = substr($param, 0, -2);
12438a47c60SMark Prins			}
12538a47c60SMark Prins			unset ($gmap['id']);
12638a47c60SMark Prins
12738a47c60SMark Prins			// create a javascript serialisation of the point data
12838a47c60SMark Prins			$poi = '';
12945c50327SMark Prins			$poitable='';
13045c50327SMark Prins			$rowId=0;
13138a47c60SMark Prins			if (!empty ($overlay)) {
13238a47c60SMark Prins				foreach ($overlay as $data) {
13338a47c60SMark Prins					list ($lat, $lon, $text, $angle, $opacity, $img) = $data;
13445c50327SMark Prins					$rowId++;
13545c50327SMark Prins					$poi .= ", {lat: $lat, lon: $lon, txt: '$text', angle: $angle, opacity: $opacity, img: '$img', rowId: $rowId}";
136*a57aaf07SMark Prins					$poitable .='<tr>'."\n".'<td class="rowId">'.$rowId.'</td>
137*a57aaf07SMark Prins							<td class="icon"><img src="/lib/plugins/openlayersmap/icons/'.$img.'" alt="icon"></td>
138*a57aaf07SMark Prins							<td class="lat" title="'.$this->getLang('olmapPOIlatTitle').'">'.$lat.'</td>
139*a57aaf07SMark Prins							<td class="lon" title="'.$this->getLang('olmapPOIlonTitle').'">'.$lon.'</td>
140*a57aaf07SMark Prins							<td class="txt">'.$text.'</td>'."\n".'</tr>';
14138a47c60SMark Prins				}
14245c50327SMark Prins				$poi = substr($poi, 2);
14338a47c60SMark Prins			}
14438a47c60SMark Prins			$js .= "createMap({" . $param . " },[$poi]);";
14538a47c60SMark Prins
14645c50327SMark Prins			return array($mapid,$style,$js,$mainLat,$mainLon,$poitable,$gmap['summary']);
14738a47c60SMark Prins		}
14838a47c60SMark Prins
14938a47c60SMark Prins		/**
15038a47c60SMark Prins		 * render html tag/output. render the content.
15138a47c60SMark Prins		 * @Override
15238a47c60SMark Prins		 */
15338a47c60SMark Prins		function render($mode, &$renderer, $data) {
15438a47c60SMark Prins			static $initialised = false; // set to true after script initialisation
15545c50327SMark Prins			list ($mapid, $style, $param, $mainLat, $mainLon, $poitable, $poitabledesc) = $data;
156554f43b5SMark Prins
15738a47c60SMark Prins			if ($mode == 'xhtml') {
15838a47c60SMark Prins				$olscript = '';
15938a47c60SMark Prins				$olEnable = false;
16038a47c60SMark Prins				$gscript = '';
16138a47c60SMark Prins				$gEnable = false;
16238a47c60SMark Prins				$vscript = '';
16338a47c60SMark Prins				$vEnable = false;
16438a47c60SMark Prins				$yscript = '';
16538a47c60SMark Prins				$yEnable = false;
16638a47c60SMark Prins
16738a47c60SMark Prins				$scriptEnable = '';
16838a47c60SMark Prins
16938a47c60SMark Prins				if (!$initialised) {
17038a47c60SMark Prins					$initialised = true;
17145c50327SMark Prins					// render necessary script tags
17238a47c60SMark Prins					$gscript = $this->getConf('googleScriptUrl');
17338a47c60SMark Prins					$gscript = $gscript ? '<script type="text/javascript" src="' . $gscript . '"></script>' : "";
17438a47c60SMark Prins
17538a47c60SMark Prins					$vscript = $this->getConf('veScriptUrl');
17638a47c60SMark Prins					$vscript = $vscript ? '<script type="text/javascript" src="' . $vscript . '"></script>' : "";
17738a47c60SMark Prins
17838a47c60SMark Prins					$yscript = $this->getConf('yahooScriptUrl');
17938a47c60SMark Prins					$yscript = $yscript ? '<script type="text/javascript" src="' . $yscript . '"></script>' : "";
18038a47c60SMark Prins
18138a47c60SMark Prins					$olscript = $this->getConf('olScriptUrl');
18238a47c60SMark Prins					$olscript = $olscript ? '<script type="text/javascript" src="' . $olscript . '"></script>' : "";
18338a47c60SMark Prins					$olscript = str_replace('DOKU_PLUGIN', DOKU_PLUGIN, $olscript);
18438a47c60SMark Prins
18545c50327SMark Prins					$scriptEnable = '<script type="text/javascript">' . "\n" . '<!--//--><![CDATA[//><!--' . "\n";
18638a47c60SMark Prins					$scriptEnable .= $olscript ? 'olEnable = true;' : 'olEnable = false;';
18738a47c60SMark Prins					$scriptEnable .= $yscript ? ' yEnable = true;' : ' yEnable = false;';
18838a47c60SMark Prins					$scriptEnable .= $vscript ? ' veEnable = true;' : ' veEnable = false;';
18938a47c60SMark Prins					$scriptEnable .= $gscript ? ' gEnable = true;' : ' gEnable = false;';
19045c50327SMark Prins					$scriptEnable .= "\n" . '//--><!]]>' . "\n" . '</script>';
19138a47c60SMark Prins				}
19238a47c60SMark Prins				$renderer->doc .= "
19338a47c60SMark Prins				$olscript
19438a47c60SMark Prins				$gscript
19538a47c60SMark Prins				$vscript
19638a47c60SMark Prins				$yscript
19738a47c60SMark Prins				$scriptEnable
19838a47c60SMark Prins			    <div id='olContainer' class='olContainer'>
19938a47c60SMark Prins			        <div id='$mapid-olToolbar' class='olToolbar'></div>
20038a47c60SMark Prins			        <div style='clear:both;'></div>
20138a47c60SMark Prins			        <div id='$mapid' $style ></div>
20238a47c60SMark Prins			        <div id='$mapid-olStatusBar' class='olStatusBarContainer'>
20338a47c60SMark Prins			            <div id='$mapid-statusbar-scale' class='olStatusBar olStatusBarScale'>scale</div>
20438a47c60SMark Prins			            <div id='$mapid-statusbar-link' class='olStatusBar olStatusBarPermalink'>
20538a47c60SMark Prins			                <a href='' id='$mapid-statusbar-link-ref'>map link</a>
20638a47c60SMark Prins			            </div>
20738a47c60SMark Prins			            <div id='$mapid-statusbar-mouseposition' class='olStatusBar olStatusBarMouseposition'></div>
20838a47c60SMark Prins			            <div id='$mapid-statusbar-projection' class='olStatusBar olStatusBarProjection'>proj</div>
20938a47c60SMark Prins			            <div id='$mapid-statusbar-text' class='olStatusBar olStatusBarText'>txt</div>
21038a47c60SMark Prins			        </div>
21138a47c60SMark Prins			    </div>
21238a47c60SMark Prins			    <p>&nbsp;</p>
21345c50327SMark Prins			    <script type='text/javascript'><!--//--><![CDATA[//><!--
21438a47c60SMark Prins			    var $mapid = $param
21545c50327SMark Prins			   //--><!]]></script>";
21645c50327SMark Prins
21745c50327SMark Prins				// render a (hidden) table of the POI for the print and a11y presentation
21845c50327SMark Prins				$renderer->doc .= '
219*a57aaf07SMark Prins 	<table class="olPOItable inline" id="'.$mapid.'-table" summary="'.$poitabledesc.'" title="'.$this->getLang('olmapPOItitle').'">
220*a57aaf07SMark Prins		<caption class="olPOITblCaption">'.$this->getLang('olmapPOItitle').'</caption>
22145c50327SMark Prins		<thead class="olPOITblHeader">
22245c50327SMark Prins			<tr>
22345c50327SMark Prins				<th class="rowId" scope="col">id</th>
224*a57aaf07SMark Prins				<th class="icon" scope="col">'.$this->getLang('olmapPOIicon').'</th>
225*a57aaf07SMark Prins				<th class="lat" scope="col" title="'.$this->getLang('olmapPOIlatTitle').'">'.$this->getLang('olmapPOIlat').'</th>
226*a57aaf07SMark Prins				<th class="lon" scope="col" title="'.$this->getLang('olmapPOIlonTitle').'">'.$this->getLang('olmapPOIlon').'</th>
227*a57aaf07SMark Prins				<th class="txt" scope="col">'.$this->getLang('olmapPOItxt').'</th>
22845c50327SMark Prins			</tr>
22945c50327SMark Prins		</thead>
230*a57aaf07SMark Prins		<tbody class="olPOITblBody">'.$poitable.'</tbody>
23145c50327SMark Prins		<tfoot class="olPOITblFooter"><tr><td colspan="5">'.$poitabledesc.'</td></tr></tfoot>
23245c50327SMark Prins	</table>';
233*a57aaf07SMark Prins				//TODO no tfoot when $poitabledesc is empty
23445c50327SMark Prins
235554f43b5SMark Prins			} elseif ($mode == 'metadata') {
236554f43b5SMark Prins				// render metadata if available
237554f43b5SMark Prins				if (!(($this->dflt['lat']==$mainLat)||($thisdflt['lon']==$mainLon))){
238554f43b5SMark Prins					// unless they are the default
239554f43b5SMark Prins					$renderer->meta['geo']['lat'] = $mainLat;
240554f43b5SMark Prins					$renderer->meta['geo']['lon'] = $mainLon;
241554f43b5SMark Prins				}
242554f43b5SMark Prins				return true;
24338a47c60SMark Prins			}
24438a47c60SMark Prins			return false;
24538a47c60SMark Prins		}
24638a47c60SMark Prins
24738a47c60SMark Prins		/**
24838a47c60SMark Prins		 * extract parameters for the map from the parameter string
24938a47c60SMark Prins		 *
25038a47c60SMark Prins		 * @param   string    $str_params   string of key="value" pairs
25138a47c60SMark Prins		 * @return  array                   associative array of parameters key=>value
25238a47c60SMark Prins		 */
25338a47c60SMark Prins		function _extract_params($str_params) {
25438a47c60SMark Prins			$param = array ();
25538a47c60SMark Prins			preg_match_all('/(\w*)="(.*?)"/us', $str_params, $param, PREG_SET_ORDER);
25638a47c60SMark Prins			// parse match for instructions, break into key value pairs
25738a47c60SMark Prins			$gmap = $this->dflt;
25838a47c60SMark Prins			foreach ($param as $kvpair) {
25938a47c60SMark Prins				list ($match, $key, $val) = $kvpair;
26038a47c60SMark Prins				$key = strtolower($key);
26138a47c60SMark Prins				if (isset ($gmap[$key])){
26245c50327SMark Prins					if ($key == 'summary'){
26345c50327SMark Prins						// preserve case for summary field
26445c50327SMark Prins						$gmap[$key] = $val;
26545c50327SMark Prins					}else {
26638a47c60SMark Prins						$gmap[$key] = strtolower($val);
26738a47c60SMark Prins					}
26838a47c60SMark Prins				}
26945c50327SMark Prins			}
27038a47c60SMark Prins			return $gmap;
27138a47c60SMark Prins		}
27238a47c60SMark Prins
27338a47c60SMark Prins		/**
27438a47c60SMark Prins		 * extract overlay points for the map from the wiki syntax data
27538a47c60SMark Prins		 *
27638a47c60SMark Prins		 * @param   string    $str_points   multi-line string of lat,lon,text triplets
27738a47c60SMark Prins		 * @return  array                   multi-dimensional array of lat,lon,text triplets
27838a47c60SMark Prins		 */
27938a47c60SMark Prins		function _extract_points($str_points) {
28038a47c60SMark Prins			$point = array ();
2817a3f422cSMark Prins			//preg_match_all('/^([+-]?[0-9].*?),\s*([+-]?[0-9].*?),(.*?),(.*?),(.*?),(.*)$/um', $str_points, $point, PREG_SET_ORDER);
2827a3f422cSMark Prins			/*
2837a3f422cSMark Prins			group 1: ([+-]?[0-9]+(?:\.[0-9]*)?)
2847a3f422cSMark Prins			group 2: ([+-]?[0-9]+(?:\.[0-9]*)?)
2857a3f422cSMark Prins			group 3: (.*?)
2867a3f422cSMark Prins			group 4: (.*?)
2877a3f422cSMark Prins			group 5: (.*?)
2887a3f422cSMark Prins			group 6: (.*)
2897a3f422cSMark Prins			*/
2907a3f422cSMark Prins			preg_match_all('/^([+-]?[0-9]+(?:\.[0-9]*)?),\s*([+-]?[0-9]+(?:\.[0-9]*)?),(.*?),(.*?),(.*?),(.*)$/um', $str_points, $point, PREG_SET_ORDER);
291554f43b5SMark Prins			// create poi array
29238a47c60SMark Prins			$overlay = array ();
29338a47c60SMark Prins			foreach ($point as $pt) {
29438a47c60SMark Prins				list ($match, $lat, $lon, $angle, $opacity, $img, $text) = $pt;
29538a47c60SMark Prins				$lat = is_numeric($lat) ? $lat : 0;
29638a47c60SMark Prins				$lon = is_numeric($lon) ? $lon : 0;
29738a47c60SMark Prins				$angle = is_numeric($angle) ? $angle : 0;
29838a47c60SMark Prins				$opacity = is_numeric($opacity) ? $opacity : 0.8;
29938a47c60SMark Prins				$img = trim($img);
30038a47c60SMark Prins				// TODO validate & set up default img?
30138a47c60SMark Prins				$text = addslashes(str_replace("\n", "", p_render("xhtml", p_get_instructions($text), $info)));
3027a3f422cSMark Prins				$overlay[] = array($lat, $lon, $text, $angle, $opacity, $img);
30338a47c60SMark Prins			}
30438a47c60SMark Prins			return $overlay;
30538a47c60SMark Prins		}
30638a47c60SMark Prins}