vendor/contao/core-bundle/src/Resources/contao/library/Contao/Combiner.php line 256

Open in your IDE?
  1. <?php
  2. /*
  3.  * This file is part of Contao.
  4.  *
  5.  * (c) Leo Feyer
  6.  *
  7.  * @license LGPL-3.0-or-later
  8.  */
  9. namespace Contao;
  10. use ScssPhp\ScssPhp\Compiler;
  11. use ScssPhp\ScssPhp\OutputStyle;
  12. /**
  13.  * Combines .css or .js files into one single file
  14.  *
  15.  * Usage:
  16.  *
  17.  *     $combiner = new Combiner();
  18.  *
  19.  *     $combiner->add('css/style.css');
  20.  *     $combiner->add('css/fonts.scss');
  21.  *     $combiner->add('css/print.less');
  22.  *
  23.  *     echo $combiner->getCombinedFile();
  24.  */
  25. class Combiner extends System
  26. {
  27.     /**
  28.      * The .css file extension
  29.      */
  30.     const CSS '.css';
  31.     /**
  32.      * The .js file extension
  33.      */
  34.     const JS '.js';
  35.     /**
  36.      * The .scss file extension
  37.      */
  38.     const SCSS '.scss';
  39.     /**
  40.      * The .less file extension
  41.      */
  42.     const LESS '.less';
  43.     /**
  44.      * Unique file key
  45.      * @var string
  46.      */
  47.     protected $strKey '';
  48.     /**
  49.      * Operation mode
  50.      * @var string
  51.      */
  52.     protected $strMode;
  53.     /**
  54.      * Files
  55.      * @var array
  56.      */
  57.     protected $arrFiles = array();
  58.     /**
  59.      * Root dir
  60.      * @var string
  61.      */
  62.     protected $strRootDir;
  63.     /**
  64.      * Web dir relative to $this->strRootDir
  65.      * @var string
  66.      */
  67.     protected $strWebDir;
  68.     /**
  69.      * Public constructor required
  70.      */
  71.     public function __construct()
  72.     {
  73.         $container System::getContainer();
  74.         $this->strRootDir $container->getParameter('kernel.project_dir');
  75.         $this->strWebDir StringUtil::stripRootDir($container->getParameter('contao.web_dir'));
  76.     }
  77.     /**
  78.      * Add a file to the combined file
  79.      *
  80.      * @param string $strFile    The file to be added
  81.      * @param string $strVersion An optional version number
  82.      * @param string $strMedia   The media type of the file (.css only)
  83.      *
  84.      * @throws \InvalidArgumentException If $strFile is invalid
  85.      * @throws \LogicException           If different file types are mixed
  86.      */
  87.     public function add($strFile$strVersion=null$strMedia='all')
  88.     {
  89.         $strType strrchr($strFile'.');
  90.         // Check the file type
  91.         if ($strType != self::CSS && $strType != self::JS && $strType != self::SCSS && $strType != self::LESS)
  92.         {
  93.             throw new \InvalidArgumentException("Invalid file $strFile");
  94.         }
  95.         $strMode = ($strType == self::JS) ? self::JS self::CSS;
  96.         // Set the operation mode
  97.         if ($this->strMode === null)
  98.         {
  99.             $this->strMode $strMode;
  100.         }
  101.         elseif ($this->strMode != $strMode)
  102.         {
  103.             throw new \LogicException('You cannot mix different file types. Create another Combiner object instead.');
  104.         }
  105.         // Check the source file
  106.         if (!file_exists($this->strRootDir '/' $strFile))
  107.         {
  108.             // Handle public bundle resources in the contao.web_dir folder
  109.             if (file_exists($this->strRootDir '/' $this->strWebDir '/' $strFile))
  110.             {
  111.                 $strFile $this->strWebDir '/' $strFile;
  112.             }
  113.             else
  114.             {
  115.                 return;
  116.             }
  117.         }
  118.         // Prevent duplicates
  119.         if (isset($this->arrFiles[$strFile]))
  120.         {
  121.             return;
  122.         }
  123.         // Default version
  124.         if ($strVersion === null)
  125.         {
  126.             $strVersion filemtime($this->strRootDir '/' $strFile);
  127.         }
  128.         // Store the file
  129.         $arrFile = array
  130.         (
  131.             'name' => $strFile,
  132.             'version' => $strVersion,
  133.             'media' => $strMedia,
  134.             'extension' => $strType
  135.         );
  136.         $this->arrFiles[$strFile] = $arrFile;
  137.         $this->strKey .= '-f' $strFile '-v' $strVersion '-m' $strMedia;
  138.     }
  139.     /**
  140.      * Add multiple files from an array
  141.      *
  142.      * @param array  $arrFiles   An array of files to be added
  143.      * @param string $strVersion An optional version number
  144.      * @param string $strMedia   The media type of the file (.css only)
  145.      */
  146.     public function addMultiple(array $arrFiles$strVersion=null$strMedia='screen')
  147.     {
  148.         foreach ($arrFiles as $strFile)
  149.         {
  150.             $this->add($strFile$strVersion$strMedia);
  151.         }
  152.     }
  153.     /**
  154.      * Check whether files have been added
  155.      *
  156.      * @return boolean True if there are files
  157.      */
  158.     public function hasEntries()
  159.     {
  160.         return !empty($this->arrFiles);
  161.     }
  162.     /**
  163.      * Generates the files and returns the URLs.
  164.      *
  165.      * @param string $strUrl An optional URL to prepend
  166.      *
  167.      * @return array The file URLs
  168.      */
  169.     public function getFileUrls($strUrl=null)
  170.     {
  171.         if ($strUrl === null)
  172.         {
  173.             $strUrl System::getContainer()->get('contao.assets.assets_context')->getStaticUrl();
  174.         }
  175.         $return = array();
  176.         $strTarget substr($this->strMode1);
  177.         $blnDebug System::getContainer()->getParameter('kernel.debug');
  178.         foreach ($this->arrFiles as $arrFile)
  179.         {
  180.             // Compile SCSS/LESS files into temporary files
  181.             if ($arrFile['extension'] == self::SCSS || $arrFile['extension'] == self::LESS)
  182.             {
  183.                 $strPath 'assets/' $strTarget '/' str_replace('/''_'$arrFile['name']) . $this->strMode;
  184.                 if ($blnDebug || !file_exists($this->strRootDir '/' $strPath))
  185.                 {
  186.                     $objFile = new File($strPath);
  187.                     $objFile->write($this->handleScssLess(file_get_contents($this->strRootDir '/' $arrFile['name']), $arrFile));
  188.                     $objFile->close();
  189.                 }
  190.                 $return[] = $strUrl $strPath '|' $arrFile['version'];
  191.             }
  192.             else
  193.             {
  194.                 $name $arrFile['name'];
  195.                 // Strip the contao.web_dir directory prefix (see #328)
  196.                 if (strncmp($name$this->strWebDir '/'\strlen($this->strWebDir) + 1) === 0)
  197.                 {
  198.                     $name substr($name\strlen($this->strWebDir) + 1);
  199.                 }
  200.                 // Add the media query (see #7070)
  201.                 if ($this->strMode == self::CSS && $arrFile['media'] && $arrFile['media'] != 'all' && !$this->hasMediaTag($arrFile['name']))
  202.                 {
  203.                     $name .= '|' $arrFile['media'];
  204.                 }
  205.                 $return[] = $strUrl $name '|' $arrFile['version'];
  206.             }
  207.         }
  208.         return $return;
  209.     }
  210.     /**
  211.      * Generate the combined file and return its path
  212.      *
  213.      * @param string $strUrl An optional URL to prepend
  214.      *
  215.      * @return string The path to the combined file
  216.      */
  217.     public function getCombinedFile($strUrl=null)
  218.     {
  219.         if (System::getContainer()->getParameter('kernel.debug'))
  220.         {
  221.             return $this->getDebugMarkup($strUrl);
  222.         }
  223.         return $this->getCombinedFileUrl($strUrl);
  224.     }
  225.     /**
  226.      * Generates the debug markup.
  227.      *
  228.      * @param string $strUrl An optional URL to prepend
  229.      *
  230.      * @return string The debug markup
  231.      */
  232.     protected function getDebugMarkup($strUrl)
  233.     {
  234.         $return $this->getFileUrls($strUrl);
  235.         foreach ($return as $k=>$v)
  236.         {
  237.             $options StringUtil::resolveFlaggedUrl($v);
  238.             $return[$k] = $v;
  239.             if ($options->mtime)
  240.             {
  241.                 $return[$k] .= '?v=' substr(md5($options->mtime), 08);
  242.             }
  243.             if ($options->media)
  244.             {
  245.                 $return[$k] .= '" media="' $options->media;
  246.             }
  247.         }
  248.         if ($this->strMode == self::JS)
  249.         {
  250.             return implode('"></script><script src="'$return);
  251.         }
  252.         return implode('"><link rel="stylesheet" href="'$return);
  253.     }
  254.     /**
  255.      * Generate the combined file and return its path
  256.      *
  257.      * @param string $strUrl An optional URL to prepend
  258.      *
  259.      * @return string The path to the combined file
  260.      */
  261.     protected function getCombinedFileUrl($strUrl=null)
  262.     {
  263.         if ($strUrl === null)
  264.         {
  265.             $strUrl System::getContainer()->get('contao.assets.assets_context')->getStaticUrl();
  266.         }
  267.         $arrPrefix = array();
  268.         $strTarget substr($this->strMode1);
  269.         foreach ($this->arrFiles as $arrFile)
  270.         {
  271.             $arrPrefix[] = basename($arrFile['name']);
  272.         }
  273.         $strKey StringUtil::substr(implode(','$arrPrefix), 64'...') . '-' substr(md5($this->strKey), 08);
  274.         // Load the existing file
  275.         if (file_exists($this->strRootDir '/assets/' $strTarget '/' $strKey $this->strMode))
  276.         {
  277.             return $strUrl 'assets/' $strTarget '/' $strKey $this->strMode;
  278.         }
  279.         // Create the file
  280.         $objFile = new File('assets/' $strTarget '/' $strKey $this->strMode);
  281.         $objFile->truncate();
  282.         foreach ($this->arrFiles as $arrFile)
  283.         {
  284.             $content file_get_contents($this->strRootDir '/' $arrFile['name']);
  285.             // Remove UTF-8 BOM
  286.             if (strncmp($content"\xEF\xBB\xBF"3) === 0)
  287.             {
  288.                 $content substr($content3);
  289.             }
  290.             // HOOK: modify the file content
  291.             if (isset($GLOBALS['TL_HOOKS']['getCombinedFile']) && \is_array($GLOBALS['TL_HOOKS']['getCombinedFile']))
  292.             {
  293.                 foreach ($GLOBALS['TL_HOOKS']['getCombinedFile'] as $callback)
  294.                 {
  295.                     $this->import($callback[0]);
  296.                     $content $this->{$callback[0]}->{$callback[1]}($content$strKey$this->strMode$arrFile);
  297.                 }
  298.             }
  299.             if ($arrFile['extension'] == self::CSS)
  300.             {
  301.                 $content $this->handleCss($content$arrFile);
  302.             }
  303.             elseif ($arrFile['extension'] == self::SCSS || $arrFile['extension'] == self::LESS)
  304.             {
  305.                 $content $this->handleScssLess($content$arrFile);
  306.             }
  307.             $objFile->append($content);
  308.         }
  309.         unset($content);
  310.         $objFile->close();
  311.         return $strUrl 'assets/' $strTarget '/' $strKey $this->strMode;
  312.     }
  313.     /**
  314.      * Handle CSS files
  315.      *
  316.      * @param string $content The file content
  317.      * @param array  $arrFile The file array
  318.      *
  319.      * @return string The modified file content
  320.      */
  321.     protected function handleCss($content$arrFile)
  322.     {
  323.         $content $this->fixPaths($content$arrFile);
  324.         // Add the media type if there is no @media command in the code
  325.         if ($arrFile['media'] && $arrFile['media'] != 'all' && strpos($content'@media') === false)
  326.         {
  327.             $content '@media ' $arrFile['media'] . "{\n" $content "\n}";
  328.         }
  329.         return $content;
  330.     }
  331.     /**
  332.      * Handle SCSS/LESS files
  333.      *
  334.      * @param string $content The file content
  335.      * @param array  $arrFile The file array
  336.      *
  337.      * @return string The modified file content
  338.      */
  339.     protected function handleScssLess($content$arrFile)
  340.     {
  341.         $blnDebug System::getContainer()->getParameter('kernel.debug');
  342.         if ($arrFile['extension'] == self::SCSS)
  343.         {
  344.             $objCompiler = new Compiler();
  345.             $objCompiler->setImportPaths($this->strRootDir '/' \dirname($arrFile['name']));
  346.             $objCompiler->setOutputStyle(($blnDebug OutputStyle::EXPANDED OutputStyle::COMPRESSED));
  347.             if ($blnDebug)
  348.             {
  349.                 $objCompiler->setSourceMap(Compiler::SOURCE_MAP_INLINE);
  350.             }
  351.             return $this->fixPaths($objCompiler->compileString($content$this->strRootDir '/' $arrFile['name'])->getCss(), $arrFile);
  352.         }
  353.         $strPath \dirname($arrFile['name']);
  354.         $arrOptions = array
  355.         (
  356.             'strictMath' => true,
  357.             'compress' => !$blnDebug,
  358.             'import_dirs' => array($this->strRootDir '/' $strPath => $strPath)
  359.         );
  360.         $objParser = new \Less_Parser();
  361.         $objParser->SetOptions($arrOptions);
  362.         $objParser->parse($content);
  363.         return $this->fixPaths($objParser->getCss(), $arrFile);
  364.     }
  365.     /**
  366.      * Fix the paths
  367.      *
  368.      * @param string $content The file content
  369.      * @param array  $arrFile The file array
  370.      *
  371.      * @return string The modified file content
  372.      */
  373.     protected function fixPaths($content$arrFile)
  374.     {
  375.         $strName $arrFile['name'];
  376.         // Strip the contao.web_dir directory prefix
  377.         if (strpos($strName$this->strWebDir '/') === 0)
  378.         {
  379.             $strName substr($strName\strlen($this->strWebDir) + 1);
  380.         }
  381.         $strDirname \dirname($strName);
  382.         $strGlue = ($strDirname != '.') ? $strDirname '/' '';
  383.         return preg_replace_callback(
  384.             '/url\(("[^"\n]+"|\'[^\'\n]+\'|[^"\'\s()]+)\)/',
  385.             static function ($matches) use ($strDirname$strGlue)
  386.             {
  387.                 $strData $matches[1];
  388.                 if ($strData[0] == '"' || $strData[0] == "'")
  389.                 {
  390.                     $strData substr($strData1, -1);
  391.                 }
  392.                 // Skip absolute links and embedded images (see #5082)
  393.                 if ($strData[0] == '/' || $strData[0] == '#' || strncmp($strData'data:'5) === || strncmp($strData'http://'7) === || strncmp($strData'https://'8) === || strncmp($strData'assets/css3pie/'15) === 0)
  394.                 {
  395.                     return $matches[0];
  396.                 }
  397.                 // Make the paths relative to the root (see #4161)
  398.                 if (strncmp($strData'../'3) !== 0)
  399.                 {
  400.                     $strData '../../' $strGlue $strData;
  401.                 }
  402.                 else
  403.                 {
  404.                     $dir $strDirname;
  405.                     // Remove relative paths
  406.                     while (strncmp($strData'../'3) === 0)
  407.                     {
  408.                         $dir \dirname($dir);
  409.                         $strData substr($strData3);
  410.                     }
  411.                     $glue = ($dir != '.') ? $dir '/' '';
  412.                     $strData '../../' $glue $strData;
  413.                 }
  414.                 $strQuote '';
  415.                 if ($matches[1][0] == "'" || $matches[1][0] == '"')
  416.                 {
  417.                     $strQuote $matches[1][0];
  418.                 }
  419.                 if (preg_match('/[(),\s"\']/'$strData))
  420.                 {
  421.                     if ($matches[1][0] == "'")
  422.                     {
  423.                         $strData str_replace("'""\\'"$strData);
  424.                     }
  425.                     else
  426.                     {
  427.                         $strQuote '"';
  428.                         $strData str_replace('"''\"'$strData);
  429.                     }
  430.                 }
  431.                 return 'url(' $strQuote $strData $strQuote ')';
  432.             },
  433.             $content
  434.         );
  435.     }
  436.     /**
  437.      * Check if the file has a @media tag
  438.      *
  439.      * @param string $strFile
  440.      *
  441.      * @return boolean True if the file has a @media tag
  442.      */
  443.     protected function hasMediaTag($strFile)
  444.     {
  445.         $return false;
  446.         $fh fopen($this->strRootDir '/' $strFile'r');
  447.         while (($line fgets($fh)) !== false)
  448.         {
  449.             if (strpos($line'@media') !== false)
  450.             {
  451.                 $return true;
  452.                 break;
  453.             }
  454.         }
  455.         fclose($fh);
  456.         return $return;
  457.     }
  458. }
  459. class_alias(Combiner::class, 'Combiner');