xref: /dokuwiki/lib/plugins/config/admin.php (revision c6639e6a6a4b11d65ecbc19f1bbbf2d9b32d0c19)
1<?php
2/**
3 * Configuration Manager admin plugin
4 *
5 * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
6 * @author     Christopher Smith <chris@jalakai.co.uk>
7 * @author     Ben Coburn <btcoburn@silicodon.net>
8 */
9
10use dokuwiki\plugin\config\core\Configuration;
11
12/**
13 * All DokuWiki plugins to extend the admin function
14 * need to inherit from this class
15 */
16class admin_plugin_config extends DokuWiki_Admin_Plugin {
17
18    const METADATA = __DIR__ . 'settings/config.metadata.php';
19    const IMGDIR = DOKU_BASE.'lib/plugins/config/images/';
20
21    protected $_config = null;
22    protected $_input = null;
23    protected $_changed = false;          // set to true if configuration has altered
24    protected $_error = false;
25    protected $_session_started = false;
26    protected $_localised_prompts = false;
27
28    /**
29     * @return int
30     */
31    public function getMenuSort() { return 100; }
32
33    /**
34     * handle user request
35     */
36    public function handle() {
37        global $ID, $INPUT;
38
39        if(!$this->_restore_session() || $INPUT->int('save') != 1 || !checkSecurityToken()) {
40            $this->_close_session();
41            return;
42        }
43
44        if(is_null($this->_config)) {
45            $this->_config = new Configuration(self::METADATA);
46        }
47
48        // don't go any further if the configuration is locked
49        if($this->_config->locked) {
50            $this->_close_session();
51            return;
52        }
53
54        $this->_input = $INPUT->arr('config');
55
56        foreach ($this->_config->setting as $key => $value){
57            $input = isset($this->_input[$key]) ? $this->_input[$key] : null;
58            if ($this->_config->setting[$key]->update($input)) {
59                $this->_changed = true;
60            }
61            if ($this->_config->setting[$key]->error()) $this->_error = true;
62        }
63
64        if ($this->_changed  && !$this->_error) {
65            $this->_config->save_settings($this->getPluginName());
66
67            // save state & force a page reload to get the new settings to take effect
68            $_SESSION['PLUGIN_CONFIG'] = array('state' => 'updated', 'time' => time());
69            $this->_close_session();
70            send_redirect(wl($ID,array('do'=>'admin','page'=>'config'),true,'&'));
71            exit();
72        } elseif(!$this->_error) {
73            $this->_config->touch_settings(); // just touch to refresh cache
74        }
75
76        $this->_close_session();
77    }
78
79    /**
80     * output appropriate html
81     */
82    public function html() {
83        $allow_debug = $GLOBALS['conf']['allowdebug']; // avoid global $conf; here.
84        global $lang;
85        global $ID;
86
87        if (is_null($this->_config)) { $this->_config = new Configuration(self::METADATA); }
88        $this->setupLocale(true);
89
90        print $this->locale_xhtml('intro');
91
92        ptln('<div id="config__manager">');
93
94        if ($this->_config->locked)
95            ptln('<div class="info">'.$this->getLang('locked').'</div>');
96        elseif ($this->_error)
97            ptln('<div class="error">'.$this->getLang('error').'</div>');
98        elseif ($this->_changed)
99            ptln('<div class="success">'.$this->getLang('updated').'</div>');
100
101        // POST to script() instead of wl($ID) so config manager still works if
102        // rewrite config is broken. Add $ID as hidden field to remember
103        // current ID in most cases.
104        ptln('<form action="'.script().'" method="post">');
105        ptln('<div class="no"><input type="hidden" name="id" value="'.$ID.'" /></div>');
106        formSecurityToken();
107        $this->_print_h1('dokuwiki_settings', $this->getLang('_header_dokuwiki'));
108
109        /** @var setting[] $undefined_settings */
110        $undefined_settings = array();
111        $in_fieldset = false;
112        $first_plugin_fieldset = true;
113        $first_template_fieldset = true;
114        foreach($this->_config->setting as $setting) {
115            if (is_a($setting, 'setting_hidden')) {
116                // skip hidden (and undefined) settings
117                if ($allow_debug && is_a($setting, 'setting_undefined')) {
118                    $undefined_settings[] = $setting;
119                } else {
120                    continue;
121                }
122            } else if (is_a($setting, 'setting_fieldset')) {
123                // config setting group
124                if ($in_fieldset) {
125                    ptln('  </table>');
126                    ptln('  </div>');
127                    ptln('  </fieldset>');
128                } else {
129                    $in_fieldset = true;
130                }
131                if ($first_plugin_fieldset && substr($setting->_key, 0, 10)=='plugin'.Configuration::KEYMARKER) {
132                    $this->_print_h1('plugin_settings', $this->getLang('_header_plugin'));
133                    $first_plugin_fieldset = false;
134                } else if ($first_template_fieldset && substr($setting->_key, 0, 7)=='tpl'.Configuration::KEYMARKER) {
135                    $this->_print_h1('template_settings', $this->getLang('_header_template'));
136                    $first_template_fieldset = false;
137                }
138                ptln('  <fieldset id="'.$setting->_key.'">');
139                ptln('  <legend>'.$setting->prompt($this).'</legend>');
140                ptln('  <div class="table">');
141                ptln('  <table class="inline">');
142            } else {
143                // config settings
144                list($label,$input) = $setting->html($this, $this->_error);
145
146                $class = $setting->is_default()
147                    ? ' class="default"'
148                    : ($setting->is_protected() ? ' class="protected"' : '');
149                $error = $setting->error()
150                    ? ' class="value error"'
151                    : ' class="value"';
152                $icon = $setting->caution()
153                    ? '<img src="'.self::IMGDIR.$setting->caution().'.png" '.
154                      'alt="'.$setting->caution().'" title="'.$this->getLang($setting->caution()).'" />'
155                    : '';
156
157                ptln('    <tr'.$class.'>');
158                ptln('      <td class="label">');
159                ptln('        <span class="outkey">'.$setting->_out_key(true, true).'</span>');
160                ptln('        '.$icon.$label);
161                ptln('      </td>');
162                ptln('      <td'.$error.'>'.$input.'</td>');
163                ptln('    </tr>');
164            }
165        }
166
167        ptln('  </table>');
168        ptln('  </div>');
169        if ($in_fieldset) {
170            ptln('  </fieldset>');
171        }
172
173        // show undefined settings list
174        if ($allow_debug && !empty($undefined_settings)) {
175            /**
176             * Callback for sorting settings
177             *
178             * @param setting $a
179             * @param setting $b
180             * @return int if $a is lower/equal/higher than $b
181             */
182            function _setting_natural_comparison($a, $b) {
183                return strnatcmp($a->_key, $b->_key);
184            }
185
186            usort($undefined_settings, '_setting_natural_comparison');
187            $this->_print_h1('undefined_settings', $this->getLang('_header_undefined'));
188            ptln('<fieldset>');
189            ptln('<div class="table">');
190            ptln('<table class="inline">');
191            $undefined_setting_match = array();
192            foreach($undefined_settings as $setting) {
193                if (
194                    preg_match(
195                        '/^(?:plugin|tpl)'.Configuration::KEYMARKER.'.*?'.Configuration::KEYMARKER.'(.*)$/',
196                        $setting->_key,
197                        $undefined_setting_match
198                    )
199                ) {
200                    $undefined_setting_key = $undefined_setting_match[1];
201                } else {
202                    $undefined_setting_key = $setting->_key;
203                }
204                ptln('  <tr>');
205                ptln('    <td class="label"><span title="$meta[\''.$undefined_setting_key.'\']">$'.
206                     $this->_config->_name.'[\''.$setting->_out_key().'\']</span></td>');
207                ptln('    <td>'.$this->getLang('_msg_'.get_class($setting)).'</td>');
208                ptln('  </tr>');
209            }
210            ptln('</table>');
211            ptln('</div>');
212            ptln('</fieldset>');
213        }
214
215        // finish up form
216        ptln('<p>');
217        ptln('  <input type="hidden" name="do"     value="admin" />');
218        ptln('  <input type="hidden" name="page"   value="config" />');
219
220        if (!$this->_config->locked) {
221            ptln('  <input type="hidden" name="save"   value="1" />');
222            ptln('  <button type="submit" name="submit" accesskey="s">'.$lang['btn_save'].'</button>');
223            ptln('  <button type="reset">'.$lang['btn_reset'].'</button>');
224        }
225
226        ptln('</p>');
227
228        ptln('</form>');
229        ptln('</div>');
230    }
231
232    /**
233     * @return boolean   true - proceed with handle, false - don't proceed
234     */
235    protected function _restore_session() {
236
237        // dokuwiki closes the session before act_dispatch. $_SESSION variables are all set,
238        // however they can't be changed without starting the session again
239        if (!headers_sent()) {
240            session_start();
241            $this->_session_started = true;
242        }
243
244        if (!isset($_SESSION['PLUGIN_CONFIG'])) return true;
245
246        $session = $_SESSION['PLUGIN_CONFIG'];
247        unset($_SESSION['PLUGIN_CONFIG']);
248
249        // still valid?
250        if (time() - $session['time'] > 120) return true;
251
252        switch ($session['state']) {
253            case 'updated' :
254                $this->_changed = true;
255                return false;
256        }
257
258        return true;
259    }
260
261    protected function _close_session() {
262      if ($this->_session_started) session_write_close();
263    }
264
265    /**
266     * @param bool $prompts
267     */
268    public function setupLocale($prompts=false) {
269
270        parent::setupLocale();
271        if (!$prompts || $this->_localised_prompts) return;
272
273        $this->_setup_localised_plugin_prompts();
274        $this->_localised_prompts = true;
275
276    }
277
278    /**
279     * @return bool
280     */
281    protected function _setup_localised_plugin_prompts() {
282        global $conf;
283
284        $langfile   = '/lang/'.$conf['lang'].'/settings.php';
285        $enlangfile = '/lang/en/settings.php';
286
287        if ($dh = opendir(DOKU_PLUGIN)) {
288            while (false !== ($plugin = readdir($dh))) {
289                if ($plugin == '.' || $plugin == '..' || $plugin == 'tmp' || $plugin == 'config') continue;
290                if (is_file(DOKU_PLUGIN.$plugin)) continue;
291
292                if (file_exists(DOKU_PLUGIN.$plugin.$enlangfile)){
293                    $lang = array();
294                    @include(DOKU_PLUGIN.$plugin.$enlangfile);
295                    if ($conf['lang'] != 'en') @include(DOKU_PLUGIN.$plugin.$langfile);
296                    foreach ($lang as $key => $value){
297                        $this->lang['plugin'.Configuration::KEYMARKER.$plugin.Configuration::KEYMARKER.$key] = $value;
298                    }
299                }
300
301                // fill in the plugin name if missing (should exist for plugins with settings)
302                if (!isset($this->lang['plugin'.Configuration::KEYMARKER.$plugin.Configuration::KEYMARKER.'plugin_settings_name'])) {
303                    $this->lang['plugin'.Configuration::KEYMARKER.$plugin.Configuration::KEYMARKER.'plugin_settings_name'] =
304                      ucwords(str_replace('_', ' ', $plugin));
305                }
306            }
307            closedir($dh);
308      }
309
310        // the same for the active template
311        $tpl = $conf['template'];
312
313        if (file_exists(tpl_incdir().$enlangfile)){
314            $lang = array();
315            @include(tpl_incdir().$enlangfile);
316            if ($conf['lang'] != 'en') @include(tpl_incdir().$langfile);
317            foreach ($lang as $key => $value){
318                $this->lang['tpl'.Configuration::KEYMARKER.$tpl.Configuration::KEYMARKER.$key] = $value;
319            }
320        }
321
322        // fill in the template name if missing (should exist for templates with settings)
323        if (!isset($this->lang['tpl'.Configuration::KEYMARKER.$tpl.Configuration::KEYMARKER.'template_settings_name'])) {
324            $this->lang['tpl'.Configuration::KEYMARKER.$tpl.Configuration::KEYMARKER.'template_settings_name'] =
325              ucwords(str_replace('_', ' ', $tpl));
326        }
327
328        return true;
329    }
330
331    /**
332     * Generates a two-level table of contents for the config plugin.
333     *
334     * @author Ben Coburn <btcoburn@silicodon.net>
335     *
336     * @return array
337     */
338    public function getTOC() {
339        if (is_null($this->_config)) { $this->_config = new Configuration(self::METADATA); }
340        $this->setupLocale(true);
341
342        $allow_debug = $GLOBALS['conf']['allowdebug']; // avoid global $conf; here.
343
344        // gather toc data
345        $has_undefined = false;
346        $toc = array('conf'=>array(), 'plugin'=>array(), 'template'=>null);
347        foreach($this->_config->setting as $setting) {
348            if (is_a($setting, 'setting_fieldset')) {
349                if (substr($setting->_key, 0, 10)=='plugin'.Configuration::KEYMARKER) {
350                    $toc['plugin'][] = $setting;
351                } else if (substr($setting->_key, 0, 7)=='tpl'.Configuration::KEYMARKER) {
352                    $toc['template'] = $setting;
353                } else {
354                    $toc['conf'][] = $setting;
355                }
356            } else if (!$has_undefined && is_a($setting, 'setting_undefined')) {
357                $has_undefined = true;
358            }
359        }
360
361        // build toc
362        $t = array();
363
364        $check = false;
365        $title = $this->getLang('_configuration_manager');
366        $t[] = html_mktocitem(sectionID($title, $check), $title, 1);
367        $t[] = html_mktocitem('dokuwiki_settings', $this->getLang('_header_dokuwiki'), 1);
368        /** @var setting $setting */
369        foreach($toc['conf'] as $setting) {
370            $name = $setting->prompt($this);
371            $t[] = html_mktocitem($setting->_key, $name, 2);
372        }
373        if (!empty($toc['plugin'])) {
374            $t[] = html_mktocitem('plugin_settings', $this->getLang('_header_plugin'), 1);
375        }
376        foreach($toc['plugin'] as $setting) {
377            $name = $setting->prompt($this);
378            $t[] = html_mktocitem($setting->_key, $name, 2);
379        }
380        if (isset($toc['template'])) {
381            $t[] = html_mktocitem('template_settings', $this->getLang('_header_template'), 1);
382            $setting = $toc['template'];
383            $name = $setting->prompt($this);
384            $t[] = html_mktocitem($setting->_key, $name, 2);
385        }
386        if ($has_undefined && $allow_debug) {
387            $t[] = html_mktocitem('undefined_settings', $this->getLang('_header_undefined'), 1);
388        }
389
390        return $t;
391    }
392
393    /**
394     * @param string $id
395     * @param string $text
396     */
397    protected function _print_h1($id, $text) {
398        ptln('<h1 id="'.$id.'">'.$text.'</h1>');
399    }
400
401    /**
402     * Adds a translation to this plugin's language array
403     *
404     * @param string $key
405     * @param string $value
406     */
407    public function addLang($key, $value) {
408        if (!$this->localised) $this->setupLocale();
409        $this->lang[$key] = $value;
410    }
411}
412