xref: /dokuwiki/inc/template.php (revision 340756e4d13bbcffbd9cba79aef7fc7a6ca2ecb8)
16b13307fSandi<?php
26b13307fSandi/**
36b13307fSandi * DokuWiki template functions
46b13307fSandi *
56b13307fSandi * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
66b13307fSandi * @author     Andreas Gohr <andi@splitbrain.org>
76b13307fSandi */
86b13307fSandi
96b13307fSandi  if(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../').'/');
106b13307fSandi  require_once(DOKU_INC.'conf/dokuwiki.php');
116b13307fSandi
126b13307fSandi/**
136b13307fSandi * Wrapper around htmlspecialchars()
146b13307fSandi *
156b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
166b13307fSandi * @see    htmlspecialchars()
176b13307fSandi */
186b13307fSandifunction hsc($string){
196b13307fSandi  return htmlspecialchars($string);
206b13307fSandi}
216b13307fSandi
226b13307fSandi/**
236b13307fSandi * print a newline terminated string
246b13307fSandi *
256b13307fSandi * You can give an indention as optional parameter
266b13307fSandi *
276b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
286b13307fSandi */
296b13307fSandifunction ptln($string,$intend=0){
306b13307fSandi  for($i=0; $i<$intend; $i++) print ' ';
316b13307fSandi  print"$string\n";
326b13307fSandi}
336b13307fSandi
346b13307fSandi/**
356b13307fSandi * Print the content
366b13307fSandi *
376b13307fSandi * This function is used for printing all the usual content
386b13307fSandi * (defined by the global $ACT var) by calling the appropriate
396b13307fSandi * outputfunction(s) from html.php
406b13307fSandi *
416b13307fSandi * Everything that doesn't use the default template isn't
426b13307fSandi * handled by this function. ACL stuff is not done either.
436b13307fSandi *
446b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
456b13307fSandi */
466b13307fSandifunction tpl_content(){
476b13307fSandi  global $ACT;
486b13307fSandi  global $TEXT;
496b13307fSandi  global $PRE;
506b13307fSandi  global $SUF;
516b13307fSandi  global $SUM;
526b13307fSandi  global $IDX;
536b13307fSandi
546b13307fSandi  switch($ACT){
556b13307fSandi    case 'show':
566b13307fSandi      html_show();
576b13307fSandi      break;
58ea67f144Sandi    case 'preview':
596b13307fSandi      html_edit($TEXT);
606b13307fSandi      html_show($TEXT);
616b13307fSandi      break;
626b13307fSandi    case 'edit':
636b13307fSandi      html_edit();
646b13307fSandi      break;
656b13307fSandi    case 'wordblock':
666b13307fSandi      html_edit($TEXT,'wordblock');
676b13307fSandi      break;
686b13307fSandi    case 'search':
696b13307fSandi      html_search();
706b13307fSandi      break;
716b13307fSandi    case 'revisions':
726b13307fSandi      html_revisions();
736b13307fSandi      break;
746b13307fSandi    case 'diff':
756b13307fSandi      html_diff();
766b13307fSandi      break;
776b13307fSandi    case 'recent':
786b13307fSandi      html_recent();
796b13307fSandi      break;
806b13307fSandi    case 'index':
816b13307fSandi      html_index($IDX); #FIXME can this be pulled from globals? is it sanitized correctly?
826b13307fSandi      break;
836b13307fSandi    case 'backlink':
846b13307fSandi      html_backlinks();
856b13307fSandi      break;
866b13307fSandi    case 'conflict':
876b13307fSandi      html_conflict(con($PRE,$TEXT,$SUF),$SUM);
886b13307fSandi      html_diff(con($PRE,$TEXT,$SUF),false);
896b13307fSandi      break;
906b13307fSandi    case 'locked':
916b13307fSandi      html_locked($lockedby);
926b13307fSandi      break;
936b13307fSandi    case 'login':
946b13307fSandi      html_login();
956b13307fSandi      break;
966b13307fSandi    case 'register':
976b13307fSandi      html_register();
986b13307fSandi      break;
99820fa24bSandi    case 'denied':
100820fa24bSandi      print parsedLocale('denied');
101820fa24bSandi			break;
102c19fe9c0Sandi    case 'admin':
103c19fe9c0Sandi      tpl_admin();
104c19fe9c0Sandi      break;
1056b13307fSandi    default:
106ea67f144Sandi			msg("Failed to handle command: ".hsc($ACT),-1);
1076b13307fSandi  }
1086b13307fSandi}
1096b13307fSandi
110c19fe9c0Sandi/**
111c19fe9c0Sandi * Handle the admin page contents
112c19fe9c0Sandi *
113c19fe9c0Sandi * @author Andreas Gohr <andi@splitbrain.org>
114c19fe9c0Sandi */
115c19fe9c0Sandifunction tpl_admin(){
116c19fe9c0Sandi  switch($_REQUEST['page']){
117c19fe9c0Sandi		case 'acl':
118c19fe9c0Sandi			admin_acl_html();
119c19fe9c0Sandi			break;
120c19fe9c0Sandi    default:
121c19fe9c0Sandi			html_admin();
122c19fe9c0Sandi	}
123c19fe9c0Sandi}
1246b13307fSandi
1256b13307fSandi/**
1266b13307fSandi * Print the correct HTML meta headers
1276b13307fSandi *
1286b13307fSandi * This has to go into the head section of your template.
1296b13307fSandi *
1306b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
1316b13307fSandi */
1326b13307fSandifunction tpl_metaheaders(){
1336b13307fSandi  global $ID;
1346b13307fSandi  global $INFO;
1356b13307fSandi  global $ACT;
1366b13307fSandi  global $lang;
1376b13307fSandi  $it=2;
1386b13307fSandi
1396b13307fSandi  // the usual stuff
1406b13307fSandi  ptln('<meta name="generator" content="DokuWiki '.getVersion().'" />',$it);
1416b13307fSandi  ptln('<link rel="start" href="'.DOKU_BASE.'" />',$it);
1426b13307fSandi  ptln('<link rel="contents" href="'.wl($ID,'do=index').'" title="'.$lang['index'].'" />',$it);
1436b13307fSandi  ptln('<link rel="alternate" type="application/rss+xml" title="Recent Changes" href="'.DOKU_BASE.'feed.php" />',$it);
1446b13307fSandi  ptln('<link rel="alternate" type="application/rss+xml" title="Current Namespace" href="'.DOKU_BASE.'feed.php?mode=list&amp;ns='.$INFO['namespace'].'" />',$it);
1456b13307fSandi  ptln('<link rel="alternate" type="text/html" title="Plain HTML" href="'.wl($ID,'do=export_html').'" />',$it);
1466b13307fSandi  ptln('<link rel="alternate" type="text/plain" title="Wiki Markup" href="'.wl($ID, 'do=export_raw').'" />',$it);
1476b13307fSandi  ptln('<link rel="stylesheet" media="screen" type="text/css" href="'.DOKU_BASE.'style.css" />',$it);
1486b13307fSandi
1496b13307fSandi  // setup robot tags apropriate for different modes
1506b13307fSandi  if( ($ACT=='show' || $ACT=='export_html') && !$REV){
1516b13307fSandi    if($INFO['exists']){
1526b13307fSandi      ptln('<meta name="date" content="'.date('Y-m-d\TH:i:sO',$INFO['lastmod']).'" />',$it);
1536b13307fSandi      //delay indexing:
1546b13307fSandi      if((time() - $INFO['lastmod']) >= $conf['indexdelay']){
1556b13307fSandi        ptln('<meta name="robots" content="index,follow" />',$it);
1566b13307fSandi      }else{
1576b13307fSandi        ptln('<meta name="robots" content="noindex,nofollow" />',$it);
1586b13307fSandi      }
1596b13307fSandi    }else{
1606b13307fSandi      ptln('<meta name="robots" content="noindex,follow" />',$it);
1616b13307fSandi    }
1626b13307fSandi  }else{
1636b13307fSandi    ptln('<meta name="robots" content="noindex,nofollow" />',$it);
1646b13307fSandi  }
1656b13307fSandi
1666b13307fSandi  // include some JavaScript language strings
1676b13307fSandi  ptln('<script language="JavaScript" type="text/javascript">',$it);
1686b13307fSandi  ptln("  var alertText   = '".$lang['qb_alert']."'",$it);
1696b13307fSandi  ptln("  var notSavedYet = '".$lang['notsavedyet']."'",$it);
1706b13307fSandi  ptln("  var DOKU_BASE   = '".DOKU_BASE."'",$it);
1716b13307fSandi  ptln('</script>',$it);
1726b13307fSandi
1736b13307fSandi  // load the default JavaScript file
1746b13307fSandi  ptln('<script language="JavaScript" type="text/javascript" src="'.DOKU_BASE.'script.js"></script>',$it);
1756b13307fSandi
1766b13307fSandi
1776b13307fSandi  //FIXME include some default CSS ? IE FIX?
1786b13307fSandi}
1796b13307fSandi
1806b13307fSandi/**
1816b13307fSandi * Print a link
1826b13307fSandi *
1836b13307fSandi * Just builds a link but adds additional JavaScript needed for
1846b13307fSandi * the unsaved data check needed in the edit form.
1856b13307fSandi *
1866b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
1876b13307fSandi */
1886b13307fSandifunction tpl_link($url,$name,$more=''){
1896b13307fSandi  print '<a href="'.$url.'" onclick="return svchk()" onkeypress="return svchk()"';
1906b13307fSandi  if ($more) print ' '.$more;
1916b13307fSandi  print ">$name</a>";
1926b13307fSandi}
1936b13307fSandi
1946b13307fSandi/**
1956b13307fSandi * Print one of the buttons
1966b13307fSandi *
1976b13307fSandi * Available Buttons are
1986b13307fSandi *
1996b13307fSandi *  edit    - edit/create/show button
2006b13307fSandi *  history - old revisions
2016b13307fSandi *  recent  - recent changes
2026b13307fSandi *  login   - login/logout button - if ACL enabled
2036b13307fSandi *  index   - The index
204c19fe9c0Sandi *  admin   - admin page - if enough rights
2056b13307fSandi *  top     - a back to top button
2066b13307fSandi *
2076b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
2086b13307fSandi */
2096b13307fSandifunction tpl_button($type){
2106b13307fSandi  global $ID;
211c19fe9c0Sandi  global $INFO;
2126b13307fSandi  global $conf;
2136b13307fSandi
2146b13307fSandi  switch($type){
2156b13307fSandi    case 'edit':
2166b13307fSandi      print html_editbutton();
2176b13307fSandi      break;
2186b13307fSandi    case 'history':
2196b13307fSandi      print html_btn(revs,$ID,'o',array('do' => 'revisions'));
2206b13307fSandi      break;
2216b13307fSandi    case 'recent':
2226b13307fSandi      print html_btn(recent,'','r',array('do' => 'recent'));
2236b13307fSandi      break;
2246b13307fSandi    case 'index':
2256b13307fSandi      print html_btn(index,$ID,'x',array('do' => 'index'));
2266b13307fSandi      break;
2276b13307fSandi    case 'top':
2286b13307fSandi      print html_topbtn();
2296b13307fSandi      break;
2306b13307fSandi    case 'login':
2316b13307fSandi      if($conf['useacl']){
2326b13307fSandi        if($_SERVER['REMOTE_USER']){
2336b13307fSandi          print html_btn('logout',$ID,'',array('do' => 'logout',));
2346b13307fSandi        }else{
2356b13307fSandi          print html_btn('login',$ID,'',array('do' => 'login'));
2366b13307fSandi        }
2376b13307fSandi      }
2386b13307fSandi      break;
239c19fe9c0Sandi    case 'admin':
240c19fe9c0Sandi      if($INFO['perm'] == AUTH_ADMIN)
241c19fe9c0Sandi        print html_btn(admin,$ID,'',array('do' => 'admin'));
242c19fe9c0Sandi      break;
243c19fe9c0Sandi		default:
244c19fe9c0Sandi			print '[unknown button type]';
2456b13307fSandi  }
2466b13307fSandi}
2476b13307fSandi
2486b13307fSandi/**
2496b13307fSandi * Print the search form
2506b13307fSandi *
2516b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
2526b13307fSandi */
2536b13307fSandifunction tpl_searchform(){
2546b13307fSandi  global $lang;
255af182434Sandi  print '<form action="'.wl().'" accept-charset="utf-8" class="search" onsubmit="return svchk()">';
2566b13307fSandi  print '<input type="hidden" name="do" value="search" />';
2576b13307fSandi  print '<input type="text" accesskey="f" name="id" class="edit" />';
2586b13307fSandi  print '<input type="submit" value="'.$lang['btn_search'].'" class="button" />';
2596b13307fSandi  print '</form>';
2606b13307fSandi}
2616b13307fSandi
2626b13307fSandi/**
2636b13307fSandi * Print the breadcrumbs trace
2646b13307fSandi *
2656b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
2666b13307fSandi */
2676b13307fSandifunction tpl_breadcrumbs(){
2686b13307fSandi  global $lang;
2696b13307fSandi  global $conf;
2706b13307fSandi
2716b13307fSandi  //check if enabled
2726b13307fSandi  if(!$conf['breadcrumbs']) return;
2736b13307fSandi
2746b13307fSandi  $crumbs = breadcrumbs(); //setup crumb trace
2756b13307fSandi  print $lang['breadcrumb'].':';
276a77f5846Sjan  foreach ($crumbs as $id => $name){
2776b13307fSandi    print ' &raquo; ';
278a77f5846Sjan    tpl_link(wl($id),$name,'class="breadcrumbs" title="'.$id.'"');
2796b13307fSandi  }
2806b13307fSandi}
2816b13307fSandi
2826b13307fSandi/**
2831734437eSandi * Hierarchical breadcrumbs
2841734437eSandi *
2851734437eSandi * This code was suggested as replacement for the usual breadcrumbs
2861734437eSandi * trail in the Wiki and was modified by me.
2871734437eSandi * It only makes sense with a deep site structure.
2881734437eSandi *
2891734437eSandi * @author Andreas Gohr <andi@splitbrain.org>
2901734437eSandi * @link   http://wiki.splitbrain.org/wiki:tipsandtricks:hierarchicalbreadcrumbs
2911734437eSandi */
2921734437eSandifunction tpl_youarehere(){
2931734437eSandi  global $conf;
2941734437eSandi  global $ID;
2951734437eSandi  global $lang;
2961734437eSandi
2971734437eSandi
2981734437eSandi  $parts     = explode(':', $ID);
2991734437eSandi
3001734437eSandi  print $lang['breadcrumb'].': ';
3011734437eSandi
3021734437eSandi  //always print the startpage
3031734437eSandi  if( $a_part[0] != $conf['start'] )
3041734437eSandi    tpl_link(wl($conf['start']),$conf['start'],'title="'.$conf['start'].'"');
3051734437eSandi
3061734437eSandi  $page = '';
3071734437eSandi  foreach ($parts as $part){
3081734437eSandi	  print ' &raquo; ';
3091734437eSandi    $page .= $part;
3101734437eSandi
3111734437eSandi    if(file_exists(wikiFN($page))){
3121734437eSandi      tpl_link(wl($page),$part,'title="'.$page.'"');
3131734437eSandi    }else{
3141734437eSandi      print $page;
3151734437eSandi    }
3161734437eSandi
3171734437eSandi    $page .= ':';
3181734437eSandi  }
3191734437eSandi}
3201734437eSandi
3211734437eSandi/**
3226b13307fSandi * Print info if the user is logged in
3236b13307fSandi *
3246b13307fSandi * Could be enhanced with a profile link in future?
3256b13307fSandi *
3266b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
3276b13307fSandi */
3286b13307fSandifunction tpl_userinfo(){
3296b13307fSandi  global $lang;
3306b13307fSandi  if($_SERVER['REMOTE_USER'])
3316b13307fSandi    print $lang['loggedinas'].': '.$_SERVER['REMOTE_USER'];
3326b13307fSandi}
3336b13307fSandi
3346b13307fSandi/**
3356b13307fSandi * Print some info about the current page
3366b13307fSandi *
3376b13307fSandi * @author Andreas Gohr <andi@splitbrain.org>
3386b13307fSandi */
3396b13307fSandifunction tpl_pageinfo(){
3406b13307fSandi  global $conf;
3416b13307fSandi  global $lang;
3426b13307fSandi  global $INFO;
3436b13307fSandi  global $REV;
3446b13307fSandi
3456b13307fSandi  // prepare date and path
3466b13307fSandi  $fn = $INFO['filepath'];
3476b13307fSandi  if(!$conf['fullpath']){
3486b13307fSandi    if($REV){
3496b13307fSandi      $fn = str_replace(realpath($conf['olddir']).DIRECTORY_SEPARATOR,'',$fn);
3506b13307fSandi    }else{
3516b13307fSandi      $fn = str_replace(realpath($conf['datadir']).DIRECTORY_SEPARATOR,'',$fn);
3526b13307fSandi    }
3536b13307fSandi  }
354bee6dc82Sandi  $fn = utf8_decodeFN($fn);
3556b13307fSandi  $date = date($conf['dformat'],$INFO['lastmod']);
3566b13307fSandi
3576b13307fSandi  // print it
3586b13307fSandi  if($INFO['exists']){
3596b13307fSandi    print $fn;
3606b13307fSandi    print ' &middot; ';
3616b13307fSandi    print $lang['lastmod'];
3626b13307fSandi    print ': ';
3636b13307fSandi    print $date;
3646b13307fSandi    if($INFO['editor']){
3656b13307fSandi      print ' '.$lang['by'].' ';
3666b13307fSandi      print $INFO['editor'];
3676b13307fSandi    }
3686b13307fSandi    if($INFO['locked']){
3696b13307fSandi      print ' &middot; ';
3706b13307fSandi      print $lang['lockedby'];
3716b13307fSandi      print ': ';
3726b13307fSandi      print $INFO['locked'];
3736b13307fSandi    }
3746b13307fSandi  }
3756b13307fSandi}
3766b13307fSandi
377820fa24bSandi/**
378820fa24bSandi * Print a list of namespaces containing media files
379820fa24bSandi *
380820fa24bSandi * @author Andreas Gohr <andi@splitbrain.org>
381820fa24bSandi */
382820fa24bSandifunction tpl_medianamespaces(){
383820fa24bSandi	global $conf;
384820fa24bSandi
385820fa24bSandi  $data = array();
386820fa24bSandi  search($data,$conf['mediadir'],'search_namespaces',array());
387820fa24bSandi  print html_buildlist($data,'idx',media_html_list_namespaces);
388820fa24bSandi}
389820fa24bSandi
390820fa24bSandi/**
391820fa24bSandi * Print a list of mediafiles in the current namespace
392820fa24bSandi *
393820fa24bSandi * @author Andreas Gohr <andi@splitbrain.org>
394820fa24bSandi */
395820fa24bSandifunction tpl_mediafilelist(){
396820fa24bSandi  global $conf;
397820fa24bSandi  global $lang;
398820fa24bSandi  global $NS;
399820fa24bSandi  $dir = utf8_encodeFN(str_replace(':','/',$NS));
400820fa24bSandi
401820fa24bSandi  $data = array();
402820fa24bSandi  search($data,$conf['mediadir'],'search_media',array(),$dir);
403820fa24bSandi
404820fa24bSandi  if(!count($data)){
405820fa24bSandi    ptln('<div class="nothing">'.$lang['nothingfound'].'<div>');
406820fa24bSandi    return;
407820fa24bSandi  }
408820fa24bSandi
409820fa24bSandi  ptln('<ul>',2);
410820fa24bSandi  foreach($data as $item){
411820fa24bSandi    ptln('<li>',4);
412820fa24bSandi    ptln('<a href="javascript:mediaSelect(\''.$item['id'].'\')">'.
413820fa24bSandi         utf8_decodeFN($item['file']).
414820fa24bSandi         '</a>',6);
415820fa24bSandi    if($item['isimg']){
416820fa24bSandi      ptln('('.$item['info'][0].'&#215;'.$item['info'][1].
417820fa24bSandi           ' '.filesize_h($item['size']).')<br />',6);
418820fa24bSandi
419820fa24bSandi      # build thumbnail
420820fa24bSandi      $link=array();
421820fa24bSandi      $link['name']=$item['id'];
422820fa24bSandi      if($item['info'][0]>120) $link['name'] .= '?120';
423820fa24bSandi      $link = format_link_media($link);
424820fa24bSandi      ptln($link['name'],6);
425820fa24bSandi
426820fa24bSandi    }else{
427820fa24bSandi      ptln ('('.filesize_h($item['size']).')',6);
428820fa24bSandi    }
429820fa24bSandi    ptln('</li>',4);
430820fa24bSandi  }
431820fa24bSandi  ptln('</ul>',2);
432820fa24bSandi}
433820fa24bSandi
434820fa24bSandi/**
435820fa24bSandi * Print the media upload form if permissions are correct
436820fa24bSandi *
437820fa24bSandi * @author Andreas Gohr <andi@splitbrain.org>
438820fa24bSandi */
439820fa24bSandifunction tpl_mediauploadform(){
440820fa24bSandi  global $NS;
441820fa24bSandi  global $UPLOADOK;
442820fa24bSandi  global $lang;
443820fa24bSandi
444820fa24bSandi  if(!$UPLOADOK) return;
445820fa24bSandi
446820fa24bSandi  ptln('<form action="'.$_SERVER['PHP_SELF'].'" name="upload"'.
447820fa24bSandi       ' method="post" enctype="multipart/form-data">',2);
448820fa24bSandi  ptln($lang['txt_upload'].':<br />',4);
449820fa24bSandi  ptln('<input type="file" name="upload" class="edit" onchange="suggestWikiname();" />',4);
450820fa24bSandi  ptln('<input type="hidden" name="ns" value="'.hsc($NS).'" /><br />',4);
451820fa24bSandi  ptln($lang['txt_filename'].'<br />',4);
452820fa24bSandi  ptln('<input type="text" name="id" class="edit" />',4);
453820fa24bSandi  ptln('<input type="submit" class="button" value="'.$lang['btn_upload'].'" accesskey="s" />',4);
454820fa24bSandi  ptln('</form>',2);
455820fa24bSandi}
456820fa24bSandi
457*340756e4Sandi
458*340756e4Sandi//Setup VIM: ex: et ts=2 enc=utf-8 :
459