xref: /template/ad-hominem/my_template.php (revision 98fa74e35999cbd96f4678cfe0e188a0128ce464)
1<?php
2/**
3 * Overwriting DokuWiki template functions
4 *
5 * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
6 * @author     Sascha Leib <sascha@leib.be>
7 * @author     Andreas Gohr <andi@splitbrain.org>
8 */
9
10use dokuwiki\Extension\Event;
11
12/**
13 * Print the specific HTML meta headers
14 *
15 * Overrides the original version by modifying the headers and the way it is printed
16 *
17 * @author Sascha Leib <sascha@leib.be>
18 * @author Andreas Gohr <andi@splitbrain.org>
19 *
20 * @triggers TPL_METAHEADER_OUTPUT
21 * @param  bool $alt Should feeds and alternative format links be added?
22 * @return bool
23 */
24function my_metaheaders($alt = true) {
25    global $ID;
26    global $REV;
27    global $INFO;
28    global $JSINFO;
29    global $ACT;
30    global $QUERY;
31    global $lang;
32    global $conf;
33    global $updateVersion;
34    /** @var Input $INPUT */
35    global $INPUT;
36
37    // prepare the head array
38    $head = array();
39
40    // prepare seed for js and css
41    $tseed   = $updateVersion;
42    $depends = getConfigFiles('main');
43    $depends[] = DOKU_CONF."tpl/".$conf['template']."/style.ini";
44    foreach($depends as $f) $tseed .= @filemtime($f);
45    $tseed   = md5($tseed);
46
47	// Open Graph information
48	$meta = p_get_metadata($ID);
49	if ($meta['title'] !== null) {
50		$head['meta'][] = array('property' => 'og:title', 'content' => tpl_pagetitle($ID, true));
51		$head['meta'][] = array('property' => 'og:site_name ', 'content' => $conf['title']);
52		$head['meta'][] = array('property' => 'og:type', 'content' => 'website');
53		$head['meta'][] = array('property' => 'og:url', 'content' => wl($ID, '', true, '&'));
54
55		$parts = explode("\n", $meta['description']['abstract']);
56		$head['meta'][] = array('property' => 'og:description', 'content' => $parts[2]);
57	}
58
59    // the usual stuff
60    $head['meta'][] = array('name'=> 'generator', 'content'=> 'DokuWiki');
61    if(actionOK('search')) {
62        $head['link'][] = array(
63            'rel' => 'search', 'type'=> 'application/opensearchdescription+xml',
64            'href'=> DOKU_BASE.'lib/exe/opensearch.php', 'title'=> $conf['title']
65        );
66    }
67
68    $head['link'][] = array('rel'=> 'start', 'href'=> DOKU_BASE);
69    if(actionOK('index')) {
70        $head['link'][] = array(
71            'rel'  => 'contents', 'href'=> wl($ID, 'do=index', false, '&'),
72            'title'=> $lang['btn_index']
73        );
74    }
75
76    if (actionOK('manifest')) {
77        $head['link'][] = array('rel'=> 'manifest', 'href'=> DOKU_BASE.'lib/exe/manifest.php');
78    }
79
80    $styleUtil = new \dokuwiki\StyleUtils();
81    $styleIni = $styleUtil->cssStyleini();
82    $replacements = $styleIni['replacements'];
83    if (!empty($replacements['__theme_color__'])) {
84        $head['meta'][] = array('name' => 'theme-color', 'content' => $replacements['__theme_color__']);
85    }
86
87    if($alt) {
88        if(actionOK('rss')) {
89            $head['link'][] = array(
90                'rel'  => 'alternate', 'type'=> 'application/rss+xml',
91                'title'=> $lang['btn_recent'], 'href'=> DOKU_BASE.'feed.php'
92            );
93            $head['link'][] = array(
94                'rel'  => 'alternate', 'type'=> 'application/rss+xml',
95                'title'=> $lang['currentns'],
96                'href' => DOKU_BASE.'feed.php?mode=list&ns='.(isset($INFO) ? $INFO['namespace'] : '')
97            );
98        }
99        if(($ACT == 'show' || $ACT == 'search') && $INFO['writable']) {
100            $head['link'][] = array(
101                'rel'  => 'edit',
102                'title'=> $lang['btn_edit'],
103                'href' => wl($ID, 'do=edit', false, '&')
104            );
105        }
106
107        if(actionOK('rss') && $ACT == 'search') {
108            $head['link'][] = array(
109                'rel'  => 'alternate', 'type'=> 'application/rss+xml',
110                'title'=> $lang['searchresult'],
111                'href' => DOKU_BASE.'feed.php?mode=search&q='.$QUERY
112            );
113        }
114
115        if(actionOK('export_xhtml')) {
116            $head['link'][] = array(
117                'rel' => 'alternate', 'type'=> 'text/html', 'title'=> $lang['plainhtml'],
118                'href'=> exportlink($ID, 'xhtml', '', false, '&')
119            );
120        }
121
122        if(actionOK('export_raw')) {
123            $head['link'][] = array(
124                'rel' => 'alternate', 'type'=> 'text/plain', 'title'=> $lang['wikimarkup'],
125                'href'=> exportlink($ID, 'raw', '', false, '&')
126            );
127        }
128    }
129
130    // setup robot tags apropriate for different modes
131    if(($ACT == 'show' || $ACT == 'export_xhtml') && !$REV) {
132        if($INFO['exists']) {
133            //delay indexing:
134            if((time() - $INFO['lastmod']) >= $conf['indexdelay'] && !isHiddenPage($ID) ) {
135                $head['meta'][] = array('name'=> 'robots', 'content'=> 'index,follow');
136            } else {
137                $head['meta'][] = array('name'=> 'robots', 'content'=> 'noindex,nofollow');
138            }
139            $canonicalUrl = wl($ID, '', true, '&');
140            if ($ID == $conf['start']) {
141                $canonicalUrl = DOKU_URL;
142            }
143            $head['link'][] = array('rel'=> 'canonical', 'href'=> $canonicalUrl);
144        } else {
145            $head['meta'][] = array('name'=> 'robots', 'content'=> 'noindex,follow');
146        }
147    } elseif(defined('DOKU_MEDIADETAIL')) {
148        $head['meta'][] = array('name'=> 'robots', 'content'=> 'index,follow');
149    } else {
150        $head['meta'][] = array('name'=> 'robots', 'content'=> 'noindex,nofollow');
151    }
152
153    // set metadata
154    if($ACT == 'show' || $ACT == 'export_xhtml') {
155        // keywords (explicit or implicit)
156        if(!empty($INFO['meta']['subject'])) {
157            $head['meta'][] = array('name'=> 'keywords', 'content'=> join(',', $INFO['meta']['subject']));
158        } else {
159            $head['meta'][] = array('name'=> 'keywords', 'content'=> str_replace(':', ',', $ID));
160        }
161    }
162
163    // load stylesheets
164    $head['link'][] = array(
165        'rel' => 'stylesheet',
166        'href'=> DOKU_BASE . 'lib/exe/css.php?t='.rawurlencode($conf['template']).'&tseed='.$tseed
167    );
168
169    $script = "var NS='".(isset($INFO)?$INFO['namespace']:'')."';\n\t\t";
170    if($conf['useacl'] && $INPUT->server->str('REMOTE_USER')) {
171        $script .= "var SIG=".toolbar_signature().";\n\t\t";
172    }
173
174    if($conf['basedir']) {
175        $script .= 'var BASEDIR="'.$conf['basedir']."\";\n\t\t";
176    }
177
178    jsinfo();
179    $script .= 'var JSINFO = ' . json_encode($JSINFO).';';
180    $head['script'][] = array('_data'=> $script);
181
182    // load jquery
183    $jquery = getCdnUrls();
184    foreach($jquery as $src) {
185        $head['script'][] = array(
186            /* 'charset' => 'utf-8', -- obsolete */
187            '_data' => '',
188            'src' => $src,
189        ) + ($conf['defer_js'] ? [ 'defer' => 'defer'] : []);
190    }
191
192    // load our javascript dispatcher
193    $head['script'][] = array(
194        /* 'charset'=> 'utf-8', -- obsolete */
195		'_data'=> '',
196        'src' => DOKU_BASE . 'lib/exe/js.php'.'?t='.rawurlencode($conf['template']).'&tseed='.$tseed,
197    ) + ($conf['defer_js'] ? [ 'defer' => 'defer'] : []);
198
199    // trigger event here
200    Event::createAndTrigger('TPL_METAHEADER_OUTPUT', $head, '_my_metaheaders_action', true);
201    return true;
202}
203
204/**
205 * prints the array build by my_metaheaders
206 *
207 * Overrides the original version by adding a tab before each line for neater HTML code
208 *
209 * @author Sascha Leib <sascha@leib.be>
210 * @author Andreas Gohr <andi@splitbrain.org>
211 *
212 * @param array $data
213 */
214function _my_metaheaders_action($data) {
215    foreach($data as $tag => $inst) {
216        foreach($inst as $attr) {
217            if ( empty($attr) ) { continue; }
218            echo "\t<", $tag, ' ', buildAttributes($attr);
219            if(isset($attr['_data']) || $tag == 'script') {
220                if($tag == 'script' && $attr['_data'])
221                    $attr['_data'] = "/*<![CDATA[*/".
222                        $attr['_data'].
223                        "\n/*!]]>*/";
224
225                echo '>', $attr['_data'], '</', $tag, '>';
226            } else {
227                echo '/>';
228            }
229            echo "\n";
230        }
231    }
232}
233
234/**
235 * Print the breadcrumbs trace
236 *
237 * Cleanup of the original code to create neater and more accessible HTML
238 *
239 * @author Sascha Leib <sascha@leib.be>
240 * @author Andreas Gohr <andi@splitbrain.org>
241 *
242 * @param string $prefix inserted before each line
243 *
244 * @return void
245 */
246function my_breadcrumbs($prefix = '') {
247    global $lang;
248    global $conf;
249
250    //check if enabled
251    if(!$conf['breadcrumbs']) return false;
252
253    $crumbs = breadcrumbs(); //setup crumb trace
254
255	/* begin listing */
256	echo $prefix . "<nav id=\"navBreadCrumbs\">\n";
257	echo $prefix . "\t<h4>" . $lang['breadcrumb'] . "</h4>\n";
258	echo $prefix . "\t<ol reversed>\n";
259
260    $last = count($crumbs);
261    $i    = 0;
262    foreach($crumbs as $id => $name) {
263        $i++;
264		echo $prefix . "\t\t<li" . ($i == $last ? ' class="current"' : '') . '><bdi>' . tpl_link(wl($id), hsc($name), '', true) .  "</bdi></li>\n";
265    }
266	echo $prefix . "\t</ol>\n";
267	echo $prefix . "</nav>\n";
268}
269
270/**
271 * Hierarchical breadcrumbs
272 *
273 * Cleanup of the original code to create neater and more accessible HTML
274 *
275 * @author Sascha Leib <sascha@leib.be>
276 * @author Andreas Gohr <andi@splitbrain.org>
277 * @author Nigel McNie <oracle.shinoda@gmail.com>
278 * @author Sean Coates <sean@caedmon.net>
279 * @author <fredrik@averpil.com>
280 *
281 * @param  string $prefix to be added before each line
282 *
283 */
284function my_youarehere($prefix = '') {
285    global $conf;
286    global $ID;
287    global $lang;
288
289    // check if enabled
290    if(!$conf['youarehere']) return false;
291
292    $parts = explode(':', $ID);
293    $count = count($parts);
294	$isdir = ( $parts[$count-1] == $conf['start']);
295
296	echo $prefix . "<nav id=\"navYouAreHere\">\n";
297	echo $prefix . "\t<h4>" . $lang['youarehere'] . "</h4>\n";
298	echo $prefix . "\t<ol>\n";
299
300    // always print the startpage
301    echo $prefix . "\t\t<li class=\"home\">" . tpl_pagelink(':'.$conf['start'], null, true) . "</li>\n";
302
303    // print intermediate namespace links
304    $part = '';
305    for($i = 0; $i < $count - 1; $i++) {
306        $part .= $parts[$i].':';
307        $page = $part;
308        //if($page !== $conf['start']) { // Skip startpage
309
310			if ($i == $count-2 && $isdir)  break; // Skip last if it is an index page
311
312			echo $prefix . "\t\t<li>" . tpl_pagelink($page, null, true) . "</li>\n";
313		//}
314    }
315
316    // chould the current page be included in the listing?
317	$trail = tpl_getConf('navtrail');
318
319	if ($trail !== 'none' && $trail !== '') {
320		resolve_pageid('', $page, $exists);
321		//if ( !(isset($page) && $page == $part.$parts[$i]) || !($page == $conf['start']) ) {
322			echo $prefix . "\t\t<li>";
323			if ($trail == 'text') {
324				echo tpl_pagetitle(null, true);
325			} else if ($trail == 'link') {
326				echo tpl_pagelink($ID, null, true);
327			}
328			echo "</li>\n";
329		//}
330	}
331
332	echo $prefix . "\t</ol>\n";
333	echo $prefix . "</nav>\n";
334}
335
336/**
337 * My implementation of the basic userinfo (in the global banner)
338 *
339 *
340 * @author Sascha Leib <sascha@leib.be>
341 *
342 * @param  string $prefix to be added before each line
343 *
344 * @return void
345 */
346function my_userinfo($prefix = '') {
347    global $lang;
348    global $INPUT;
349
350	// add login/logout button:
351	$items = (new \dokuwiki\Menu\UserMenu())->getItems();
352	foreach($items as $it) {
353		$typ = $it->getType();
354		if ($typ === 'profile') {
355			echo $prefix . "<li class=\"action $typ\"><span class=\"sronly\">" . $lang['loggedinas'] . ' </span>' . userlink() . "</li>\n";
356		} else {
357			echo $prefix . "<li class=\"action $typ\"><a href=\"" . htmlentities($it->getLink()) . '" title="' . $it->getTitle() . '">' . $it->getLabel() . "</a></li>\n";
358		}
359	}
360
361}
362
363/**
364 *Inserts a cleaner version of the TOC
365 *
366 * This is an update of the original function that renders the TOC directly.
367 *
368 * @author Sascha Leib <sascha@leib.be>
369 * @author Andreas Gohr <andi@splitbrain.org>
370 *
371 * @param  string $prefix to be added before each line
372 *
373 * @return void
374 */
375function my_toc($prefix = '') {
376    global $TOC;
377    global $ACT;
378    global $ID;
379    global $REV;
380    global $INFO;
381    global $conf;
382    global $lang;
383    $toc = array();
384
385    if(is_array($TOC)) {
386        // if a TOC was prepared in global scope, always use it
387        $toc = $TOC;
388    } elseif(($ACT == 'show' || substr($ACT, 0, 6) == 'export') && !$REV && $INFO['exists']) {
389        // get TOC from metadata, render if neccessary
390        $meta = p_get_metadata($ID, '', METADATA_RENDER_USING_CACHE);
391        if(isset($meta['internal']['toc'])) {
392            $tocok = $meta['internal']['toc'];
393        } else {
394            $tocok = true;
395        }
396        $toc = isset($meta['description']['tableofcontents']) ? $meta['description']['tableofcontents'] : null;
397        if(!$tocok || !is_array($toc) || !$conf['tocminheads'] || count($toc) < $conf['tocminheads']) {
398            $toc = array();
399        }
400    } elseif($ACT == 'admin') {
401        // try to load admin plugin TOC
402        /** @var $plugin AdminPlugin */
403        if ($plugin = plugin_getRequestAdminPlugin()) {
404            $toc = $plugin->getTOC();
405            $TOC = $toc; // avoid later rebuild
406        }
407    }
408
409	/* Build the hierarchical list of headline links: */
410	if (count($toc) >= intval($conf['tocminheads'])) {
411		echo $prefix . "<aside id=\"toc\" class=\"toggle hide\">\n";
412		echo $prefix . "\t<h3 class=\"tg_button\" title=\"" . htmlentities($lang['toc']) . '"><span>' . htmlentities($lang['toc']) . "</span></h3>\n" . $prefix . "\t<div class=\"tg_content\">";
413		$level = intval("0");
414		foreach($toc as $it) {
415
416			$nl = intval($it['level']);
417			$cp = ($nl <=> $level);
418
419			if ($cp > 0) {
420				echo "\n" . $prefix . str_repeat("\t", $level*2 + 2) . "<ol>\n";
421			} else if ($cp < 0) {
422				echo "\n" . $prefix . str_repeat("\t", $level*2) . "</ol></li>\n";
423			} else {
424				echo "</li>\n";
425			}
426
427			$href = $it['link'] . ( $it['hid'] == '' ? '' : '#' . $it['hid'] );
428
429			echo $prefix . str_repeat("\t", $nl*2 + 1) . '<li><a href="' . $href . '">' . htmlentities($it['title']) . "</a>";
430			$level = $nl;
431		}
432
433		for ($i = $level-1; $i > 0; $i--) {
434			echo "</li>\n" . $prefix . str_repeat("\t", $i*2 + 1) . "</ol>";
435		}
436
437		echo "</li>\n" . $prefix . "\t\t</ol>\n" . $prefix . "\t</div>\n" . $prefix . "</aside>\n";
438	}
439}
440
441/**
442 * Print last change date
443 *
444 * @author Sascha Leib <sascha@leib.be>
445 *
446 * @param  string $prefix to be added before each line
447 *
448 * @return void
449 */
450function my_lastchange($prefix = '') {
451
452    global $lang;
453    global $INFO;
454
455	$format = '%Y-%m-%dT%T%z';	/* e.g. 2021-21-05T16:45:12+02:00 */
456
457	$date = $INFO['lastmod'];
458
459	echo $prefix . '<bdi>' . $lang['lastmod'] . "</bdi>\n";
460	echo $prefix . '<time datetime="' . strftime($format, $date) . '">' . dformat($date) . "</time>\n";
461
462	/* user name for last change (is this really interesting to the visitor?) */
463	/* echo $prefix .'<span class="editorname" tabindex="0">' . $lang['by'] . ' <bdi>' . editorinfo($INFO['editor']) . "</bdi></span>\n"; */
464}
465
466/**
467 * Returns a description list of the metatags of the current image
468 *
469 * @return string html of description list
470 */
471function my_img_meta($prefix = '') {
472    global $lang;
473
474	$format = '%Y-%m-%dT%T%z';	/* e.g. 2021-21-05T16:45:12+02:00 */
475
476    $tags = tpl_get_img_meta();
477
478    foreach($tags as $tag) {
479        $label = $lang[$tag['langkey']];
480        if(!$label) $label = $tag['langkey'] . ':';
481
482        echo $prefix . '<tr><th>'.$label.'</th><td>';
483        if ($tag['type'] == 'date') {
484            echo '<time datetime="' . strftime($format, $tag['value']) . '">' . dformat($tag['value']) . '</time>';
485        } else {
486            echo hsc($tag['value']);
487        }
488        echo "</td></tr>\n";
489    }
490}
491
492/**
493 * Creates the Site logo image link
494 *
495 */
496function my_sitelogo() {
497    global $conf;
498
499	// get logo either out of the template images folder or data/media folder
500	$logoSize = array();
501	$logo = tpl_getMediaFile(array(':logo.svg', ':wiki:logo.svg', ':logo.png', ':wiki:logo.png', 'images/sitelogo.svg'), false, $logoSize);
502	tpl_link( wl(),
503		'<img src="'.$logo.'" ' . $logoSize[3] . ' alt="' . htmlentities($conf['title']) . '" />', 'accesskey="h" title="[H]" class="logo"');
504}
505
506/**
507 * Creates the various favicon and similar links:
508 *
509 * @param  string $color overwrite the theme color.
510 *
511 * @return null
512 */
513function my_favicons($color = null) {
514
515	$logoSize = array();
516
517	// Theme color:
518	if ($color == null) {
519
520		/* get the style config */
521		$styleUtil = new \dokuwiki\StyleUtils();
522		$styleIni = $styleUtil->cssStyleini();
523		$replacements = $styleIni['replacements'];
524		$color = $replacements['__theme_color__'];
525
526		if ($color== null) { $color = '#2b73b7'; }
527	}
528	echo "\t<meta name=\"theme-color\" content=\"" . $color . "\" />\n";
529
530	// get the favicon:
531	$link = tpl_getMediaFile(array(':favicon.ico', ':favicon.png', ':favicon.svg', ':wiki:favicon.ico', ':wiki:favicon.png', ':wiki:favicon.svg'), false, $logoSize);
532	echo "\t<link rel=\"icon\" href=\"" . $link . "\" />\n";
533
534	// Apple Touch Icon
535	$logoSize = array();
536	$link = tpl_getMediaFile(array(':apple-touch-icon.png', ':wiki:apple-touch-icon.png', 'images/apple-touch-icon.png'), false, $logoSize);
537	echo "\t<link rel=\"apple-touch-icon\" href=\"" . $link . "\" />\n";
538
539}