xref: /dokuwiki/lib/plugins/config/admin.php (revision 5d85efc6906bd57ed94bd1fceb7fe09d6a795866)
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