[BUGFIX] Fix several typos in php comments
[Packages/TYPO3.CMS.git] / typo3 / sysext / core / Classes / Page / PageRenderer.php
index 181d050..249344c 100644 (file)
@@ -16,28 +16,32 @@ namespace TYPO3\CMS\Core\Page;
 
 use TYPO3\CMS\Backend\Routing\Router;
 use TYPO3\CMS\Backend\Routing\UriBuilder;
 
 use TYPO3\CMS\Backend\Routing\Router;
 use TYPO3\CMS\Backend\Routing\UriBuilder;
-use TYPO3\CMS\Backend\Utility\BackendUtility;
+use TYPO3\CMS\Core\Cache\CacheManager;
+use TYPO3\CMS\Core\Cache\Frontend\FrontendInterface;
+use TYPO3\CMS\Core\Core\Environment;
+use TYPO3\CMS\Core\Localization\Locales;
 use TYPO3\CMS\Core\Localization\LocalizationFactory;
 use TYPO3\CMS\Core\Localization\LocalizationFactory;
+use TYPO3\CMS\Core\MetaTag\MetaTagManagerRegistry;
+use TYPO3\CMS\Core\Package\PackageManager;
+use TYPO3\CMS\Core\Resource\ResourceCompressor;
 use TYPO3\CMS\Core\Service\MarkerBasedTemplateService;
 use TYPO3\CMS\Core\Service\MarkerBasedTemplateService;
+use TYPO3\CMS\Core\SingletonInterface;
 use TYPO3\CMS\Core\Utility\GeneralUtility;
 use TYPO3\CMS\Core\Utility\GeneralUtility;
-use TYPO3\CMS\Frontend\Controller\TypoScriptFrontendController;
+use TYPO3\CMS\Core\Utility\PathUtility;
 
 /**
 
 /**
- * TYPO3 pageRender class (new in TYPO3 4.3.0)
+ * TYPO3 pageRender class
  * This class render the HTML of a webpage, usable for BE and FE
  */
  * This class render the HTML of a webpage, usable for BE and FE
  */
