xref: /dokuwiki/lib/plugins/config/admin.php (revision 0ea51e63908793de4c5d5fa2b4d82c2769fec559)
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    function getMenuSort() { return 100; }
3610449332Schris
3710449332Schris    /**
3810449332Schris     * handle user request
3910449332Schris     */
4010449332Schris    function handle() {
41392c9b52SHakan Sandell        global $ID, $INPUT;
4210449332Schris
4310449332Schris        if (!$this->_restore_session()) return $this->_close_session();
44392c9b52SHakan Sandell        if ($INPUT->int('save') != 1) return $this->_close_session();
45aea87c78SAndreas Gohr        if (!checkSecurityToken()) return $this->_close_session();
4610449332Schris
4710449332Schris        if (is_null($this->_config)) { $this->_config = new configuration($this->_file); }
4810449332Schris
4910449332Schris        // don't go any further if the configuration is locked
5010449332Schris        if ($this->_config->_locked) return $this->_close_session();
5110449332Schris
52392c9b52SHakan Sandell        $this->_input = $INPUT->arr('config');
5310449332Schris
5410449332Schris        while (list($key) = each($this->_config->setting)) {
55*0ea51e63SMatt Perry            $input = isset($this->_input[$key]) ? $this->_input[$key] : null;
5610449332Schris            if ($this->_config->setting[$key]->update($input)) {
5710449332Schris                $this->_changed = true;
5810449332Schris            }
5910449332Schris            if ($this->_config->setting[$key]->error()) $this->_error = true;
6010449332Schris        }
6110449332Schris
6210449332Schris        if ($this->_changed  && !$this->_error) {
6310449332Schris            $this->_config->save_settings($this->getPluginName());
6410449332Schris
6510449332Schris            // save state & force a page reload to get the new settings to take effect
6610449332Schris            $_SESSION['PLUGIN_CONFIG'] = array('state' => 'updated', 'time' => time());
6710449332Schris            $this->_close_session();
683295f40aSAndreas Gohr            send_redirect(wl($ID,array('do'=>'admin','page'=>'config'),true,'&'));
6910449332Schris            exit();
70e0e514e1SAndreas Gohr        } elseif(!$this->_error) {
71e0e514e1SAndreas Gohr            $this->_config->touch_settings(); // just touch to refresh cache
7210449332Schris        }
7310449332Schris
7410449332Schris        $this->_close_session();
7510449332Schris    }
7610449332Schris
7710449332Schris    /**
7810449332Schris     * output appropriate html
7910449332Schris     */
8010449332Schris    function html() {
81685bdd2eSBen Coburn        $allow_debug = $GLOBALS['conf']['allowdebug']; // avoid global $conf; here.
8210449332Schris        global $lang;
83400497e1Schris        global $ID;
8410449332Schris
8510449332Schris        if (is_null($this->_config)) { $this->_config = new configuration($this->_file); }
8689d74a2fSchris        $this->setupLocale(true);
8710449332Schris
8810449332Schris        print $this->locale_xhtml('intro');
8910449332Schris
9024a33b42SAndreas Gohr        ptln('<div id="config__manager">');
9110449332Schris
9210449332Schris        if ($this->_config->locked)
93e4a98f5cSAnika Henke            ptln('<div class="info">'.$this->getLang('locked').'</div>');
9410449332Schris        elseif ($this->_error)
95e4a98f5cSAnika Henke            ptln('<div class="error">'.$this->getLang('error').'</div>');
9610449332Schris        elseif ($this->_changed)
97e4a98f5cSAnika Henke            ptln('<div class="success">'.$this->getLang('updated').'</div>');
9810449332Schris
995d85efc6SAdrian Lang        // POST to script() instead of wl($ID) so config manager still works if
1005d85efc6SAdrian Lang        // rewrite config is broken. Add $ID as hidden field to remember
1015d85efc6SAdrian Lang        // current ID in most cases.
102f338aa80SAndreas Gohr        ptln('<form action="'.script().'" method="post">');
103f24af591SAnika Henke        ptln('<div class="no"><input type="hidden" name="id" value="'.$ID.'" /></div>');
104634d7150SAndreas Gohr        formSecurityToken();
1054fa2dffcSBen Coburn        $this->_print_h1('dokuwiki_settings', $this->getLang('_header_dokuwiki'));
10610449332Schris
107685bdd2eSBen Coburn        $undefined_settings = array();
1084fa2dffcSBen Coburn        $in_fieldset = false;
1094fa2dffcSBen Coburn        $first_plugin_fieldset = true;
1104fa2dffcSBen Coburn        $first_template_fieldset = true;
11110449332Schris        foreach($this->_config->setting as $setting) {
112685bdd2eSBen Coburn            if (is_a($setting, 'setting_hidden')) {
113685bdd2eSBen Coburn                // skip hidden (and undefined) settings
114685bdd2eSBen Coburn                if ($allow_debug && is_a($setting, 'setting_undefined')) {
115685bdd2eSBen Coburn                    $undefined_settings[] = $setting;
116685bdd2eSBen Coburn                } else {
117685bdd2eSBen Coburn                    continue;
118685bdd2eSBen Coburn                }
119685bdd2eSBen Coburn            } else if (is_a($setting, 'setting_fieldset')) {
1204fa2dffcSBen Coburn                // config setting group
1214fa2dffcSBen Coburn                if ($in_fieldset) {
1224fa2dffcSBen Coburn                    ptln('  </table>');
123e260f93bSAnika Henke                    ptln('  </div>');
1244fa2dffcSBen Coburn                    ptln('  </fieldset>');
1254fa2dffcSBen Coburn                } else {
1264fa2dffcSBen Coburn                    $in_fieldset = true;
1274fa2dffcSBen Coburn                }
1284fa2dffcSBen Coburn                if ($first_plugin_fieldset && substr($setting->_key, 0, 10)=='plugin'.CM_KEYMARKER) {
1294fa2dffcSBen Coburn                    $this->_print_h1('plugin_settings', $this->getLang('_header_plugin'));
1304fa2dffcSBen Coburn                    $first_plugin_fieldset = false;
1314fa2dffcSBen Coburn                } else if ($first_template_fieldset && substr($setting->_key, 0, 7)=='tpl'.CM_KEYMARKER) {
1324fa2dffcSBen Coburn                    $this->_print_h1('template_settings', $this->getLang('_header_template'));
1334fa2dffcSBen Coburn                    $first_template_fieldset = false;
1344fa2dffcSBen Coburn                }
135685bdd2eSBen Coburn                ptln('  <fieldset id="'.$setting->_key.'">');
1364fa2dffcSBen Coburn                ptln('  <legend>'.$setting->prompt($this).'</legend>');
137c7b28ffdSAnika Henke                ptln('  <div class="table">');
1384fa2dffcSBen Coburn                ptln('  <table class="inline">');
1394fa2dffcSBen Coburn            } else {
1404fa2dffcSBen Coburn                // config settings
14110449332Schris                list($label,$input) = $setting->html($this, $this->_error);
14210449332Schris
14310449332Schris                $class = $setting->is_default() ? ' class="default"' : ($setting->is_protected() ? ' class="protected"' : '');
1444fa2dffcSBen Coburn                $error = $setting->error() ? ' class="value error"' : ' class="value"';
1457ed33b2aSAnika Henke                $icon = $setting->caution() ? '<img src="'.DOKU_PLUGIN_IMAGES.$setting->caution().'.png" alt="'.$setting->caution().'" title="'.$this->getLang($setting->caution()).'" />' : '';
14610449332Schris
14710449332Schris                ptln('    <tr'.$class.'>');
148dde31035SAndreas Gohr                ptln('      <td class="label">');
1497ed33b2aSAnika Henke                ptln('        <span class="outkey">'.$setting->_out_key(true, true).'</span>');
150bccdba06SAndreas Gohr                ptln('        '.$icon.$label);
151dde31035SAndreas Gohr                ptln('      </td>');
15210449332Schris                ptln('      <td'.$error.'>'.$input.'</td>');
15310449332Schris                ptln('    </tr>');
15410449332Schris            }
1554fa2dffcSBen Coburn        }
15610449332Schris
15710449332Schris        ptln('  </table>');
158c7b28ffdSAnika Henke        ptln('  </div>');
1594fa2dffcSBen Coburn        if ($in_fieldset) {
1604fa2dffcSBen Coburn            ptln('  </fieldset>');
1614fa2dffcSBen Coburn        }
16210449332Schris
163685bdd2eSBen Coburn        // show undefined settings list
164685bdd2eSBen Coburn        if ($allow_debug && !empty($undefined_settings)) {
165685bdd2eSBen Coburn            function _setting_natural_comparison($a, $b) { return strnatcmp($a->_key, $b->_key); }
166685bdd2eSBen Coburn            usort($undefined_settings, '_setting_natural_comparison');
167685bdd2eSBen Coburn            $this->_print_h1('undefined_settings', $this->getLang('_header_undefined'));
168685bdd2eSBen Coburn            ptln('<fieldset>');
169c7b28ffdSAnika Henke            ptln('<div class="table">');
170685bdd2eSBen Coburn            ptln('<table class="inline">');
171685bdd2eSBen Coburn            $undefined_setting_match = array();
172685bdd2eSBen Coburn            foreach($undefined_settings as $setting) {
173685bdd2eSBen Coburn                if (preg_match('/^(?:plugin|tpl)'.CM_KEYMARKER.'.*?'.CM_KEYMARKER.'(.*)$/', $setting->_key, $undefined_setting_match)) {
174685bdd2eSBen Coburn                    $undefined_setting_key = $undefined_setting_match[1];
175685bdd2eSBen Coburn                } else {
176685bdd2eSBen Coburn                    $undefined_setting_key = $setting->_key;
177685bdd2eSBen Coburn                }
178685bdd2eSBen Coburn                ptln('  <tr>');
179dde31035SAndreas Gohr                ptln('    <td class="label"><span title="$meta[\''.$undefined_setting_key.'\']">$'.$this->_config->_name.'[\''.$setting->_out_key().'\']</span></td>');
180685bdd2eSBen Coburn                ptln('    <td>'.$this->getLang('_msg_'.get_class($setting)).'</td>');
181685bdd2eSBen Coburn                ptln('  </tr>');
182685bdd2eSBen Coburn            }
183685bdd2eSBen Coburn            ptln('</table>');
184c7b28ffdSAnika Henke            ptln('</div>');
185685bdd2eSBen Coburn            ptln('</fieldset>');
186685bdd2eSBen Coburn        }
187685bdd2eSBen Coburn
188685bdd2eSBen Coburn        // finish up form
18910449332Schris        ptln('<p>');
19010449332Schris        ptln('  <input type="hidden" name="do"     value="admin" />');
19110449332Schris        ptln('  <input type="hidden" name="page"   value="config" />');
19210449332Schris
19310449332Schris        if (!$this->_config->locked) {
19410449332Schris            ptln('  <input type="hidden" name="save"   value="1" />');
19591f04971SAndreas Gohr            ptln('  <input type="submit" name="submit" class="button" value="'.$lang['btn_save'].'" accesskey="s" />');
196e4a98f5cSAnika Henke            ptln('  <input type="reset" class="button" value="'.$lang['btn_reset'].'" />');
19710449332Schris        }
19810449332Schris
19910449332Schris        ptln('</p>');
20010449332Schris
20110449332Schris        ptln('</form>');
20210449332Schris        ptln('</div>');
20310449332Schris    }
20410449332Schris
20510449332Schris    /**
20610449332Schris     * @return boolean   true - proceed with handle, false - don't proceed
20710449332Schris     */
20810449332Schris    function _restore_session() {
20910449332Schris
21010449332Schris        // dokuwiki closes the session before act_dispatch. $_SESSION variables are all set,
21110449332Schris        // however they can't be changed without starting the session again
21210449332Schris        if (!headers_sent()) {
21310449332Schris            session_start();
21410449332Schris            $this->_session_started = true;
21510449332Schris        }
21610449332Schris
21710449332Schris        if (!isset($_SESSION['PLUGIN_CONFIG'])) return true;
21810449332Schris
21910449332Schris        $session = $_SESSION['PLUGIN_CONFIG'];
22010449332Schris        unset($_SESSION['PLUGIN_CONFIG']);
22110449332Schris
22210449332Schris        // still valid?
22310449332Schris        if (time() - $session['time'] > 120) return true;
22410449332Schris
22510449332Schris        switch ($session['state']) {
22610449332Schris            case 'updated' :
22710449332Schris                $this->_changed = true;
22810449332Schris                return false;
22910449332Schris        }
23010449332Schris
23110449332Schris        return true;
23210449332Schris    }
23310449332Schris
23410449332Schris    function _close_session() {
23510449332Schris      if ($this->_session_started) session_write_close();
23610449332Schris    }
23710449332Schris
23889d74a2fSchris    function setupLocale($prompts=false) {
23989d74a2fSchris
24089d74a2fSchris        parent::setupLocale();
24189d74a2fSchris        if (!$prompts || $this->_localised_prompts) return;
24289d74a2fSchris
24389d74a2fSchris        $this->_setup_localised_plugin_prompts();
24489d74a2fSchris        $this->_localised_prompts = true;
24589d74a2fSchris
24689d74a2fSchris    }
24789d74a2fSchris
248a954fae3SBen Coburn    function _setup_localised_plugin_prompts() {
24989d74a2fSchris        global $conf;
25089d74a2fSchris
251746855cfSBen Coburn        $langfile   = '/lang/'.$conf['lang'].'/settings.php';
25289d74a2fSchris        $enlangfile = '/lang/en/settings.php';
25389d74a2fSchris
25489d74a2fSchris        if ($dh = opendir(DOKU_PLUGIN)) {
25589d74a2fSchris            while (false !== ($plugin = readdir($dh))) {
25689d74a2fSchris                if ($plugin == '.' || $plugin == '..' || $plugin == 'tmp' || $plugin == 'config') continue;
25789d74a2fSchris                if (is_file(DOKU_PLUGIN.$plugin)) continue;
25889d74a2fSchris
25989d74a2fSchris                if (@file_exists(DOKU_PLUGIN.$plugin.$enlangfile)){
260f1f77134SEsther Brunner                    $lang = array();
26189d74a2fSchris                    @include(DOKU_PLUGIN.$plugin.$enlangfile);
26289d74a2fSchris                    if ($conf['lang'] != 'en') @include(DOKU_PLUGIN.$plugin.$langfile);
263f1f77134SEsther Brunner                    foreach ($lang as $key => $value){
264f1f77134SEsther Brunner                        $this->lang['plugin'.CM_KEYMARKER.$plugin.CM_KEYMARKER.$key] = $value;
265f1f77134SEsther Brunner                    }
26689d74a2fSchris                }
2674fa2dffcSBen Coburn
2684fa2dffcSBen Coburn                // fill in the plugin name if missing (should exist for plugins with settings)
2694fa2dffcSBen Coburn                if (!isset($this->lang['plugin'.CM_KEYMARKER.$plugin.CM_KEYMARKER.'plugin_settings_name'])) {
2704fa2dffcSBen Coburn                    $this->lang['plugin'.CM_KEYMARKER.$plugin.CM_KEYMARKER.'plugin_settings_name'] =
271cd3ed83cSMatthias Schulte                      ucwords(str_replace('_', ' ', $plugin));
2724fa2dffcSBen Coburn                }
27389d74a2fSchris            }
27489d74a2fSchris            closedir($dh);
27589d74a2fSchris      }
27689d74a2fSchris
2774a778400SEsther Brunner        // the same for the active template
2784a778400SEsther Brunner        $tpl = $conf['template'];
2794a778400SEsther Brunner
280bc9d46afSAndreas Gohr        if (@file_exists(tpl_incdir().$enlangfile)){
2814a778400SEsther Brunner            $lang = array();
282bc9d46afSAndreas Gohr            @include(tpl_incdir().$enlangfile);
283bc9d46afSAndreas Gohr            if ($conf['lang'] != 'en') @include(tpl_incdir().$langfile);
2844a778400SEsther Brunner            foreach ($lang as $key => $value){
2854a778400SEsther Brunner                $this->lang['tpl'.CM_KEYMARKER.$tpl.CM_KEYMARKER.$key] = $value;
2864a778400SEsther Brunner            }
2874a778400SEsther Brunner        }
2884a778400SEsther Brunner
2894fa2dffcSBen Coburn        // fill in the template name if missing (should exist for templates with settings)
2904fa2dffcSBen Coburn        if (!isset($this->lang['tpl'.CM_KEYMARKER.$tpl.CM_KEYMARKER.'template_settings_name'])) {
2914fa2dffcSBen Coburn            $this->lang['tpl'.CM_KEYMARKER.$tpl.CM_KEYMARKER.'template_settings_name'] =
292cd3ed83cSMatthias Schulte              ucwords(str_replace('_', ' ', $tpl));
2934fa2dffcSBen Coburn        }
2944fa2dffcSBen Coburn
29589d74a2fSchris        return true;
29689d74a2fSchris    }
29789d74a2fSchris
2984fa2dffcSBen Coburn    /**
2994fa2dffcSBen Coburn     * Generates a two-level table of contents for the config plugin.
3004fa2dffcSBen Coburn     *
3014fa2dffcSBen Coburn     * @author Ben Coburn <btcoburn@silicodon.net>
3024fa2dffcSBen Coburn     */
303b8595a66SAndreas Gohr    function getTOC() {
304b8595a66SAndreas Gohr        if (is_null($this->_config)) { $this->_config = new configuration($this->_file); }
305b8595a66SAndreas Gohr        $this->setupLocale(true);
306b8595a66SAndreas Gohr
307685bdd2eSBen Coburn        $allow_debug = $GLOBALS['conf']['allowdebug']; // avoid global $conf; here.
308685bdd2eSBen Coburn
3094fa2dffcSBen Coburn        // gather toc data
310685bdd2eSBen Coburn        $has_undefined = false;
3114fa2dffcSBen Coburn        $toc = array('conf'=>array(), 'plugin'=>array(), 'template'=>null);
3124fa2dffcSBen Coburn        foreach($this->_config->setting as $setting) {
3134fa2dffcSBen Coburn            if (is_a($setting, 'setting_fieldset')) {
3144fa2dffcSBen Coburn                if (substr($setting->_key, 0, 10)=='plugin'.CM_KEYMARKER) {
3154fa2dffcSBen Coburn                    $toc['plugin'][] = $setting;
3164fa2dffcSBen Coburn                } else if (substr($setting->_key, 0, 7)=='tpl'.CM_KEYMARKER) {
3174fa2dffcSBen Coburn                    $toc['template'] = $setting;
3184fa2dffcSBen Coburn                } else {
3194fa2dffcSBen Coburn                    $toc['conf'][] = $setting;
3204fa2dffcSBen Coburn                }
321685bdd2eSBen Coburn            } else if (!$has_undefined && is_a($setting, 'setting_undefined')) {
322685bdd2eSBen Coburn                $has_undefined = true;
3234fa2dffcSBen Coburn            }
3244fa2dffcSBen Coburn        }
3254fa2dffcSBen Coburn
326e1b31a95SBen Coburn        // build toc
327b8595a66SAndreas Gohr        $t = array();
328b8595a66SAndreas Gohr
329b8595a66SAndreas Gohr        $t[] = html_mktocitem('configuration_manager', $this->getLang('_configuration_manager'), 1);
330b8595a66SAndreas Gohr        $t[] = html_mktocitem('dokuwiki_settings', $this->getLang('_header_dokuwiki'), 1);
331e1b31a95SBen Coburn        foreach($toc['conf'] as $setting) {
332e1b31a95SBen Coburn            $name = $setting->prompt($this);
333b8595a66SAndreas Gohr            $t[] = html_mktocitem($setting->_key, $name, 2);
334e1b31a95SBen Coburn        }
335e1b31a95SBen Coburn        if (!empty($toc['plugin'])) {
336b8595a66SAndreas Gohr            $t[] = html_mktocitem('plugin_settings', $this->getLang('_header_plugin'), 1);
337e1b31a95SBen Coburn        }
338e1b31a95SBen Coburn        foreach($toc['plugin'] as $setting) {
339e1b31a95SBen Coburn            $name = $setting->prompt($this);
340b8595a66SAndreas Gohr            $t[] = html_mktocitem($setting->_key, $name, 2);
341e1b31a95SBen Coburn        }
342e1b31a95SBen Coburn        if (isset($toc['template'])) {
343b8595a66SAndreas Gohr            $t[] = html_mktocitem('template_settings', $this->getLang('_header_template'), 1);
344e1b31a95SBen Coburn            $setting = $toc['template'];
345e1b31a95SBen Coburn            $name = $setting->prompt($this);
346b8595a66SAndreas Gohr            $t[] = html_mktocitem($setting->_key, $name, 2);
347e1b31a95SBen Coburn        }
348e1b31a95SBen Coburn        if ($has_undefined && $allow_debug) {
349b8595a66SAndreas Gohr            $t[] = html_mktocitem('undefined_settings', $this->getLang('_header_undefined'), 1);
350e1b31a95SBen Coburn        }
351e1b31a95SBen Coburn
352b8595a66SAndreas Gohr        return $t;
3534fa2dffcSBen Coburn    }
3544fa2dffcSBen Coburn
3554fa2dffcSBen Coburn    function _print_h1($id, $text) {
35663e967bdSAnika Henke        ptln('<h1 id="'.$id.'">'.$text.'</h1>');
3574fa2dffcSBen Coburn    }
3584fa2dffcSBen Coburn
3594fa2dffcSBen Coburn
36010449332Schris}
361