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}