-class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
+class PageRenderer implements SingletonInterface
 {
     // Constants for the part to be rendered
     const PART_COMPLETE = 0;
     const PART_HEADER = 1;
     const PART_FOOTER = 2;
 {
     // Constants for the part to be rendered
     const PART_COMPLETE = 0;
     const PART_HEADER = 1;
     const PART_FOOTER = 2;
-    // jQuery Core version that is shipped with TYPO3
-    const JQUERY_VERSION_LATEST = '2.1.4';
-    // jQuery namespace options
-    const JQUERY_NAMESPACE_NONE = 'none';
-    const JQUERY_NAMESPACE_DEFAULT = 'jQuery';
-    const JQUERY_NAMESPACE_DEFAULT_NOCONFLICT = 'defaultNoConflict';
+
+    const REQUIREJS_SCOPE_CONFIG = 'config';
+    const REQUIREJS_SCOPE_RESOLVE = 'resolve';
 
     /**
      * @var bool
 
     /**
      * @var bool
@@ -57,11 +61,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * @var bool
      */
     /**
      * @var bool
      */
-    protected $concatenateFiles = false;
-
-    /**
-     * @var bool
-     */
     protected $concatenateJavascript = false;
 
     /**
     protected $concatenateJavascript = false;
 
     /**
@@ -75,12 +74,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     protected $moveJsFromHeaderToFooter = false;
 
     /**
     protected $moveJsFromHeaderToFooter = false;
 
     /**
-     * @var \TYPO3\CMS\Core\Charset\CharsetConverter
-     */
-    protected $csConvObj;
-
-    /**
-     * @var \TYPO3\CMS\Core\Localization\Locales
+     * @var Locales
      */
     protected $locales;
 
      */
     protected $locales;
 
@@ -98,10 +92,10 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      *
      * @var array
      */
      *
      * @var array
      */
-    protected $languageDependencies = array();
+    protected $languageDependencies = [];
 
     /**
 
     /**
-     * @var \TYPO3\CMS\Core\Resource\ResourceCompressor
+     * @var ResourceCompressor
      */
     protected $compressor;
 
      */
     protected $compressor;
 
@@ -109,32 +103,32 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * @var array
      */
     /**
      * @var array
      */
-    protected $jsFiles = array();
+    protected $jsFiles = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $jsFooterFiles = array();
+    protected $jsFooterFiles = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $jsLibs = array();
+    protected $jsLibs = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $jsFooterLibs = array();
+    protected $jsFooterLibs = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $cssFiles = array();
+    protected $cssFiles = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $cssLibs = array();
+    protected $cssLibs = [];
 
     /**
      * The title of the page
 
     /**
      * The title of the page
@@ -174,22 +168,22 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * @var array
      */
     /**
      * @var array
      */
-    protected $metaTags = array();
+    protected $metaTags = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $inlineComments = array();
+    protected $inlineComments = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $headerData = array();
+    protected $headerData = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $footerData = array();
+    protected $footerData = [];
 
     /**
      * @var string
 
     /**
      * @var string
@@ -230,22 +224,17 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * @var array
      */
     /**
      * @var array
      */
-    protected $jsInline = array();
-
-    /**
-     * @var array
-     */
-    protected $jsFooterInline = array();
+    protected $jsInline = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $extOnReadyCode = array();
+    protected $jsFooterInline = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $cssInline = array();
+    protected $cssInline = [];
 
     /**
      * @var string
 
     /**
      * @var string
@@ -257,133 +246,63 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      */
     protected $templateFile;
 
      */
     protected $templateFile;
 
-    /**
-     * @var array
-     */
-    protected $jsLibraryNames = array('extjs');
-
-    // Paths to contibuted libraries
+    // Paths to contributed libraries
 
     /**
      * default path to the requireJS library, relative to the typo3/ directory
      * @var string
      */
 
     /**
      * default path to the requireJS library, relative to the typo3/ directory
      * @var string
      */
-    protected $requireJsPath = 'sysext/core/Resources/Public/JavaScript/Contrib/';
-
-    /**
-     * @var string
-     */
-    protected $extJsPath = 'sysext/core/Resources/Public/JavaScript/Contrib/extjs/';
-
-    /**
-     * The local directory where one can find jQuery versions and plugins
-     *
-     * @var string
-     */
-    protected $jQueryPath = 'sysext/core/Resources/Public/JavaScript/Contrib/jquery/';
+    protected $requireJsPath = 'EXT:core/Resources/Public/JavaScript/Contrib/';
 
     // Internal flags for JS-libraries
     /**
 
     // Internal flags for JS-libraries
     /**
-     * This array holds all jQuery versions that should be included in the
-     * current page.
-     * Each version is described by "source", "version" and "namespace"
-     *
-     * The namespace of every particular version is the key
-     * of that array, because only one version per namespace can exist.
-     *
-     * The type "source" describes where the jQuery core should be included from
-     * currently, TYPO3 supports "local" (make use of jQuery path), "google",
-     * "jquery", "msn" and "cloudflare".
-     *
-     * Currently there are downsides to "local" which supports only the latest/shipped
-     * jQuery core out of the box.
-     *
-     * @var array
-     */
-    protected $jQueryVersions = array();
-
-    /**
-     * Array of jQuery version numbers shipped with the core
-     *
-     * @var array
-     */
-    protected $availableLocalJqueryVersions = array(
-        self::JQUERY_VERSION_LATEST
-    );
-
-    /**
-     * Array of jQuery CDNs with placeholders
-     *
-     * @var array
-     */
-    protected $jQueryCdnUrls = array(
-        'google' => 'https://ajax.googleapis.com/ajax/libs/jquery/%1$s/jquery%2$s.js',
-        'msn' => 'https://ajax.aspnetcdn.com/ajax/jQuery/jquery-%1$s%2$s.js',
-        'jquery' => 'https://code.jquery.com/jquery-%1$s%2$s.js',
-        'cloudflare' => 'https://cdnjs.cloudflare.com/ajax/libs/jquery/%1$s/jquery%2$s.js'
-    );
-
-    /**
      * if set, the requireJS library is included
      * @var bool
      */
     protected $addRequireJs = false;
 
     /**
      * if set, the requireJS library is included
      * @var bool
      */
     protected $addRequireJs = false;
 
     /**
-     * inline configuration for requireJS
+     * Inline configuration for requireJS (internal)
      * @var array
      */
      * @var array
      */
-    protected $requireJsConfig = array();
-
-    /**
-     * @var bool
-     */
-    protected $addExtJS = false;
-
-    /**
-     * @var bool
-     */
-    protected $extDirectCodeAdded = false;
+    protected $requireJsConfig = [];
 
     /**
 
     /**
-     * @var bool
-     */
-    protected $enableExtJsDebug = false;
-
-    /**
-     * @var bool
+     * Module names of internal requireJS 'paths'
+     * @var array
      */
      */
-    protected $enableJqueryDebug = false;
+    protected $internalRequireJsPathModuleNames = [];
 
     /**
 
     /**
-     * @var bool
+     * Inline configuration for requireJS (public)
+     * @var array
      */
      */
-    protected $extJStheme = true;
+    protected $publicRequireJsConfig = [];
 
     /**
 
     /**
-     * @var bool
+     * @var array
      */
      */
-    protected $extJScss = true;
+    protected $inlineLanguageLabels = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $inlineLanguageLabels = array();
+    protected $inlineLanguageLabelFiles = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $inlineLanguageLabelFiles = array();
+    protected $inlineSettings = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $inlineSettings = array();
+    protected $inlineJavascriptWrap = [];
 
     /**
      * @var array
      */
 
     /**
      * @var array
      */
-    protected $inlineJavascriptWrap = array();
+    protected $inlineCssWrap = [];
 
     /**
      * Saves error messages generated during compression
 
     /**
      * Saves error messages generated during compression
@@ -400,56 +319,73 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     protected $endingSlash = '';
 
     /**
     protected $endingSlash = '';
 
     /**
-     * Used by BE modules
-     *
-     * @var null|string
+     * @var MetaTagManagerRegistry
      */
      */
-    public $backPath;
+    protected $metaTagRegistry;
+
+    /**
+     * @var FrontendInterface
+     */
+    protected static $cache = null;
 
     /**
      * @param string $templateFile Declare the used template file. Omit this parameter will use default template
 
     /**
      * @param string $templateFile Declare the used template file. Omit this parameter will use default template
-     * @param string $backPath Relative path to typo3-folder. It varies for BE modules, in FE it will be typo3/
      */
      */
-    public function __construct($templateFile = '', $backPath = null)
+    public function __construct($templateFile = '')
     {
         $this->reset();
     {
         $this->reset();
-        $this->csConvObj = GeneralUtility::makeInstance(\TYPO3\CMS\Core\Charset\CharsetConverter::class);
-        $this->locales = GeneralUtility::makeInstance(\TYPO3\CMS\Core\Localization\Locales::class);
+        $this->locales = GeneralUtility::makeInstance(Locales::class);
         if ($templateFile !== '') {
             $this->templateFile = $templateFile;
         }
         if ($templateFile !== '') {
             $this->templateFile = $templateFile;
         }
-        $this->backPath = isset($backPath) ? $backPath : $GLOBALS['BACK_PATH'];
-        $this->inlineJavascriptWrap = array(
+        $this->inlineJavascriptWrap = [
             '<script type="text/javascript">' . LF . '/*<![CDATA[*/' . LF,
             '/*]]>*/' . LF . '</script>' . LF
             '<script type="text/javascript">' . LF . '/*<![CDATA[*/' . LF,
             '/*]]>*/' . LF . '</script>' . LF
-        );
-        $this->inlineCssWrap = array(
+        ];
+        $this->inlineCssWrap = [
             '<style type="text/css">' . LF . '/*<![CDATA[*/' . LF . '<!-- ' . LF,
             '-->' . LF . '/*]]>*/' . LF . '</style>' . LF
             '<style type="text/css">' . LF . '/*<![CDATA[*/' . LF . '<!-- ' . LF,
             '-->' . LF . '/*]]>*/' . LF . '</style>' . LF
-        );
+        ];
+
+        $this->metaTagRegistry = GeneralUtility::makeInstance(MetaTagManagerRegistry::class);
+        $this->setMetaTag('name', 'generator', 'TYPO3 CMS');
+    }
+
+    /**
+     * Set restored meta tag managers as singletons
+     * so that uncached plugins can use them to add or remove meta tags
+     */
+    public function __wakeup()
+    {
+        GeneralUtility::setSingletonInstance(get_class($this->metaTagRegistry), $this->metaTagRegistry);
+    }
+
+    /**
+     * @param FrontendInterface $cache
+     */
+    public static function setCache(FrontendInterface $cache)
+    {
+        static::$cache = $cache;
     }
 
     /**
      * Reset all vars to initial values
     }
 
     /**
      * Reset all vars to initial values
-     *
-     * @return void
      */
     protected function reset()
     {
         $this->templateFile = 'EXT:core/Resources/Private/Templates/PageRenderer.html';
      */
     protected function reset()
     {
         $this->templateFile = 'EXT:core/Resources/Private/Templates/PageRenderer.html';
-        $this->jsFiles = array();
-        $this->jsFooterFiles = array();
-        $this->jsInline = array();
-        $this->jsFooterInline = array();
-        $this->jsLibs = array();
-        $this->cssFiles = array();
-        $this->cssInline = array();
-        $this->metaTags = array();
-        $this->inlineComments = array();
-        $this->headerData = array();
-        $this->footerData = array();
-        $this->extOnReadyCode = array();
-        $this->jQueryVersions = array();
+        $this->jsFiles = [];
+        $this->jsFooterFiles = [];
+        $this->jsInline = [];
+        $this->jsFooterInline = [];
+        $this->jsLibs = [];
+        $this->cssFiles = [];
+        $this->cssInline = [];
+        $this->metaTags = [];
+        $this->metaTagsByAPI = [];
+        $this->inlineComments = [];
+        $this->headerData = [];
+        $this->footerData = [];
     }
 
     /*****************************************************/
     }
 
     /*****************************************************/
@@ -462,7 +398,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets the title
      *
      * @param string $title    title of webpage
      * Sets the title
      *
      * @param string $title    title of webpage
-     * @return void
      */
     public function setTitle($title)
     {
      */
     public function setTitle($title)
     {
@@ -473,7 +408,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Enables/disables rendering of XHTML code
      *
      * @param bool $enable Enable XHTML
      * Enables/disables rendering of XHTML code
      *
      * @param bool $enable Enable XHTML
-     * @return void
      */
     public function setRenderXhtml($enable)
     {
      */
     public function setRenderXhtml($enable)
     {
@@ -484,7 +418,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets xml prolog and docType
      *
      * @param string $xmlPrologAndDocType Complete tags for xml prolog and docType
      * Sets xml prolog and docType
      *
      * @param string $xmlPrologAndDocType Complete tags for xml prolog and docType
-     * @return void
      */
     public function setXmlPrologAndDocType($xmlPrologAndDocType)
     {
      */
     public function setXmlPrologAndDocType($xmlPrologAndDocType)
     {
@@ -495,7 +428,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets meta charset
      *
      * @param string $charSet Used charset
      * Sets meta charset
      *
      * @param string $charSet Used charset
-     * @return void
      */
     public function setCharSet($charSet)
     {
      */
     public function setCharSet($charSet)
     {
@@ -506,12 +438,11 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets language
      *
      * @param string $lang Used language
      * Sets language
      *
      * @param string $lang Used language
-     * @return void
      */
     public function setLanguage($lang)
     {
         $this->lang = $lang;
      */
     public function setLanguage($lang)
     {
         $this->lang = $lang;
-        $this->languageDependencies = array();
+        $this->languageDependencies = [];
 
         // Language is found. Configure it:
         if (in_array($this->lang, $this->locales->getLocales())) {
 
         // Language is found. Configure it:
         if (in_array($this->lang, $this->locales->getLocales())) {
@@ -526,7 +457,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Set the meta charset tag
      *
      * @param string $metaCharsetTag
      * Set the meta charset tag
      *
      * @param string $metaCharsetTag
-     * @return void
      */
     public function setMetaCharsetTag($metaCharsetTag)
     {
      */
     public function setMetaCharsetTag($metaCharsetTag)
     {
@@ -537,7 +467,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets html tag
      *
      * @param string $htmlTag Html tag
      * Sets html tag
      *
      * @param string $htmlTag Html tag
-     * @return void
      */
     public function setHtmlTag($htmlTag)
     {
      */
     public function setHtmlTag($htmlTag)
     {
@@ -548,7 +477,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets HTML head tag
      *
      * @param string $headTag HTML head tag
      * Sets HTML head tag
      *
      * @param string $headTag HTML head tag
-     * @return void
      */
     public function setHeadTag($headTag)
     {
      */
     public function setHeadTag($headTag)
     {
@@ -559,7 +487,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets favicon
      *
      * @param string $favIcon
      * Sets favicon
      *
      * @param string $favIcon
-     * @return void
      */
     public function setFavIcon($favIcon)
     {
      */
     public function setFavIcon($favIcon)
     {
@@ -570,7 +497,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets icon mime type
      *
      * @param string $iconMimeType
      * Sets icon mime type
      *
      * @param string $iconMimeType
-     * @return void
      */
     public function setIconMimeType($iconMimeType)
     {
      */
     public function setIconMimeType($iconMimeType)
     {
@@ -581,7 +507,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets HTML base URL
      *
      * @param string $baseUrl HTML base URL
      * Sets HTML base URL
      *
      * @param string $baseUrl HTML base URL
-     * @return void
      */
     public function setBaseUrl($baseUrl)
     {
      */
     public function setBaseUrl($baseUrl)
     {
@@ -592,7 +517,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets template file
      *
      * @param string $file
      * Sets template file
      *
      * @param string $file
-     * @return void
      */
     public function setTemplateFile($file)
     {
      */
     public function setTemplateFile($file)
     {
@@ -600,21 +524,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
-     * Sets back path
-     *
-     * @param string $backPath
-     * @return void
-     */
-    public function setBackPath($backPath)
-    {
-        $this->backPath = $backPath;
-    }
-
-    /**
      * Sets Content for Body
      *
      * @param string $content
      * Sets Content for Body
      *
      * @param string $content
-     * @return void
      */
     public function setBodyContent($content)
     {
      */
     public function setBodyContent($content)
     {
@@ -625,7 +537,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Sets path to requireJS library (relative to typo3 directory)
      *
      * @param string $path Path to requireJS library
      * Sets path to requireJS library (relative to typo3 directory)
      *
      * @param string $path Path to requireJS library
-     * @return void
      */
     public function setRequireJsPath($path)
     {
      */
     public function setRequireJsPath($path)
     {
@@ -633,14 +544,31 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
-     * Sets Path for ExtJs library (relative to typo3 directory)
-     *
-     * @param string $path
-     * @return void
+     * @param string $scope
+     * @return array
      */
      */
-    public function setExtJsPath($path)
-    {
-        $this->extJsPath = $path;
+    public function getRequireJsConfig(string $scope = null): array
+    {
+        // return basic RequireJS configuration without shim, paths and packages
+        if ($scope === static::REQUIREJS_SCOPE_CONFIG) {
+            return array_replace_recursive(
+                $this->publicRequireJsConfig,
+                $this->filterArrayKeys(
+                    $this->requireJsConfig,
+                    ['shim', 'paths', 'packages'],
+                    false
+                )
+            );
+        }
+        // return RequireJS configuration for resolving only shim, paths and packages
+        if ($scope === static::REQUIREJS_SCOPE_RESOLVE) {
+            return $this->filterArrayKeys(
+                $this->requireJsConfig,
+                ['shim', 'paths', 'packages'],
+                true
+            );
+        }
+        return [];
     }
 
     /*****************************************************/
     }
 
     /*****************************************************/
@@ -651,8 +579,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /*****************************************************/
     /**
      * Enables MoveJsFromHeaderToFooter
     /*****************************************************/
     /**
      * Enables MoveJsFromHeaderToFooter
-     *
-     * @return void
      */
     public function enableMoveJsFromHeaderToFooter()
     {
      */
     public function enableMoveJsFromHeaderToFooter()
     {
@@ -661,8 +587,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Disables MoveJsFromHeaderToFooter
 
     /**
      * Disables MoveJsFromHeaderToFooter
-     *
-     * @return void
      */
     public function disableMoveJsFromHeaderToFooter()
     {
      */
     public function disableMoveJsFromHeaderToFooter()
     {
@@ -671,8 +595,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Enables compression of javascript
 
     /**
      * Enables compression of javascript
-     *
-     * @return void
      */
     public function enableCompressJavascript()
     {
      */
     public function enableCompressJavascript()
     {
@@ -681,8 +603,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Disables compression of javascript
 
     /**
      * Disables compression of javascript
-     *
-     * @return void
      */
     public function disableCompressJavascript()
     {
      */
     public function disableCompressJavascript()
     {
@@ -691,8 +611,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Enables compression of css
 
     /**
      * Enables compression of css
-     *
-     * @return void
      */
     public function enableCompressCss()
     {
      */
     public function enableCompressCss()
     {
@@ -701,8 +619,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Disables compression of css
 
     /**
      * Disables compression of css
-     *
-     * @return void
      */
     public function disableCompressCss()
     {
      */
     public function disableCompressCss()
     {
@@ -710,29 +626,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
-     * Enables concatenation of js and css files
-     *
-     * @return void
-     */
-    public function enableConcatenateFiles()
-    {
-        $this->concatenateFiles = true;
-    }
-
-    /**
-     * Disables concatenation of js and css files
-     *
-     * @return void
-     */
-    public function disableConcatenateFiles()
-    {
-        $this->concatenateFiles = false;
-    }
-
-    /**
      * Enables concatenation of js files
      * Enables concatenation of js files
-     *
-     * @return void
      */
     public function enableConcatenateJavascript()
     {
      */
     public function enableConcatenateJavascript()
     {
@@ -741,8 +635,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Disables concatenation of js files
 
     /**
      * Disables concatenation of js files
-     *
-     * @return void
      */
     public function disableConcatenateJavascript()
     {
      */
     public function disableConcatenateJavascript()
     {
@@ -751,8 +643,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Enables concatenation of css files
 
     /**
      * Enables concatenation of css files
-     *
-     * @return void
      */
     public function enableConcatenateCss()
     {
      */
     public function enableConcatenateCss()
     {
@@ -761,8 +651,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Disables concatenation of css files
 
     /**
      * Disables concatenation of css files
-     *
-     * @return void
      */
     public function disableConcatenateCss()
     {
      */
     public function disableConcatenateCss()
     {
@@ -771,8 +659,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Sets removal of all line breaks in template
 
     /**
      * Sets removal of all line breaks in template
-     *
-     * @return void
      */
     public function enableRemoveLineBreaksFromTemplate()
     {
      */
     public function enableRemoveLineBreaksFromTemplate()
     {
@@ -781,8 +667,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Unsets removal of all line breaks in template
 
     /**
      * Unsets removal of all line breaks in template
-     *
-     * @return void
      */
     public function disableRemoveLineBreaksFromTemplate()
     {
      */
     public function disableRemoveLineBreaksFromTemplate()
     {
@@ -792,17 +676,14 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * Enables Debug Mode
      * This is a shortcut to switch off all compress/concatenate features to enable easier debug
     /**
      * Enables Debug Mode
      * This is a shortcut to switch off all compress/concatenate features to enable easier debug
-     *
-     * @return void
      */
     public function enableDebugMode()
     {
         $this->compressJavascript = false;
         $this->compressCss = false;
      */
     public function enableDebugMode()
     {
         $this->compressJavascript = false;
         $this->compressCss = false;
-        $this->concatenateFiles = false;
+        $this->concatenateCss = false;
+        $this->concatenateJavascript = false;
         $this->removeLineBreaksFromTemplate = false;
         $this->removeLineBreaksFromTemplate = false;
-        $this->enableExtJsDebug = true;
-        $this->enableJqueryDebug = true;
     }
 
     /*****************************************************/
     }
 
     /*****************************************************/
@@ -952,16 +833,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
-     * Gets concatenate of js and css files
-     *
-     * @return bool
-     */
-    public function getConcatenateFiles()
-    {
-        return $this->concatenateFiles;
-    }
-
-    /**
      * Gets concatenate of js files
      *
      * @return bool
      * Gets concatenate of js files
      *
      * @return bool
@@ -1002,16 +873,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
-     * Gets Path for ExtJs library (relative to typo3 directory)
-     *
-     * @return string
-     */
-    public function getExtJsPath()
-    {
-        return $this->extJsPath;
-    }
-
-    /**
      * Gets the inline language labels.
      *
      * @return array The inline language labels
      * Gets the inline language labels.
      *
      * @return array The inline language labels
@@ -1037,24 +898,79 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /*                                                   */
     /*                                                   */
     /*****************************************************/
     /*                                                   */
     /*                                                   */
     /*****************************************************/
+
     /**
     /**
-     * Adds meta data
+     * Sets a given meta tag
      *
      *
-     * @param string $meta Meta data (complete metatag)
-     * @return void
+     * @param string $type The type of the meta tag. Allowed values are property, name or http-equiv
+     * @param string $name The name of the property to add
+     * @param string $content The content of the meta tag
+     * @param array $subProperties Subproperties of the meta tag (like e.g. og:image:width)
+     * @param bool $replace Replace earlier set meta tag
+     * @throws \InvalidArgumentException
      */
      */
-    public function addMetaTag($meta)
+    public function setMetaTag(string $type, string $name, string $content, array $subProperties = [], $replace = true)
     {
     {
-        if (!in_array($meta, $this->metaTags)) {
-            $this->metaTags[] = $meta;
+        // Lowercase all the things
+        $type = strtolower($type);
+        $name = strtolower($name);
+        if (!in_array($type, ['property', 'name', 'http-equiv'], true)) {
+            throw new \InvalidArgumentException(
+                'When setting a meta tag the only types allowed are property, name or http-equiv. "' . $type . '" given.',
+                1496402460
+            );
         }
         }
+        $manager = $this->metaTagRegistry->getManagerForProperty($name);
+        $manager->addProperty($name, $content, $subProperties, $replace, $type);
+    }
+
+    /**
+     * Returns the requested meta tag
+     *
+     * @param string $type
+     * @param string $name
+     *
+     * @return array
+     */
+    public function getMetaTag(string $type, string $name): array
+    {
+        // Lowercase all the things
+        $type = strtolower($type);
+        $name = strtolower($name);
+
+        $manager = $this->metaTagRegistry->getManagerForProperty($name);
+        $propertyContent = $manager->getProperty($name, $type);
+
+        if (!empty($propertyContent[0])) {
+            return [
+                'type' => $type,
+                'name' => $name,
+                'content' => $propertyContent[0]['content']
+            ];
+        }
+        return [];
+    }
+
+    /**
+     * Unset the requested meta tag
+     *
+     * @param string $type
+     * @param string $name
+     */
+    public function removeMetaTag(string $type, string $name)
+    {
+        // Lowercase all the things
+        $type = strtolower($type);
+        $name = strtolower($name);
+
+        $manager = $this->metaTagRegistry->getManagerForProperty($name);
+        $manager->removeProperty($name, $type);
     }
 
     /**
      * Adds inline HTML comment
      *
      * @param string $comment
     }
 
     /**
      * Adds inline HTML comment
      *
      * @param string $comment
-     * @return void
      */
     public function addInlineComment($comment)
     {
      */
     public function addInlineComment($comment)
     {
@@ -1067,7 +983,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Adds header data
      *
      * @param string $data Free header data for HTML header
      * Adds header data
      *
      * @param string $data Free header data for HTML header
-     * @return void
      */
     public function addHeaderData($data)
     {
      */
     public function addHeaderData($data)
     {
@@ -1080,7 +995,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Adds footer data
      *
      * @param string $data Free header data for HTML header
      * Adds footer data
      *
      * @param string $data Free header data for HTML header
-     * @return void
      */
     public function addFooterData($data)
     {
      */
     public function addFooterData($data)
     {
@@ -1102,15 +1016,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
-     * @return void
+     * @param bool $defer Flag if property 'defer="defer"' should be added to JavaScript tags
+     * @param string $crossorigin CORS settings attribute
      */
      */
-    public function addJsLibrary($name, $file, $type = 'text/javascript', $compress = false, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '')
+    public function addJsLibrary($name, $file, $type = '', $compress = false, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '', $defer = false, $crossorigin = '')
     {
     {
-        if (!$type) {
-            $type = 'text/javascript';
-        }
         if (!in_array(strtolower($name), $this->jsLibs)) {
         if (!in_array(strtolower($name), $this->jsLibs)) {
-            $this->jsLibs[strtolower($name)] = array(
+            $this->jsLibs[strtolower($name)] = [
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_HEADER,
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_HEADER,
@@ -1121,7 +1033,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
-            );
+                'defer' => $defer,
+                'crossorigin' => $crossorigin,
+            ];
         }
     }
 
         }
     }
 
@@ -1138,15 +1052,14 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
-     * @return void
+     * @param bool $defer Flag if property 'defer="defer"' should be added to JavaScript tags
+     * @param string $crossorigin CORS settings attribute
      */
      */
-    public function addJsFooterLibrary($name, $file, $type = 'text/javascript', $compress = false, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '')
+    public function addJsFooterLibrary($name, $file, $type = '', $compress = false, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '', $defer = false, $crossorigin = '')
     {
     {
-        if (!$type) {
-            $type = 'text/javascript';
-        }
+        $name .= '_jsFooterLibrary';
         if (!in_array(strtolower($name), $this->jsLibs)) {
         if (!in_array(strtolower($name), $this->jsLibs)) {
-            $this->jsLibs[strtolower($name)] = array(
+            $this->jsLibs[strtolower($name)] = [
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_FOOTER,
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_FOOTER,
@@ -1157,7 +1070,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
-            );
+                'defer' => $defer,
+                'crossorigin' => $crossorigin,
+            ];
         }
     }
 
         }
     }
 
@@ -1173,15 +1088,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
-     * @return void
+     * @param bool $defer Flag if property 'defer="defer"' should be added to JavaScript tags
+     * @param string $crossorigin CORS settings attribute
      */
      */
-    public function addJsFile($file, $type = 'text/javascript', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '')
+    public function addJsFile($file, $type = '', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '', $defer = false, $crossorigin = '')
     {
     {
-        if (!$type) {
-            $type = 'text/javascript';
-        }
         if (!isset($this->jsFiles[$file])) {
         if (!isset($this->jsFiles[$file])) {
-            $this->jsFiles[$file] = array(
+            $this->jsFiles[$file] = [
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_HEADER,
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_HEADER,
@@ -1192,7 +1105,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
-            );
+                'defer' => $defer,
+                'crossorigin' => $crossorigin,
+            ];
         }
     }
 
         }
     }
 
@@ -1208,15 +1123,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param bool $async Flag if property 'async="async"' should be added to JavaScript tags
      * @param string $integrity Subresource Integrity (SRI)
-     * @return void
+     * @param bool $defer Flag if property 'defer="defer"' should be added to JavaScript tags
+     * @param string $crossorigin CORS settings attribute
      */
      */
-    public function addJsFooterFile($file, $type = 'text/javascript', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '')
+    public function addJsFooterFile($file, $type = '', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $async = false, $integrity = '', $defer = false, $crossorigin = '')
     {
     {
-        if (!$type) {
-            $type = 'text/javascript';
-        }
         if (!isset($this->jsFiles[$file])) {
         if (!isset($this->jsFiles[$file])) {
-            $this->jsFiles[$file] = array(
+            $this->jsFiles[$file] = [
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_FOOTER,
                 'file' => $file,
                 'type' => $type,
                 'section' => self::PART_FOOTER,
@@ -1227,7 +1140,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
                 'splitChar' => $splitChar,
                 'async' => $async,
                 'integrity' => $integrity,
-            );
+                'defer' => $defer,
+                'crossorigin' => $crossorigin,
+            ];
         }
     }
 
         }
     }
 
@@ -1238,17 +1153,16 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $block
      * @param bool $compress
      * @param bool $forceOnTop
      * @param string $block
      * @param bool $compress
      * @param bool $forceOnTop
-     * @return void
      */
     public function addJsInlineCode($name, $block, $compress = true, $forceOnTop = false)
     {
         if (!isset($this->jsInline[$name]) && !empty($block)) {
      */
     public function addJsInlineCode($name, $block, $compress = true, $forceOnTop = false)
     {
         if (!isset($this->jsInline[$name]) && !empty($block)) {
-            $this->jsInline[$name] = array(
+            $this->jsInline[$name] = [
                 'code' => $block . LF,
                 'section' => self::PART_HEADER,
                 'compress' => $compress,
                 'forceOnTop' => $forceOnTop
                 'code' => $block . LF,
                 'section' => self::PART_HEADER,
                 'compress' => $compress,
                 'forceOnTop' => $forceOnTop
-            );
+            ];
         }
     }
 
         }
     }
 
@@ -1259,167 +1173,17 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $block
      * @param bool $compress
      * @param bool $forceOnTop
      * @param string $block
      * @param bool $compress
      * @param bool $forceOnTop
-     * @return void
      */
     public function addJsFooterInlineCode($name, $block, $compress = true, $forceOnTop = false)
     {
         if (!isset($this->jsInline[$name]) && !empty($block)) {
      */
     public function addJsFooterInlineCode($name, $block, $compress = true, $forceOnTop = false)
     {
         if (!isset($this->jsInline[$name]) && !empty($block)) {
-            $this->jsInline[$name] = array(
+            $this->jsInline[$name] = [
                 'code' => $block . LF,
                 'section' => self::PART_FOOTER,
                 'compress' => $compress,
                 'forceOnTop' => $forceOnTop
                 'code' => $block . LF,
                 'section' => self::PART_FOOTER,
                 'compress' => $compress,
                 'forceOnTop' => $forceOnTop
-            );
-        }
-    }
-
-    /**
-     * Adds Ext.onready code, which will be wrapped in Ext.onReady(function() {...});
-     *
-     * @param string $block Javascript code
-     * @param bool $forceOnTop Position of the javascript code (TRUE for putting it on top, default is FALSE = bottom)
-     * @return void
-     */
-    public function addExtOnReadyCode($block, $forceOnTop = false)
-    {
-        if (!in_array($block, $this->extOnReadyCode)) {
-            if ($forceOnTop) {
-                array_unshift($this->extOnReadyCode, $block);
-            } else {
-                $this->extOnReadyCode[] = $block;
-            }
-        }
-    }
-
-    /**
-     * Adds the ExtDirect code
-     *
-     * @param array $filterNamespaces Limit the output to defined namespaces. If empty, all namespaces are generated
-     * @return void
-     */
-    public function addExtDirectCode(array $filterNamespaces = array())
-    {
-        if ($this->extDirectCodeAdded) {
-            return;
-        }
-        $this->extDirectCodeAdded = true;
-        if (empty($filterNamespaces)) {
-            $filterNamespaces = array('TYPO3');
+            ];
         }
         }
-        // @deprecated since TYPO3 CMS 7, will be removed in TYPO3 CMS 8
-        // add compatibility mapping for the old flashmessage API
-        $this->addJsFile(GeneralUtility::resolveBackPath($this->backPath .
-            'sysext/backend/Resources/Public/JavaScript/flashmessage_compatibility.js'));
-
-        // Add language labels for ExtDirect
-        if (TYPO3_MODE === 'FE') {
-            $this->addInlineLanguageLabelArray(array(
-                'extDirect_timeoutHeader' => $this->getTypoScriptFrontendController()->sL('LLL:EXT:lang/locallang_misc.xlf:extDirect_timeoutHeader'),
-                'extDirect_timeoutMessage' => $this->getTypoScriptFrontendController()->sL('LLL:EXT:lang/locallang_misc.xlf:extDirect_timeoutMessage')
-            ));
-        } else {
-            $this->addInlineLanguageLabelArray(array(
-                'extDirect_timeoutHeader' => $this->getLanguageService()->sL('LLL:EXT:lang/locallang_misc.xlf:extDirect_timeoutHeader'),
-                'extDirect_timeoutMessage' => $this->getLanguageService()->sL('LLL:EXT:lang/locallang_misc.xlf:extDirect_timeoutMessage')
-            ));
-        }
-
-        $token = ($api = '');
-        if (TYPO3_MODE === 'BE') {
-            $formprotection = \TYPO3\CMS\Core\FormProtection\FormProtectionFactory::get();
-            $token = $formprotection->generateToken('extDirect');
-
-            // Debugger Console strings
-            $this->addInlineLanguageLabelFile('EXT:core/Resources/Private/Language/debugger.xlf');
-        }
-        /** @var $extDirect \TYPO3\CMS\Core\ExtDirect\ExtDirectApi */
-        $extDirect = GeneralUtility::makeInstance(\TYPO3\CMS\Core\ExtDirect\ExtDirectApi::class);
-        $api = $extDirect->getApiPhp($filterNamespaces);
-        if ($api) {
-            $this->addJsInlineCode('TYPO3ExtDirectAPI', $api, false);
-        }
-        // Note: we need to iterate thru the object, because the addProvider method
-        // does this only with multiple arguments
-        $this->addExtOnReadyCode('
-                       (function() {
-                               TYPO3.ExtDirectToken = "' . $token . '";
-                               for (var api in Ext.app.ExtDirectAPI) {
-                                       var provider = Ext.Direct.addProvider(Ext.app.ExtDirectAPI[api]);
-                                       provider.on("beforecall", function(provider, transaction, meta) {
-                                               if (transaction.data) {
-                                                       transaction.data[transaction.data.length] = TYPO3.ExtDirectToken;
-                                               } else {
-                                                       transaction.data = [TYPO3.ExtDirectToken];
-                                               }
-                                       });
-
-                                       provider.on("call", function(provider, transaction, meta) {
-                                               if (transaction.isForm) {
-                                                       transaction.params.securityToken = TYPO3.ExtDirectToken;
-                                               }
-                                       });
-                               }
-                       })();
-
-                       var extDirectDebug = function(message, header, group) {
-                               var TYPO3ViewportInstance = null;
-
-                               if (top && top.TYPO3 && typeof top.TYPO3.Backend === "object") {
-                                       TYPO3ViewportInstance = top.TYPO3.Backend;
-                               } else if (typeof TYPO3 === "object" && typeof TYPO3.Backend === "object") {
-                                       TYPO3ViewportInstance = TYPO3.Backend;
-                               }
-
-                               if (TYPO3ViewportInstance !== null) {
-                                       TYPO3ViewportInstance.DebugConsole.addTab(message, header, group);
-                               } else if (typeof console === "object") {
-                                       console.log(message);
-                               } else {
-                                       document.write(message);
-                               }
-                       };
-
-                       Ext.Direct.on("exception", function(event) {
-                               if (event.code === Ext.Direct.exceptions.TRANSPORT && !event.where) {
-                                       top.TYPO3.Notification.error(
-                                               TYPO3.l10n.localize("extDirect_timeoutHeader"),
-                                               TYPO3.l10n.localize("extDirect_timeoutMessage")
-                                       );
-                               } else {
-                                       var backtrace = "";
-                                       if (event.code === "parse") {
-                                               extDirectDebug(
-                                                       "<p>" + event.xhr.responseText + "<\\/p>",
-                                                       event.type,
-                                                       "ExtDirect - Exception"
-                                               );
-                                       } else if (event.code === "router") {
-                                               top.TYPO3.Notification.error(
-                                                       event.code,
-                                                       event.message
-                                               );
-                                       } else if (event.where) {
-                                               backtrace = "<p style=\\"margin-top: 20px;\\">" +
-                                                       "<strong>Backtrace:<\\/strong><br \\/>" +
-                                                       event.where.replace(/#/g, "<br \\/>#") +
-                                                       "<\\/p>";
-                                               extDirectDebug(
-                                                       "<p>" + event.message + "<\\/p>" + backtrace,
-                                                       event.method,
-                                                       "ExtDirect - Exception"
-                                               );
-                                       }
-
-
-                               }
-                       });
-
-                       Ext.Direct.on("event", function(event, provider) {
-                               if (typeof event.debug !== "undefined" && event.debug !== "") {
-                                       extDirectDebug(event.debug, event.method, "ExtDirect - Debug");
-                               }
-                       });
-                       ', true);
     }
 
     /**
     }
 
     /**
@@ -1434,12 +1198,12 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $allWrap
      * @param bool $excludeFromConcatenation
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param string $allWrap
      * @param bool $excludeFromConcatenation
      * @param string $splitChar The char used to split the allWrap value, default is "|"
-     * @return void
+     * @param bool $inline
      */
      */
-    public function addCssFile($file, $rel = 'stylesheet', $media = 'all', $title = '', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|')
+    public function addCssFile($file, $rel = 'stylesheet', $media = 'all', $title = '', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $inline = false)
     {
         if (!isset($this->cssFiles[$file])) {
     {
         if (!isset($this->cssFiles[$file])) {
-            $this->cssFiles[$file] = array(
+            $this->cssFiles[$file] = [
                 'file' => $file,
                 'rel' => $rel,
                 'media' => $media,
                 'file' => $file,
                 'rel' => $rel,
                 'media' => $media,
@@ -1448,8 +1212,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                 'forceOnTop' => $forceOnTop,
                 'allWrap' => $allWrap,
                 'excludeFromConcatenation' => $excludeFromConcatenation,
                 'forceOnTop' => $forceOnTop,
                 'allWrap' => $allWrap,
                 'excludeFromConcatenation' => $excludeFromConcatenation,
-                'splitChar' => $splitChar
-            );
+                'splitChar' => $splitChar,
+                'inline' => $inline
+            ];
         }
     }
 
         }
     }
 
@@ -1465,12 +1230,12 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $allWrap
      * @param bool $excludeFromConcatenation
      * @param string $splitChar The char used to split the allWrap value, default is "|"
      * @param string $allWrap
      * @param bool $excludeFromConcatenation
      * @param string $splitChar The char used to split the allWrap value, default is "|"
-     * @return void
+     * @param bool $inline
      */
      */
-    public function addCssLibrary($file, $rel = 'stylesheet', $media = 'all', $title = '', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|')
+    public function addCssLibrary($file, $rel = 'stylesheet', $media = 'all', $title = '', $compress = true, $forceOnTop = false, $allWrap = '', $excludeFromConcatenation = false, $splitChar = '|', $inline = false)
     {
         if (!isset($this->cssLibs[$file])) {
     {
         if (!isset($this->cssLibs[$file])) {
-            $this->cssLibs[$file] = array(
+            $this->cssLibs[$file] = [
                 'file' => $file,
                 'rel' => $rel,
                 'media' => $media,
                 'file' => $file,
                 'rel' => $rel,
                 'media' => $media,
@@ -1479,8 +1244,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                 'forceOnTop' => $forceOnTop,
                 'allWrap' => $allWrap,
                 'excludeFromConcatenation' => $excludeFromConcatenation,
                 'forceOnTop' => $forceOnTop,
                 'allWrap' => $allWrap,
                 'excludeFromConcatenation' => $excludeFromConcatenation,
-                'splitChar' => $splitChar
-            );
+                'splitChar' => $splitChar,
+                'inline' => $inline
+            ];
         }
     }
 
         }
     }
 
@@ -1491,103 +1257,133 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $block
      * @param bool $compress
      * @param bool $forceOnTop
      * @param string $block
      * @param bool $compress
      * @param bool $forceOnTop
-     * @return void
      */
     public function addCssInlineBlock($name, $block, $compress = false, $forceOnTop = false)
     {
         if (!isset($this->cssInline[$name]) && !empty($block)) {
      */
     public function addCssInlineBlock($name, $block, $compress = false, $forceOnTop = false)
     {
         if (!isset($this->cssInline[$name]) && !empty($block)) {
-            $this->cssInline[$name] = array(
+            $this->cssInline[$name] = [
                 'code' => $block,
                 'compress' => $compress,
                 'forceOnTop' => $forceOnTop
                 'code' => $block,
                 'compress' => $compress,
                 'forceOnTop' => $forceOnTop
-            );
+            ];
         }
     }
 
     /**
         }
     }
 
     /**
-     * Call this function if you need to include the jQuery library
-     *
-     * @param null|string $version The jQuery version that should be included, either "latest" or any available version
-     * @param null|string $source The location of the jQuery source, can be "local", "google", "msn", "jquery" or just an URL to your jQuery lib
-     * @param string $namespace The namespace in which the jQuery object of the specific version should be stored.
-     * @return void
-     * @throws \UnexpectedValueException
+     * Call function if you need the requireJS library
+     * this automatically adds the JavaScript path of all loaded extensions in the requireJS path option
+     * so it resolves names like TYPO3/CMS/MyExtension/MyJsFile to EXT:MyExtension/Resources/Public/JavaScript/MyJsFile.js
+     * when using requireJS
      */
      */
-    public function loadJquery($version = null, $source = null, $namespace = self::JQUERY_NAMESPACE_DEFAULT)
+    public function loadRequireJs()
     {
     {
-        // Set it to the version that is shipped with the TYPO3 core
-        if ($version === null || $version === 'latest') {
-            $version = self::JQUERY_VERSION_LATEST;
-        }
-        // Check if the source is set, otherwise set it to "default"
-        if ($source === null) {
-            $source = 'local';
-        }
-        if ($source === 'local' && !in_array($version, $this->availableLocalJqueryVersions)) {
-            throw new \UnexpectedValueException('The requested jQuery version is not available in the local filesystem.', 1341505305);
+        $this->addRequireJs = true;
+        if (!empty($this->requireJsConfig) && !empty($this->publicRequireJsConfig)) {
+            return;
         }
         }
-        if (!preg_match('/^[a-zA-Z0-9]+$/', $namespace)) {
-            throw new \UnexpectedValueException('The requested namespace contains non alphanumeric characters.', 1341571604);
+
+        $packages = GeneralUtility::makeInstance(PackageManager::class)->getActivePackages();
+        $isDevelopment = GeneralUtility::getApplicationContext()->isDevelopment();
+        $cacheIdentifier = 'requireJS_' . md5(implode(',', array_keys($packages)) . ($isDevelopment ? ':dev' : '') . GeneralUtility::getIndpEnv('TYPO3_REQUEST_SCRIPT'));
+        /** @var FrontendInterface $cache */
+        $cache = static::$cache ?? GeneralUtility::makeInstance(CacheManager::class)->getCache('assets');
+        $requireJsConfig = $cache->get($cacheIdentifier);
+
+        // if we did not get a configuration from the cache, compute and store it in the cache
+        if (!isset($requireJsConfig['internal']) || !isset($requireJsConfig['public'])) {
+            $requireJsConfig = $this->computeRequireJsConfig($isDevelopment, $packages);
+            $cache->set($cacheIdentifier, $requireJsConfig);
         }
         }
-        $this->jQueryVersions[$namespace] = array(
-            'version' => $version,
-            'source' => $source
-        );
+
+        $this->requireJsConfig = $requireJsConfig['internal'];
+        $this->publicRequireJsConfig = $requireJsConfig['public'];
+        $this->internalRequireJsPathModuleNames = $requireJsConfig['internalNames'];
     }
 
     /**
     }
 
     /**
-     * Call function if you need the requireJS library
-     * this automatically adds the JavaScript path of all loaded extensions in the requireJS path option
-     * so it resolves names like TYPO3/CMS/MyExtension/MyJsFile to EXT:MyExtension/Resources/Public/JavaScript/MyJsFile.js
-     * when using requireJS
+     * Computes the RequireJS configuration, mainly consisting of the paths to the core and all extension JavaScript
+     * resource folders plus some additional generic configuration.
      *
      *
-     * @return void
+     * @param bool $isDevelopment
+     * @param array $packages
+     * @return array The RequireJS configuration
      */
      */
-    public function loadRequireJs()
+    protected function computeRequireJsConfig($isDevelopment, array $packages)
     {
     {
-
         // load all paths to map to package names / namespaces
         // load all paths to map to package names / namespaces
-        if (empty($this->requireJsConfig)) {
-            // In order to avoid browser caching of JS files, adding a GET parameter to the files loaded via requireJS
-            if (GeneralUtility::getApplicationContext()->isDevelopment()) {
-                $this->requireJsConfig['urlArgs'] = 'bust=' . $GLOBALS['EXEC_TIME'];
-            } else {
-                $this->requireJsConfig['urlArgs'] = 'bust=' . GeneralUtility::hmac(TYPO3_version . PATH_site);
+        $requireJsConfig = [
+            'public' => [],
+            'internal' => [],
+            'internalNames' => [],
+        ];
+
+        $corePath = $packages['core']->getPackagePath() . 'Resources/Public/JavaScript/Contrib/';
+        $corePath = PathUtility::getAbsoluteWebPath($corePath);
+        // first, load all paths for the namespaces, and configure contrib libs.
+        $requireJsConfig['public']['paths'] = [
+            'jquery' => $corePath . '/jquery/jquery',
+            'jquery-ui' => $corePath . 'jquery-ui',
+            'datatables' => $corePath . 'jquery.dataTables',
+            'nprogress' => $corePath . 'nprogress',
+            'moment' => $corePath . 'moment',
+            'cropper' => $corePath . 'cropper.min',
+            'imagesloaded' => $corePath . 'imagesloaded.pkgd.min',
+            'bootstrap' => $corePath . 'bootstrap/bootstrap',
+            'twbs/bootstrap-datetimepicker' => $corePath . 'bootstrap-datetimepicker',
+            'autosize' => $corePath . 'autosize',
+            'taboverride' => $corePath . 'taboverride.min',
+            'twbs/bootstrap-slider' => $corePath . 'bootstrap-slider.min',
+            'jquery/autocomplete' => $corePath . 'jquery.autocomplete',
+            'd3' => $corePath . 'd3/d3',
+            'Sortable' => $corePath . 'Sortable.min',
+            'broadcastchannel' => $corePath . '/broadcastchannel-polyfill',
+        ];
+        $requireJsConfig['public']['waitSeconds'] = 30;
+        $requireJsConfig['public']['typo3BaseUrl'] = false;
+        $publicPackageNames = ['core', 'frontend', 'backend'];
+        $requireJsExtensionVersions = [];
+        foreach ($packages as $packageName => $package) {
+            $absoluteJsPath = $package->getPackagePath() . 'Resources/Public/JavaScript/';
+            $fullJsPath = PathUtility::getAbsoluteWebPath($absoluteJsPath);
+            $fullJsPath = rtrim($fullJsPath, '/');
+            if (!empty($fullJsPath) && file_exists($absoluteJsPath)) {
+                $type = in_array($packageName, $publicPackageNames, true) ? 'public' : 'internal';
+                $requireJsConfig[$type]['paths']['TYPO3/CMS/' . GeneralUtility::underscoredToUpperCamelCase($packageName)] = $fullJsPath;
+                $requireJsExtensionVersions[] = $package->getPackageKey() . ':' . $package->getPackageMetadata()->getVersion();
             }
             }
-            // first, load all paths for the namespaces, and configure contrib libs.
-            $this->requireJsConfig['paths'] = array(
-                'jquery-ui' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/jquery-ui',
-                'datatables' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/jquery.dataTables',
-                'nprogress' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/nprogress',
-                'moment' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/moment',
-                'cropper' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/cropper.min',
-                'imagesloaded' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/imagesloaded.pkgd.min',
-                'bootstrap' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/bootstrap/bootstrap',
-                'twbs/bootstrap-datetimepicker' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/bootstrap-datetimepicker',
-                'autosize' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/autosize',
-                'taboverride' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/taboverride.min',
-                'twbs/bootstrap-slider' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/bootstrap-slider.min',
-                'jquery/autocomplete' => $this->backPath . 'sysext/core/Resources/Public/JavaScript/Contrib/jquery.autocomplete',
+        }
+        // sanitize module names in internal 'paths'
+        $internalPathModuleNames = array_keys($requireJsConfig['internal']['paths'] ?? []);
+        $sanitizedInternalPathModuleNames = array_map(
+            function ($moduleName) {
+                // trim spaces and slashes & add ending slash
+                return trim($moduleName, ' /') . '/';
+            },
+            $internalPathModuleNames
+        );
+        $requireJsConfig['internalNames'] = array_combine(
+            $sanitizedInternalPathModuleNames,
+            $internalPathModuleNames
+        );
+
+        // Add a GET parameter to the files loaded via requireJS in order to avoid browser caching of JS files
+        if ($isDevelopment) {
+            $requireJsConfig['public']['urlArgs'] = 'bust=' . $GLOBALS['EXEC_TIME'];
+        } else {
+            $requireJsConfig['public']['urlArgs'] = 'bust=' . GeneralUtility::hmac(
+                Environment::getProjectPath() . implode('|', $requireJsExtensionVersions)
             );
             );
-            // get all extensions that are loaded
-            $loadedExtensions = \TYPO3\CMS\Core\Utility\ExtensionManagementUtility::getLoadedExtensionListArray();
-            foreach ($loadedExtensions as $packageName) {
-                $fullJsPath = 'EXT:' . $packageName . '/Resources/Public/JavaScript/';
-                $fullJsPath = GeneralUtility::getFileAbsFileName($fullJsPath);
-                $fullJsPath = \TYPO3\CMS\Core\Utility\PathUtility::getRelativePath(PATH_typo3, $fullJsPath);
-                $fullJsPath = rtrim($fullJsPath, '/');
-                if ($fullJsPath) {
-                    $this->requireJsConfig['paths']['TYPO3/CMS/' . GeneralUtility::underscoredToUpperCamelCase($packageName)] = $this->backPath . $fullJsPath;
-                }
-            }
+        }
 
 
-            // check if additional AMD modules need to be loaded if a single AMD module is initialized
-            if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['RequireJS']['postInitializationModules'])) {
-                $this->addInlineSettingArray('RequireJS.PostInitializationModules', $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['RequireJS']['postInitializationModules']);
-            }
+        // check if additional AMD modules need to be loaded if a single AMD module is initialized
+        if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['RequireJS']['postInitializationModules'] ?? false)) {
+            $this->addInlineSettingArray(
+                'RequireJS.PostInitializationModules',
+                $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['RequireJS']['postInitializationModules']
+            );
         }
 
         }
 
-        $this->addRequireJs = true;
+        return $requireJsConfig;
     }
 
     /**
     }
 
     /**
@@ -1601,14 +1397,81 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      *      ),
      *
      * @param array $configuration The configuration that will be merged with existing one.
      *      ),
      *
      * @param array $configuration The configuration that will be merged with existing one.
-     * @return void
      */
     public function addRequireJsConfiguration(array $configuration)
     {
      */
     public function addRequireJsConfiguration(array $configuration)
     {
+        if (TYPO3_MODE === 'BE') {
+            // Load RequireJS in backend context at first. Doing this in FE could break the output
+            $this->loadRequireJs();
+        }
         \TYPO3\CMS\Core\Utility\ArrayUtility::mergeRecursiveWithOverrule($this->requireJsConfig, $configuration);
     }
 
     /**
         \TYPO3\CMS\Core\Utility\ArrayUtility::mergeRecursiveWithOverrule($this->requireJsConfig, $configuration);
     }
 
     /**
+     * Generates RequireJS loader HTML markup.
+     *
+     * @return string
+     * @throws \TYPO3\CMS\Backend\Routing\Exception\RouteNotFoundException
+     */
+    protected function getRequireJsLoader(): string
+    {
+        $html = '';
+        $backendRequest = TYPO3_REQUESTTYPE & TYPO3_REQUESTTYPE_BE;
+        $backendUserLoggedIn = !empty($GLOBALS['BE_USER']->user['uid']);
+
+        // no backend request - basically frontend
+        if (!$backendRequest) {
+            $requireJsConfig = $this->getRequireJsConfig(static::REQUIREJS_SCOPE_CONFIG);
+            $requireJsConfig['typo3BaseUrl'] = GeneralUtility::getIndpEnv('TYPO3_SITE_PATH') . '?eID=requirejs';
+        // backend request, but no backend user logged in
+        } elseif (!$backendUserLoggedIn) {
+            $uriBuilder = GeneralUtility::makeInstance(UriBuilder::class);
+            $requireJsConfig = $this->getRequireJsConfig(static::REQUIREJS_SCOPE_CONFIG);
+            $requireJsConfig['typo3BaseUrl'] = (string)$uriBuilder->buildUriFromRoute('ajax_core_requirejs');
+        // backend request, having backend user logged in
+        } else {
+            $requireJsConfig = array_replace_recursive(
+                $this->publicRequireJsConfig,
+                $this->requireJsConfig
+            );
+        }
+
+        // add (probably filtered) RequireJS configuration
+        $html .= GeneralUtility::wrapJS('var require = ' . json_encode($requireJsConfig)) . LF;
+        // directly after that, include the require.js file
+        $html .= '<script src="'
+            . $this->processJsFile($this->requireJsPath . 'require.js')
+            . '"></script>' . LF;
+
+        if (!empty($requireJsConfig['typo3BaseUrl'])) {
+            $html .= '<script src="'
+                . $this->processJsFile(
+                    'EXT:core/Resources/Public/JavaScript/requirejs-loader.js'
+                )
+                . '"></script>' . LF;
+        }
+
+        return $html;
+    }
+
+    /**
+     * @param array $array
+     * @param string[] $keys
+     * @param bool $keep
+     * @return array
+     */
+    protected function filterArrayKeys(array $array, array $keys, bool $keep = true): array
+    {
+        return array_filter(
+            $array,
+            function (string $key) use ($keys, $keep) {
+                return in_array($key, $keys, true) === $keep;
+            },
+            ARRAY_FILTER_USE_KEY
+        );
+    }
+
+    /**
      * includes an AMD-compatible JS file by resolving the ModuleName, and then requires the file via a requireJS request,
      * additionally allowing to execute JavaScript code afterwards
      *
      * includes an AMD-compatible JS file by resolving the ModuleName, and then requires the file via a requireJS request,
      * additionally allowing to execute JavaScript code afterwards
      *
@@ -1623,14 +1486,19 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      *
      * @param string $mainModuleName Must be in the form of "TYPO3/CMS/PackageName/ModuleName" e.g. "TYPO3/CMS/Backend/FormEngine"
      * @param string $callBackFunction loaded right after the requireJS loading, must be wrapped in function() {}
      *
      * @param string $mainModuleName Must be in the form of "TYPO3/CMS/PackageName/ModuleName" e.g. "TYPO3/CMS/Backend/FormEngine"
      * @param string $callBackFunction loaded right after the requireJS loading, must be wrapped in function() {}
-     * @return void
      */
     public function loadRequireJsModule($mainModuleName, $callBackFunction = null)
     {
         $inlineCodeKey = $mainModuleName;
         // make sure requireJS is initialized
         $this->loadRequireJs();
      */
     public function loadRequireJsModule($mainModuleName, $callBackFunction = null)
     {
         $inlineCodeKey = $mainModuleName;
         // make sure requireJS is initialized
         $this->loadRequireJs();
-
+        // move internal module path definition to public module definition
+        // (since loading a module ends up disclosing the existence anyway)
+        $baseModuleName = $this->findRequireJsBaseModuleName($mainModuleName);
+        if ($baseModuleName !== null && isset($this->requireJsConfig['paths'][$baseModuleName])) {
+            $this->publicRequireJsConfig['paths'][$baseModuleName] = $this->requireJsConfig['paths'][$baseModuleName];
+            unset($this->requireJsConfig['paths'][$baseModuleName]);
+        }
         // execute the main module, and load a possible callback function
         $javaScriptCode = 'require(["' . $mainModuleName . '"]';
         if ($callBackFunction !== null) {
         // execute the main module, and load a possible callback function
         $javaScriptCode = 'require(["' . $mainModuleName . '"]';
         if ($callBackFunction !== null) {
@@ -1641,39 +1509,30 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
         $this->addJsInlineCode('RequireJS-Module-' . $inlineCodeKey, $javaScriptCode);
     }
 
         $this->addJsInlineCode('RequireJS-Module-' . $inlineCodeKey, $javaScriptCode);
     }
 
-
     /**
     /**
-     * call this function if you need the extJS library
+     * Determines requireJS base module name (if defined).
      *
      *
-     * @param bool $css Flag, if set the ext-css will be loaded
-     * @param bool $theme Flag, if set the ext-theme "grey" will be loaded
-     * @return void
+     * @param string $moduleName
+     * @return string|null
      */
      */
-    public function loadExtJS($css = true, $theme = true)
+    protected function findRequireJsBaseModuleName(string $moduleName)
     {
     {
-        $this->addExtJS = true;
-        $this->extJStheme = $theme;
-        $this->extJScss = $css;
-    }
-
-    /**
-     * Call this function to load debug version of ExtJS. Use this for development only
-     *
-     * @return void
-     */
-    public function enableExtJsDebug()
-    {
-        $this->enableExtJsDebug = true;
+        // trim spaces and slashes & add ending slash
+        $sanitizedModuleName = trim($moduleName, ' /') . '/';
+        foreach ($this->internalRequireJsPathModuleNames as $sanitizedBaseModuleName => $baseModuleName) {
+            if (strpos($sanitizedModuleName, $sanitizedBaseModuleName) === 0) {
+                return $baseModuleName;
+            }
+        }
+        return null;
     }
 
     /**
      * Adds Javascript Inline Label. This will occur in TYPO3.lang - object
      * The label can be used in scripts with TYPO3.lang.<key>
     }
 
     /**
      * Adds Javascript Inline Label. This will occur in TYPO3.lang - object
      * The label can be used in scripts with TYPO3.lang.<key>
-     * Need extJs loaded
      *
      * @param string $key
      * @param string $value
      *
      * @param string $key
      * @param string $value
-     * @return void
      */
     public function addInlineLanguageLabel($key, $value)
     {
      */
     public function addInlineLanguageLabel($key, $value)
     {
@@ -1684,24 +1543,11 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Adds Javascript Inline Label Array. This will occur in TYPO3.lang - object
      * The label can be used in scripts with TYPO3.lang.<key>
      * Array will be merged with existing array.
      * Adds Javascript Inline Label Array. This will occur in TYPO3.lang - object
      * The label can be used in scripts with TYPO3.lang.<key>
      * Array will be merged with existing array.
-     * Need extJs loaded
      *
      * @param array $array
      *
      * @param array $array
-     * @param bool $parseWithLanguageService
-     * @return void
      */
      */
-    public function addInlineLanguageLabelArray(array $array, $parseWithLanguageService = false)
+    public function addInlineLanguageLabelArray(array $array)
     {
     {
-        if ($parseWithLanguageService === true) {
-            foreach ($array as $key => $value) {
-                if (TYPO3_MODE === 'FE') {
-                    $array[$key] = $this->getTypoScriptFrontendController()->sL($value);
-                } else {
-                    $array[$key] = $this->getLanguageService()->sL($value);
-                }
-            }
-        }
-
         $this->inlineLanguageLabels = array_merge($this->inlineLanguageLabels, $array);
     }
 
         $this->inlineLanguageLabels = array_merge($this->inlineLanguageLabels, $array);
     }
 
@@ -1711,31 +1557,26 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $fileRef Input is a file-reference (see GeneralUtility::getFileAbsFileName). That file is expected to be a 'locallang.xlf' file containing a valid XML TYPO3 language structure.
      * @param string $selectionPrefix Prefix to select the correct labels (default: '')
      * @param string $stripFromSelectionName String to be removed from the label names in the output. (default: '')
      * @param string $fileRef Input is a file-reference (see GeneralUtility::getFileAbsFileName). That file is expected to be a 'locallang.xlf' file containing a valid XML TYPO3 language structure.
      * @param string $selectionPrefix Prefix to select the correct labels (default: '')
      * @param string $stripFromSelectionName String to be removed from the label names in the output. (default: '')
-     * @param int $errorMode Error mode (when file could not be found): 0 - syslog entry, 1 - do nothing, 2 - throw an exception
-     * @return void
      */
      */
-    public function addInlineLanguageLabelFile($fileRef, $selectionPrefix = '', $stripFromSelectionName = '', $errorMode = 0)
+    public function addInlineLanguageLabelFile($fileRef, $selectionPrefix = '', $stripFromSelectionName = '')
     {
         $index = md5($fileRef . $selectionPrefix . $stripFromSelectionName);
         if ($fileRef && !isset($this->inlineLanguageLabelFiles[$index])) {
     {
         $index = md5($fileRef . $selectionPrefix . $stripFromSelectionName);
         if ($fileRef && !isset($this->inlineLanguageLabelFiles[$index])) {
-            $this->inlineLanguageLabelFiles[$index] = array(
+            $this->inlineLanguageLabelFiles[$index] = [
                 'fileRef' => $fileRef,
                 'selectionPrefix' => $selectionPrefix,
                 'fileRef' => $fileRef,
                 'selectionPrefix' => $selectionPrefix,
-                'stripFromSelectionName' => $stripFromSelectionName,
-                'errorMode' => $errorMode
-            );
+                'stripFromSelectionName' => $stripFromSelectionName
+            ];
         }
     }
 
     /**
      * Adds Javascript Inline Setting. This will occur in TYPO3.settings - object
      * The label can be used in scripts with TYPO3.setting.<key>
         }
     }
 
     /**
      * Adds Javascript Inline Setting. This will occur in TYPO3.settings - object
      * The label can be used in scripts with TYPO3.setting.<key>
-     * Need extJs loaded
      *
      * @param string $namespace
      * @param string $key
      *
      * @param string $namespace
      * @param string $key
-     * @param string $value
-     * @return void
+     * @param mixed $value
      */
     public function addInlineSetting($namespace, $key, $value)
     {
      */
     public function addInlineSetting($namespace, $key, $value)
     {
@@ -1759,11 +1600,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Adds Javascript Inline Setting. This will occur in TYPO3.settings - object
      * The label can be used in scripts with TYPO3.setting.<key>
      * Array will be merged with existing array.
      * Adds Javascript Inline Setting. This will occur in TYPO3.settings - object
      * The label can be used in scripts with TYPO3.setting.<key>
      * Array will be merged with existing array.
-     * Need extJs loaded
      *
      * @param string $namespace
      * @param array $array
      *
      * @param string $namespace
      * @param array $array
-     * @return void
      */
     public function addInlineSettingArray($namespace, array $array)
     {
      */
     public function addInlineSettingArray($namespace, array $array)
     {
@@ -1787,7 +1626,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Adds content to body content
      *
      * @param string $content
      * Adds content to body content
      *
      * @param string $content
-     * @return void
      */
     public function addBodyContent($content)
     {
      */
     public function addBodyContent($content)
     {
@@ -1809,7 +1647,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     {
         $this->prepareRendering();
         list($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs) = $this->renderJavaScriptAndCss();
     {
         $this->prepareRendering();
         list($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs) = $this->renderJavaScriptAndCss();
-        $metaTags = implode(LF, $this->metaTags);
+        $metaTags = implode(LF, array_merge($this->metaTags, $this->renderMetaTagsFromAPI()));
         $markerArray = $this->getPreparedMarkerArray($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs, $metaTags);
         $template = $this->getTemplateForPart($part);
 
         $markerArray = $this->getPreparedMarkerArray($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs, $metaTags);
         $template = $this->getTemplateForPart($part);
 
@@ -1822,10 +1660,29 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
+     * Renders metaTags based on tags added via the API
+     *
+     * @return array
+     */
+    protected function renderMetaTagsFromAPI()
+    {
+        $metaTags = [];
+        $metaTagManagers = $this->metaTagRegistry->getAllManagers();
+
+        foreach ($metaTagManagers as $manager => $managerObject) {
+            $properties = $managerObject->renderAllProperties();
+            if (!empty($properties)) {
+                $metaTags[] = $properties;
+            }
+        }
+        return $metaTags;
+    }
+
+    /**
      * Render the page but not the JavaScript and CSS Files
      *
      * Render the page but not the JavaScript and CSS Files
      *
-     * @param string $substituteHash The hash that is used for the placehoder markers
-     * @access private
+     * @param string $substituteHash The hash that is used for the placeholder markers
+     * @internal
      * @return string Content of rendered section
      */
     public function renderPageWithUncachedObjects($substituteHash)
      * @return string Content of rendered section
      */
     public function renderPageWithUncachedObjects($substituteHash)
@@ -1842,8 +1699,8 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * of uncached content objects (USER_INT, COA_INT)
      *
      * @param string $cachedPageContent
      * of uncached content objects (USER_INT, COA_INT)
      *
      * @param string $cachedPageContent
-     * @param string $substituteHash The hash that is used for the placehoder markers
-     * @access private
+     * @param string $substituteHash The hash that is used for the variables
+     * @internal
      * @return string
      */
     public function renderJavaScriptAndCssForProcessingOfUncachedContentObjects($cachedPageContent, $substituteHash)
      * @return string
      */
     public function renderJavaScriptAndCssForProcessingOfUncachedContentObjects($cachedPageContent, $substituteHash)
@@ -1851,7 +1708,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
         $this->prepareRendering();
         list($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs) = $this->renderJavaScriptAndCss();
         $title = $this->title ? str_replace('|', htmlspecialchars($this->title), $this->titleTag) : '';
         $this->prepareRendering();
         list($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs) = $this->renderJavaScriptAndCss();
         $title = $this->title ? str_replace('|', htmlspecialchars($this->title), $this->titleTag) : '';
-        $markerArray = array(
+        $markerArray = [
             '<!-- ###TITLE' . $substituteHash . '### -->' => $title,
             '<!-- ###CSS_LIBS' . $substituteHash . '### -->' => $cssLibs,
             '<!-- ###CSS_INCLUDE' . $substituteHash . '### -->' => $cssFiles,
             '<!-- ###TITLE' . $substituteHash . '### -->' => $title,
             '<!-- ###CSS_LIBS' . $substituteHash . '### -->' => $cssLibs,
             '<!-- ###CSS_INCLUDE' . $substituteHash . '### -->' => $cssFiles,
@@ -1859,12 +1716,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             '<!-- ###JS_INLINE' . $substituteHash . '### -->' => $jsInline,
             '<!-- ###JS_INCLUDE' . $substituteHash . '### -->' => $jsFiles,
             '<!-- ###JS_LIBS' . $substituteHash . '### -->' => $jsLibs,
             '<!-- ###JS_INLINE' . $substituteHash . '### -->' => $jsInline,
             '<!-- ###JS_INCLUDE' . $substituteHash . '### -->' => $jsFiles,
             '<!-- ###JS_LIBS' . $substituteHash . '### -->' => $jsLibs,
+            '<!-- ###META' . $substituteHash . '### -->' => implode(LF, array_merge($this->metaTags, $this->renderMetaTagsFromAPI())),
             '<!-- ###HEADERDATA' . $substituteHash . '### -->' => implode(LF, $this->headerData),
             '<!-- ###FOOTERDATA' . $substituteHash . '### -->' => implode(LF, $this->footerData),
             '<!-- ###JS_LIBS_FOOTER' . $substituteHash . '### -->' => $jsFooterLibs,
             '<!-- ###JS_INCLUDE_FOOTER' . $substituteHash . '### -->' => $jsFooterFiles,
             '<!-- ###JS_INLINE_FOOTER' . $substituteHash . '### -->' => $jsFooterInline
             '<!-- ###HEADERDATA' . $substituteHash . '### -->' => implode(LF, $this->headerData),
             '<!-- ###FOOTERDATA' . $substituteHash . '### -->' => implode(LF, $this->footerData),
             '<!-- ###JS_LIBS_FOOTER' . $substituteHash . '### -->' => $jsFooterLibs,
             '<!-- ###JS_INCLUDE_FOOTER' . $substituteHash . '### -->' => $jsFooterFiles,
             '<!-- ###JS_INLINE_FOOTER' . $substituteHash . '### -->' => $jsFooterInline
-        );
+        ];
         foreach ($markerArray as $placeHolder => $content) {
             $cachedPageContent = str_replace($placeHolder, $content, $cachedPageContent);
         }
         foreach ($markerArray as $placeHolder => $content) {
             $cachedPageContent = str_replace($placeHolder, $content, $cachedPageContent);
         }
@@ -1874,10 +1732,8 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Remove ending slashes from static header block
 
     /**
      * Remove ending slashes from static header block
-     * if the page is beeing rendered as html (not xhtml)
+     * if the page is being rendered as html (not xhtml)
      * and define property $this->endingSlash for further use
      * and define property $this->endingSlash for further use
-     *
-     * @return void
      */
     protected function prepareRendering()
     {
      */
     protected function prepareRendering()
     {
@@ -1894,13 +1750,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * Renders all JavaScript and CSS
      *
     /**
      * Renders all JavaScript and CSS
      *
-     * @return array<string>
+     * @return array|string[]
      */
     protected function renderJavaScriptAndCss()
     {
         $this->executePreRenderHook();
         $mainJsLibs = $this->renderMainJavaScriptLibraries();
      */
     protected function renderJavaScriptAndCss()
     {
         $this->executePreRenderHook();
         $mainJsLibs = $this->renderMainJavaScriptLibraries();
-        if ($this->concatenateFiles || $this->concatenateJavascript || $this->concatenateCss) {
+        if ($this->concatenateJavascript || $this->concatenateCss) {
             // Do the file concatenation
             $this->doConcatenate();
         }
             // Do the file concatenation
             $this->doConcatenate();
         }
@@ -1925,27 +1781,27 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             $jsInline = '';
         }
         $this->executePostRenderHook($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs);
             $jsInline = '';
         }
         $this->executePostRenderHook($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs);
-        return array($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs);
+        return [$jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs];
     }
 
     /**
      * Fills the marker array with the given strings and trims each value
      *
     }
 
     /**
      * Fills the marker array with the given strings and trims each value
      *
-     * @param $jsLibs string
-     * @param $jsFiles string
-     * @param $jsFooterFiles string
-     * @param $cssLibs string
-     * @param $cssFiles string
-     * @param $jsInline string
-     * @param $cssInline string
-     * @param $jsFooterInline string
-     * @param $jsFooterLibs string
-     * @param $metaTags string
+     * @param string $jsLibs
+     * @param string $jsFiles
+     * @param string $jsFooterFiles
+     * @param string $cssLibs
+     * @param string $cssFiles
+     * @param string $jsInline
+     * @param string $cssInline
+     * @param string $jsFooterInline
+     * @param string $jsFooterLibs
+     * @param string $metaTags
      * @return array Marker array
      */
     protected function getPreparedMarkerArray($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs, $metaTags)
     {
      * @return array Marker array
      */
     protected function getPreparedMarkerArray($jsLibs, $jsFiles, $jsFooterFiles, $cssLibs, $cssFiles, $jsInline, $cssInline, $jsFooterInline, $jsFooterLibs, $metaTags)
     {
-        $markerArray = array(
+        $markerArray = [
             'XMLPROLOG_DOCTYPE' => $this->xmlPrologAndDocType,
             'HTMLTAG' => $this->htmlTag,
             'HEADTAG' => $this->headTag,
             'XMLPROLOG_DOCTYPE' => $this->xmlPrologAndDocType,
             'HTMLTAG' => $this->htmlTag,
             'HEADTAG' => $this->headTag,
@@ -1967,7 +1823,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             'JS_INCLUDE_FOOTER' => $jsFooterFiles,
             'JS_INLINE_FOOTER' => $jsFooterInline,
             'BODY' => $this->bodyContent
             'JS_INCLUDE_FOOTER' => $jsFooterFiles,
             'JS_INLINE_FOOTER' => $jsFooterInline,
             'BODY' => $this->bodyContent
-        );
+        ];
         $markerArray = array_map('trim', $markerArray);
         return $markerArray;
     }
         $markerArray = array_map('trim', $markerArray);
         return $markerArray;
     }
@@ -1975,12 +1831,12 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * Fills the marker array with the given strings and trims each value
      *
     /**
      * Fills the marker array with the given strings and trims each value
      *
-     * @param string $substituteHash The hash that is used for the placehoder markers
+     * @param string $substituteHash The hash that is used for the placeholder markers
      * @return array Marker array
      */
     protected function getPreparedMarkerArrayForPageWithUncachedObjects($substituteHash)
     {
      * @return array Marker array
      */
     protected function getPreparedMarkerArrayForPageWithUncachedObjects($substituteHash)
     {
-        $markerArray = array(
+        $markerArray = [
             'XMLPROLOG_DOCTYPE' => $this->xmlPrologAndDocType,
             'HTMLTAG' => $this->htmlTag,
             'HEADTAG' => $this->headTag,
             'XMLPROLOG_DOCTYPE' => $this->xmlPrologAndDocType,
             'HTMLTAG' => $this->htmlTag,
             'HEADTAG' => $this->headTag,
@@ -1988,7 +1844,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             'INLINECOMMENT' => $this->inlineComments ? LF . LF . '<!-- ' . LF . implode(LF, $this->inlineComments) . '-->' . LF . LF : '',
             'BASEURL' => $this->baseUrl ? str_replace('|', $this->baseUrl, $this->baseUrlTag) : '',
             'SHORTCUT' => $this->favIcon ? sprintf($this->shortcutTag, htmlspecialchars($this->favIcon), $this->iconMimeType) : '',
             'INLINECOMMENT' => $this->inlineComments ? LF . LF . '<!-- ' . LF . implode(LF, $this->inlineComments) . '-->' . LF . LF : '',
             'BASEURL' => $this->baseUrl ? str_replace('|', $this->baseUrl, $this->baseUrlTag) : '',
             'SHORTCUT' => $this->favIcon ? sprintf($this->shortcutTag, htmlspecialchars($this->favIcon), $this->iconMimeType) : '',
-            'META' => implode(LF, $this->metaTags),
+            'META' => '<!-- ###META' . $substituteHash . '### -->',
             'BODY' => $this->bodyContent,
             'TITLE' => '<!-- ###TITLE' . $substituteHash . '### -->',
             'CSS_LIBS' => '<!-- ###CSS_LIBS' . $substituteHash . '### -->',
             'BODY' => $this->bodyContent,
             'TITLE' => '<!-- ###TITLE' . $substituteHash . '### -->',
             'CSS_LIBS' => '<!-- ###CSS_LIBS' . $substituteHash . '### -->',
@@ -2002,7 +1858,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             'JS_LIBS_FOOTER' => '<!-- ###JS_LIBS_FOOTER' . $substituteHash . '### -->',
             'JS_INCLUDE_FOOTER' => '<!-- ###JS_INCLUDE_FOOTER' . $substituteHash . '### -->',
             'JS_INLINE_FOOTER' => '<!-- ###JS_INLINE_FOOTER' . $substituteHash . '### -->'
             'JS_LIBS_FOOTER' => '<!-- ###JS_LIBS_FOOTER' . $substituteHash . '### -->',
             'JS_INCLUDE_FOOTER' => '<!-- ###JS_INCLUDE_FOOTER' . $substituteHash . '### -->',
             'JS_INLINE_FOOTER' => '<!-- ###JS_INLINE_FOOTER' . $substituteHash . '### -->'
-        );
+        ];
         $markerArray = array_map('trim', $markerArray);
         return $markerArray;
     }
         $markerArray = array_map('trim', $markerArray);
         return $markerArray;
     }
@@ -2015,21 +1871,25 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      */
     protected function getTemplateForPart($part)
     {
      */
     protected function getTemplateForPart($part)
     {
-        $templateFile = GeneralUtility::getFileAbsFileName($this->templateFile, true);
-        $template = GeneralUtility::getUrl($templateFile);
-        if ($this->removeLineBreaksFromTemplate) {
-            $template = strtr($template, array(LF => '', CR => ''));
-        }
-        if ($part !== self::PART_COMPLETE) {
-            $templatePart = explode('###BODY###', $template);
-            $template = $templatePart[$part - 1];
+        $templateFile = GeneralUtility::getFileAbsFileName($this->templateFile);
+        if (is_file($templateFile)) {
+            $template = file_get_contents($templateFile);
+            if ($this->removeLineBreaksFromTemplate) {
+                $template = strtr($template, [LF => '', CR => '']);
+            }
+            if ($part !== self::PART_COMPLETE) {
+                $templatePart = explode('###BODY###', $template);
+                $template = $templatePart[$part - 1];
+            }
+        } else {
+            $template = '';
         }
         return $template;
     }
 
     /**
      * Helper function for render the main JavaScript libraries,
         }
         return $template;
     }
 
     /**
      * Helper function for render the main JavaScript libraries,
-     * currently: RequireJS, jQuery, ExtJS
+     * currently: RequireJS
      *
      * @return string Content with JavaScript libraries
      */
      *
      * @return string Content with JavaScript libraries
      */
@@ -2039,139 +1899,106 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
         // Include RequireJS
         if ($this->addRequireJs) {
 
         // Include RequireJS
         if ($this->addRequireJs) {
-            // load the paths of the requireJS configuration
-            $out .= GeneralUtility::wrapJS('var require = ' . json_encode($this->requireJsConfig)) . LF;
-                // directly after that, include the require.js file
-            $out .= '<script src="' . $this->processJsFile(($this->backPath . $this->requireJsPath . 'require.js')) . '" type="text/javascript"></script>' . LF;
+            $out .= $this->getRequireJsLoader();
         }
 
         }
 
-        // Include jQuery Core for each namespace, depending on the version and source
-        if (!empty($this->jQueryVersions)) {
-            foreach ($this->jQueryVersions as $namespace => $jQueryVersion) {
-                $out .= $this->renderJqueryScriptTag($jQueryVersion['version'], $jQueryVersion['source'], $namespace);
-            }
-        }
-        // Include extJS
-        if ($this->addExtJS) {
-            // Use the base adapter all the time
-            $out .= '<script src="' . $this->processJsFile(($this->backPath . $this->extJsPath . 'adapter/ext-base' . ($this->enableExtJsDebug ? '-debug' : '') . '.js')) . '" type="text/javascript"></script>' . LF;
-            $out .= '<script src="' . $this->processJsFile(($this->backPath . $this->extJsPath . 'ext-all' . ($this->enableExtJsDebug ? '-debug' : '') . '.js')) . '" type="text/javascript"></script>' . LF;
-            // Add extJS localization
-            // Load standard ISO mapping and modify for use with ExtJS
-            $localeMap = $this->locales->getIsoMapping();
-            $localeMap[''] = 'en';
-            $localeMap['default'] = 'en';
-            // Greek
-            $localeMap['gr'] = 'el_GR';
-            // Norwegian Bokmaal
-            $localeMap['no'] = 'no_BO';
-            // Swedish
-            $localeMap['se'] = 'se_SV';
-            $extJsLang = isset($localeMap[$this->lang]) ? $localeMap[$this->lang] : $this->lang;
-            // @todo autoconvert file from UTF8 to current BE charset if necessary!!!!
-            $extJsLocaleFile = $this->extJsPath . 'locale/ext-lang-' . $extJsLang . '.js';
-            if (file_exists(PATH_typo3 . $extJsLocaleFile)) {
-                $out .= '<script src="' . $this->processJsFile(($this->backPath . $extJsLocaleFile)) . '" type="text/javascript" charset="utf-8"></script>' . LF;
-            }
-            // Remove extjs from JScodeLibArray
-            unset($this->jsFiles[$this->backPath . $this->extJsPath . 'ext-all.js'], $this->jsFiles[$this->backPath . $this->extJsPath . 'ext-all-debug.js']);
-        }
         $this->loadJavaScriptLanguageStrings();
         if (TYPO3_MODE === 'BE') {
         $this->loadJavaScriptLanguageStrings();
         if (TYPO3_MODE === 'BE') {
-            $this->addAjaxUrlsToInlineSettings();
+            $noBackendUserLoggedIn = empty($GLOBALS['BE_USER']->user['uid']);
+            $this->addAjaxUrlsToInlineSettings($noBackendUserLoggedIn);
+        }
+        $inlineSettings = '';
+        $languageLabels = $this->parseLanguageLabelsForJavaScript();
+        if (!empty($languageLabels)) {
+            $inlineSettings .= 'TYPO3.lang = ' . json_encode($languageLabels) . ';';
         }
         }
-        $inlineSettings = $this->inlineLanguageLabels ? 'TYPO3.lang = ' . json_encode($this->inlineLanguageLabels) . ';' : '';
         $inlineSettings .= $this->inlineSettings ? 'TYPO3.settings = ' . json_encode($this->inlineSettings) . ';' : '';
         $inlineSettings .= $this->inlineSettings ? 'TYPO3.settings = ' . json_encode($this->inlineSettings) . ';' : '';
-        if ($this->addExtJS) {
-            // Set clear.gif, move it on top, add handler code
-            $code = '';
-            if (!empty($this->extOnReadyCode)) {
-                foreach ($this->extOnReadyCode as $block) {
-                    $code .= $block;
-                }
-            }
-            $out .= $this->inlineJavascriptWrap[0] . '
-                               Ext.ns("TYPO3");
-                               Ext.BLANK_IMAGE_URL = "' . htmlspecialchars(GeneralUtility::locationHeaderUrl($this->backPath . 'sysext/t3skin/icons/gfx/clear.gif')) . '";
-                               Ext.SSL_SECURE_URL = "' . htmlspecialchars(GeneralUtility::locationHeaderUrl($this->backPath . 'sysext/t3skin/icons/gfx/clear.gif')) . '";' . LF
-                . $inlineSettings
-                . 'Ext.onReady(function() {'
-                    . $code
-                . ' });'
-                . $this->inlineJavascriptWrap[1];
-            $this->extOnReadyCode = array();
-            // Include TYPO3.l10n object
-            if (TYPO3_MODE === 'BE') {
-                $out .= '<script src="' . $this->processJsFile(($this->backPath . 'sysext/lang/Resources/Public/JavaScript/Typo3Lang.js')) . '" type="text/javascript" charset="utf-8"></script>' . LF;
-            }
-            if ($this->extJScss) {
-                if (isset($GLOBALS['TBE_STYLES']['extJS']['all'])) {
-                    $this->addCssLibrary($this->backPath . $GLOBALS['TBE_STYLES']['extJS']['all'], 'stylesheet', 'all', '', true);
-                } else {
-                    $this->addCssLibrary($this->backPath . $this->extJsPath . 'resources/css/ext-all-notheme.css', 'stylesheet', 'all', '', true);
-                }
-            }
-            if ($this->extJStheme) {
-                if (isset($GLOBALS['TBE_STYLES']['extJS']['theme'])) {
-                    $this->addCssLibrary($this->backPath . $GLOBALS['TBE_STYLES']['extJS']['theme'], 'stylesheet', 'all', '', true);
-                } else {
-                    $this->addCssLibrary($this->backPath . $this->extJsPath . 'resources/css/xtheme-blue.css', 'stylesheet', 'all', '', true);
-                }
-            }
-        } else {
-            // no extJS loaded, but still inline settings
-            if ($inlineSettings !== '') {
-                // make sure the global TYPO3 is available
-                $inlineSettings = 'var TYPO3 = TYPO3 || {};' . CRLF . $inlineSettings;
-                $out .= $this->inlineJavascriptWrap[0] . $inlineSettings . $this->inlineJavascriptWrap[1];
-                // Add language module only if also jquery is guaranteed to be there
-                if (TYPO3_MODE === 'BE' && !empty($this->jQueryVersions)) {
-                    $this->loadRequireJsModule('TYPO3/CMS/Lang/Lang');
-                }
-            }
+
+        if ($inlineSettings !== '') {
+            // make sure the global TYPO3 is available
+            $inlineSettings = 'var TYPO3 = TYPO3 || {};' . CRLF . $inlineSettings;
+            $out .= $this->inlineJavascriptWrap[0] . $inlineSettings . $this->inlineJavascriptWrap[1];
         }
         }
+
         return $out;
     }
 
     /**
         return $out;
     }
 
     /**
+     * Converts the language labels for usage in JavaScript
+     *
+     * @return array
+     */
+    protected function parseLanguageLabelsForJavaScript(): array
+    {
+        if (empty($this->inlineLanguageLabels)) {
+            return [];
+        }
+
+        $labels = [];
+        foreach ($this->inlineLanguageLabels as $key => $translationUnit) {
+            if (is_array($translationUnit)) {
+                $translationUnit = current($translationUnit);
+                $labels[$key] = $translationUnit['target'] ?? $translationUnit['source'];
+            } else {
+                $labels[$key] = $translationUnit;
+            }
+        }
+
+        return $labels;
+    }
+
+    /**
      * Load the language strings into JavaScript
      */
     protected function loadJavaScriptLanguageStrings()
     {
         if (!empty($this->inlineLanguageLabelFiles)) {
             foreach ($this->inlineLanguageLabelFiles as $languageLabelFile) {
      * Load the language strings into JavaScript
      */
     protected function loadJavaScriptLanguageStrings()
     {
         if (!empty($this->inlineLanguageLabelFiles)) {
             foreach ($this->inlineLanguageLabelFiles as $languageLabelFile) {
-                $this->includeLanguageFileForInline($languageLabelFile['fileRef'], $languageLabelFile['selectionPrefix'], $languageLabelFile['stripFromSelectionName'], $languageLabelFile['errorMode']);
+                $this->includeLanguageFileForInline($languageLabelFile['fileRef'], $languageLabelFile['selectionPrefix'], $languageLabelFile['stripFromSelectionName']);
             }
         }
             }
         }
-        $this->inlineLanguageLabelFiles = array();
-        // Convert labels/settings back to UTF-8 since json_encode() only works with UTF-8:
-        if (TYPO3_MODE === 'FE' && $this->getCharSet() !== 'utf-8') {
-            if ($this->inlineLanguageLabels) {
-                $this->csConvObj->convArray($this->inlineLanguageLabels, $this->getCharSet(), 'utf-8');
-            }
-            if ($this->inlineSettings) {
-                $this->csConvObj->convArray($this->inlineSettings, $this->getCharSet(), 'utf-8');
-            }
+        $this->inlineLanguageLabelFiles = [];
+        // Convert settings back to UTF-8 since json_encode() only works with UTF-8:
+        if ($this->getCharSet() && $this->getCharSet() !== 'utf-8' && is_array($this->inlineSettings)) {
+            $this->convertCharsetRecursivelyToUtf8($this->inlineSettings, $this->getCharSet());
         }
     }
 
     /**
         }
     }
 
     /**
-     * Make URLs to all backend ajax handlers available as inline setting.
+     * Small helper function to convert charsets for arrays into utf-8
+     *
+     * @param mixed $data given by reference (string/array usually)
+     * @param string $fromCharset convert FROM this charset
      */
      */
-    protected function addAjaxUrlsToInlineSettings()
+    protected function convertCharsetRecursivelyToUtf8(&$data, string $fromCharset)
     {
     {
-        $ajaxUrls = array();
-        foreach ($GLOBALS['TYPO3_CONF_VARS']['BE']['AJAX'] as $ajaxHandler => $_) {
-            $ajaxUrls[$ajaxHandler] = BackendUtility::getAjaxUrl($ajaxHandler);
+        foreach ($data as $key => $value) {
+            if (is_array($data[$key])) {
+                $this->convertCharsetRecursivelyToUtf8($data[$key], $fromCharset);
+            } elseif (is_string($data[$key])) {
+                $data[$key] = mb_convert_encoding($data[$key], 'utf-8', $fromCharset);
+            }
         }
         }
+    }
 
 
-        // also add the ajax-based routes
+    /**
+     * Make URLs to all backend ajax handlers available as inline setting.
+     *
+     * @param bool $publicRoutesOnly
+     */
+    protected function addAjaxUrlsToInlineSettings(bool $publicRoutesOnly = false)
+    {
+        $ajaxUrls = [];
+        // Add the ajax-based routes
         /** @var UriBuilder $uriBuilder */
         $uriBuilder = GeneralUtility::makeInstance(UriBuilder::class);
         /** @var Router $router */
         $router = GeneralUtility::makeInstance(Router::class);
         $routes = $router->getRoutes();
         foreach ($routes as $routeIdentifier => $route) {
         /** @var UriBuilder $uriBuilder */
         $uriBuilder = GeneralUtility::makeInstance(UriBuilder::class);
         /** @var Router $router */
         $router = GeneralUtility::makeInstance(Router::class);
         $routes = $router->getRoutes();
         foreach ($routes as $routeIdentifier => $route) {
+            if ($publicRoutesOnly && $route->getOption('access') !== 'public') {
+                continue;
+            }
             if ($route->getOption('ajax')) {
                 $uri = (string)$uriBuilder->buildUriFromRoute($routeIdentifier);
                 // use the shortened value in order to use this in JavaScript
             if ($route->getOption('ajax')) {
                 $uri = (string)$uriBuilder->buildUriFromRoute($routeIdentifier);
                 // use the shortened value in order to use this in JavaScript
@@ -2184,53 +2011,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
-     * Renders the HTML script tag for the given jQuery version.
-     *
-     * @param string $version The jQuery version that should be included, either "latest" or any available version
-     * @param string $source The location of the jQuery source, can be "local", "google", "msn" or "jquery
-     * @param string $namespace The namespace in which the jQuery object of the specific version should be stored
-     * @return string
-     */
-    protected function renderJqueryScriptTag($version, $source, $namespace)
-    {
-        switch (true) {
-            case isset($this->jQueryCdnUrls[$source]):
-                if ($this->enableJqueryDebug) {
-                    $minifyPart = '';
-                } else {
-                    $minifyPart = '.min';
-                }
-                $jQueryFileName = sprintf($this->jQueryCdnUrls[$source], $version, $minifyPart);
-                break;
-            case $source === 'local':
-                $jQueryFileName = $this->backPath . $this->jQueryPath . 'jquery-' . rawurlencode($version);
-                if ($this->enableJqueryDebug) {
-                    $jQueryFileName .= '.js';
-                } else {
-                    $jQueryFileName .= '.min.js';
-                }
-                break;
-            default:
-                $jQueryFileName = $source;
-        }
-        // Include the jQuery Core
-        $scriptTag = '<script src="' . htmlspecialchars($jQueryFileName) . '" type="text/javascript"></script>' . LF;
-        // Set the noConflict mode to be available via "TYPO3.jQuery" in all installations
-        switch ($namespace) {
-            case self::JQUERY_NAMESPACE_DEFAULT_NOCONFLICT:
-                $scriptTag .= GeneralUtility::wrapJS('jQuery.noConflict();') . LF;
-                break;
-            case self::JQUERY_NAMESPACE_NONE:
-                break;
-            case self::JQUERY_NAMESPACE_DEFAULT:
-
-            default:
-                $scriptTag .= GeneralUtility::wrapJS('var TYPO3 = TYPO3 || {}; TYPO3.' . $namespace . ' = jQuery.noConflict(true);') . LF;
-        }
-        return $scriptTag;
-    }
-
-    /**
      * Render CSS library files
      *
      * @return string
      * Render CSS library files
      *
      * @return string
@@ -2240,18 +2020,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
         $cssFiles = '';
         if (!empty($this->cssLibs)) {
             foreach ($this->cssLibs as $file => $properties) {
         $cssFiles = '';
         if (!empty($this->cssLibs)) {
             foreach ($this->cssLibs as $file => $properties) {
-                $file = GeneralUtility::resolveBackPath($file);
-                $file = GeneralUtility::createVersionNumberedFilename($file);
-                $tag = '<link rel="' . htmlspecialchars($properties['rel'])
-                    . '" type="text/css" href="' . htmlspecialchars($file)
-                    . '" media="' . htmlspecialchars($properties['media']) . '"'
-                    . ($properties['title'] ? ' title="' . htmlspecialchars($properties['title']) . '"' : '')
-                    . $this->endingSlash . '>';
-                if ($properties['allWrap']) {
-                    $wrapArr = explode($properties['splitChar'] ?: '|', $properties['allWrap'], 2);
-                    $tag = $wrapArr[0] . $tag . $wrapArr[1];
-                }
-                $tag .= LF;
+                $tag = $this->createCssTag($properties, $file);
                 if ($properties['forceOnTop']) {
                     $cssFiles = $tag . $cssFiles;
                 } else {
                 if ($properties['forceOnTop']) {
                     $cssFiles = $tag . $cssFiles;
                 } else {
@@ -2272,18 +2041,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
         $cssFiles = '';
         if (!empty($this->cssFiles)) {
             foreach ($this->cssFiles as $file => $properties) {
         $cssFiles = '';
         if (!empty($this->cssFiles)) {
             foreach ($this->cssFiles as $file => $properties) {
-                $file = GeneralUtility::resolveBackPath($file);
-                $file = GeneralUtility::createVersionNumberedFilename($file);
-                $tag = '<link rel="' . htmlspecialchars($properties['rel'])
-                    . '" type="text/css" href="' . htmlspecialchars($file)
-                    . '" media="' . htmlspecialchars($properties['media']) . '"'
-                    . ($properties['title'] ? ' title="' . htmlspecialchars($properties['title']) . '"' : '')
-                    . $this->endingSlash . '>';
-                if ($properties['allWrap']) {
-                    $wrapArr = explode($properties['splitChar'] ?: '|', $properties['allWrap'], 2);
-                    $tag = $wrapArr[0] . $tag . $wrapArr[1];
-                }
-                $tag .= LF;
+                $tag = $this->createCssTag($properties, $file);
                 if ($properties['forceOnTop']) {
                     $cssFiles = $tag . $cssFiles;
                 } else {
                 if ($properties['forceOnTop']) {
                     $cssFiles = $tag . $cssFiles;
                 } else {
@@ -2295,6 +2053,34 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
+     * Create link (inline=0) or style (inline=1) tag
+     *
+     * @param array $properties
+     * @param string $file
+     * @return string
+     */
+    private function createCssTag(array $properties, string $file): string
+    {
+        if ($properties['inline'] && @is_file($file)) {
+            $tag = $this->createInlineCssTagFromFile($file, $properties);
+        } else {
+            $href = $this->getStreamlinedFileName($file);
+            $tag = '<link rel="' . htmlspecialchars($properties['rel'])
+                . '" type="text/css" href="' . htmlspecialchars($href)
+                . '" media="' . htmlspecialchars($properties['media']) . '"'
+                . ($properties['title'] ? ' title="' . htmlspecialchars($properties['title']) . '"' : '')
+                . $this->endingSlash . '>';
+        }
+        if ($properties['allWrap']) {
+            $wrapArr = explode($properties['splitChar'] ?: '|', $properties['allWrap'], 2);
+            $tag = $wrapArr[0] . $tag . $wrapArr[1];
+        }
+        $tag .= LF;
+
+        return $tag;
+    }
+
+    /**
      * Render inline CSS
      *
      * @return string
      * Render inline CSS
      *
      * @return string
@@ -2317,9 +2103,9 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     }
 
     /**
     }
 
     /**
-     * Render JavaScipt libraries
+     * Render JavaScript libraries
      *
      *
-     * @return array<string> jsLibs and jsFooterLibs strings
+     * @return array|string[] jsLibs and jsFooterLibs strings
      */
     protected function renderAdditionalJavaScriptLibraries()
     {
      */
     protected function renderAdditionalJavaScriptLibraries()
     {
@@ -2327,11 +2113,12 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
         $jsFooterLibs = '';
         if (!empty($this->jsLibs)) {
             foreach ($this->jsLibs as $properties) {
         $jsFooterLibs = '';
         if (!empty($this->jsLibs)) {
             foreach ($this->jsLibs as $properties) {
-                $properties['file'] = GeneralUtility::resolveBackPath($properties['file']);
-                $properties['file'] = GeneralUtility::createVersionNumberedFilename($properties['file']);
-                $async = ($properties['async']) ? ' async="async"' : '';
-                $integrity = ($properties['integrity']) ? ' integrity="' . htmlspecialchars($properties['integrity']) . '"' : '';
-                $tag = '<script src="' . htmlspecialchars($properties['file']) . '" type="' . htmlspecialchars($properties['type']) . '"' . $async . $integrity . '></script>';
+                $properties['file'] = $this->getStreamlinedFileName($properties['file']);
+                $async = $properties['async'] ? ' async="async"' : '';
+                $defer = $properties['defer'] ? ' defer="defer"' : '';
+                $integrity = $properties['integrity'] ? ' integrity="' . htmlspecialchars($properties['integrity']) . '"' : '';
+                $crossorigin = $properties['crossorigin'] ? ' crossorigin="' . htmlspecialchars($properties['crossorigin']) . '"' : '';
+                $tag = '<script src="' . htmlspecialchars($properties['file']) . '" type="' . htmlspecialchars($properties['type']) . '"' . $async . $defer . $integrity . $crossorigin . '></script>';
                 if ($properties['allWrap']) {
                     $wrapArr = explode($properties['splitChar'] ?: '|', $properties['allWrap'], 2);
                     $tag = $wrapArr[0] . $tag . $wrapArr[1];
                 if ($properties['allWrap']) {
                     $wrapArr = explode($properties['splitChar'] ?: '|', $properties['allWrap'], 2);
                     $tag = $wrapArr[0] . $tag . $wrapArr[1];
@@ -2356,13 +2143,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             $jsFooterLibs = $jsLibs . LF . $jsFooterLibs;
             $jsLibs = '';
         }
             $jsFooterLibs = $jsLibs . LF . $jsFooterLibs;
             $jsLibs = '';
         }
-        return array($jsLibs, $jsFooterLibs);
+        return [$jsLibs, $jsFooterLibs];
     }
 
     /**
      * Render JavaScript files
      *
     }
 
     /**
      * Render JavaScript files
      *
-     * @return array<string> jsFiles and jsFooterFiles strings
+     * @return array|string[] jsFiles and jsFooterFiles strings
      */
     protected function renderJavaScriptFiles()
     {
      */
     protected function renderJavaScriptFiles()
     {
@@ -2370,11 +2157,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
         $jsFooterFiles = '';
         if (!empty($this->jsFiles)) {
             foreach ($this->jsFiles as $file => $properties) {
         $jsFooterFiles = '';
         if (!empty($this->jsFiles)) {
             foreach ($this->jsFiles as $file => $properties) {
-                $file = GeneralUtility::resolveBackPath($file);
-                $file = GeneralUtility::createVersionNumberedFilename($file);
-                $async = ($properties['async']) ? ' async="async"' : '';
-                $integrity = ($properties['integrity']) ? ' integrity="' . htmlspecialchars($properties['integrity']) . '"' : '';
-                $tag = '<script src="' . htmlspecialchars($file) . '" type="' . htmlspecialchars($properties['type']) . '"' . $async . $integrity . '></script>';
+                $file = $this->getStreamlinedFileName($file);
+                $type = $properties['type'] ? ' type="' . htmlspecialchars($properties['type']) . '"' : '';
+                $async = $properties['async'] ? ' async="async"' : '';
+                $defer = $properties['defer'] ? ' defer="defer"' : '';
+                $integrity = $properties['integrity'] ? ' integrity="' . htmlspecialchars($properties['integrity']) . '"' : '';
+                $crossorigin = $properties['crossorigin'] ? ' crossorigin="' . htmlspecialchars($properties['crossorigin']) . '"' : '';
+                $tag = '<script src="' . htmlspecialchars($file) . '"' . $type . $async . $defer . $integrity . $crossorigin . '></script>';
                 if ($properties['allWrap']) {
                     $wrapArr = explode($properties['splitChar'] ?: '|', $properties['allWrap'], 2);
                     $tag = $wrapArr[0] . $tag . $wrapArr[1];
                 if ($properties['allWrap']) {
                     $wrapArr = explode($properties['splitChar'] ?: '|', $properties['allWrap'], 2);
                     $tag = $wrapArr[0] . $tag . $wrapArr[1];
@@ -2399,13 +2188,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             $jsFooterFiles = $jsFiles . $jsFooterFiles;
             $jsFiles = '';
         }
             $jsFooterFiles = $jsFiles . $jsFooterFiles;
             $jsFiles = '';
         }
-        return array($jsFiles, $jsFooterFiles);
+        return [$jsFiles, $jsFooterFiles];
     }
 
     /**
      * Render inline JavaScript
      *
     }
 
     /**
      * Render inline JavaScript
      *
-     * @return array<string> jsInline and jsFooterInline string
+     * @return array|string[] jsInline and jsFooterInline string
      */
     protected function renderInlineJavaScript()
     {
      */
     protected function renderInlineJavaScript()
     {
@@ -2439,7 +2228,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
             $jsFooterInline = $jsInline . $jsFooterInline;
             $jsInline = '';
         }
             $jsFooterInline = $jsInline . $jsFooterInline;
             $jsInline = '';
         }
-        return array($jsInline, $jsFooterInline);
+        return [$jsInline, $jsFooterInline];
     }
 
     /**
     }
 
     /**
@@ -2448,17 +2237,15 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * @param string $fileRef
      * @param string $selectionPrefix
      * @param string $stripFromSelectionName
      * @param string $fileRef
      * @param string $selectionPrefix
      * @param string $stripFromSelectionName
-     * @param int $errorMode
-     * @return void
      * @throws \RuntimeException
      */
      * @throws \RuntimeException
      */
-    protected function includeLanguageFileForInline($fileRef, $selectionPrefix = '', $stripFromSelectionName = '', $errorMode = 0)
+    protected function includeLanguageFileForInline($fileRef, $selectionPrefix = '', $stripFromSelectionName = '')
     {
         if (!isset($this->lang) || !isset($this->charSet)) {
             throw new \RuntimeException('Language and character encoding are not set.', 1284906026);
         }
     {
         if (!isset($this->lang) || !isset($this->charSet)) {
             throw new \RuntimeException('Language and character encoding are not set.', 1284906026);
         }
-        $labelsFromFile = array();
-        $allLabels = $this->readLLfile($fileRef, $errorMode);
+        $labelsFromFile = [];
+        $allLabels = $this->readLLfile($fileRef);
         if ($allLabels !== false) {
             // Merge language specific translations:
             if ($this->lang !== 'default' && isset($allLabels[$this->lang])) {
         if ($allLabels !== false) {
             // Merge language specific translations:
             if ($this->lang !== 'default' && isset($allLabels[$this->lang])) {
@@ -2483,12 +2270,11 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      * Reads a locallang file.
      *
      * @param string $fileRef Reference to a relative filename to include.
      * Reads a locallang file.
      *
      * @param string $fileRef Reference to a relative filename to include.
-     * @param int $errorMode Error mode (when file could not be found): 0 - syslog entry, 1 - do nothing, 2 - throw an exception
      * @return array Returns the $LOCAL_LANG array found in the file. If no array found, returns empty array.
      */
      * @return array Returns the $LOCAL_LANG array found in the file. If no array found, returns empty array.
      */
-    protected function readLLfile($fileRef, $errorMode = 0)
+    protected function readLLfile($fileRef)
     {
     {
-        /** @var $languageFactory LocalizationFactory */
+        /** @var LocalizationFactory $languageFactory */
         $languageFactory = GeneralUtility::makeInstance(LocalizationFactory::class);
 
         if ($this->lang !== 'default') {
         $languageFactory = GeneralUtility::makeInstance(LocalizationFactory::class);
 
         if ($this->lang !== 'default') {
@@ -2498,12 +2284,12 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                 $languages[] = 'default';
             }
         } else {
                 $languages[] = 'default';
             }
         } else {
-            $languages = array('default');
+            $languages = ['default'];
         }
 
         }
 
-        $localLanguage = array();
+        $localLanguage = [];
         foreach ($languages as $language) {
         foreach ($languages as $language) {
-            $tempLL = $languageFactory->getParsedData($fileRef, $language, $this->charSet, $errorMode);
+            $tempLL = $languageFactory->getParsedData($fileRef, $language);
 
             $localLanguage['default'] = $tempLL['default'];
             if (!isset($localLanguage[$this->lang])) {
 
             $localLanguage['default'] = $tempLL['default'];
             if (!isset($localLanguage[$this->lang])) {
@@ -2519,7 +2305,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
         return $localLanguage;
     }
 
         return $localLanguage;
     }
 
-
     /*****************************************************/
     /*                                                   */
     /*  Tools                                            */
     /*****************************************************/
     /*                                                   */
     /*  Tools                                            */
@@ -2528,8 +2313,6 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * Concatenate files into one file
      * registered handler
     /**
      * Concatenate files into one file
      * registered handler
-     *
-     * @return void
      */
     protected function doConcatenate()
     {
      */
     protected function doConcatenate()
     {
@@ -2539,21 +2322,19 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Concatenate JavaScript files according to the configuration.
 
     /**
      * Concatenate JavaScript files according to the configuration.
-     *
-     * @return void
      */
     protected function doConcatenateJavaScript()
     {
      */
     protected function doConcatenateJavaScript()
     {
-        if ($this->concatenateFiles || $this->concatenateJavascript) {
+        if ($this->concatenateJavascript) {
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsConcatenateHandler'])) {
                 // use external concatenation routine
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsConcatenateHandler'])) {
                 // use external concatenation routine
-                $params = array(
+                $params = [
                     'jsLibs' => &$this->jsLibs,
                     'jsFiles' => &$this->jsFiles,
                     'jsFooterFiles' => &$this->jsFooterFiles,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
                     'jsLibs' => &$this->jsLibs,
                     'jsFiles' => &$this->jsFiles,
                     'jsFooterFiles' => &$this->jsFooterFiles,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
-                );
+                ];
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsConcatenateHandler'], $params, $this);
             } else {
                 $this->jsLibs = $this->getCompressor()->concatenateJsFiles($this->jsLibs);
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsConcatenateHandler'], $params, $this);
             } else {
                 $this->jsLibs = $this->getCompressor()->concatenateJsFiles($this->jsLibs);
@@ -2565,36 +2346,28 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Concatenate CSS files according to configuration.
 
     /**
      * Concatenate CSS files according to configuration.
-     *
-     * @return void
      */
     protected function doConcatenateCss()
     {
      */
     protected function doConcatenateCss()
     {
-        if ($this->concatenateFiles || $this->concatenateCss) {
+        if ($this->concatenateCss) {
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssConcatenateHandler'])) {
                 // use external concatenation routine
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssConcatenateHandler'])) {
                 // use external concatenation routine
-                $params = array(
+                $params = [
                     'cssFiles' => &$this->cssFiles,
                     'cssLibs' => &$this->cssLibs,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
                     'cssFiles' => &$this->cssFiles,
                     'cssLibs' => &$this->cssLibs,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
-                );
+                ];
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssConcatenateHandler'], $params, $this);
             } else {
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssConcatenateHandler'], $params, $this);
             } else {
-                $cssOptions = array();
-                if (TYPO3_MODE === 'BE') {
-                    $cssOptions = array('baseDirectories' => $GLOBALS['TBE_TEMPLATE']->getSkinStylesheetDirectories());
-                }
-                $this->cssLibs = $this->getCompressor()->concatenateCssFiles($this->cssLibs, $cssOptions);
-                $this->cssFiles = $this->getCompressor()->concatenateCssFiles($this->cssFiles, $cssOptions);
+                $this->cssLibs = $this->getCompressor()->concatenateCssFiles($this->cssLibs);
+                $this->cssFiles = $this->getCompressor()->concatenateCssFiles($this->cssFiles);
             }
         }
     }
 
     /**
      * Compresses inline code
             }
         }
     }
 
     /**
      * Compresses inline code
-     *
-     * @return void
      */
     protected function doCompress()
     {
      */
     protected function doCompress()
     {
@@ -2604,21 +2377,19 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Compresses CSS according to configuration.
 
     /**
      * Compresses CSS according to configuration.
-     *
-     * @return void
      */
     protected function doCompressCss()
     {
         if ($this->compressCss) {
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssCompressHandler'])) {
                 // Use external compression routine
      */
     protected function doCompressCss()
     {
         if ($this->compressCss) {
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssCompressHandler'])) {
                 // Use external compression routine
-                $params = array(
+                $params = [
                     'cssInline' => &$this->cssInline,
                     'cssFiles' => &$this->cssFiles,
                     'cssLibs' => &$this->cssLibs,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
                     'cssInline' => &$this->cssInline,
                     'cssFiles' => &$this->cssFiles,
                     'cssLibs' => &$this->cssLibs,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
-                );
+                ];
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssCompressHandler'], $params, $this);
             } else {
                 $this->cssLibs = $this->getCompressor()->compressCssFiles($this->cssLibs);
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['cssCompressHandler'], $params, $this);
             } else {
                 $this->cssLibs = $this->getCompressor()->compressCssFiles($this->cssLibs);
@@ -2629,15 +2400,13 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
 
     /**
      * Compresses JavaScript according to configuration.
 
     /**
      * Compresses JavaScript according to configuration.
-     *
-     * @return void
      */
     protected function doCompressJavaScript()
     {
         if ($this->compressJavascript) {
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsCompressHandler'])) {
                 // Use external compression routine
      */
     protected function doCompressJavaScript()
     {
         if ($this->compressJavascript) {
             if (!empty($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsCompressHandler'])) {
                 // Use external compression routine
-                $params = array(
+                $params = [
                     'jsInline' => &$this->jsInline,
                     'jsFooterInline' => &$this->jsFooterInline,
                     'jsLibs' => &$this->jsLibs,
                     'jsInline' => &$this->jsInline,
                     'jsFooterInline' => &$this->jsFooterInline,
                     'jsLibs' => &$this->jsLibs,
@@ -2645,7 +2414,7 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
                     'jsFooterFiles' => &$this->jsFooterFiles,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
                     'jsFooterFiles' => &$this->jsFooterFiles,
                     'headerData' => &$this->headerData,
                     'footerData' => &$this->footerData
-                );
+                ];
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsCompressHandler'], $params, $this);
             } else {
                 // Traverse the arrays, compress files
                 GeneralUtility::callUserFunction($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['jsCompressHandler'], $params, $this);
             } else {
                 // Traverse the arrays, compress files
@@ -2670,12 +2439,12 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /**
      * Returns instance of \TYPO3\CMS\Core\Resource\ResourceCompressor
      *
     /**
      * Returns instance of \TYPO3\CMS\Core\Resource\ResourceCompressor
      *
-     * @return \TYPO3\CMS\Core\Resource\ResourceCompressor
+     * @return ResourceCompressor
      */
     protected function getCompressor()
     {
         if ($this->compressor === null) {
      */
     protected function getCompressor()
     {
         if ($this->compressor === null) {
-            $this->compressor = GeneralUtility::makeInstance(\TYPO3\CMS\Core\Resource\ResourceCompressor::class);
+            $this->compressor = GeneralUtility::makeInstance(ResourceCompressor::class);
         }
         return $this->compressor;
     }
         }
         return $this->compressor;
     }
@@ -2690,41 +2459,59 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
      */
     protected function processJsFile($filename)
     {
      */
     protected function processJsFile($filename)
     {
-        switch (TYPO3_MODE) {
-            case 'FE':
-                if ($this->compressJavascript) {
-                    $filename = $this->getCompressor()->compressJsFile($filename);
-                } else {
-                    $filename = GeneralUtility::createVersionNumberedFilename($filename);
-                }
-                break;
-            case 'BE':
-                if ($this->compressJavascript) {
-                    $filename = $this->getCompressor()->compressJsFile($filename);
-                }
-                break;
+        $filename = $this->getStreamlinedFileName($filename, false);
+        if ($this->compressJavascript) {
+            $filename = $this->getCompressor()->compressJsFile($filename);
+        } elseif (TYPO3_MODE === 'FE') {
+            $filename = GeneralUtility::createVersionNumberedFilename($filename);
         }
         }
-        return $filename;
+        return $this->getAbsoluteWebPath($filename);
     }
 
     /**
     }
 
     /**
-     * Returns global frontend controller
+     * This function acts as a wrapper to allow relative and paths starting with EXT: to be dealt with
+     * in this very case to always return the absolute web path to be included directly before output.
+     *
+     * This is mainly added so the EXT: syntax can be resolved for PageRenderer in one central place,
+     * and hopefully removed in the future by one standard API call.
      *
      *
-     * @return TypoScriptFrontendController
+     * @param string $file the filename to process
+     * @param bool $prepareForOutput whether the file should be prepared as version numbered file and prefixed as absolute webpath
+     * @return string
+     * @internal
      */
      */
-    protected function getTypoScriptFrontendController()
+    protected function getStreamlinedFileName($file, $prepareForOutput = true)
     {
     {
-        return $GLOBALS['TSFE'];
+        if (strpos($file, 'EXT:') === 0) {
+            $file = GeneralUtility::getFileAbsFileName($file);
+            // as the path is now absolute, make it "relative" to the current script to stay compatible
+            $file = PathUtility::getRelativePathTo($file);
+            $file = rtrim($file, '/');
+        } else {
+            $file = GeneralUtility::resolveBackPath($file);
+        }
+        if ($prepareForOutput) {
+            $file = GeneralUtility::createVersionNumberedFilename($file);
+            $file = $this->getAbsoluteWebPath($file);
+        }
+        return $file;
     }
 
     /**
     }
 
     /**
-     * Returns global language service instance
+     * Gets absolute web path of filename for backend disposal.
+     * Resolving the absolute path in the frontend with conflict with
+     * applying config.absRefPrefix in frontend rendering process.
      *
      *
-     * @return \TYPO3\CMS\Lang\LanguageService
+     * @param string $file
+     * @return string
+     * @see \TYPO3\CMS\Frontend\Controller\TypoScriptFrontendController::setAbsRefPrefix()
      */
      */
-    protected function getLanguageService()
+    protected function getAbsoluteWebPath(string $file): string
     {
     {
-        return $GLOBALS['LANG'];
+        if (TYPO3_MODE === 'FE') {
+            return $file;
+        }
+        return PathUtility::getAbsoluteWebPath($file);
     }
 
     /*****************************************************/
     }
 
     /*****************************************************/
@@ -2734,104 +2521,127 @@ class PageRenderer implements \TYPO3\CMS\Core\SingletonInterface
     /*****************************************************/
     /**
      * Execute PreRenderHook for possible manipulation
     /*****************************************************/
     /**
      * Execute PreRenderHook for possible manipulation
-     *
-     * @return void
      */
     protected function executePreRenderHook()
     {
      */
     protected function executePreRenderHook()
     {
-        if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-preProcess'])) {
-            $params = array(
-                'jsLibs' => &$this->jsLibs,
-                'jsFooterLibs' => &$this->jsFooterLibs,
-                'jsFiles' => &$this->jsFiles,
-                'jsFooterFiles' => &$this->jsFooterFiles,
-                'cssFiles' => &$this->cssFiles,
-                'headerData' => &$this->headerData,
-                'footerData' => &$this->footerData,
-                'jsInline' => &$this->jsInline,
-                'jsFooterInline' => &$this->jsFooterInline,
-                'cssInline' => &$this->cssInline
-            );
-            foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-preProcess'] as $hook) {
-                GeneralUtility::callUserFunction($hook, $params, $this);
-            }
+        $hooks = $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-preProcess'] ?? false;
+        if (!$hooks) {
+            return;
+        }
+        $params = [
+            'jsLibs' => &$this->jsLibs,
+            'jsFooterLibs' => &$this->jsFooterLibs,
+            'jsFiles' => &$this->jsFiles,
+            'jsFooterFiles' => &$this->jsFooterFiles,
+            'cssLibs' => &$this->cssLibs,
+            'cssFiles' => &$this->cssFiles,
+            'headerData' => &$this->headerData,
+            'footerData' => &$this->footerData,
+            'jsInline' => &$this->jsInline,
+            'jsFooterInline' => &$this->jsFooterInline,
+            'cssInline' => &$this->cssInline
+        ];
+        foreach ($hooks as $hook) {
+            GeneralUtility::callUserFunction($hook, $params, $this);
         }
     }
 
     /**
      * PostTransform for possible manipulation of concatenated and compressed files
         }
     }
 
     /**
      * PostTransform for possible manipulation of concatenated and compressed files
-     *
-     * @return void
      */
     protected function executeRenderPostTransformHook()
     {
      */
     protected function executeRenderPostTransformHook()
     {
-        if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-postTransform'])) {
-            $params = array(
-                'jsLibs' => &$this->jsLibs,
-                'jsFooterLibs' => &$this->jsFooterLibs,
-                'jsFiles' => &$this->jsFiles,
-                'jsFooterFiles' => &$this->jsFooterFiles,
-                'cssFiles' => &$this->cssFiles,
-                'headerData' => &$this->headerData,
-                'footerData' => &$this->footerData,
-                'jsInline' => &$this->jsInline,
-                'jsFooterInline' => &$this->jsFooterInline,
-                'cssInline' => &$this->cssInline
-            );
-            foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-postTransform'] as $hook) {
-                GeneralUtility::callUserFunction($hook, $params, $this);
-            }
+        $hooks = $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-postTransform'] ?? false;
+        if (!$hooks) {
+            return;
+        }
+        $params = [
+            'jsLibs' => &$this->jsLibs,
+            'jsFooterLibs' => &$this->jsFooterLibs,
+            'jsFiles' => &$this->jsFiles,
+            'jsFooterFiles' => &$this->jsFooterFiles,
+            'cssLibs' => &$this->cssLibs,
+            'cssFiles' => &$this->cssFiles,
+            'headerData' => &$this->headerData,
+            'footerData' => &$this->footerData,
+            'jsInline' => &$this->jsInline,
+            'jsFooterInline' => &$this->jsFooterInline,
+            'cssInline' => &$this->cssInline
+        ];
+        foreach ($hooks as $hook) {
+            GeneralUtility::callUserFunction($hook, $params, $this);
         }
     }
 
     /**
      * Execute postRenderHook for possible manipulation
      *
         }
     }
 
     /**
      * Execute postRenderHook for possible manipulation
      *
-     * @param $jsLibs string
-     * @param $jsFiles string
-     * @param $jsFooterFiles string
-     * @param $cssLibs string
-     * @param $cssFiles string
-     * @param $jsInline string
-     * @param $cssInline string
-     * @param $jsFooterInline string
-     * @param $jsFooterLibs string
-     * @return void
+     * @param string $jsLibs
+     * @param string $jsFiles
+     * @param string $jsFooterFiles
+     * @param string $cssLibs
+     * @param string $cssFiles
+     * @param string $jsInline
+     * @param string $cssInline
+     * @param string $jsFooterInline
+     * @param string $jsFooterLibs
      */
     protected function executePostRenderHook(&$jsLibs, &$jsFiles, &$jsFooterFiles, &$cssLibs, &$cssFiles, &$jsInline, &$cssInline, &$jsFooterInline, &$jsFooterLibs)
     {
      */
     protected function executePostRenderHook(&$jsLibs, &$jsFiles, &$jsFooterFiles, &$cssLibs, &$cssFiles, &$jsInline, &$cssInline, &$jsFooterInline, &$jsFooterLibs)
     {
-        if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-postProcess'])) {
-            $params = array(
-                'jsLibs' => &$jsLibs,
-                'jsFiles' => &$jsFiles,
-                'jsFooterFiles' => &$jsFooterFiles,
-                'cssLibs' => &$cssLibs,
-                'cssFiles' => &$cssFiles,
-                'headerData' => &$this->headerData,
-                'footerData' => &$this->footerData,
-                'jsInline' => &$jsInline,
-                'cssInline' => &$cssInline,
-                'xmlPrologAndDocType' => &$this->xmlPrologAndDocType,
-                'htmlTag' => &$this->htmlTag,
-                'headTag' => &$this->headTag,
-                'charSet' => &$this->charSet,
-                'metaCharsetTag' => &$this->metaCharsetTag,
-                'shortcutTag' => &$this->shortcutTag,
-                'inlineComments' => &$this->inlineComments,
-                'baseUrl' => &$this->baseUrl,
-                'baseUrlTag' => &$this->baseUrlTag,
-                'favIcon' => &$this->favIcon,
-                'iconMimeType' => &$this->iconMimeType,
-                'titleTag' => &$this->titleTag,
-                'title' => &$this->title,
-                'metaTags' => &$this->metaTags,
-                'jsFooterInline' => &$jsFooterInline,
-                'jsFooterLibs' => &$jsFooterLibs,
-                'bodyContent' => &$this->bodyContent
-            );
-            foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-postProcess'] as $hook) {
-                GeneralUtility::callUserFunction($hook, $params, $this);
-            }
+        $hooks = $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['t3lib/class.t3lib_pagerenderer.php']['render-postProcess'] ?? false;
+        if (!$hooks) {
+            return;
         }
         }
+        $params = [
+            'jsLibs' => &$jsLibs,
+            'jsFiles' => &$jsFiles,
+            'jsFooterFiles' => &$jsFooterFiles,
+            'cssLibs' => &$cssLibs,
+            'cssFiles' => &$cssFiles,
+            'headerData' => &$this->headerData,
+            'footerData' => &$this->footerData,
+            'jsInline' => &$jsInline,
+            'cssInline' => &$cssInline,
+            'xmlPrologAndDocType' => &$this->xmlPrologAndDocType,
+            'htmlTag' => &$this->htmlTag,
+            'headTag' => &$this->headTag,
+            'charSet' => &$this->charSet,
+            'metaCharsetTag' => &$this->metaCharsetTag,
+            'shortcutTag' => &$this->shortcutTag,
+            'inlineComments' => &$this->inlineComments,
+            'baseUrl' => &$this->baseUrl,
+            'baseUrlTag' => &$this->baseUrlTag,
+            'favIcon' => &$this->favIcon,
+            'iconMimeType' => &$this->iconMimeType,
+            'titleTag' => &$this->titleTag,
+            'title' => &$this->title,
+            'metaTags' => &$this->metaTags,
+            'jsFooterInline' => &$jsFooterInline,
+            'jsFooterLibs' => &$jsFooterLibs,
+            'bodyContent' => &$this->bodyContent
+        ];
+        foreach ($hooks as $hook) {
+            GeneralUtility::callUserFunction($hook, $params, $this);
+        }
+    }
+
+    /**
+     * Creates a CSS inline tag
+     *
+     * @param string $file the filename to process
+     * @param array $properties
+     * @return string
+     */
+    protected function createInlineCssTagFromFile(string $file, array $properties): string
+    {
+        $cssInline = file_get_contents($file);
+
+        return '<style type="text/css"'
+            . ' media="' . htmlspecialchars($properties['media']) . '"'
+            . ($properties['title'] ? ' title="' . htmlspecialchars($properties['title']) . '"' : '')
+            . '>' . LF
+            . '/*<![CDATA[*/' . LF . '<!-- ' . LF
+            . $cssInline
+            . '-->' . LF . '/*]]>*/' . LF . '</style>' . LF;
     }
 }
     }
 }