system/modules/sg-elements/core/library/SgElements/SgElements.php line 44

Open in your IDE?
  1. <?php
  2. /**
  3.  * Contao Open Source CMS
  4.  *
  5.  * Copyright (c) 2005-2015 Leo Feyer
  6.  *
  7.  * @package   SgElements
  8.  * @author    Sg-Medien GmbH <info@sg-medien.com>
  9.  * @license   EULA
  10.  * @copyright Sg-Medien GmbH 2015
  11.  */
  12. /**
  13.  * Set custom namespace
  14.  */
  15. namespace SgM;
  16. use Contao\StringUtil;
  17. /**
  18.  * Class SgElements
  19.  *
  20.  * Base class for "SgElements".
  21.  *
  22.  * @package   SgElements
  23.  * @author    Sg-Medien GmbH <info@sg-medien.com>
  24.  * @copyright Sg-Medien GmbH 2015
  25.  */
  26. class SgElements
  27. {
  28.     /**
  29.      * Contao hook to add specific classes to a content element or an article.
  30.      *
  31.      * @param object $objElement The contao content element object or the article object.
  32.      * @param string $strBuffer The parsed html code of the contao content element or nothing for an article.
  33.      *
  34.      * @return string The modified html code of the contao content element
  35.      *
  36.      */
  37.     public function addClasses($objElement$strBuffer=false)
  38.     {
  39.         if('FE' !== TL_MODE)
  40.         {
  41.             return $strBuffer;
  42.         }
  43.         // get classes
  44.         $classes = array();
  45.         $cssID StringUtil::deserialize($objElement->cssIDtrue);
  46.         if(!empty($cssID)) {
  47.             $classes explode(' '$cssID[1]);
  48.         }
  49.         $startClassChecksum md5(serialize($classes));
  50.         // if contao content element
  51.         if ($strBuffer)
  52.         {
  53.             // add space class if set and not "defspace"
  54.             if ($objElement->space_class)
  55.             {
  56.                 $classes[] = $objElement->space_class;
  57.             }
  58.             // add withbg class if set
  59.             if ($objElement->withbg_class)
  60.             {
  61.                 $classes[] = 'withbg';
  62.             }
  63.             // remove duplicate classes
  64.             #$classes = array_unique($classes, SORT_REGULAR);
  65.             $endClassChecksum md5(serialize($classes));
  66.             // only if classes changed
  67.             if ($startClassChecksum!=$endClassChecksum)
  68.             {
  69.                 // try to use the real object to inject the space class
  70.                 $strCte '';
  71.                 foreach($GLOBALS['TL_CTE'] AS $node => $arrTypes)
  72.                 {
  73.                     foreach($arrTypes AS $type => $class)
  74.                     {
  75.                         if ($type == $objElement->type)
  76.                         {
  77.                             $strCte $GLOBALS['TL_CTE'][$node][$type];
  78.                         }
  79.                     }
  80.                 }
  81.                 // if real object found
  82.                 if(strlen($strCte) > 0)
  83.                 {
  84.                     $objCte = new $strCte($objElement);
  85.                     $cssID[1] = trim(implode(' '$classes));
  86.                     $objCte->cssID $cssID;
  87.                     
  88.                     try {
  89.                         $strBuffer $objCte->generate();
  90.                     } catch (\Exception $objError) {
  91.                         //
  92.                     }
  93.                 }
  94.                 // fallback: inject classes with rewriting the class attribute
  95.                 else
  96.                 {
  97.                     $preg preg_match('/class="(.*?)\"/'$strBuffer$result);
  98.                     if($preg)
  99.                     {
  100.                        $class explode(' '$result[1]);
  101.                        $classes array_unique(array_merge($classes$class));
  102.                     }
  103.                     $strBuffer preg_replace('/class="(.*?)\"/''class="'.trim(implode(' '$classes)).'"'$strBuffer);
  104.                 }
  105.             }
  106.             return $strBuffer;
  107.         }
  108.         // or if an article
  109.         // add space class if set and not "nospace"
  110.         if ($objElement->space_class && $objElement->space_class !== 'nospace')
  111.         {
  112.             $classes[] = $objElement->space_class;
  113.         }
  114.         // add withbg class if set
  115.         if ($objElement->withbg_class)
  116.         {
  117.             $classes[] = 'withbg';
  118.         }
  119.         // remove duplicate classes
  120.         $classes          array_unique($classes);
  121.         $endClassChecksum md5(serialize($classes));
  122.         // only if classes changed
  123.         if ($startClassChecksum != $endClassChecksum)
  124.         {
  125.             // write classes back
  126.             $cssID[1] = trim(implode(' '$classes));
  127.             $objElement->cssID $cssID;
  128.         }
  129.         return $strBuffer;
  130.     }
  131.     /**
  132.      * Contao hook to check if the element is a wrapper and if yes, get id of wrapper for the child elements.
  133.      *
  134.      * @param object $objElement The contao content element object.
  135.      * @param string $strBuffer The parsed html code of the contao content element.
  136.      *
  137.      * @return string The same html code of the contao content element.
  138.      *
  139.      */
  140.     public function checkWrapper($objElement$strBuffer)
  141.     {
  142.         if (!isset($GLOBALS['TL_WRAPPERS']) OR !is_array($GLOBALS['TL_WRAPPERS']['start']))
  143.         {
  144.             return $strBuffer;
  145.         }
  146.         if (in_array($objElement->type$GLOBALS['TL_WRAPPERS']['start']))
  147.         {
  148.             $GLOBALS['SG_ELEMENTS']['wrapper'][$objElement->type][] = $objElement->id;
  149.         }
  150.         if (in_array($objElement->type$GLOBALS['TL_WRAPPERS']['stop']))
  151.         {
  152.             // try to find start element belonging to current stop element
  153.             $arrIndex array_search($objElement->type$GLOBALS['TL_WRAPPERS']['stop']);
  154.             if (isset($GLOBALS['TL_WRAPPERS']['start'][$arrIndex]))
  155.             {
  156.                 $startWrapper $GLOBALS['TL_WRAPPERS']['start'][$arrIndex];
  157.                 if (isset($GLOBALS['SG_ELEMENTS']['wrapper'][$startWrapper]) AND !empty($GLOBALS['SG_ELEMENTS']['wrapper'][$startWrapper]))
  158.                 {
  159.                     // remove last start wrapper
  160.                     end($GLOBALS['SG_ELEMENTS']['wrapper'][$startWrapper]);
  161.                     $lwk key($GLOBALS['SG_ELEMENTS']['wrapper'][$startWrapper]);
  162.                     unset($GLOBALS['SG_ELEMENTS']['wrapper'][$startWrapper][$lwk]);
  163.                 }
  164.             }
  165.         }
  166.         return $strBuffer;
  167.     }
  168.     /**
  169.      * Method to get the id of current wrapper from a child element.
  170.      *
  171.      * @param string $wrapperName The name of the start wrapper.
  172.      *
  173.      * @return integer|boolean The id of the current wrapper or false if no wrapper found.
  174.      *
  175.      */
  176.     public static function getWrapperId($wrapperName)
  177.     {
  178.         if (!empty($wrapperName) AND isset($GLOBALS['SG_ELEMENTS']['wrapper'][$wrapperName]) AND !empty($GLOBALS['SG_ELEMENTS']['wrapper'][$wrapperName]))
  179.         {
  180.             return end($GLOBALS['SG_ELEMENTS']['wrapper'][$wrapperName]);
  181.         }
  182.         return false;
  183.     }
  184.     /**
  185.      * Add dynamic assets if required.
  186.      *
  187.      * @param string $assetsName The name of the assets lib defined in the config
  188.      *
  189.      * @return void
  190.      */
  191.     public static function addDynamicAssets($assetsName=false)
  192.     {
  193.         if (!$assetsName OR empty($assetsName) OR !is_string($assetsName))
  194.         {
  195.             return;
  196.         }
  197.         // add dynamic css libs
  198.         if (isset($GLOBALS['SG_ELEMENTS']['css']) AND isset($GLOBALS['SG_ELEMENTS']['css'][$assetsName]) AND is_array($GLOBALS['SG_ELEMENTS']['css'][$assetsName]))
  199.         {
  200.             foreach($GLOBALS['SG_ELEMENTS']['css'][$assetsName] AS $css)
  201.             {
  202.                 $GLOBALS['TL_CSS'][] = $css.'||static';
  203.             }
  204.         }
  205.         // add dynamic javascript libs
  206.         if (isset($GLOBALS['SG_ELEMENTS']['js']) AND isset($GLOBALS['SG_ELEMENTS']['js'][$assetsName]) AND is_array($GLOBALS['SG_ELEMENTS']['js'][$assetsName]))
  207.         {
  208.             foreach($GLOBALS['SG_ELEMENTS']['js'][$assetsName] AS $js)
  209.             {
  210.                 $GLOBALS['TL_JAVASCRIPT'][] = $js.'|static';
  211.             }
  212.         }
  213.     }
  214.     /**
  215.      * Get Vimeo-Video-HTML.
  216.      *
  217.      * @param string $vimeoID The ID of the Vimeo-Video.
  218.      * @param string $playerSize The width and the height in html attributes (e.g. width="640" height="360").
  219.      * @param string $playerColor The player color as hex code without a the hash tag (default: #00adef).
  220.      * @param boolean $showTitle Show the title of the video (default: false).
  221.      * @param boolean $showUserImage Show the user's portrait on the video (default: false).
  222.      * @param boolean $showUserByline Show the user’s byline on the video (default: false).
  223.      * @param boolean $autoplay Activate "autoplay" for video (default: false).
  224.      * @param boolean $playerID A unique player id (default: false).
  225.      * @param boolean $activateApi Activate the vimeo api (default: false).
  226.      *
  227.      * @return string The Vimeo-HTML-Code
  228.      */
  229.     public static function getVimeoVideo($vimeoID=false$playerSize=false$playerColor=false$showTitle=false$showUserImage=false$showUserByline=false$autoplay=false$playerID=false$activateApi=false)
  230.     {
  231.         // only if Vimeo-ID given
  232.         if (!$vimeoID OR empty($vimeoID) OR !is_string($vimeoID))
  233.         {
  234.             return false;
  235.         }
  236.         // video options array
  237.         $videoOptions = array();
  238.         // set player color
  239.         if (is_string($playerColor) AND !empty($playerColor))
  240.         {
  241.             $videoOptions[] = 'color='.urlencode($playerColor);
  242.         }
  243.         // show no video title
  244.         if (!$showTitle)
  245.         {
  246.             $videoOptions[] = 'title=0';
  247.         }
  248.         // show no user portrait
  249.         if (!$showUserImage)
  250.         {
  251.             $videoOptions[] = 'portrait=0';
  252.         }
  253.         // show no user byline
  254.         if (!$showUserImage)
  255.         {
  256.             $videoOptions[] = 'byline=0';
  257.         }
  258.         // autoplay?
  259.         if ($autoplay)
  260.         {
  261.             $videoOptions[] = 'autoplay=1';
  262.         }
  263.         // use a unique player id
  264.         if ($playerID)
  265.         {
  266.             $videoOptions[] = 'player_id='.urlencode($playerID);
  267.         }
  268.         // activate the vimeo api
  269.         if ($activateApi)
  270.         {
  271.             $videoOptions[] = 'api=1';
  272.         }
  273.         // return final Vimeo-HTML-Code
  274.         return '<iframe'.(($playerSize AND !empty($playerSize)) ? $playerSize '').' src="//player.vimeo.com/video/'.$vimeoID.(count($videoOptions) ? '?'.implode('&amp;'$videoOptions)  : '').'" frameborder="0" webkitallowfullscreen mozallowfullscreen allowfullscreen></iframe>';
  275.     }
  276.     /**
  277.      * Add a request string to an URL.
  278.      *
  279.      * @param string  $strRequest The string to be added
  280.      * @param array   $arrUnset An optional array of keys to unset
  281.      * @param string  $strUrl An optional url instead of the current url
  282.      * @param boolean $blnAddRef Add the referer ID
  283.      * @param boolean $blnAddCurQueries Add the current queries
  284.      *
  285.      * @return string The new URL
  286.      */
  287.     public static function addToUrl($strRequest$arrUnset=array(), $strUrl=false$blnAddRef=false$blnAddCurQueries=true)
  288.     {
  289.         $strRequest preg_replace('/^&(amp;)?/i'''$strRequest);
  290.         if ($strRequest != '' && $blnAddRef)
  291.         {
  292.             $strRequest .= '&amp;ref=' TL_REFERER_ID;
  293.         }
  294.         $queries $blnAddCurQueries preg_split('/&(amp;)?/i'\Environment::get('queryString')) : array();
  295.         // overwrite existing parameters
  296.         foreach ($queries as $k=>$v)
  297.         {
  298.             list($key) = explode('='$v);
  299.             if (in_array($key$arrUnset) OR preg_match('/(^|&(amp;)?)' preg_quote($key'/') . '=/i'$strRequest) OR empty($v))
  300.             {
  301.                 unset($queries[$k]);
  302.             }
  303.         }
  304.         $href '?';
  305.         if (!empty($queries))
  306.         {
  307.             $href .= implode('&amp;'$queries) . '&amp;';
  308.         }
  309.         return preg_replace('/((&(amp;)?)|\?)$/'''trim(((is_string($strUrl) AND !empty($strUrl)) ? $strUrl ampersand(preg_replace('/\?.*?$/'''\Environment::get('request')))).$href.str_replace(' ''%20'$strRequest)));
  310.     }
  311.     /**
  312.      * Sort a multidimensional array according to a defined field (ascending or descending).
  313.      *
  314.      * @param array $multiArray
  315.      * @param string $fieldName
  316.      * @param string $sortDirection
  317.      *
  318.      * @return array
  319.      */
  320.     public static function arrayMultiSort($multiArray$fieldName$sortDirection='asc')
  321.     {
  322.         $sortMultiArray = array();
  323.         foreach($multiArray AS $key => $data)
  324.         {
  325.             ${$fieldName}[$key] = $data[$fieldName];
  326.         }
  327.         if (${$fieldName}!==null)
  328.         {
  329.             if ($sortDirection!=='asc')
  330.             {
  331.                 arsort(${$fieldName});
  332.             }
  333.             else
  334.             {
  335.                 asort(${$fieldName});
  336.             }
  337.             foreach (${$fieldName} AS $key => $value)
  338.             {
  339.                 $sortMultiArray[] = $multiArray[$key];
  340.             }
  341.         }
  342.         return $sortMultiArray;
  343.     }
  344.     /**
  345.      * Process backend ajax requests.
  346.      *
  347.      * @param string $strAction The name of the ajax action.
  348.      * @param object $dc The data container object.
  349.      *
  350.      * @return void
  351.      */
  352.     public function ajaxRequest($strAction\DataContainer $dc)
  353.     {
  354.         switch ($strAction)
  355.         {
  356.             // file upload
  357.             case 'sg_ajaxupload':
  358.                 // get dca data
  359.                 $intId = ($dc->id!==null AND is_integer((int)$dc->id) AND (int)$dc->id>0) ? (int)$dc->id 0;
  360.                 $strTable = ($dc->table!==null AND trim($dc->table)!='' AND isset($GLOBALS['TL_DCA'][trim($dc->table)])) ? trim($dc->table) : false;
  361.                 $strField = ($strTable AND \Input::post('name')!==null AND trim(\Input::post('name'))!='') ? trim(\Input::post('name')) : false;
  362.                 $strName $strField;
  363.                 // handle the keys in "edit multiple" mode
  364.                 if ($strField!==false AND \Input::get('act') == 'editAll')
  365.                 {
  366.                     $intId = (int)preg_replace('/.*_([0-9a-zA-Z]+)$/''$1'$strField);
  367.                     $strField preg_replace('/(.*)_[0-9a-zA-Z]+$/''$1'$strField);
  368.                     $strName $strField.'_'.$intId;
  369.                 }
  370.                 // check if strField exists
  371.                 $strField = isset($GLOBALS['TL_DCA'][$strTable]['fields'][$strField]) ? $strField false;
  372.                 // if error with the dca data, die...
  373.                 if ($intId===false OR $strTable===false OR $strField===false)
  374.                 {
  375.                     // if no id given
  376.                     if ($intId===false)
  377.                     {
  378.                         $this->log('No ID given for DCA "'.($strTable===false 'undefined' $strTable).'"'__METHOD__TL_ERROR);
  379.                     }
  380.                     // otherwise...
  381.                     else
  382.                     {
  383.                         $this->log('Field "'.($strField===false 'undefined' $strField).'" does not exist in DCA "'.($strTable===false 'undefined' $strTable).'"'__METHOD__TL_ERROR);
  384.                     }
  385.                     header('HTTP/1.1 400 Bad Request');
  386.                     die('Bad Request');
  387.                 }
  388.                 // get current field value
  389.                 $objRow \Database::getInstance()->prepare("SELECT ".$strField." FROM ".$strTable." WHERE id=?")->limit(1)->execute($intId);
  390.                 // if field value not exists, die...
  391.                 if ($objRow===null OR !$objRow->count())
  392.                 {
  393.                     $this->log('Could not load value for field "'.$strField.'" in DCA "'.$strTable.'" (given ID was: '.$intId.')'__METHOD__TL_ERROR);
  394.                     header('HTTP/1.1 400 Bad Request');
  395.                     die('Bad Request');
  396.                 }
  397.                 // get current value
  398.                 $varValue deserialize($objRow->{$strField});
  399.                 // initialize the widget
  400.                 $class $GLOBALS['BE_FFL']['ajax_upload'];
  401.                 $objWidget = new $class($class::getAttributesFromDca($GLOBALS['TL_DCA'][$strTable]['fields'][$strField], $strName$varValue$strField$strTable$dc));
  402.                 // upload file
  403.                 $fileUpload self::uploadFile($objWidget);
  404.                 if ($fileUpload['error'])
  405.                 {
  406.                     $arrResponse = array
  407.                     (
  408.                         'success' => false,
  409.                         'error' => $fileUpload['error'],
  410.                         'preventRetry' => true
  411.                     );
  412.                 }
  413.                 else
  414.                 {
  415.                     $arrResponse = array
  416.                     (
  417.                         'success' => true,
  418.                         'file' => $fileUpload['file'],
  419.                         'html' => $fileUpload['html']
  420.                     );
  421.                 }
  422.                 // return json response
  423.                 echo json_encode($arrResponse); exit;
  424.                 break;
  425.         }
  426.     }
  427.     /**
  428.      * Process a file upload.
  429.      *
  430.      * @param object $objWidget The widget object of the file upload.
  431.      *
  432.      * @return array The result of the file upload.
  433.      */
  434.     public static function uploadFile(\Widget $objWidget)
  435.     {
  436.         // validate and upload file
  437.         $strFile $objWidget->validateUpload();
  438.         // default return
  439.         $arrReturn = array
  440.         (
  441.             'success' => false,
  442.             'error' => false,
  443.             'file' => false,
  444.             'html' => false
  445.         );
  446.         // check uploaded file
  447.         if ($objWidget->hasErrors())
  448.         {
  449.             $arrReturn['error'] = $objWidget->getErrorAsString();
  450.         }
  451.         else
  452.         {
  453.             // generate the file item
  454.             $generatedFile $objWidget->generateFileItem(array('path' => $strFile));
  455.             // if error on file item generation
  456.             if (!$generatedFile)
  457.             {
  458.                 $arrReturn['error'] = $GLOBALS['TL_LANG']['MSC']['ajax_upload_errors']['unknown'];
  459.             }
  460.             else
  461.             {
  462.                 $arrReturn['success'] = true;
  463.                 $arrReturn['file'] = $strFile;
  464.                 $arrReturn['html'] = $generatedFile['html'];
  465.             }
  466.         }
  467.         // return response
  468.         return $arrReturn;
  469.     }
  470.     /**
  471.      * Add the required javascript files if an ajax upload field was added to the page.
  472.      *
  473.      * @param string $strName The name of the ajax upload field.
  474.      *
  475.      * @return void
  476.      */
  477.     public function addUploadAssets($strName)
  478.     {
  479.         if (isset($GLOBALS['TL_DCA'][$strName]) AND is_array($GLOBALS['TL_DCA'][$strName]['fields']))
  480.         {
  481.             foreach($GLOBALS['TL_DCA'][$strName]['fields'] AS $name => $field)
  482.             {
  483.                 if (isset($field['inputType']) AND $field['inputType']=='ajax_upload')
  484.                 {
  485.                     // add dynamic assets
  486.                     \SgElements::addDynamicAssets('ajax_upload');
  487.                     break;
  488.                 }
  489.             }
  490.         }
  491.     }
  492.     /**
  493.      * Form validators.
  494.      *
  495.      * @param string $strRegexp The name of the validation method.
  496.      * @param string $varValue The value of the field.
  497.      * @param object $objWidget A widget object.
  498.      *
  499.      * @return boolean If validation found true, otherwise false.
  500.      */
  501.     public function validation($strRegexp$varValue\Widget $objWidget)
  502.     {
  503.         if ($strRegexp=='integer')
  504.         {
  505.             if (!preg_match('/^[0-9]+$/'$varValue))
  506.             {
  507.                 $objWidget->addError(str_replace('%v'$varValuestr_replace('%s'$objWidget->label$GLOBALS['TL_LANG']['MSC']['validationErrors']['integer'])));
  508.             }
  509.             return true;
  510.         }
  511.         else if ($strRegexp=='year')
  512.         {
  513.             if (!preg_match('/^[0-9]{4}$/'$varValue) OR (int)$varValue<1000)
  514.             {
  515.                 $objWidget->addError(str_replace('%v'$varValuestr_replace('%s'$objWidget->label$GLOBALS['TL_LANG']['MSC']['validationErrors']['year'])));
  516.             }
  517.             else if ($objWidget->year_from!==null AND (int)$varValue<(int)$objWidget->year_from)
  518.             {
  519.                 $objWidget->addError(str_replace('%e'$objWidget->year_fromstr_replace('%v'$varValuestr_replace('%s'$objWidget->label$GLOBALS['TL_LANG']['MSC']['validationErrors']['year_from']))));
  520.             }
  521.             else if ($objWidget->year_to!==null AND (int)$varValue>(int)$objWidget->year_to)
  522.             {
  523.                 $objWidget->addError(str_replace('%e'$objWidget->year_tostr_replace('%v'$varValuestr_replace('%s'$objWidget->label$GLOBALS['TL_LANG']['MSC']['validationErrors']['year_to']))));
  524.             }
  525.             return true;
  526.         }
  527.         return false;
  528.     }
  529.     /**
  530.      * Transliterate a string.
  531.      *
  532.      * @param string $string The string to transliterate.
  533.      *
  534.      * @return string|boolean The transliteraded string or false.
  535.      */
  536.     public static function transliterate($string=false)
  537.     {
  538.         if (is_string($string))
  539.         {
  540.             $transliteration_array = array
  541.             (
  542.                 'ä' => 'ae',
  543.                 'Ä' => 'Ae',
  544.                 'ë' => 'e',
  545.                 'Ë' => 'e',
  546.                 'ï' => 'i',
  547.                 'Ï' => 'I',
  548.                 'ö' => 'oe',
  549.                 'Ö' => 'Oe',
  550.                 'ü' => 'ue',
  551.                 'Ü' => 'Ue',
  552.                 'ÿ' => 'y',
  553.                 'Ÿ' => 'Y',
  554.                 'ß' => 'ss',
  555.                 'á' => 'a',
  556.                 'Á' => 'A',
  557.                 'é' => 'e',
  558.                 'É' => 'E',
  559.                 'í' => 'i',
  560.                 'Í' => 'I',
  561.                 'ó' => 'o',
  562.                 'Ó' => 'O',
  563.                 'ú' => 'u',
  564.                 'Ú' => 'U',
  565.                 'ñ' => 'n',
  566.                 'Ñ' => 'N',
  567.                 'à' => 'a',
  568.                 'À' => 'A',
  569.                 'è' => 'e',
  570.                 'È' => 'E',
  571.                 'ì' => 'i',
  572.                 'Ì' => 'I',
  573.                 'ò' => 'o',
  574.                 'Ò' => 'O',
  575.                 'ù' => 'u',
  576.                 'Ù' => 'U',
  577.                 'â' => 'a',
  578.                 'Â' => 'A',
  579.                 'ê' => 'e',
  580.                 'Ê' => 'E',
  581.                 'î' => 'i',
  582.                 'Î' => 'I',
  583.                 'ô' => 'o',
  584.                 'Ô' => 'O',
  585.                 'û' => 'u',
  586.                 'Û' => 'U',
  587.                 'æ' => 'ae',
  588.                 'Æ' => 'oe',
  589.                 'œ' => 'ae',
  590.                 'Œ' => 'oe',
  591.                 'ç' => 'c',
  592.                 'Ç' => 'C',
  593.                 'ø' => 'oe',
  594.                 'Ø' => 'Oe',
  595.                 'č' => 'c',
  596.                 'Č' => 'C',
  597.                 'ď' => 'd',
  598.                 'Ď' => 'D',
  599.                 'ě' => 'e',
  600.                 'Ě' => 'E',
  601.                 'ň' => 'n',
  602.                 'Ň' => 'N',
  603.                 'ř' => 'r',
  604.                 'Ř' => 'R',
  605.                 'š' => 's',
  606.                 'Š' => 'S',
  607.                 'ť' => 't',
  608.                 'Ť' => 'T',
  609.                 'ů' => 'u',
  610.                 'Ů' => 'U',
  611.                 'ý' => 'y',
  612.                 'Ý' => 'Y',
  613.                 'ž' => 'z',
  614.                 'Ž' => 'Z',
  615.                 'ą' => 'a',
  616.                 'Ą' => 'A',
  617.                 'ć' => 'c',
  618.                 'Ć' => 'C',
  619.                 'ę' => 'e',
  620.                 'Ę' => 'E',
  621.                 'ł' => 'l',
  622.                 'Ł' => 'L',
  623.                 'ń' => 'n',
  624.                 'Ń' => 'N',
  625.                 'ś' => 's',
  626.                 'Ś' => 'S',
  627.                 'ź' => 'z',
  628.                 'Ź' => 'Z',
  629.                 'ż' => 'z',
  630.                 'Ż' => 'Z',
  631.                 'ğ' => 'g',
  632.                 'Ğ' => 'G',
  633.                 'i' => 'i',
  634.                 'İ' => 'I',
  635.                 'ı' => 'i',
  636.                 'I' => 'I',
  637.                 'ş' => 's',
  638.                 'Ş' => 'S',
  639.                 'а' => 'a',
  640.                 'А' => 'A',
  641.                 'б' => 'b',
  642.                 'Б' => 'B',
  643.                 'в' => 'v',
  644.                 'В' => 'V',
  645.                 'г' => 'g',
  646.                 'Г' => 'G',
  647.                 'д' => 'd',
  648.                 'Д' => 'D',
  649.                 'ё' => 'jo',
  650.                 'Ё' => 'Jo',
  651.                 'Е' => 'E',
  652.                 'е' => 'e',
  653.                 'Ж' => 'Zh',
  654.                 'ж' => 'zh',
  655.                 'З' => 'Z',
  656.                 'з' => 'z',
  657.                 'И' => 'I',
  658.                 'и' => 'i',
  659.                 'Й' => 'J',
  660.                 'й' => 'j',
  661.                 'К' => 'K',
  662.                 'к' => 'k',
  663.                 'Л' => 'L',
  664.                 'л' => 'l',
  665.                 'М' => 'M',
  666.                 'м' => 'm',
  667.                 'Н' => 'N',
  668.                 'н' => 'n',
  669.                 'О' => 'O',
  670.                 'о' => 'o',
  671.                 'П' => 'P',
  672.                 'п' => 'p',
  673.                 'Р' => 'R',
  674.                 'р' => 'r',
  675.                 'С' => 'S',
  676.                 'с' => 's',
  677.                 'Т' => 'T',
  678.                 'т' => 't',
  679.                 'У' => 'U',
  680.                 'у' => 'u',
  681.                 'ф' => 'f',
  682.                 'Ф' => 'F',
  683.                 'Х' => 'H',
  684.                 'х' => 'h',
  685.                 'Ц' => 'C',
  686.                 'ц' => 'c',
  687.                 'Ч' => 'Ch',
  688.                 'ч' => 'ch',
  689.                 'Ш' => 'Sh',
  690.                 'ш' => 'sh',
  691.                 'Щ' => 'Shh',
  692.                 'щ' => 'shh',
  693.                 'Ъ' => '',
  694.                 'ъ' => '',
  695.                 'ы' => 'y',
  696.                 'Ы' => 'Y',
  697.                 'Ь' => '',
  698.                 'ь' => '',
  699.                 'Э' => 'Je',
  700.                 'э' => 'je',
  701.                 'Ю' => 'Ju',
  702.                 'ю' => 'ju',
  703.                 'Я' => 'Ja',
  704.                 'я' => 'ja',
  705.                 'Ґ' => 'G',
  706.                 'ґ' => 'g',
  707.                 'Ў' => 'U',
  708.                 'ў' => 'u',
  709.                 'Є' => 'Je',
  710.                 'є' => 'je',
  711.                 'Ї' => 'Ji',
  712.                 'ї' => 'ji',
  713.                 'І' => 'I',
  714.                 'і' => 'i'
  715.             );
  716.             return strtr($string$transliteration_array);
  717.         }
  718.         return false;
  719.     }
  720.     /**
  721.      * Convert a string to a valid alias.
  722.      *
  723.      * @param string $string The string to convert.
  724.      * @param boolean $strtolower Convert the string to lowercase?
  725.      * @param integer|boolean $length Limit the length of the string?
  726.      *
  727.      * @return string|boolean The converted string.
  728.      */
  729.     public static function strToValidAlias($string=false$strtolower=true$length=200)
  730.     {
  731.         $string self::transliterate(preg_replace('/(\\r|\\n)+/s'' '$string));
  732.         if (is_string($string))
  733.         {
  734.             $replace_from = array
  735.             (
  736.                 '/(–|–|‐|‐|‐|‐|\s|\/|\\\|,|;|:|\*|#|\+)+/',
  737.                 '/([^\.^a-z^0-9^-])/i',
  738.                 '/-+/',
  739.                 '/-+$/',
  740.                 '/^-+/'
  741.             );
  742.             $replace_to = array
  743.             (
  744.                 '-',
  745.                 '',
  746.                 '-',
  747.                 '',
  748.                 ''
  749.             );
  750.             $string preg_replace($replace_from$replace_totrim($string));
  751.             $string $strtolower strtolower($string) : $string;
  752.             return ($length AND is_numeric($length)) ? substr($string0$length) : $string;
  753.         }
  754.         return false;
  755.     }
  756.     /**
  757.      * Convert a string to a valid file name.
  758.      *
  759.      * @param string $string The string to convert.
  760.      * @param integer|boolean $length Limit the length of the string?
  761.      *
  762.      * @return string|boolean The converted string.
  763.      */
  764.     public static function strToValidFilename($string=false$length=200)
  765.     {
  766.         if (is_string($string))
  767.         {
  768.             $string_array = @explode('.'$string);
  769.             if (is_array($string_array) AND count($string_array))
  770.             {
  771.                 foreach($string_array AS $k => $s)
  772.                 {
  773.                         $string_array[$k] = ($ps self::strToValidAlias($s)) ? $ps '';
  774.                 }
  775.                 $string = @implode('.'$string_array);
  776.                 if ($length AND is_numeric($length))
  777.                 {
  778.                     if (($c=count($string_array))>1)
  779.                     {
  780.                         $ext $string_array[$c-1];
  781.                         unset($string_array[$c-1]);
  782.                         $string substr(@implode('.'$string_array), 0, (($nl=$length-(strlen($ext)+1))>$nl 0)).'.'.$ext;
  783.                     }
  784.                     return substr($string0$length);
  785.                 }
  786.                 return $string;
  787.             }
  788.         }
  789.         return false;
  790.     }
  791.     /**
  792.      * Get all style sheet file paths (even the combined)
  793.      *
  794.      * @return array The style sheet files.
  795.      */
  796.     public static function getStyleSheetFiles()
  797.     {
  798.         $cssFiles = array();
  799.         // new combiner
  800.         $objCombiner = new \Combiner();
  801.         // add the CSS framework style sheets
  802.         if (!empty($GLOBALS['TL_FRAMEWORK_CSS']) AND is_array($GLOBALS['TL_FRAMEWORK_CSS']))
  803.         {
  804.             foreach (array_unique($GLOBALS['TL_FRAMEWORK_CSS']) AS $stylesheet)
  805.             {
  806.                 if (!\Config::get('debugMode'))
  807.                 {
  808.                     $objCombiner->add($stylesheet);
  809.                 }
  810.                 else
  811.                 {
  812.                     $cssFiles[] = \Controller::addStaticUrlTo($stylesheet);
  813.                 }
  814.             }
  815.         }
  816.         // add the internal style sheets
  817.         if (!empty($GLOBALS['TL_CSS']) AND is_array($GLOBALS['TL_CSS']))
  818.         {
  819.             foreach (array_unique($GLOBALS['TL_CSS']) AS $stylesheet)
  820.             {
  821.                 $options \String::resolveFlaggedUrl($stylesheet);
  822.                 if ($options->static AND !\Config::get('debugMode'))
  823.                 {
  824.                     $objCombiner->add($stylesheetfilemtime(TL_ROOT '/' $stylesheet), $options->media);
  825.                 }
  826.                 else
  827.                 {
  828.                     $cssFiles[] = \Controller::addStaticUrlTo($stylesheet);
  829.                 }
  830.             }
  831.         }
  832.         // add the user style sheets
  833.         if (!empty($GLOBALS['TL_USER_CSS']) AND is_array($GLOBALS['TL_USER_CSS']))
  834.         {
  835.             foreach (array_unique($GLOBALS['TL_USER_CSS']) AS $stylesheet)
  836.             {
  837.                 $options \String::resolveFlaggedUrl($stylesheet);
  838.                 if ($options->static AND !\Config::get('debugMode'))
  839.                 {
  840.                     if ($options->mtime === null)
  841.                     {
  842.                         $options->mtime filemtime(TL_ROOT '/' $stylesheet);
  843.                     }
  844.                     $objCombiner->add($stylesheet$options->mtime$options->media);
  845.                 }
  846.                 else
  847.                 {
  848.                     $cssFiles[] = \Controller::addStaticUrlTo($stylesheet);
  849.                 }
  850.             }
  851.         }
  852.         // get the aggregated style sheet file
  853.         if ($objCombiner->hasEntries())
  854.         {
  855.             $cssFiles[] = $objCombiner->getCombinedFile();
  856.         }
  857.         // return unique style sheet files
  858.         return array_unique($cssFiles);
  859.     }
  860.     /**
  861.      * Get all inline css
  862.      *
  863.      * @return string The inline css.
  864.      */
  865.     public static function getInlineCSS()
  866.     {
  867.         $inlineCss '';
  868.         foreach(self::getStyleSheetFiles() AS $cssFile)
  869.         {
  870.             if (file_exists(TL_ROOT.'/'.$cssFile))
  871.             {
  872.                 $inlineCss .= file_get_contents(TL_ROOT.'/'.$cssFile);
  873.             }
  874.         }
  875.         $inlineCss trim($inlineCss);
  876.         if (!empty($inlineCss))
  877.         {
  878.             $inlineCss '<style>'.$inlineCss.'</style>';
  879.         }
  880.         return $inlineCss;
  881.     }
  882. }