110449332Schris<?php 210449332Schris/** 310449332Schris * Configuration Manager admin plugin 410449332Schris * 510449332Schris * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 610449332Schris * @author Christopher Smith <chris@jalakai.co.uk> 7685bdd2eSBen Coburn * @author Ben Coburn <btcoburn@silicodon.net> 810449332Schris */ 9e04f1f16Schris// must be run within Dokuwiki 10e04f1f16Schrisif(!defined('DOKU_INC')) die(); 1110449332Schris 1210449332Schrisdefine('CM_KEYMARKER','____'); // used for settings with multiple dimensions of array indices 1310449332Schris 1410449332Schrisdefine('PLUGIN_SELF',dirname(__FILE__).'/'); 1510449332Schrisdefine('PLUGIN_METADATA',PLUGIN_SELF.'settings/config.metadata.php'); 167ed33b2aSAnika Henkeif(!defined('DOKU_PLUGIN_IMAGES')) define('DOKU_PLUGIN_IMAGES',DOKU_BASE.'lib/plugins/config/images/'); 1710449332Schris 1810449332Schrisrequire_once(PLUGIN_SELF.'settings/config.class.php'); // main configuration class and generic settings classes 1910449332Schrisrequire_once(PLUGIN_SELF.'settings/extra.class.php'); // settings classes specific to these settings 2010449332Schris 2110449332Schris/** 2210449332Schris * All DokuWiki plugins to extend the admin function 2310449332Schris * need to inherit from this class 2410449332Schris */ 2510449332Schrisclass admin_plugin_config extends DokuWiki_Admin_Plugin { 2610449332Schris 2710449332Schris var $_file = PLUGIN_METADATA; 2810449332Schris var $_config = null; 2910449332Schris var $_input = null; 3010449332Schris var $_changed = false; // set to true if configuration has altered 3110449332Schris var $_error = false; 3210449332Schris var $_session_started = false; 3389d74a2fSchris var $_localised_prompts = false; 3410449332Schris 3510449332Schris /** 3610449332Schris * return some info 3710449332Schris */ 3810449332Schris function getInfo(){ 3910449332Schris 4010449332Schris return array( 4110449332Schris 'author' => 'Christopher Smith', 4210449332Schris 'email' => 'chris@jalakai.co.uk', 43b8595a66SAndreas Gohr 'date' => '2007-08-05', 4410449332Schris 'name' => 'Configuration Manager', 4510449332Schris 'desc' => "Manage Dokuwiki's Configuration Settings", 46f46c9e83SAnika Henke 'url' => 'http://dokuwiki.org/plugin:config', 4710449332Schris ); 4810449332Schris } 4910449332Schris 5010449332Schris function getMenuSort() { return 100; } 5110449332Schris 5210449332Schris /** 5310449332Schris * handle user request 5410449332Schris */ 5510449332Schris function handle() { 56400497e1Schris global $ID; 5710449332Schris 5810449332Schris if (!$this->_restore_session()) return $this->_close_session(); 5910449332Schris if (!isset($_REQUEST['save']) || ($_REQUEST['save'] != 1)) return $this->_close_session(); 60aea87c78SAndreas Gohr if (!checkSecurityToken()) return $this->_close_session(); 6110449332Schris 6210449332Schris if (is_null($this->_config)) { $this->_config = new configuration($this->_file); } 6310449332Schris 6410449332Schris // don't go any further if the configuration is locked 6510449332Schris if ($this->_config->_locked) return $this->_close_session(); 6610449332Schris 6710449332Schris $this->_input = $_REQUEST['config']; 6810449332Schris 6910449332Schris while (list($key) = each($this->_config->setting)) { 7010449332Schris $input = isset($this->_input[$key]) ? $this->_input[$key] : NULL; 7110449332Schris if ($this->_config->setting[$key]->update($input)) { 7210449332Schris $this->_changed = true; 7310449332Schris } 7410449332Schris if ($this->_config->setting[$key]->error()) $this->_error = true; 7510449332Schris } 7610449332Schris 7710449332Schris if ($this->_changed && !$this->_error) { 7810449332Schris $this->_config->save_settings($this->getPluginName()); 7910449332Schris 8010449332Schris // save state & force a page reload to get the new settings to take effect 8110449332Schris $_SESSION['PLUGIN_CONFIG'] = array('state' => 'updated', 'time' => time()); 8210449332Schris $this->_close_session(); 83b174aeaeSchris header("Location: ".wl($ID,array('do'=>'admin','page'=>'config'),true,'&')); 8410449332Schris exit(); 8510449332Schris } 8610449332Schris 8710449332Schris $this->_close_session(); 8810449332Schris } 8910449332Schris 9010449332Schris /** 9110449332Schris * output appropriate html 9210449332Schris */ 9310449332Schris function html() { 94685bdd2eSBen Coburn $allow_debug = $GLOBALS['conf']['allowdebug']; // avoid global $conf; here. 9510449332Schris global $lang; 96400497e1Schris global $ID; 9710449332Schris 9810449332Schris if (is_null($this->_config)) { $this->_config = new configuration($this->_file); } 9989d74a2fSchris $this->setupLocale(true); 10010449332Schris 10110449332Schris print $this->locale_xhtml('intro'); 10210449332Schris 10324a33b42SAndreas Gohr ptln('<div id="config__manager">'); 10410449332Schris 10510449332Schris if ($this->_config->locked) 106e4a98f5cSAnika Henke ptln('<div class="info">'.$this->getLang('locked').'</div>'); 10710449332Schris elseif ($this->_error) 108e4a98f5cSAnika Henke ptln('<div class="error">'.$this->getLang('error').'</div>'); 10910449332Schris elseif ($this->_changed) 110e4a98f5cSAnika Henke ptln('<div class="success">'.$this->getLang('updated').'</div>'); 11110449332Schris 112*5d85efc6SAdrian Lang // POST to script() instead of wl($ID) so config manager still works if 113*5d85efc6SAdrian Lang // rewrite config is broken. Add $ID as hidden field to remember 114*5d85efc6SAdrian Lang // current ID in most cases. 115f338aa80SAndreas Gohr ptln('<form action="'.script().'" method="post">'); 11664726aa4STom N Harris ptln('<input type="hidden" name="id" value="'.$ID.'" />'); 117634d7150SAndreas Gohr formSecurityToken(); 1184fa2dffcSBen Coburn $this->_print_h1('dokuwiki_settings', $this->getLang('_header_dokuwiki')); 11910449332Schris 120685bdd2eSBen Coburn $undefined_settings = array(); 1214fa2dffcSBen Coburn $in_fieldset = false; 1224fa2dffcSBen Coburn $first_plugin_fieldset = true; 1234fa2dffcSBen Coburn $first_template_fieldset = true; 12410449332Schris foreach($this->_config->setting as $setting) { 125685bdd2eSBen Coburn if (is_a($setting, 'setting_hidden')) { 126685bdd2eSBen Coburn // skip hidden (and undefined) settings 127685bdd2eSBen Coburn if ($allow_debug && is_a($setting, 'setting_undefined')) { 128685bdd2eSBen Coburn $undefined_settings[] = $setting; 129685bdd2eSBen Coburn } else { 130685bdd2eSBen Coburn continue; 131685bdd2eSBen Coburn } 132685bdd2eSBen Coburn } else if (is_a($setting, 'setting_fieldset')) { 1334fa2dffcSBen Coburn // config setting group 1344fa2dffcSBen Coburn if ($in_fieldset) { 1354fa2dffcSBen Coburn ptln(' </table>'); 1364fa2dffcSBen Coburn ptln(' </fieldset>'); 1374fa2dffcSBen Coburn } else { 1384fa2dffcSBen Coburn $in_fieldset = true; 1394fa2dffcSBen Coburn } 1404fa2dffcSBen Coburn if ($first_plugin_fieldset && substr($setting->_key, 0, 10)=='plugin'.CM_KEYMARKER) { 1414fa2dffcSBen Coburn $this->_print_h1('plugin_settings', $this->getLang('_header_plugin')); 1424fa2dffcSBen Coburn $first_plugin_fieldset = false; 1434fa2dffcSBen Coburn } else if ($first_template_fieldset && substr($setting->_key, 0, 7)=='tpl'.CM_KEYMARKER) { 1444fa2dffcSBen Coburn $this->_print_h1('template_settings', $this->getLang('_header_template')); 1454fa2dffcSBen Coburn $first_template_fieldset = false; 1464fa2dffcSBen Coburn } 147685bdd2eSBen Coburn ptln(' <fieldset id="'.$setting->_key.'">'); 1484fa2dffcSBen Coburn ptln(' <legend>'.$setting->prompt($this).'</legend>'); 1494fa2dffcSBen Coburn ptln(' <table class="inline">'); 1504fa2dffcSBen Coburn } else { 1514fa2dffcSBen Coburn // config settings 15210449332Schris list($label,$input) = $setting->html($this, $this->_error); 15310449332Schris 15410449332Schris $class = $setting->is_default() ? ' class="default"' : ($setting->is_protected() ? ' class="protected"' : ''); 1554fa2dffcSBen Coburn $error = $setting->error() ? ' class="value error"' : ' class="value"'; 1567ed33b2aSAnika Henke $icon = $setting->caution() ? '<img src="'.DOKU_PLUGIN_IMAGES.$setting->caution().'.png" alt="'.$setting->caution().'" title="'.$this->getLang($setting->caution()).'" />' : ''; 15710449332Schris 15810449332Schris ptln(' <tr'.$class.'>'); 159dde31035SAndreas Gohr ptln(' <td class="label">'); 1607ed33b2aSAnika Henke ptln(' <span class="outkey">'.$setting->_out_key(true, true).'</span>'); 161bccdba06SAndreas Gohr ptln(' '.$icon.$label); 162dde31035SAndreas Gohr ptln(' </td>'); 16310449332Schris ptln(' <td'.$error.'>'.$input.'</td>'); 16410449332Schris ptln(' </tr>'); 16510449332Schris } 1664fa2dffcSBen Coburn } 16710449332Schris 16810449332Schris ptln(' </table>'); 1694fa2dffcSBen Coburn if ($in_fieldset) { 1704fa2dffcSBen Coburn ptln(' </fieldset>'); 1714fa2dffcSBen Coburn } 17210449332Schris 173685bdd2eSBen Coburn // show undefined settings list 174685bdd2eSBen Coburn if ($allow_debug && !empty($undefined_settings)) { 175685bdd2eSBen Coburn function _setting_natural_comparison($a, $b) { return strnatcmp($a->_key, $b->_key); } 176685bdd2eSBen Coburn usort($undefined_settings, '_setting_natural_comparison'); 177685bdd2eSBen Coburn $this->_print_h1('undefined_settings', $this->getLang('_header_undefined')); 178685bdd2eSBen Coburn ptln('<fieldset>'); 179685bdd2eSBen Coburn ptln('<table class="inline">'); 180685bdd2eSBen Coburn $undefined_setting_match = array(); 181685bdd2eSBen Coburn foreach($undefined_settings as $setting) { 182685bdd2eSBen Coburn if (preg_match('/^(?:plugin|tpl)'.CM_KEYMARKER.'.*?'.CM_KEYMARKER.'(.*)$/', $setting->_key, $undefined_setting_match)) { 183685bdd2eSBen Coburn $undefined_setting_key = $undefined_setting_match[1]; 184685bdd2eSBen Coburn } else { 185685bdd2eSBen Coburn $undefined_setting_key = $setting->_key; 186685bdd2eSBen Coburn } 187685bdd2eSBen Coburn ptln(' <tr>'); 188dde31035SAndreas Gohr ptln(' <td class="label"><span title="$meta[\''.$undefined_setting_key.'\']">$'.$this->_config->_name.'[\''.$setting->_out_key().'\']</span></td>'); 189685bdd2eSBen Coburn ptln(' <td>'.$this->getLang('_msg_'.get_class($setting)).'</td>'); 190685bdd2eSBen Coburn ptln(' </tr>'); 191685bdd2eSBen Coburn } 192685bdd2eSBen Coburn ptln('</table>'); 193685bdd2eSBen Coburn ptln('</fieldset>'); 194685bdd2eSBen Coburn } 195685bdd2eSBen Coburn 196685bdd2eSBen Coburn // finish up form 19710449332Schris ptln('<p>'); 19810449332Schris ptln(' <input type="hidden" name="do" value="admin" />'); 19910449332Schris ptln(' <input type="hidden" name="page" value="config" />'); 20010449332Schris 20110449332Schris if (!$this->_config->locked) { 20210449332Schris ptln(' <input type="hidden" name="save" value="1" />'); 20391f04971SAndreas Gohr ptln(' <input type="submit" name="submit" class="button" value="'.$lang['btn_save'].'" accesskey="s" />'); 204e4a98f5cSAnika Henke ptln(' <input type="reset" class="button" value="'.$lang['btn_reset'].'" />'); 20510449332Schris } 20610449332Schris 20710449332Schris ptln('</p>'); 20810449332Schris 20910449332Schris ptln('</form>'); 21010449332Schris ptln('</div>'); 21110449332Schris } 21210449332Schris 21310449332Schris /** 21410449332Schris * @return boolean true - proceed with handle, false - don't proceed 21510449332Schris */ 21610449332Schris function _restore_session() { 21710449332Schris 21810449332Schris // dokuwiki closes the session before act_dispatch. $_SESSION variables are all set, 21910449332Schris // however they can't be changed without starting the session again 22010449332Schris if (!headers_sent()) { 22110449332Schris session_start(); 22210449332Schris $this->_session_started = true; 22310449332Schris } 22410449332Schris 22510449332Schris if (!isset($_SESSION['PLUGIN_CONFIG'])) return true; 22610449332Schris 22710449332Schris $session = $_SESSION['PLUGIN_CONFIG']; 22810449332Schris unset($_SESSION['PLUGIN_CONFIG']); 22910449332Schris 23010449332Schris // still valid? 23110449332Schris if (time() - $session['time'] > 120) return true; 23210449332Schris 23310449332Schris switch ($session['state']) { 23410449332Schris case 'updated' : 23510449332Schris $this->_changed = true; 23610449332Schris return false; 23710449332Schris } 23810449332Schris 23910449332Schris return true; 24010449332Schris } 24110449332Schris 24210449332Schris function _close_session() { 24310449332Schris if ($this->_session_started) session_write_close(); 24410449332Schris } 24510449332Schris 24689d74a2fSchris function setupLocale($prompts=false) { 24789d74a2fSchris 24889d74a2fSchris parent::setupLocale(); 24989d74a2fSchris if (!$prompts || $this->_localised_prompts) return; 25089d74a2fSchris 25189d74a2fSchris $this->_setup_localised_plugin_prompts(); 25289d74a2fSchris $this->_localised_prompts = true; 25389d74a2fSchris 25489d74a2fSchris } 25589d74a2fSchris 256a954fae3SBen Coburn function _setup_localised_plugin_prompts() { 25789d74a2fSchris global $conf; 25889d74a2fSchris 259746855cfSBen Coburn $langfile = '/lang/'.$conf['lang'].'/settings.php'; 26089d74a2fSchris $enlangfile = '/lang/en/settings.php'; 26189d74a2fSchris 26289d74a2fSchris if ($dh = opendir(DOKU_PLUGIN)) { 26389d74a2fSchris while (false !== ($plugin = readdir($dh))) { 26489d74a2fSchris if ($plugin == '.' || $plugin == '..' || $plugin == 'tmp' || $plugin == 'config') continue; 26589d74a2fSchris if (is_file(DOKU_PLUGIN.$plugin)) continue; 26689d74a2fSchris 26789d74a2fSchris if (@file_exists(DOKU_PLUGIN.$plugin.$enlangfile)){ 268f1f77134SEsther Brunner $lang = array(); 26989d74a2fSchris @include(DOKU_PLUGIN.$plugin.$enlangfile); 27089d74a2fSchris if ($conf['lang'] != 'en') @include(DOKU_PLUGIN.$plugin.$langfile); 271f1f77134SEsther Brunner foreach ($lang as $key => $value){ 272f1f77134SEsther Brunner $this->lang['plugin'.CM_KEYMARKER.$plugin.CM_KEYMARKER.$key] = $value; 273f1f77134SEsther Brunner } 27489d74a2fSchris } 2754fa2dffcSBen Coburn 2764fa2dffcSBen Coburn // fill in the plugin name if missing (should exist for plugins with settings) 2774fa2dffcSBen Coburn if (!isset($this->lang['plugin'.CM_KEYMARKER.$plugin.CM_KEYMARKER.'plugin_settings_name'])) { 2784fa2dffcSBen Coburn $this->lang['plugin'.CM_KEYMARKER.$plugin.CM_KEYMARKER.'plugin_settings_name'] = 2794fa2dffcSBen Coburn ucwords(str_replace('_', ' ', $plugin)).' '.$this->getLang('_plugin_sufix'); 2804fa2dffcSBen Coburn } 28189d74a2fSchris } 28289d74a2fSchris closedir($dh); 28389d74a2fSchris } 28489d74a2fSchris 2854a778400SEsther Brunner // the same for the active template 2864a778400SEsther Brunner $tpl = $conf['template']; 2874a778400SEsther Brunner 2884a778400SEsther Brunner if (@file_exists(DOKU_TPLINC.$enlangfile)){ 2894a778400SEsther Brunner $lang = array(); 2904a778400SEsther Brunner @include(DOKU_TPLINC.$enlangfile); 2914a778400SEsther Brunner if ($conf['lang'] != 'en') @include(DOKU_TPLINC.$langfile); 2924a778400SEsther Brunner foreach ($lang as $key => $value){ 2934a778400SEsther Brunner $this->lang['tpl'.CM_KEYMARKER.$tpl.CM_KEYMARKER.$key] = $value; 2944a778400SEsther Brunner } 2954a778400SEsther Brunner } 2964a778400SEsther Brunner 2974fa2dffcSBen Coburn // fill in the template name if missing (should exist for templates with settings) 2984fa2dffcSBen Coburn if (!isset($this->lang['tpl'.CM_KEYMARKER.$tpl.CM_KEYMARKER.'template_settings_name'])) { 2994fa2dffcSBen Coburn $this->lang['tpl'.CM_KEYMARKER.$tpl.CM_KEYMARKER.'template_settings_name'] = 3004fa2dffcSBen Coburn ucwords(str_replace('_', ' ', $tpl)).' '.$this->getLang('_template_sufix'); 3014fa2dffcSBen Coburn } 3024fa2dffcSBen Coburn 30389d74a2fSchris return true; 30489d74a2fSchris } 30589d74a2fSchris 3064fa2dffcSBen Coburn /** 3074fa2dffcSBen Coburn * Generates a two-level table of contents for the config plugin. 3084fa2dffcSBen Coburn * 3094fa2dffcSBen Coburn * @author Ben Coburn <btcoburn@silicodon.net> 3104fa2dffcSBen Coburn */ 311b8595a66SAndreas Gohr function getTOC() { 312b8595a66SAndreas Gohr if (is_null($this->_config)) { $this->_config = new configuration($this->_file); } 313b8595a66SAndreas Gohr $this->setupLocale(true); 314b8595a66SAndreas Gohr 315685bdd2eSBen Coburn $allow_debug = $GLOBALS['conf']['allowdebug']; // avoid global $conf; here. 316685bdd2eSBen Coburn 3174fa2dffcSBen Coburn // gather toc data 318685bdd2eSBen Coburn $has_undefined = false; 3194fa2dffcSBen Coburn $toc = array('conf'=>array(), 'plugin'=>array(), 'template'=>null); 3204fa2dffcSBen Coburn foreach($this->_config->setting as $setting) { 3214fa2dffcSBen Coburn if (is_a($setting, 'setting_fieldset')) { 3224fa2dffcSBen Coburn if (substr($setting->_key, 0, 10)=='plugin'.CM_KEYMARKER) { 3234fa2dffcSBen Coburn $toc['plugin'][] = $setting; 3244fa2dffcSBen Coburn } else if (substr($setting->_key, 0, 7)=='tpl'.CM_KEYMARKER) { 3254fa2dffcSBen Coburn $toc['template'] = $setting; 3264fa2dffcSBen Coburn } else { 3274fa2dffcSBen Coburn $toc['conf'][] = $setting; 3284fa2dffcSBen Coburn } 329685bdd2eSBen Coburn } else if (!$has_undefined && is_a($setting, 'setting_undefined')) { 330685bdd2eSBen Coburn $has_undefined = true; 3314fa2dffcSBen Coburn } 3324fa2dffcSBen Coburn } 3334fa2dffcSBen Coburn 334e1b31a95SBen Coburn // build toc 335b8595a66SAndreas Gohr $t = array(); 336b8595a66SAndreas Gohr 337b8595a66SAndreas Gohr $t[] = html_mktocitem('configuration_manager', $this->getLang('_configuration_manager'), 1); 338b8595a66SAndreas Gohr $t[] = html_mktocitem('dokuwiki_settings', $this->getLang('_header_dokuwiki'), 1); 339e1b31a95SBen Coburn foreach($toc['conf'] as $setting) { 340e1b31a95SBen Coburn $name = $setting->prompt($this); 341b8595a66SAndreas Gohr $t[] = html_mktocitem($setting->_key, $name, 2); 342e1b31a95SBen Coburn } 343e1b31a95SBen Coburn if (!empty($toc['plugin'])) { 344b8595a66SAndreas Gohr $t[] = html_mktocitem('plugin_settings', $this->getLang('_header_plugin'), 1); 345e1b31a95SBen Coburn } 346e1b31a95SBen Coburn foreach($toc['plugin'] as $setting) { 347e1b31a95SBen Coburn $name = $setting->prompt($this); 348b8595a66SAndreas Gohr $t[] = html_mktocitem($setting->_key, $name, 2); 349e1b31a95SBen Coburn } 350e1b31a95SBen Coburn if (isset($toc['template'])) { 351b8595a66SAndreas Gohr $t[] = html_mktocitem('template_settings', $this->getLang('_header_template'), 1); 352e1b31a95SBen Coburn $setting = $toc['template']; 353e1b31a95SBen Coburn $name = $setting->prompt($this); 354b8595a66SAndreas Gohr $t[] = html_mktocitem($setting->_key, $name, 2); 355e1b31a95SBen Coburn } 356e1b31a95SBen Coburn if ($has_undefined && $allow_debug) { 357b8595a66SAndreas Gohr $t[] = html_mktocitem('undefined_settings', $this->getLang('_header_undefined'), 1); 358e1b31a95SBen Coburn } 359e1b31a95SBen Coburn 360b8595a66SAndreas Gohr return $t; 3614fa2dffcSBen Coburn } 3624fa2dffcSBen Coburn 3634fa2dffcSBen Coburn function _print_h1($id, $text) { 3644fa2dffcSBen Coburn ptln('<h1><a name="'.$id.'" id="'.$id.'">'.$text.'</a></h1>'); 3654fa2dffcSBen Coburn } 3664fa2dffcSBen Coburn 3674fa2dffcSBen Coburn 36810449332Schris} 369