Seditio Source
Root |
./othercms/dotclear-2.22/inc/admin/lib.dc.page.php
<?php
/**
 * @package Dotclear
 * @subpackage Backend
 *
 * @copyright Olivier Meunier & Association Dotclear
 * @copyright GPL-2.0-only
 */
if (!defined('DC_RC_PATH')) {
    return;
}

define('DC_AUTH_PAGE', 'auth.php');

class
dcPage
{
    private static
$loaded_js     = [];
    private static
$loaded_css    = [];
    private static
$preloaded     = [];
    private static
$xframe_loaded = false;

    private static function
getCore()
    {
        return
$GLOBALS['core'];
    }

   
/**
     * Auth check
     *
     * @param      string  $permissions  The permissions
     * @param      bool    $home         The home
     */
   
public static function check($permissions, $home = false)
    {
       
$core = self::getCore();

        if (
$core->blog && $core->auth->check($permissions, $core->blog->id)) {
            return;
        }

       
// Check if dashboard is not the current page et if it is granted for the user
       
if (!$home && $core->blog && $core->auth->check('usage,contentadmin', $core->blog->id)) {
           
// Go back to the dashboard
           
http::redirect(DC_ADMIN_URL);
        }

        if (
session_id()) {
           
$core->session->destroy();
        }
       
http::redirect(DC_AUTH_PAGE);
    }

   
/**
     * Check super admin
     *
     * @param      bool  $home   The home
     */
   
public static function checkSuper($home = false)
    {
       
$core = self::getCore();

        if (!
$core->auth->isSuperAdmin()) {
           
// Check if dashboard is not the current page et if it is granted for the user
           
if (!$home && $core->blog && $core->auth->check('usage,contentadmin', $core->blog->id)) {
               
// Go back to the dashboard
               
http::redirect(DC_ADMIN_URL);
            }

            if (
session_id()) {
               
$core->session->destroy();
            }
           
http::redirect(DC_AUTH_PAGE);
        }
    }

   
/**
     * Top of admin page
     *
     * @param      string  $title       The title
     * @param      string  $head        The head
     * @param      string  $breadcrumb  The breadcrumb
     * @param      array   $options     The options
     */
   
public static function open($title = '', $head = '', $breadcrumb = '', $options = [])
    {
       
$core = self::getCore();
       
$js   = [];

       
# List of user's blogs
       
if ($core->auth->getBlogCount() == 1 || $core->auth->getBlogCount() > 20) {
           
$blog_box = '<p>' . __('Blog:') . ' <strong title="' . html::escapeHTML($core->blog->url) . '">' .
           
html::escapeHTML($core->blog->name) . '</strong>';

            if (
$core->auth->getBlogCount() > 20) {
               
$blog_box .= ' - <a href="' . $core->adminurl->get('admin.blogs') . '">' . __('Change blog') . '</a>';
            }
           
$blog_box .= '</p>';
        } else {
           
$rs_blogs = $core->getBlogs(['order' => 'LOWER(blog_name)', 'limit' => 20]);
           
$blogs    = [];
            while (
$rs_blogs->fetch()) {
               
$blogs[html::escapeHTML($rs_blogs->blog_name . ' - ' . $rs_blogs->blog_url)] = $rs_blogs->blog_id;
            }
           
$blog_box = '<p><label for="switchblog" class="classic">' . __('Blogs:') . '</label> ' .
           
$core->formNonce() . form::combo('switchblog', $blogs, $core->blog->id) .
           
form::hidden(['redir'], $_SERVER['REQUEST_URI']) .
           
'<input type="submit" value="' . __('ok') . '" class="hidden-if-js" /></p>';
        }

       
$safe_mode = isset($_SESSION['sess_safe_mode']) && $_SESSION['sess_safe_mode'];

       
# Display
       
$headers = new ArrayObject([]);

       
# Content-Type
       
$headers['content-type'] = 'Content-Type: text/html; charset=UTF-8';

       
# Referrer Policy for admin pages
       
$headers['referrer'] = 'Referrer-Policy: strict-origin';

       
# Prevents Clickjacking as far as possible
       
if (isset($options['x-frame-allow'])) {
           
self::setXFrameOptions($headers, $options['x-frame-allow']);
        } else {
           
self::setXFrameOptions($headers);
        }

       
# Content-Security-Policy (only if safe mode if not active, it may help)
       
if (!$safe_mode && $core->blog->settings->system->csp_admin_on) {
           
// Get directives from settings if exist, else set defaults
           
$csp = new ArrayObject([]);

           
// SQlite Clearbricks driver does not allow using single quote at beginning or end of a field value
                                                                                // so we have to use neutral values (localhost and 127.0.0.1) for some CSP directives
           
$csp_prefix = $core->con->syntax() == 'sqlite' ? 'localhost ' : ''; // Hack for SQlite Clearbricks syntax
           
$csp_suffix = $core->con->syntax() == 'sqlite' ? ' 127.0.0.1' : ''; // Hack for SQlite Clearbricks syntax

           
$csp['default-src'] = $core->blog->settings->system->csp_admin_default ?:
           
$csp_prefix . "'self'" . $csp_suffix;
           
$csp['script-src'] = $core->blog->settings->system->csp_admin_script ?:
           
$csp_prefix . "'self' 'unsafe-eval'" . $csp_suffix;
           
$csp['style-src'] = $core->blog->settings->system->csp_admin_style ?:
           
$csp_prefix . "'self' 'unsafe-inline'" . $csp_suffix;
           
$csp['img-src'] = $core->blog->settings->system->csp_admin_img ?:
           
$csp_prefix . "'self' data: https://media.dotaddict.org blob:";

           
# Cope with blog post preview (via public URL in iframe)
           
if (!is_null($core->blog->host)) {
               
$csp['default-src'] .= ' ' . parse_url($core->blog->host, PHP_URL_HOST);
               
$csp['script-src']  .= ' ' . parse_url($core->blog->host, PHP_URL_HOST);
               
$csp['style-src']   .= ' ' . parse_url($core->blog->host, PHP_URL_HOST);
            }
           
# Cope with media display in media manager (via public URL)
           
if (!is_null($core->media)) {
               
$csp['img-src'] .= ' ' . parse_url($core->media->root_url, PHP_URL_HOST);
            } elseif (!
is_null($core->blog->host)) {
               
// Let's try with the blog URL
               
$csp['img-src'] .= ' ' . parse_url($core->blog->host, PHP_URL_HOST);
            }
           
# Allow everything in iframe (used by editors to preview public content)
           
$csp['frame-src'] = '*';

           
# --BEHAVIOR-- adminPageHTTPHeaderCSP
           
$core->callBehavior('adminPageHTTPHeaderCSP', $csp);

           
// Construct CSP header
           
$directives = [];
            foreach (
$csp as $key => $value) {
                if (
$value) {
                   
$directives[] = $key . ' ' . $value;
                }
            }
            if (
count($directives)) {
               
$directives[]   = 'report-uri ' . DC_ADMIN_URL . 'csp_report.php';
               
$report_only    = ($core->blog->settings->system->csp_admin_report_only) ? '-Report-Only' : '';
               
$headers['csp'] = 'Content-Security-Policy' . $report_only . ': ' . implode(' ; ', $directives);
            }
        }

       
# --BEHAVIOR-- adminPageHTTPHeaders
       
$core->callBehavior('adminPageHTTPHeaders', $headers);
        foreach (
$headers as $key => $value) {
           
header($value);
        }

       
$data_theme = $core->auth->user_prefs->interface->theme;

        echo
       
'<!DOCTYPE html>' .
       
'<html lang="' . $core->auth->getInfo('user_lang') . '" data-theme="' . $data_theme . '">' . "\n" .
       
"<head>\n" .
       
'  <meta charset="UTF-8" />' . "\n" .
       
'  <meta name="ROBOTS" content="NOARCHIVE,NOINDEX,NOFOLLOW" />' . "\n" .
       
'  <meta name="GOOGLEBOT" content="NOSNIPPET" />' . "\n" .
       
'  <meta name="viewport" content="width=device-width, initial-scale=1.0" />' . "\n" .
       
'  <title>' . $title . ' - ' . html::escapeHTML($core->blog->name) . ' - ' . html::escapeHTML(DC_VENDOR_NAME) . ' - ' . DC_VERSION . '</title>' . "\n";

        echo
self::preload('style/default.css') . self::cssLoad('style/default.css');

        if (
$rtl = (l10n::getLanguageTextDirection($GLOBALS['_lang']) == 'rtl')) {
            echo
self::cssLoad('style/default-rtl.css');
        }

       
$core->auth->user_prefs->addWorkspace('interface');
        if (!
$core->auth->user_prefs->interface->hide_std_favicon) {
            echo
               
'<link rel="icon" type="image/png" href="images/favicon96-login.png" />' . "\n" .
               
'<link rel="shortcut icon" href="images/favicon.ico" type="image/x-icon" />' . "\n";
        }
        if (
$core->auth->user_prefs->interface->htmlfontsize) {
           
$js['htmlFontSize'] = $core->auth->user_prefs->interface->htmlfontsize;
        }
       
$js['hideMoreInfo']   = (bool) $core->auth->user_prefs->interface->hidemoreinfo;
       
$js['showAjaxLoader'] = (bool) $core->auth->user_prefs->interface->showajaxloader;

       
$core->auth->user_prefs->addWorkspace('accessibility');
       
$js['noDragDrop'] = (bool) $core->auth->user_prefs->accessibility->nodragdrop;

       
$js['debug'] = !!DC_DEBUG;  // @phpstan-ignore-line

       
$js['showIp'] = $core->blog && $core->blog->id ? $core->auth->check('contentadmin', $core->blog->id) : false;

       
// Set some JSON data
       
echo dcUtils::jsJson('dotclear_init', $js);

        echo
       
self::jsCommon() .
       
self::jsToggles() .
           
$head;

       
# --BEHAVIOR-- adminPageHTMLHead
       
$core->callBehavior('adminPageHTMLHead');

        echo
       
"</head>\n" .
       
'<body id="dotclear-admin" class="no-js' .
        (
$rtl ? ' rtl ' : '') .
        (
$safe_mode ? ' safe-mode' : '') .
        (
DC_DEBUG ? // @phpstan-ignore-line
           
' debug-mode' :
           
'') .
       
'">' . "\n" .

       
'<ul id="prelude">' .
       
'<li><a href="#content">' . __('Go to the content') . '</a></li>' .
       
'<li><a href="#main-menu">' . __('Go to the menu') . '</a></li>' .
       
'<li><a href="#help">' . __('Go to help') . '</a></li>' .
       
'</ul>' . "\n" .
       
'<header id="header" role="banner">' .
       
'<h1><a href="' . $core->adminurl->get('admin.home') . '"><span class="hidden">' . DC_VENDOR_NAME . '</span></a></h1>' . "\n";

        echo
       
'<form action="' . $core->adminurl->get('admin.home') . '" method="post" id="top-info-blog">' .
       
$blog_box .
       
'<p><a href="' . $core->blog->url . '" class="outgoing" title="' . __('Go to site') .
       
'">' . __('Go to site') . '<img src="images/outgoing-link.svg" alt="" /></a>' .
       
'</p></form>' .
       
'<ul id="top-info-user">' .
       
'<li><a class="' . (preg_match('/' . preg_quote($core->adminurl->get('admin.home')) . '$/', $_SERVER['REQUEST_URI']) ? ' active' : '') . '" href="' . $core->adminurl->get('admin.home') . '">' . __('My dashboard') . '</a></li>' .
       
'<li><a class="smallscreen' . (preg_match('/' . preg_quote($core->adminurl->get('admin.user.preferences')) . '(\?.*)?$/', $_SERVER['REQUEST_URI']) ? ' active' : '') .
       
'" href="' . $core->adminurl->get('admin.user.preferences') . '">' . __('My preferences') . '</a></li>' .
       
'<li><a href="' . $core->adminurl->get('admin.home', ['logout' => 1]) . '" class="logout"><span class="nomobile">' . sprintf(__('Logout %s'), $core->auth->userID()) .
           
'</span><img src="images/logout.svg" alt="" /></a></li>' .
           
'</ul>' .
           
'</header>'; // end header

       
echo
       
'<div id="wrapper" class="clearfix">' . "\n" .
       
'<div class="hidden-if-no-js collapser-box"><button type="button" id="collapser" class="void-btn">' .
       
'<img class="collapse-mm visually-hidden" src="images/collapser-hide.png" alt="' . __('Hide main menu') . '" />' .
       
'<img class="expand-mm visually-hidden" src="images/collapser-show.png" alt="' . __('Show main menu') . '" />' .
           
'</button></div>' .
           
'<main id="main" role="main">' . "\n" .
           
'<div id="content" class="clearfix">' . "\n";

       
# Safe mode
       
if ($safe_mode) {
            echo
           
'<div class="warning" role="alert"><h3>' . __('Safe mode') . '</h3>' .
           
'<p>' . __('You are in safe mode. All plugins have been temporarily disabled. Remind to log out then log in again normally to get back all functionalities') . '</p>' .
               
'</div>';
        }

       
// Display breadcrumb (if given) before any error messages
       
echo $breadcrumb;

       
// Display notices and errors
       
echo dcAdminNotices::getNotices();
    }

    public static function
notices()
    {
        return
dcAdminNotices::getNotices();
    }

    public static function
addNotice($type, $message, $options = [])
    {
       
dcAdminNotices::addNotice($type, $message, $options);
    }

    public static function
addSuccessNotice($message, $options = [])
    {
       
self::addNotice('success', $message, $options);
    }

    public static function
addWarningNotice($message, $options = [])
    {
       
self::addNotice('warning', $message, $options);
    }

    public static function
addErrorNotice($message, $options = [])
    {
       
self::addNotice('error', $message, $options);
    }

   
/**
     * The end
     */
   
public static function close()
    {
       
$core = self::getCore();

        if (!
$GLOBALS['__resources']['ctxhelp']) {
            if (!
$core->auth->user_prefs->interface->hidehelpbutton) {
                echo
               
'<p id="help-button"><a href="' . $core->adminurl->get('admin.help') . '" class="outgoing" title="' .
               
__('Global help') . '">' . __('Global help') . '</a></p>';
            }
        }

       
$menu = &$GLOBALS['_menu'];

        echo
       
"</div>\n" .  // End of #content
       
"</main>\n" . // End of #main

       
'<nav id="main-menu" role="navigation">' . "\n" .

       
'<form id="search-menu" action="' . $core->adminurl->get('admin.search') . '" method="get" role="search">' .
       
'<p><label for="qx" class="hidden">' . __('Search:') . ' </label>' . form::field('qx', 30, 255, '') .
       
'<input type="submit" value="' . __('OK') . '" /></p>' .
           
'</form>';

        foreach (
$menu as $k => $v) {
            echo
$menu[$k]->draw();
        }

       
$text = sprintf(__('Thank you for using %s.'), 'Dotclear ' . DC_VERSION);

       
# --BEHAVIOR-- adminPageFooter
       
$textAlt = $core->callBehavior('adminPageFooter', $core, $text);
        if (
$textAlt != '') {
           
$text = $textAlt;
        }
       
$text = html::escapeHTML($text);

        echo
       
'</nav>' . "\n" . // End of #main-menu
       
"</div>\n";       // End of #wrapper

       
echo '<p id="gototop"><a href="#wrapper">' . __('Page top') . '</a></p>' . "\n";

       
$figure = "\n" .
       
"\n" .
       
'          (╯°□°)╯︵ ┻━┻' . "\n" .
       
"\n";

        echo
           
'<footer id="footer" role="contentinfo">' .
           
'<a href="https://dotclear.org/" title="' . $text . '">' .
           
'<img src="style/dc_logos/w-dotclear90.png" alt="' . $text . '" /></a></footer>' . "\n" .
           
'<!-- ' . "\n" .
           
$figure .
           
' -->' . "\n";

        if (
defined('DC_DEV') && DC_DEV === true) {
            echo
self::debugInfo();
        }

        echo
           
'</body></html>';
    }

   
/**
     * The top of a popup.
     *
     * @param      string  $title       The title
     * @param      string  $head        The head
     * @param      string  $breadcrumb  The breadcrumb
     */
   
public static function openPopup($title = '', $head = '', $breadcrumb = '')
    {
       
$core = self::getCore();
       
$js   = [];

       
$safe_mode = isset($_SESSION['sess_safe_mode']) && $_SESSION['sess_safe_mode'];

       
# Display
       
header('Content-Type: text/html; charset=UTF-8');

       
# Referrer Policy for admin pages
       
header('Referrer-Policy: strict-origin');

       
# Prevents Clickjacking as far as possible
       
header('X-Frame-Options: SAMEORIGIN'); // FF 3.6.9+ Chrome 4.1+ IE 8+ Safari 4+ Opera 10.5+

       
$data_theme = $core->auth->user_prefs->interface->theme;

        echo
       
'<!DOCTYPE html>' .
       
'<html lang="' . $core->auth->getInfo('user_lang') . '" data-theme="' . $data_theme . '">' . "\n" .
       
"<head>\n" .
       
'  <meta charset="UTF-8" />' . "\n" .
       
'  <meta name="viewport" content="width=device-width, initial-scale=1.0" />' . "\n" .
       
'  <title>' . $title . ' - ' . html::escapeHTML($core->blog->name) . ' - ' . html::escapeHTML(DC_VENDOR_NAME) . ' - ' . DC_VERSION . '</title>' . "\n" .
           
'  <meta name="ROBOTS" content="NOARCHIVE,NOINDEX,NOFOLLOW" />' . "\n" .
           
'  <meta name="GOOGLEBOT" content="NOSNIPPET" />' . "\n";

        echo
self::preload('style/default.css') . self::cssLoad('style/default.css');

        if (
$rtl = (l10n::getLanguageTextDirection($GLOBALS['_lang']) == 'rtl')) {
            echo
self::cssLoad('style/default-rtl.css');
        }

       
$core->auth->user_prefs->addWorkspace('interface');
        if (
$core->auth->user_prefs->interface->htmlfontsize) {
           
$js['htmlFontSize'] = $core->auth->user_prefs->interface->htmlfontsize;
        }
       
$js['hideMoreInfo']   = (bool) $core->auth->user_prefs->interface->hidemoreinfo;
       
$js['showAjaxLoader'] = (bool) $core->auth->user_prefs->interface->showajaxloader;

       
$core->auth->user_prefs->addWorkspace('accessibility');
       
$js['noDragDrop'] = (bool) $core->auth->user_prefs->accessibility->nodragdrop;

       
$js['debug'] = !!DC_DEBUG;  // @phpstan-ignore-line

        // Set JSON data
       
echo dcUtils::jsJson('dotclear_init', $js);

        echo
       
self::jsCommon() .
       
self::jsToggles() .
           
$head;

       
# --BEHAVIOR-- adminPageHTMLHead
       
$core->callBehavior('adminPageHTMLHead');

        echo
           
"</head>\n" .
           
'<body id="dotclear-admin" class="popup' .
            (
$rtl ? 'rtl' : '') .
            (
$safe_mode ? ' safe-mode' : '') .
            (
DC_DEBUG ? // @phpstan-ignore-line
               
' debug-mode' :
               
'') .
           
'">' . "\n" .

           
'<h1>' . DC_VENDOR_NAME . '</h1>' . "\n";

        echo
           
'<div id="wrapper">' . "\n" .
           
'<main id="main" role="main">' . "\n" .
           
'<div id="content">' . "\n";

       
// display breadcrumb if given
       
echo $breadcrumb;

       
// Display notices and errors
       
echo dcAdminNotices::getNotices();
    }

   
/**
     * The end of a popup.
     */
   
public static function closePopup()
    {
        echo
       
"</div>\n" .  // End of #content
       
"</main>\n" . // End of #main
       
"</div>\n" .  // End of #wrapper

       
'<p id="gototop"><a href="#wrapper">' . __('Page top') . '</a></p>' . "\n" .

           
'<footer id="footer" role="contentinfo"><p>&nbsp;</p></footer>' . "\n" .
           
'</body></html>';
    }

   
/**
     * Get breadcrumb
     *
     * @param      mixed   $elements  The elements
     * @param      array   $options   The options
     *
     * @return     string
     */
   
public static function breadcrumb($elements = null, $options = [])
    {
       
$core = self::getCore();

       
$with_home_link = $options['home_link'] ?? true;
       
$hl             = $options['hl']        ?? true;
       
$hl_pos         = $options['hl_pos']    ?? -1;
       
// First item of array elements should be blog's name, System or Plugins
       
$res = '<h2>' . ($with_home_link ?
           
'<a class="go_home" href="' . $core->adminurl->get('admin.home') . '">' .
           
'<img class="go_home light-only" src="style/dashboard.svg" alt="' . __('Go to dashboard') . '" />' .
           
'<img class="go_home dark-only" src="style/dashboard-dark.svg" alt="' . __('Go to dashboard') . '" />' .
           
'</a>' :
           
'<img class="go_home light-only" src="style/dashboard-alt.svg" alt="" />' .
           
'<img class="go_home dark-only" src="style/dashboard-alt-dark.svg" alt="" />');
       
$index = 0;
        if (
$hl_pos < 0) {
           
$hl_pos = count($elements) + $hl_pos;
        }
        foreach (
$elements as $element => $url) {
            if (
$hl && $index == $hl_pos) {
               
$element = sprintf('<span class="page-title">%s</span>', $element);
            }
           
$res .= ($with_home_link ? ($index == 1 ? ' : ' : ' &rsaquo; ') : ($index == 0 ? ' ' : ' &rsaquo; ')) .
                (
$url ? '<a href="' . $url . '">' : '') . $element . ($url ? '</a>' : '');
           
$index++;
        }
       
$res .= '</h2>';

        return
$res;
    }

    public static function
message($msg, $timestamp = true, $div = false, $echo = true, $class = 'message')
    {
        return
dcAdminNotices::message($msg, $timestamp, $div, $echo, $class);
    }

    public static function
success($msg, $timestamp = true, $div = false, $echo = true)
    {
        return
self::message($msg, $timestamp, $div, $echo, 'success');
    }

    public static function
warning($msg, $timestamp = true, $div = false, $echo = true)
    {
        return
self::message($msg, $timestamp, $div, $echo, 'warning-msg');
    }

   
/**
     * Get HTML code of debug information
     *
     * @return     string
     */
   
private static function debugInfo()
    {
       
$global_vars = implode(', ', array_keys($GLOBALS));

       
$res = '<div id="debug"><div>' .
       
'<p>memory usage: ' . memory_get_usage() . ' (' . files::size(memory_get_usage()) . ')</p>';

        if (
function_exists('xdebug_get_profiler_filename')) {
           
$res .= '<p>Elapsed time: ' . xdebug_time_index() . ' seconds</p>';

           
$prof_file = xdebug_get_profiler_filename();
            if (
$prof_file) {
               
$res .= '<p>Profiler file : ' . xdebug_get_profiler_filename() . '</p>';
            } else {
               
$prof_url = http::getSelfURI();
               
$prof_url .= (strpos($prof_url, '?') === false) ? '?' : '&';
               
$prof_url .= 'XDEBUG_PROFILE';
               
$res      .= '<p><a href="' . html::escapeURL($prof_url) . '">Trigger profiler</a></p>';
            }

           
/* xdebug configuration:
        zend_extension = /.../xdebug.so
        xdebug.auto_trace = On
        xdebug.trace_format = 0
        xdebug.trace_options = 1
        xdebug.show_mem_delta = On
        xdebug.profiler_enable = 0
        xdebug.profiler_enable_trigger = 1
        xdebug.profiler_output_dir = /tmp
        xdebug.profiler_append = 0
        xdebug.profiler_output_name = timestamp
         */
       
}

       
$res .= '<p>Global vars: ' . $global_vars . '</p>' .
           
'</div></div>';

        return
$res;
    }

    public static function
help($page, $index = '')
    {
       
# Deprecated but we keep this for plugins.
   
}

   
/**
     * Display Help block
     *
     * @param      mixed  ...$params  The parameters
     */
   
public static function helpBlock(...$params)
    {
       
$core = self::getCore();

        if (
$core->auth->user_prefs->interface->hidehelpbutton) {
            return;
        }

       
$args = new ArrayObject($params);

       
# --BEHAVIOR-- adminPageHelpBlock
       
$core->callBehavior('adminPageHelpBlock', $args);

        if (!
count($args)) {
            return;
        };

        global
$__resources;
        if (empty(
$__resources['help'])) {
            return;
        }

       
$content = '';
        foreach (
$args as $v) {
            if (
is_object($v) && isset($v->content)) {
               
$content .= $v->content;

                continue;
            }

            if (!isset(
$__resources['help'][$v])) {
                continue;
            }
           
$f = $__resources['help'][$v];
            if (!
file_exists($f) || !is_readable($f)) {
                continue;
            }

           
$fc = file_get_contents($f);
            if (
preg_match('|<body[^>]*?>(.*?)</body>|ms', $fc, $matches)) {
               
$content .= $matches[1];
            } else {
               
$content .= $fc;
            }
        }

        if (
trim($content) == '') {
            return;
        }

       
// Set contextual help global flag
       
$GLOBALS['__resources']['ctxhelp'] = true;

        echo
       
'<div id="help"><hr /><div class="help-content clear"><h3>' . __('Help about this page') . '</h3>' .
       
$content .
       
'</div>' .
       
'<div id="helplink"><hr />' .
       
'<p>' .
       
sprintf(__('See also %s'), sprintf('<a href="' . $core->adminurl->get('admin.help') . '">%s</a>', __('the global help'))) .
           
'.</p>' .
           
'</div></div>';
    }

   
/**
     * Get HTML code to preload resource
     *
     * @param      string  $src    The source
     * @param      string  $v      The version
     * @param      string  $type   The type
     *
     * @return     mixed
     */
   
public static function preload($src, $v = '', $type = 'style')
    {
       
$escaped_src = html::escapeHTML($src);
        if (!isset(
self::$preloaded[$escaped_src])) {
           
self::$preloaded[$escaped_src] = true;
           
$escaped_src                   = self::appendVersion($escaped_src, $v);

            return
'<link rel="preload" href="' . $escaped_src . '" as="' . $type . '" />' . "\n";
        }
    }

   
/**
     * Get HTML code to load CSS stylesheet
     *
     * @param      string  $src    The source
     * @param      string  $media  The media
     * @param      string  $v      The version
     *
     * @return     mixed
     */
   
public static function cssLoad($src, $media = 'screen', $v = '')
    {
       
$escaped_src = html::escapeHTML($src);
        if (!isset(
self::$loaded_css[$escaped_src])) {
           
self::$loaded_css[$escaped_src] = true;
           
$escaped_src                    = self::appendVersion($escaped_src, $v);

            return
'<link rel="stylesheet" href="' . $escaped_src . '" type="text/css" media="' . $media . '" />' . "\n";
        }
    }

   
/**
     * Wrapper for cssLoad to be used by module
     *
     * @param      string  $src    The source
     * @param      string  $media  The media
     * @param      string  $v      The version
     *
     * @return     mixed
     */
   
public static function cssModuleLoad($src, $media = 'screen', $v = '')
    {
        return
self::cssLoad(urldecode(self::getPF($src)), $media, $v);
    }

   
/**
     * Get HTML code to load JS script
     *
     * @param      string  $src    The source
     * @param      string  $v      The version
     *
     * @return     mixed
     */
   
public static function jsLoad($src, $v = '')
    {
       
$escaped_src = html::escapeHTML($src);
        if (!isset(
self::$loaded_js[$escaped_src])) {
           
self::$loaded_js[$escaped_src] = true;
           
$escaped_src                   = self::appendVersion($escaped_src, $v);

            return
'<script src="' . $escaped_src . '"></script>' . "\n";
        }
    }

   
/**
     * Wrapper for jsLoad to be used by module
     *
     * @param      string  $src    The source
     * @param      string  $v      The version
     *
     * @return     mixed
     */
   
public static function jsModuleLoad($src, $v = '')
    {
        return
self::jsLoad(urldecode(self::getPF($src)), $v);
    }

   
/**
     * Appends a version to force cache refresh if necessary.
     *
     * @param      string  $src    The source
     * @param      string  $v      The version
     *
     * @return     string
     */
   
private static function appendVersion(string $src, ?string $v = ''): string
   
{
        return
$src .
            (
strpos($src, '?') === false ? '?' : '&amp;') .
           
'v=' . (defined('DC_DEV') && DC_DEV === true ? md5(uniqid()) : ($v ?: DC_VERSION));
    }

   
/**
     * return a javascript variable definition line code
     *
     * @deprecated 2.15 use dcPage::jsJson() and dotclear.getData()/dotclear.mergeDeep() in javascript
     *
     * @param      string  $n      variable name
     * @param      mixed   $v      value
     *
     * @return     string  javascript code
     */
   
public static function jsVar($n, $v)
    {
        return
$n . " = '" . html::escapeJS($v) . "';\n";
    }

   
/**
     * return a list of javascript variables définitions code
     *
     * @deprecated 2.15 use dcPage::jsJson() and dotclear.getData()/dotclear.mergeDeep() in javascript
     *
     * @param      array  $vars   The variables
     *
     * @return     string  javascript code (inside <script… ></script>)
     */
   
public static function jsVars($vars)
    {
       
$ret = '<script>' . "\n";
        foreach (
$vars as $var => $value) {
           
$ret .= $var . ' = ' . (is_string($value) ? "'" . html::escapeJS($value) . "'" : $value) . ';' . "\n";
        }
       
$ret .= "</script>\n";

        return
$ret;
    }

   
/**
     * Get HTML code to load JS variables encoded as JSON
     *
     * @param      string  $id     The identifier
     * @param      mixed   $vars   The variables
     *
     * @return     string
     */
   
public static function jsJson($id, $vars)
    {
        return
dcUtils::jsJson($id, $vars);
    }

   
/**
     * Get HTML code to load toggles JS
     *
     * @return     string
     */
   
public static function jsToggles()
    {
       
$core = self::getCore();

       
$js = [];
        if (
$core->auth->user_prefs->toggles) {
           
$unfolded_sections = explode(',', (string) $core->auth->user_prefs->toggles->unfolded_sections);
            foreach (
$unfolded_sections as $k => &$v) {
                if (
$v !== '') {
                   
$js[$unfolded_sections[$k]] = true;
                }
            }
        }

        return
       
self::jsJson('dotclear_toggles', $js) .
       
self::jsLoad('js/toggles.js');
    }

   
/**
     * Get HTML code to load common JS for admin pages
     *
     * @return     string
     */
   
public static function jsCommon()
    {
       
$core = self::getCore();
        if (
$core->auth->user_prefs) {
           
$core->auth->user_prefs->addWorkspace('interface');
        }

       
$js = [
           
'nonce' => $core->getNonce(),

           
'img_plus_src' => 'images/expand.svg',
           
'img_plus_txt' => '▶',
           
'img_plus_alt' => __('uncover'),

           
'img_minus_src' => 'images/hide.svg',
           
'img_minus_txt' => '▼',
           
'img_minus_alt' => __('hide'),

           
'adblocker_check' => (
                (
                    !
defined('DC_ADBLOCKER_CHECK') || DC_ADBLOCKER_CHECK === true
               
) && $core->auth->user_prefs !== null && $core->auth->user_prefs->interface->nocheckadblocker !== true
           
),
        ];

       
$js_msg = [
           
'help'                                 => __('Need help?'),
           
'new_window'                           => __('new window'),
           
'help_hide'                            => __('Hide'),
           
'to_select'                            => __('Select:'),
           
'no_selection'                         => __('no selection'),
           
'select_all'                           => __('select all'),
           
'invert_sel'                           => __('Invert selection'),
           
'website'                              => __('Web site:'),
           
'email'                                => __('Email:'),
           
'ip_address'                           => __('IP address:'),
           
'error'                                => __('Error:'),
           
'entry_created'                        => __('Entry has been successfully created.'),
           
'edit_entry'                           => __('Edit entry'),
           
'view_entry'                           => __('view entry'),
           
'confirm_delete_posts'                 => __('Are you sure you want to delete selected entries (%s)?'),
           
'confirm_delete_medias'                => __('Are you sure you want to delete selected medias (%d)?'),
           
'confirm_delete_categories'            => __('Are you sure you want to delete selected categories (%s)?'),
           
'confirm_delete_post'                  => __('Are you sure you want to delete this entry?'),
           
'click_to_unlock'                      => __('Click here to unlock the field'),
           
'confirm_spam_delete'                  => __('Are you sure you want to delete all spams?'),
           
'confirm_delete_comments'              => __('Are you sure you want to delete selected comments (%s)?'),
           
'confirm_delete_comment'               => __('Are you sure you want to delete this comment?'),
           
'cannot_delete_users'                  => __('Users with posts cannot be deleted.'),
           
'confirm_delete_user'                  => __('Are you sure you want to delete selected users (%s)?'),
           
'confirm_delete_blog'                  => __('Are you sure you want to delete selected blogs (%s)?'),
           
'confirm_delete_category'              => __('Are you sure you want to delete category "%s"?'),
           
'confirm_reorder_categories'           => __('Are you sure you want to reorder all categories?'),
           
'confirm_delete_media'                 => __('Are you sure you want to remove media "%s"?'),
           
'confirm_delete_directory'             => __('Are you sure you want to remove directory "%s"?'),
           
'confirm_extract_current'              => __('Are you sure you want to extract archive in current directory?'),
           
'confirm_remove_attachment'            => __('Are you sure you want to remove attachment "%s"?'),
           
'confirm_delete_lang'                  => __('Are you sure you want to delete "%s" language?'),
           
'confirm_delete_plugin'                => __('Are you sure you want to delete "%s" plugin?'),
           
'confirm_delete_plugins'               => __('Are you sure you want to delete selected plugins?'),
           
'use_this_theme'                       => __('Use this theme'),
           
'remove_this_theme'                    => __('Remove this theme'),
           
'confirm_delete_theme'                 => __('Are you sure you want to delete "%s" theme?'),
           
'confirm_delete_themes'                => __('Are you sure you want to delete selected themes?'),
           
'confirm_delete_backup'                => __('Are you sure you want to delete this backup?'),
           
'confirm_revert_backup'                => __('Are you sure you want to revert to this backup?'),
           
'zip_file_content'                     => __('Zip file content'),
           
'xhtml_validator'                      => __('XHTML markup validator'),
           
'xhtml_valid'                          => __('XHTML content is valid.'),
           
'xhtml_not_valid'                      => __('There are XHTML markup errors.'),
           
'warning_validate_no_save_content'     => __('Attention: an audit of a content not yet registered.'),
           
'confirm_change_post_format'           => __('You have unsaved changes. Switch post format will loose these changes. Proceed anyway?'),
           
'confirm_change_post_format_noconvert' => __('Warning: post format change will not convert existing content. You will need to apply new format by yourself. Proceed anyway?'),
           
'load_enhanced_uploader'               => __('Loading enhanced uploader, please wait.'),

           
'module_author'  => __('Author:'),
           
'module_details' => __('Details'),
           
'module_support' => __('Support'),
           
'module_help'    => __('Help:'),
           
'module_section' => __('Section:'),
           
'module_tags'    => __('Tags:'),

           
'close_notice' => __('Hide this notice'),

           
'show_password' => __('Show password'),
           
'hide_password' => __('Hide password'),

           
'set_today' => __('Reset to now'),

           
'adblocker' => __('An ad blocker has been detected on this Dotclear dashboard (Ghostery, Adblock plus, uBlock origin, …) and it may interfere with some features. In this case you should disable it. Note that this detection may be disabled in your preferences.'),
        ];

        return
       
self::jsLoad('js/prepend.js') .
       
self::jsLoad('js/jquery/jquery.js') .
        (
           
DC_DEBUG ? // @phpstan-ignore-line
           
self::jsJson('dotclear_jquery', [
               
'mute' => (empty($core->blog) || $core->blog->settings->system->jquery_migrate_mute),
            ]) .
           
self::jsLoad('js/jquery-mute.js') .
           
self::jsLoad('js/jquery/jquery-migrate.js') :
           
''
       
) .

       
self::jsJson('dotclear', $js) .
       
self::jsJson('dotclear_msg', $js_msg) .

       
self::jsLoad('js/common.js') .
       
self::jsLoad('js/ads.js') .
       
self::jsLoad('js/services.js') .
       
self::jsLoad('js/prelude.js');
    }

   
/**
     * @deprecated since version 2.11
     *
     * @return     string  ( description_of_the_return_value )
     */
   
public static function jsLoadIE7()
    {
        return
'';
    }

   
/**
     * Get HTML code to load ConfirmClose JS
     *
     * @param      mixed  ...$args  The arguments
     *
     * @return     string
     */
   
public static function jsConfirmClose(...$args)
    {
       
$js = [
           
'prompt' => __('You have unsaved changes.'),
           
'forms'  => $args,
        ];

        return
       
self::jsJson('confirm_close', $js) .
       
self::jsLoad('js/confirm-close.js');
    }

   
/**
     * Get HTML code to load page tabs JS
     *
     * @param      mixed   $default  The default
     *
     * @return     string
     */
   
public static function jsPageTabs($default = null)
    {
       
$js = [
           
'default' => $default,
        ];

        return
       
self::jsJson('page_tabs', $js) .
       
self::jsLoad('js/jquery/jquery.pageTabs.js') .
       
self::jsLoad('js/page-tabs.js');
    }

   
/**
     * Get HTML code to load Magnific popup JS
     *
     * @return     string
     */
   
public static function jsModal()
    {
        return
       
self::jsLoad('js/jquery/jquery.magnific-popup.js');
    }

   
/**
     * @deprecated since version 2.16
     *
     * @return     string
     */
   
public static function jsColorPicker()
    {
        return
'';
    }

   
/**
     * Get HTML code for date picker JS utility
     *
     * @deprecated since 2.21
     *
     * @return     string
     */
   
public static function jsDatePicker()
    {
        return
'';
    }

    public static function
jsToolBar()
    {
       
# Deprecated but we keep this for plugins.
   
}

   
/**
     * Get HTML to load Upload JS utility
     *
     * @param      array   $params    The parameters
     * @param      mixed   $base_url  The base url
     *
     * @return     string
     */
   
public static function jsUpload($params = [], $base_url = null)
    {
       
$core = self::getCore();

        if (!
$base_url) {
           
$base_url = path::clean(dirname(preg_replace('/(\?.*$)?/', '', $_SERVER['REQUEST_URI']))) . '/';
        }

       
$params = array_merge($params, [
           
'sess_id=' . session_id(),
           
'sess_uid=' . $_SESSION['sess_browser_uid'],
           
'xd_check=' . $core->getNonce(),
        ]);

       
$js_msg = [
           
'enhanced_uploader_activate' => __('Temporarily activate enhanced uploader'),
           
'enhanced_uploader_disable'  => __('Temporarily disable enhanced uploader'),
        ];
       
$js = [
           
'msg' => [
               
'limit_exceeded'             => __('Limit exceeded.'),
               
'size_limit_exceeded'        => __('File size exceeds allowed limit.'),
               
'canceled'                   => __('Canceled.'),
               
'http_error'                 => __('HTTP Error:'),
               
'error'                      => __('Error:'),
               
'choose_file'                => __('Choose file'),
               
'choose_files'               => __('Choose files'),
               
'cancel'                     => __('Cancel'),
               
'clean'                      => __('Clean'),
               
'upload'                     => __('Upload'),
               
'send'                       => __('Send'),
               
'file_successfully_uploaded' => __('File successfully uploaded.'),
               
'no_file_in_queue'           => __('No file in queue.'),
               
'file_in_queue'              => __('1 file in queue.'),
               
'files_in_queue'             => __('%d files in queue.'),
               
'queue_error'                => __('Queue error:'),
            ],
           
'base_url' => $base_url,
        ];

        return
       
self::jsJson('file_upload', $js) .
       
self::jsJson('file_upload_msg', $js_msg) .
       
self::jsLoad('js/file-upload.js') .
       
self::jsLoad('js/jquery/jquery-ui.custom.js') .
       
self::jsLoad('js/jsUpload/tmpl.js') .
       
self::jsLoad('js/jsUpload/template-upload.js') .
       
self::jsLoad('js/jsUpload/template-download.js') .
       
self::jsLoad('js/jsUpload/load-image.js') .
       
self::jsLoad('js/jsUpload/jquery.iframe-transport.js') .
       
self::jsLoad('js/jsUpload/jquery.fileupload.js') .
       
self::jsLoad('js/jsUpload/jquery.fileupload-process.js') .
       
self::jsLoad('js/jsUpload/jquery.fileupload-resize.js') .
       
self::jsLoad('js/jsUpload/jquery.fileupload-ui.js');
    }

   
/**
     * Get HTML code to load meta editor
     *
     * @return     string
     */
   
public static function jsMetaEditor()
    {
        return
self::jsLoad('js/meta-editor.js');
    }

   
/**
     * Get HTML code for filters control JS utility
     *
     * @param      bool    $show   Show filters?
     *
     * @return     string
     */
   
public static function jsFilterControl($show = true)
    {
       
$core = self::getCore();
       
$js   = [
           
'show_filters'      => (bool) $show,
           
'filter_posts_list' => __('Show filters and display options'),
           
'cancel_the_filter' => __('Cancel filters and display options'),
        ];

        return
       
self::jsJson('filter_controls', $js) .
       
self::jsJson('filter_options', ['auto_filter' => $core->auth->user_prefs->interface->auto_filter]) .
       
self::jsLoad('js/filter-controls.js');
    }

   
/**
     * Get HTML code to load Codemirror
     *
     * @param      string  $theme  The theme
     * @param      bool    $multi  Is multiplex?
     * @param      array   $modes  The modes
     *
     * @return     string
     */
   
public static function jsLoadCodeMirror($theme = '', $multi = true, $modes = ['css', 'htmlmixed', 'javascript', 'php', 'xml', 'clike'])
    {
       
$ret = self::cssLoad('js/codemirror/lib/codemirror.css') .
       
self::jsLoad('js/codemirror/lib/codemirror.js');
        if (
$multi) {
           
$ret .= self::jsLoad('js/codemirror/addon/mode/multiplex.js');
        }
        foreach (
$modes as $mode) {
           
$ret .= self::jsLoad('js/codemirror/mode/' . $mode . '/' . $mode . '.js');
        }
       
$ret .= self::jsLoad('js/codemirror/addon/edit/closebrackets.js') .
       
self::jsLoad('js/codemirror/addon/edit/matchbrackets.js') .
       
self::cssLoad('js/codemirror/addon/display/fullscreen.css') .
       
self::jsLoad('js/codemirror/addon/display/fullscreen.js');
        if (
$theme != '' && $theme !== 'default') {
           
$ret .= self::cssLoad('js/codemirror/theme/' . $theme . '.css');
        }

        return
$ret;
    }

   
/**
     * Get HTML code to run Codemirror
     *
     * @param      mixed        $name   The HTML name attribute
     * @param      mixed        $id     The HTML id attribute
     * @param      mixed        $mode   The Codemirror mode
     * @param      string       $theme  The theme
     *
     * @return     string
     */
   
public static function jsRunCodeMirror($name, $id = null, $mode = null, $theme = '')
    {
        if (
is_array($name)) {
           
$js = $name;
        } else {
           
$js = [[
               
'name'  => $name,
               
'id'    => $id,
               
'mode'  => $mode,
               
'theme' => $theme ?: 'default',
            ]];
        }

       
$ret = self::jsJson('codemirror', $js) .
       
self::jsLoad('js/codemirror.js');

        return
$ret;
    }

   
/**
     * Gets the codemirror themes list.
     *
     * @return     array  The code mirror themes.
     */
   
public static function getCodeMirrorThemes()
    {
       
$themes      = [];
       
$themes_root = __DIR__ . '/../../admin' . '/js/codemirror/theme/';
        if (
is_dir($themes_root) && is_readable($themes_root)) {
            if ((
$d = @dir($themes_root)) !== false) {
                while ((
$entry = $d->read()) !== false) {
                    if (
$entry != '.' && $entry != '..' && substr($entry, 0, 1) != '.' && is_readable($themes_root . '/' . $entry)) {
                       
$themes[] = substr($entry, 0, -4); // remove .css extension
                   
}
                }
               
sort($themes);
            }
        }

        return
$themes;
    }

   
/**
     * Gets plugin file.
     *
     * @param      string  $file   The filename
     *
     * @return     string  The URL.
     */
   
public static function getPF($file)
    {
       
$core = self::getCore();

        return
$core->adminurl->get('load.plugin.file', ['pf' => $file]);
    }

   
/**
     * Gets var file.
     *
     * @param      string  $file   The filename
     *
     * @return     string  The URL.
     */
   
public static function getVF($file)
    {
       
$core = self::getCore();

        return
$core->adminurl->get('load.var.file', ['vf' => $file]);
    }

   
/**
     * Sets the x frame options.
     *
     * @param      array|ArrayObject    $headers  The headers
     * @param      mixed                $origin   The origin
     */
   
public static function setXFrameOptions($headers, $origin = null)
    {
        if (
self::$xframe_loaded) {
            return;
        }

        if (
$origin !== null) {
           
$url                        = parse_url($origin);
           
$headers['x-frame-options'] = sprintf('X-Frame-Options: %s', is_array($url) && isset($url['host']) ?
                (
'ALLOW-FROM ' . (isset($url['scheme']) ? $url['scheme'] . ':' : '') . '//' . $url['host']) :
               
'SAMEORIGIN');
        } else {
           
$headers['x-frame-options'] = 'X-Frame-Options: SAMEORIGIN'; // FF 3.6.9+ Chrome 4.1+ IE 8+ Safari 4+ Opera 10.5+
       
}
       
self::$xframe_loaded = true;
    }
}