vendor/contao/core-bundle/src/Resources/contao/library/Contao/Widget.php line 608

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 Contao\Database\Result;
  11. use Doctrine\DBAL\Types\Types;
  12. /**
  13.  * Generates and validates form fields
  14.  *
  15.  * The class functions as abstract parent class for all widget classes and
  16.  * provides methods to generate the form field markup and to validate the form
  17.  * field input.
  18.  *
  19.  * Usage:
  20.  *
  21.  *     $widget = new TextField();
  22.  *     $widget->name = 'test';
  23.  *     $widget->label = 'Test';
  24.  *
  25.  *     if ($_POST)
  26.  *     {
  27.  *         $widget->validate();
  28.  *
  29.  *         if (!$widget->hasErrors())
  30.  *         {
  31.  *             echo $widget->value;
  32.  *         }
  33.  *     }
  34.  *
  35.  * @property string        $id                 The field ID
  36.  * @property string        $type               the field type
  37.  * @property string        $name               the field name
  38.  * @property string        $label              The field label
  39.  * @property mixed         $value              The field value
  40.  * @property string        $class              One or more CSS classes
  41.  * @property string        $prefix             The CSS class prefix
  42.  * @property string        $template           The template name
  43.  * @property string        $wizard             The field wizard markup
  44.  * @property string        $alt                The alternative text
  45.  * @property string        $style              The style attribute
  46.  * @property string        $accesskey          The key to focus the field
  47.  * @property integer       $tabindex           The tabindex of the field
  48.  * @property boolean       $disabled           Adds the disabled attribute
  49.  * @property boolean       $readonly           Adds the readonly attribute
  50.  * @property boolean       $autofocus          Adds the autofocus attribute
  51.  * @property boolean       $required           Adds the required attribute
  52.  * @property string        $onblur             The blur event
  53.  * @property string        $onchange           The change event
  54.  * @property string        $onclick            The click event
  55.  * @property string        $ondblclick         The double click event
  56.  * @property string        $onfocus            The focus event
  57.  * @property string        $onmousedown        The mouse down event
  58.  * @property string        $onmousemove        The mouse move event
  59.  * @property string        $onmouseout         The mouse out event
  60.  * @property string        $onmouseover        The mouse over event
  61.  * @property string        $onmouseup          The mouse up event
  62.  * @property string        $onkeydown          The key down event
  63.  * @property string        $onkeypress         The key press event
  64.  * @property string        $onkeyup            The key up event
  65.  * @property string        $onselect           The select event
  66.  * @property boolean       $mandatory          The field value must not be empty
  67.  * @property boolean       $nospace            Do not allow whitespace characters
  68.  * @property boolean       $allowHtml          Allow HTML tags in the field value
  69.  * @property boolean       $storeFile          Store uploaded files in a given folder
  70.  * @property boolean       $useHomeDir         Store uploaded files in the user's home directory
  71.  * @property boolean       $trailingSlash      Add or remove a trailing slash
  72.  * @property boolean       $spaceToUnderscore  Convert spaces to underscores
  73.  * @property boolean       $doNotTrim          Do not trim the user input
  74.  * @property string        $forAttribute       The "for" attribute
  75.  * @property DataContainer $dataContainer      The data container object
  76.  * @property Result        $activeRecord       The active record
  77.  * @property string        $mandatoryField     The "mandatory field" label
  78.  * @property string        $customTpl          A custom template name
  79.  * @property string        $slabel             The submit button label
  80.  * @property boolean       $preserveTags       Preserve HTML tags
  81.  * @property boolean       $decodeEntities     Decode HTML entities
  82.  * @property boolean       $useRawRequestData  Use the raw request data from the Symfony request
  83.  * @property integer       $minlength          The minimum length
  84.  * @property integer       $maxlength          The maximum length
  85.  * @property integer       $minval             The minimum value
  86.  * @property integer       $maxval             The maximum value
  87.  * @property integer       $rgxp               The regular expression name
  88.  * @property boolean       $isHexColor         The field value is a hex color
  89.  * @property string        $strTable           The table name
  90.  * @property string        $strField           The field name
  91.  * @property string        $xlabel
  92.  * @property string        $customRgxp
  93.  * @property string        $errorMsg
  94.  * @property integer       $currentRecord
  95.  * @property integer       $rowClass
  96.  * @property integer       $rowClassConfirm
  97.  * @property integer       $storeValues
  98.  * @property boolean       $includeBlankOption
  99.  * @property string        $blankOptionLabel
  100.  */
  101. abstract class Widget extends Controller
  102. {
  103.     use TemplateInheritance;
  104.     /**
  105.      * Id
  106.      * @var integer
  107.      */
  108.     protected $strId;
  109.     /**
  110.      * Name
  111.      * @var string
  112.      */
  113.     protected $strName;
  114.     /**
  115.      * Label
  116.      * @var string
  117.      */
  118.     protected $strLabel;
  119.     /**
  120.      * Value
  121.      * @var mixed
  122.      */
  123.     protected $varValue;
  124.     /**
  125.      * Input callback
  126.      * @var callable
  127.      */
  128.     protected $inputCallback;
  129.     /**
  130.      * CSS class
  131.      * @var string
  132.      */
  133.     protected $strClass;
  134.     /**
  135.      * CSS class prefix
  136.      * @var string
  137.      */
  138.     protected $strPrefix;
  139.     /**
  140.      * Wizard
  141.      * @var string
  142.      */
  143.     protected $strWizard;
  144.     /**
  145.      * Errors
  146.      * @var array
  147.      */
  148.     protected $arrErrors = array();
  149.     /**
  150.      * Attributes
  151.      * @var array
  152.      */
  153.     protected $arrAttributes = array();
  154.     /**
  155.      * Configuration
  156.      * @var array
  157.      */
  158.     protected $arrConfiguration = array();
  159.     /**
  160.      * Options
  161.      * @var array
  162.      */
  163.     protected $arrOptions = array();
  164.     /**
  165.      * Submit indicator
  166.      * @var boolean
  167.      */
  168.     protected $blnSubmitInput false;
  169.     /**
  170.      * For attribute indicator
  171.      * @var boolean
  172.      */
  173.     protected $blnForAttribute false;
  174.     /**
  175.      * Data container
  176.      * @var object
  177.      */
  178.     protected $objDca;
  179.     /**
  180.      * Initialize the object
  181.      *
  182.      * @param array $arrAttributes An optional attributes array
  183.      */
  184.     public function __construct($arrAttributes=null)
  185.     {
  186.         parent::__construct();
  187.         $this->addAttributes($arrAttributes);
  188.     }
  189.     /**
  190.      * Set an object property
  191.      *
  192.      * @param string $strKey   The property name
  193.      * @param mixed  $varValue The property value
  194.      */
  195.     public function __set($strKey$varValue)
  196.     {
  197.         switch ($strKey)
  198.         {
  199.             case 'id':
  200.                 $this->strId $varValue;
  201.                 break;
  202.             case 'name':
  203.                 $this->strName $varValue;
  204.                 break;
  205.             case 'label':
  206.                 $this->strLabel $varValue;
  207.                 break;
  208.             case 'value':
  209.                 if (\is_float($varValue))
  210.                 {
  211.                     // Prevent exponential notations (see #5296)
  212.                     $this->varValue StringUtil::numberToString($varValue);
  213.                 }
  214.                 else
  215.                 {
  216.                     $this->varValue StringUtil::deserialize($varValue);
  217.                 }
  218.                 // Decrypt the value if it is encrypted
  219.                 if ($this->arrConfiguration['encrypt'] ?? null)
  220.                 {
  221.                     $this->varValue Encryption::decrypt($this->varValue);
  222.                 }
  223.                 break;
  224.             case 'class':
  225.                 if ($varValue && strpos($this->strClass ?? ''$varValue) === false)
  226.                 {
  227.                     $this->strClass trim($this->strClass ' ' $varValue);
  228.                 }
  229.                 break;
  230.             case 'prefix':
  231.                 $this->strPrefix $varValue;
  232.                 break;
  233.             case 'template':
  234.                 $this->strTemplate $varValue;
  235.                 break;
  236.             case 'wizard':
  237.                 $this->strWizard $varValue;
  238.                 break;
  239.             case 'autocomplete':
  240.             case 'autocorrect':
  241.             case 'autocapitalize':
  242.             case 'spellcheck':
  243.                 if (\is_bool($varValue))
  244.                 {
  245.                     $varValue $varValue 'on' 'off';
  246.                 }
  247.                 // no break
  248.             case 'alt':
  249.             case 'style':
  250.             case 'accesskey':
  251.             case 'onblur':
  252.             case 'onchange':
  253.             case 'onclick':
  254.             case 'ondblclick':
  255.             case 'onfocus':
  256.             case 'onmousedown':
  257.             case 'onmousemove':
  258.             case 'onmouseout':
  259.             case 'onmouseover':
  260.             case 'onmouseup':
  261.             case 'onkeydown':
  262.             case 'onkeypress':
  263.             case 'onkeyup':
  264.             case 'onselect':
  265.                 $this->arrAttributes[$strKey] = $varValue;
  266.                 break;
  267.             case 'tabindex':
  268.                 if ($varValue 0)
  269.                 {
  270.                     trigger_deprecation('contao/core-bundle''4.12''Using a tabindex value greater than 0 has been deprecated and will no longer work in Contao 5.0.');
  271.                     $this->arrAttributes['tabindex'] = $varValue;
  272.                 }
  273.                 break;
  274.             case 'disabled':
  275.             case 'readonly':
  276.                 $this->blnSubmitInput $varValue false true;
  277.                 // no break
  278.             case 'autofocus':
  279.                 if ($varValue)
  280.                 {
  281.                     $this->arrAttributes[$strKey] = $strKey;
  282.                 }
  283.                 else
  284.                 {
  285.                     unset($this->arrAttributes[$strKey]);
  286.                 }
  287.                 break;
  288.             case 'required':
  289.                 if ($varValue)
  290.                 {
  291.                     $this->strClass trim($this->strClass ' mandatory');
  292.                 }
  293.                 // no break
  294.             case 'mandatory':
  295.             case 'nospace':
  296.             case 'allowHtml':
  297.             case 'storeFile':
  298.             case 'useHomeDir':
  299.             case 'storeValues':
  300.             case 'trailingSlash':
  301.             case 'spaceToUnderscore':
  302.             case 'doNotTrim':
  303.             case 'useRawRequestData':
  304.                 $this->arrConfiguration[$strKey] = $varValue true false;
  305.                 break;
  306.             case 'forAttribute':
  307.                 $this->blnForAttribute $varValue;
  308.                 break;
  309.             case 'dataContainer':
  310.                 $this->objDca $varValue;
  311.                 break;
  312.             case strncmp($strKey'ng-'3) === 0:
  313.             case strncmp($strKey'data-'5) === 0:
  314.                 $this->arrAttributes[$strKey] = $varValue;
  315.                 break;
  316.             default:
  317.                 $this->arrConfiguration[$strKey] = $varValue;
  318.                 break;
  319.         }
  320.     }
  321.     /**
  322.      * Return an object property
  323.      *
  324.      * @param string $strKey The property name
  325.      *
  326.      * @return string The property value
  327.      */
  328.     public function __get($strKey)
  329.     {
  330.         switch ($strKey)
  331.         {
  332.             case 'id':
  333.                 return $this->strId;
  334.             case 'name':
  335.                 return $this->strName;
  336.             case 'label':
  337.                 return $this->strLabel;
  338.             case 'value':
  339.                 // Encrypt the value
  340.                 if (isset($this->arrConfiguration['encrypt']) && $this->arrConfiguration['encrypt'])
  341.                 {
  342.                     return Encryption::encrypt($this->varValue);
  343.                 }
  344.                 if ($this->varValue === '')
  345.                 {
  346.                     return $this->getEmptyStringOrNull();
  347.                 }
  348.                 return $this->varValue;
  349.             case 'class':
  350.                 return $this->strClass;
  351.             case 'prefix':
  352.                 return $this->strPrefix;
  353.             case 'template':
  354.                 return $this->strTemplate;
  355.             case 'wizard':
  356.                 return $this->strWizard;
  357.             case 'required':
  358.                 return $this->arrConfiguration[$strKey] ?? null;
  359.             case 'forAttribute':
  360.                 return $this->blnForAttribute;
  361.             case 'dataContainer':
  362.                 return $this->objDca;
  363.             case 'activeRecord':
  364.                 return $this->objDca->activeRecord;
  365.             default:
  366.                 if (isset($this->arrAttributes[$strKey]))
  367.                 {
  368.                     return $this->arrAttributes[$strKey];
  369.                 }
  370.                 if (isset($this->arrConfiguration[$strKey]))
  371.                 {
  372.                     return $this->arrConfiguration[$strKey];
  373.                 }
  374.                 break;
  375.         }
  376.         return parent::__get($strKey);
  377.     }
  378.     /**
  379.      * Check whether an object property exists
  380.      *
  381.      * @param string $strKey The property name
  382.      *
  383.      * @return boolean True if the property exists
  384.      */
  385.     public function __isset($strKey)
  386.     {
  387.         switch ($strKey)
  388.         {
  389.             case 'id':
  390.                 return isset($this->strId);
  391.             case 'name':
  392.                 return isset($this->strName);
  393.             case 'label':
  394.                 return isset($this->strLabel);
  395.             case 'value':
  396.                 return isset($this->varValue);
  397.             case 'class':
  398.                 return isset($this->strClass);
  399.             case 'template':
  400.                 return isset($this->strTemplate);
  401.             case 'wizard':
  402.                 return isset($this->strWizard);
  403.             case 'required':
  404.                 return isset($this->arrConfiguration[$strKey]);
  405.             case 'forAttribute':
  406.                 return isset($this->blnForAttribute);
  407.             case 'dataContainer':
  408.                 return isset($this->objDca);
  409.             case 'activeRecord':
  410.                 return isset($this->objDca->activeRecord);
  411.             default:
  412.                 return isset($this->arrAttributes[$strKey]) || isset($this->arrConfiguration[$strKey]);
  413.         }
  414.     }
  415.     /**
  416.      * Add an attribute
  417.      *
  418.      * @param string $strName  The attribute name
  419.      * @param mixed  $varValue The attribute value
  420.      */
  421.     public function addAttribute($strName$varValue)
  422.     {
  423.         $this->arrAttributes[$strName] = $varValue;
  424.     }
  425.     /**
  426.      * Add an error message
  427.      *
  428.      * @param string $strError The error message
  429.      */
  430.     public function addError($strError)
  431.     {
  432.         $this->class 'error';
  433.         $this->arrErrors[] = $strError;
  434.     }
  435.     /**
  436.      * Return true if the widget has errors
  437.      *
  438.      * @return boolean True if there are errors
  439.      */
  440.     public function hasErrors()
  441.     {
  442.         return !empty($this->arrErrors);
  443.     }
  444.     /**
  445.      * Return the errors array
  446.      *
  447.      * @return array An array of error messages
  448.      */
  449.     public function getErrors()
  450.     {
  451.         return $this->arrErrors;
  452.     }
  453.     /**
  454.      * Return a particular error as string
  455.      *
  456.      * @param integer $intIndex The message index
  457.      *
  458.      * @return string The corresponding error message
  459.      */
  460.     public function getErrorAsString($intIndex=0)
  461.     {
  462.         return $this->arrErrors[$intIndex];
  463.     }
  464.     /**
  465.      * Return all errors as string separated by a given separator
  466.      *
  467.      * @param string $strSeparator An optional separator (defaults to "<br>")
  468.      *
  469.      * @return string The error messages string
  470.      */
  471.     public function getErrorsAsString($strSeparator=null)
  472.     {
  473.         if ($strSeparator === null)
  474.         {
  475.             $strSeparator '<br' $this->strTagEnding "\n";
  476.         }
  477.         return $this->hasErrors() ? implode($strSeparator$this->arrErrors) : '';
  478.     }
  479.     /**
  480.      * Return a particular error as HTML string
  481.      *
  482.      * @param integer $intIndex The message index
  483.      *
  484.      * @return string The HTML markup of the corresponding error message
  485.      */
  486.     public function getErrorAsHTML($intIndex=0)
  487.     {
  488.         return $this->hasErrors() ? sprintf('<p class="%s">%s</p>', ((TL_MODE == 'BE') ? 'tl_error tl_tip' 'error'), $this->arrErrors[$intIndex]) : '';
  489.     }
  490.     /**
  491.      * Return true if the widgets submits user input
  492.      *
  493.      * @return boolean True if the widget submits user input
  494.      */
  495.     public function submitInput()
  496.     {
  497.         return $this->blnSubmitInput;
  498.     }
  499.     /**
  500.      * Parse the template file and return it as string
  501.      *
  502.      * @param array $arrAttributes An optional attributes array
  503.      *
  504.      * @return string The template markup
  505.      */
  506.     public function parse($arrAttributes=null)
  507.     {
  508.         if (!$this->strTemplate)
  509.         {
  510.             return '';
  511.         }
  512.         $this->addAttributes($arrAttributes);
  513.         $this->mandatoryField $GLOBALS['TL_LANG']['MSC']['mandatory'];
  514.         if ($this->customTpl)
  515.         {
  516.             $request System::getContainer()->get('request_stack')->getCurrentRequest();
  517.             // Use the custom template unless it is a back end request
  518.             if (!$request || !System::getContainer()->get('contao.routing.scope_matcher')->isBackendRequest($request))
  519.             {
  520.                 $this->strTemplate $this->customTpl;
  521.             }
  522.         }
  523.         $strBuffer $this->inherit();
  524.         // HOOK: add custom parse filters (see #5553)
  525.         if (isset($GLOBALS['TL_HOOKS']['parseWidget']) && \is_array($GLOBALS['TL_HOOKS']['parseWidget']))
  526.         {
  527.             foreach ($GLOBALS['TL_HOOKS']['parseWidget'] as $callback)
  528.             {
  529.                 $this->import($callback[0]);
  530.                 $strBuffer $this->{$callback[0]}->{$callback[1]}($strBuffer$this);
  531.             }
  532.         }
  533.         return $strBuffer;
  534.     }
  535.     /**
  536.      * Generate the label and return it as string
  537.      *
  538.      * @return string The label markup
  539.      */
  540.     public function generateLabel()
  541.     {
  542.         if (!$this->strLabel)
  543.         {
  544.             return '';
  545.         }
  546.         return sprintf(
  547.             '<label%s%s>%s%s%s</label>',
  548.             ($this->blnForAttribute ' for="ctrl_' $this->strId '"' ''),
  549.             ($this->strClass ' class="' $this->strClass '"' ''),
  550.             ($this->mandatory '<span class="invisible">' $GLOBALS['TL_LANG']['MSC']['mandatory'] . ' </span>' ''),
  551.             $this->strLabel,
  552.             ($this->mandatory '<span class="mandatory">*</span>' '')
  553.         );
  554.     }
  555.     /**
  556.      * Generate the widget and return it as string
  557.      *
  558.      * @return string The widget markup
  559.      */
  560.     abstract public function generate();
  561.     /**
  562.      * Generate the widget with error message and return it as string
  563.      *
  564.      * @param boolean $blnSwitchOrder If true, the error message will be shown below the field
  565.      *
  566.      * @return string The form field markup
  567.      */
  568.     public function generateWithError($blnSwitchOrder=false)
  569.     {
  570.         $strWidget $this->generate();
  571.         $strError $this->getErrorAsHTML();
  572.         return $blnSwitchOrder $strWidget $strError $strError $strWidget;
  573.     }
  574.     /**
  575.      * Return all attributes as string
  576.      *
  577.      * @param array $arrStrip An optional array with attributes to strip
  578.      *
  579.      * @return string The attributes string
  580.      */
  581.     public function getAttributes($arrStrip=array())
  582.     {
  583.         $strAttributes '';
  584.         foreach (array_keys($this->arrAttributes) as $strKey)
  585.         {
  586.             if (!\in_array($strKey$arrStrip))
  587.             {
  588.                 $strAttributes .= $this->getAttribute($strKey);
  589.             }
  590.         }
  591.         return $strAttributes;
  592.     }
  593.     /**
  594.      * Return a single attribute
  595.      *
  596.      * @param string $strKey The attribute name
  597.      *
  598.      * @return string The attribute markup
  599.      */
  600.     public function getAttribute($strKey)
  601.     {
  602.         if (!isset($this->arrAttributes[$strKey]))
  603.         {
  604.             return '';
  605.         }
  606.         $varValue $this->arrAttributes[$strKey];
  607.         // Prevent the autofocus attribute from being added multiple times (see #8281)
  608.         if ($strKey == 'autofocus')
  609.         {
  610.             unset($this->arrAttributes[$strKey]);
  611.         }
  612.         if ($strKey == 'disabled' || $strKey == 'readonly' || $strKey == 'required' || $strKey == 'autofocus' || $strKey == 'multiple')
  613.         {
  614.             return ' ' $strKey;
  615.         }
  616.         if ('' !== (string) $varValue)
  617.         {
  618.             return ' ' $strKey '="' StringUtil::specialchars($varValue) . '"';
  619.         }
  620.         return '';
  621.     }
  622.     /**
  623.      * Set a callback to fetch the widget input instead of using getPost()
  624.      *
  625.      * @param callable|null $callback The callback
  626.      *
  627.      * @return $this The widget object
  628.      */
  629.     public function setInputCallback(callable $callback=null)
  630.     {
  631.         $this->inputCallback $callback;
  632.         return $this;
  633.     }
  634.     /**
  635.      * Validate the user input and set the value
  636.      */
  637.     public function validate()
  638.     {
  639.         $varValue $this->validator($this->getPost($this->strName));
  640.         if ($this->hasErrors())
  641.         {
  642.             $this->class 'error';
  643.         }
  644.         $this->varValue $varValue;
  645.     }
  646.     /**
  647.      * Find and return a $_POST variable
  648.      *
  649.      * @param string $strKey The variable name
  650.      *
  651.      * @return mixed The variable value
  652.      */
  653.     protected function getPost($strKey)
  654.     {
  655.         if (\is_callable($this->inputCallback))
  656.         {
  657.             return ($this->inputCallback)();
  658.         }
  659.         if ($this->useRawRequestData === true)
  660.         {
  661.             $request System::getContainer()->get('request_stack')->getCurrentRequest();
  662.             return $request->request->get($strKey);
  663.         }
  664.         $strMethod $this->allowHtml 'postHtml' 'post';
  665.         if ($this->preserveTags)
  666.         {
  667.             $strMethod 'postRaw';
  668.         }
  669.         // Support arrays (thanks to Andreas Schempp)
  670.         $arrParts explode('['str_replace(']''', (string) $strKey));
  671.         $varValue Input::$strMethod(array_shift($arrParts), $this->decodeEntities);
  672.         foreach ($arrParts as $part)
  673.         {
  674.             if (!\is_array($varValue))
  675.             {
  676.                 break;
  677.             }
  678.             $varValue $varValue[$part] ?? null;
  679.         }
  680.         return $varValue;
  681.     }
  682.     /**
  683.      * Recursively validate an input variable
  684.      *
  685.      * @param mixed $varInput The user input
  686.      *
  687.      * @return mixed The original or modified user input
  688.      */
  689.     protected function validator($varInput)
  690.     {
  691.         if (\is_array($varInput))
  692.         {
  693.             foreach ($varInput as $k=>$v)
  694.             {
  695.                 $varInput[$k] = $this->validator($v);
  696.             }
  697.             return $varInput;
  698.         }
  699.         if (!$this->doNotTrim && \is_string($varInput))
  700.         {
  701.             $varInput trim($varInput);
  702.         }
  703.         if ((string) $varInput === '')
  704.         {
  705.             if (!$this->mandatory)
  706.             {
  707.                 return '';
  708.             }
  709.             if (!$this->strLabel)
  710.             {
  711.                 $this->addError($GLOBALS['TL_LANG']['ERR']['mdtryNoLabel']);
  712.             }
  713.             else
  714.             {
  715.                 $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['mandatory'], $this->strLabel));
  716.             }
  717.         }
  718.         if ($this->minlength && $varInput && mb_strlen($varInput) < $this->minlength)
  719.         {
  720.             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['minlength'], $this->strLabel$this->minlength));
  721.         }
  722.         if ($this->maxlength && $varInput && mb_strlen($varInput) > $this->maxlength)
  723.         {
  724.             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['maxlength'], $this->strLabel$this->maxlength));
  725.         }
  726.         if ($this->minval && is_numeric($varInput) && $varInput $this->minval)
  727.         {
  728.             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['minval'], $this->strLabel$this->minval));
  729.         }
  730.         if ($this->maxval && is_numeric($varInput) && $varInput $this->maxval)
  731.         {
  732.             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['maxval'], $this->strLabel$this->maxval));
  733.         }
  734.         if ($this->rgxp)
  735.         {
  736.             switch ($this->rgxp)
  737.             {
  738.                 case strncmp($this->rgxp'digit_'6) === 0:
  739.                     // Special validation rule for style sheets
  740.                     $textual explode('_'$this->rgxp);
  741.                     array_shift($textual);
  742.                     if (\in_array($varInput$textual) || strncmp($varInput'$'1) === 0)
  743.                     {
  744.                         break;
  745.                     }
  746.                     // no break
  747.                 case 'digit':
  748.                     // Support decimal commas and convert them automatically (see #3488)
  749.                     if (substr_count($varInput',') == && strpos($varInput'.') === false)
  750.                     {
  751.                         $varInput str_replace(',''.'$varInput);
  752.                     }
  753.                     if (!Validator::isNumeric($varInput))
  754.                     {
  755.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['digit'], $this->strLabel));
  756.                     }
  757.                     break;
  758.                 case 'natural':
  759.                     if (!Validator::isNatural($varInput))
  760.                     {
  761.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['natural'], $this->strLabel));
  762.                     }
  763.                     break;
  764.                 case 'alpha':
  765.                     if (!Validator::isAlphabetic($varInput))
  766.                     {
  767.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['alpha'], $this->strLabel));
  768.                     }
  769.                     break;
  770.                 case 'alnum':
  771.                     if (!Validator::isAlphanumeric($varInput))
  772.                     {
  773.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['alnum'], $this->strLabel));
  774.                     }
  775.                     break;
  776.                 case 'extnd':
  777.                     if (!Validator::isExtendedAlphanumeric(html_entity_decode($varInput)))
  778.                     {
  779.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['extnd'], $this->strLabel));
  780.                     }
  781.                     break;
  782.                 case 'date':
  783.                     if (!Validator::isDate($varInput))
  784.                     {
  785.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['date'], Date::getInputFormat(Date::getNumericDateFormat())));
  786.                     }
  787.                     else
  788.                     {
  789.                         // Validate the date (see #5086)
  790.                         try
  791.                         {
  792.                             new Date($varInputDate::getNumericDateFormat());
  793.                         }
  794.                         catch (\OutOfBoundsException $e)
  795.                         {
  796.                             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['invalidDate'], $varInput));
  797.                         }
  798.                     }
  799.                     break;
  800.                 case 'time':
  801.                     if (!Validator::isTime($varInput))
  802.                     {
  803.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['time'], Date::getInputFormat(Date::getNumericTimeFormat())));
  804.                     }
  805.                     break;
  806.                 case 'datim':
  807.                     if (!Validator::isDatim($varInput))
  808.                     {
  809.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['dateTime'], Date::getInputFormat(Date::getNumericDatimFormat())));
  810.                     }
  811.                     else
  812.                     {
  813.                         // Validate the date (see #5086)
  814.                         try
  815.                         {
  816.                             new Date($varInputDate::getNumericDatimFormat());
  817.                         }
  818.                         catch (\OutOfBoundsException $e)
  819.                         {
  820.                             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['invalidDate'], $varInput));
  821.                         }
  822.                     }
  823.                     break;
  824.                 case 'friendly':
  825.                     list ($strName$varInput) = StringUtil::splitFriendlyEmail($varInput);
  826.                     // no break
  827.                 case 'email':
  828.                     if (!Validator::isEmail($varInput))
  829.                     {
  830.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['email'], $this->strLabel));
  831.                     }
  832.                     if ($this->rgxp == 'friendly' && !empty($strName))
  833.                     {
  834.                         $varInput $strName ' [' $varInput ']';
  835.                     }
  836.                     break;
  837.                 case 'emails':
  838.                     // Check whether the current value is list of valid e-mail addresses
  839.                     $arrEmails StringUtil::trimsplit(','$varInput);
  840.                     foreach ($arrEmails as $strEmail)
  841.                     {
  842.                         $strEmail Idna::encodeEmail($strEmail);
  843.                         if (!Validator::isEmail($strEmail))
  844.                         {
  845.                             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['emails'], $this->strLabel));
  846.                             break;
  847.                         }
  848.                     }
  849.                     break;
  850.                 case 'url':
  851.                     $varInput StringUtil::specialcharsUrl($varInput);
  852.                     if ($this->decodeEntities)
  853.                     {
  854.                         $varInput StringUtil::decodeEntities($varInput);
  855.                     }
  856.                     if (!Validator::isUrl($varInput))
  857.                     {
  858.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['url'], $this->strLabel));
  859.                     }
  860.                     break;
  861.                 case 'alias':
  862.                     if (!Validator::isAlias($varInput))
  863.                     {
  864.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['alias'], $this->strLabel));
  865.                     }
  866.                     break;
  867.                 case 'folderalias':
  868.                     if (!Validator::isFolderAlias($varInput))
  869.                     {
  870.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['folderalias'], $this->strLabel));
  871.                     }
  872.                     break;
  873.                 case 'phone':
  874.                     if (!Validator::isPhone(html_entity_decode($varInput)))
  875.                     {
  876.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['phone'], $this->strLabel));
  877.                     }
  878.                     break;
  879.                 case 'prcnt':
  880.                     if (!Validator::isPercent($varInput))
  881.                     {
  882.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['prcnt'], $this->strLabel));
  883.                     }
  884.                     break;
  885.                 case 'locale':
  886.                     if (!Validator::isLocale($varInput))
  887.                     {
  888.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['locale'], $this->strLabel));
  889.                     }
  890.                     break;
  891.                 case 'language':
  892.                     if (!Validator::isLanguage($varInput))
  893.                     {
  894.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['language'], $this->strLabel));
  895.                     }
  896.                     break;
  897.                 case 'google+':
  898.                     if (!Validator::isGooglePlusId($varInput))
  899.                     {
  900.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['invalidGoogleId'], $this->strLabel));
  901.                     }
  902.                     break;
  903.                 case 'fieldname':
  904.                     if (!Validator::isFieldName($varInput))
  905.                     {
  906.                         $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['invalidFieldName'], $this->strLabel));
  907.                     }
  908.                     break;
  909.                 // HOOK: pass unknown tags to callback functions
  910.                 default:
  911.                     if (isset($GLOBALS['TL_HOOKS']['addCustomRegexp']) && \is_array($GLOBALS['TL_HOOKS']['addCustomRegexp']))
  912.                     {
  913.                         foreach ($GLOBALS['TL_HOOKS']['addCustomRegexp'] as $callback)
  914.                         {
  915.                             $this->import($callback[0]);
  916.                             $break $this->{$callback[0]}->{$callback[1]}($this->rgxp$varInput$this);
  917.                             // Stop the loop if a callback returned true
  918.                             if ($break === true)
  919.                             {
  920.                                 break;
  921.                             }
  922.                         }
  923.                     }
  924.                     break;
  925.             }
  926.         }
  927.         if ($this->isHexColor && $varInput && strncmp($varInput'$'1) !== 0)
  928.         {
  929.             $varInput preg_replace('/[^a-f0-9]+/i'''$varInput);
  930.         }
  931.         if ($this->nospace && preg_match('/[\t ]+/'$varInput))
  932.         {
  933.             $this->addError(sprintf($GLOBALS['TL_LANG']['ERR']['noSpace'], $this->strLabel));
  934.         }
  935.         if ($this->spaceToUnderscore)
  936.         {
  937.             $varInput preg_replace('/\s+/''_'trim($varInput));
  938.         }
  939.         if (\is_bool($this->trailingSlash) && $varInput)
  940.         {
  941.             $varInput preg_replace('/\/+$/'''$varInput) . ($this->trailingSlash '/' '');
  942.         }
  943.         return $varInput;
  944.     }
  945.     /**
  946.      * Take an associative array and add it to the object's attributes
  947.      *
  948.      * @param array $arrAttributes An array of attributes
  949.      */
  950.     public function addAttributes($arrAttributes)
  951.     {
  952.         if (!\is_array($arrAttributes))
  953.         {
  954.             return;
  955.         }
  956.         foreach ($arrAttributes as $k=>$v)
  957.         {
  958.             $this->$k $v;
  959.         }
  960.     }
  961.     /**
  962.      * Check whether an option is checked
  963.      *
  964.      * @param array $arrOption The options array
  965.      *
  966.      * @return string The "checked" attribute or an empty string
  967.      */
  968.     protected function isChecked($arrOption)
  969.     {
  970.         if (empty($this->varValue) && empty($_POST) && ($arrOption['default'] ?? null))
  971.         {
  972.             return static::optionChecked(11);
  973.         }
  974.         return static::optionChecked($arrOption['value'] ?? null$this->varValue);
  975.     }
  976.     /**
  977.      * Check whether an option is selected
  978.      *
  979.      * @param array $arrOption The options array
  980.      *
  981.      * @return string The "selected" attribute or an empty string
  982.      */
  983.     protected function isSelected($arrOption)
  984.     {
  985.         if (empty($this->varValue) && empty($_POST) && ($arrOption['default'] ?? null))
  986.         {
  987.             return static::optionSelected(11);
  988.         }
  989.         return static::optionSelected($arrOption['value'] ?? null$this->varValue);
  990.     }
  991.     /**
  992.      * Return a "selected" attribute if the option is selected
  993.      *
  994.      * @param string $strOption The option to check
  995.      * @param mixed  $varValues One or more values to check against
  996.      *
  997.      * @return string The attribute or an empty string
  998.      */
  999.     public static function optionSelected($strOption$varValues)
  1000.     {
  1001.         if ($strOption === '')
  1002.         {
  1003.             return '';
  1004.         }
  1005.         return (\is_array($varValues) ? \in_array($strOption$varValues) : $strOption == $varValues) ? ' selected' '';
  1006.     }
  1007.     /**
  1008.      * Return a "checked" attribute if the option is checked
  1009.      *
  1010.      * @param string $strOption The option to check
  1011.      * @param mixed  $varValues One or more values to check against
  1012.      *
  1013.      * @return string The attribute or an empty string
  1014.      */
  1015.     public static function optionChecked($strOption$varValues)
  1016.     {
  1017.         if ($strOption === '')
  1018.         {
  1019.             return '';
  1020.         }
  1021.         return (\is_array($varValues) ? \in_array($strOption$varValues) : $strOption == $varValues) ? ' checked' '';
  1022.     }
  1023.     /**
  1024.      * Check whether an input is one of the given options
  1025.      *
  1026.      * @param mixed $varInput The input string or array
  1027.      *
  1028.      * @return boolean True if the selected option exists
  1029.      */
  1030.     protected function isValidOption($varInput)
  1031.     {
  1032.         if (!\is_array($varInput))
  1033.         {
  1034.             $varInput = array($varInput);
  1035.         }
  1036.         // Check each option
  1037.         foreach ($varInput as $strInput)
  1038.         {
  1039.             $blnFound false;
  1040.             foreach ($this->arrOptions as $v)
  1041.             {
  1042.                 // Single dimensional array
  1043.                 if (\array_key_exists('value'$v))
  1044.                 {
  1045.                     if ($strInput == $v['value'])
  1046.                     {
  1047.                         $blnFound true;
  1048.                     }
  1049.                 }
  1050.                 // Multi-dimensional array
  1051.                 else
  1052.                 {
  1053.                     foreach ($v as $vv)
  1054.                     {
  1055.                         if ($strInput == $vv['value'])
  1056.                         {
  1057.                             $blnFound true;
  1058.                         }
  1059.                     }
  1060.                 }
  1061.             }
  1062.             if (!$blnFound)
  1063.             {
  1064.                 return false;
  1065.             }
  1066.         }
  1067.         return true;
  1068.     }
  1069.     /**
  1070.      * Extract the Widget attributes from a Data Container array
  1071.      *
  1072.      * @param array                     $arrData  The field configuration array
  1073.      * @param string                    $strName  The field name in the form
  1074.      * @param mixed                     $varValue The field value
  1075.      * @param string                    $strField The field name in the database
  1076.      * @param string                    $strTable The table name in the database
  1077.      * @param DataContainer|Module|null $objDca   An optional DataContainer or Module object
  1078.      *
  1079.      * @return array An attributes array that can be passed to a widget
  1080.      */
  1081.     public static function getAttributesFromDca($arrData$strName$varValue=null$strField=''$strTable=''$objDca=null)
  1082.     {
  1083.         $arrAttributes $arrData['eval'] ?? array();
  1084.         if (method_exists(System::getContainer(), 'getParameterBag'))
  1085.         {
  1086.             $objParameterBag System::getContainer()->getParameterBag();
  1087.             foreach ($arrAttributes as $strAttrKey => $varAttrValue)
  1088.             {
  1089.                 if (!\is_string($varAttrValue) || !preg_match('/%[a-z][a-z0-9_]*\.[a-z0-9_.]+%/i'$varAttrValue))
  1090.                 {
  1091.                     continue;
  1092.                 }
  1093.                 $varAttrValue $objParameterBag->resolveValue($varAttrValue);
  1094.                 $varAttrValue $objParameterBag->unescapeValue($varAttrValue);
  1095.                 $arrAttributes[$strAttrKey] = $varAttrValue;
  1096.             }
  1097.         }
  1098.         $arrAttributes['id'] = $strName;
  1099.         $arrAttributes['name'] = $strName;
  1100.         $arrAttributes['strField'] = $strField;
  1101.         $arrAttributes['strTable'] = $strTable;
  1102.         $arrAttributes['label'] = (($label \is_array($arrData['label'] ?? null) ? $arrData['label'][0] : $arrData['label'] ?? null) !== null) ? $label $strField;
  1103.         $arrAttributes['description'] = $arrData['label'][1] ?? null;
  1104.         $arrAttributes['type'] = $arrData['inputType'] ?? null;
  1105.         $arrAttributes['dataContainer'] = $objDca;
  1106.         $arrAttributes['value'] = StringUtil::deserialize($varValue);
  1107.         // Internet Explorer does not support onchange for checkboxes and radio buttons
  1108.         if ($arrData['eval']['submitOnChange'] ?? null)
  1109.         {
  1110.             if (($arrData['inputType'] ?? null) == 'checkbox' || ($arrData['inputType'] ?? null) == 'checkboxWizard' || ($arrData['inputType'] ?? null) == 'radio' || ($arrData['inputType'] ?? null) == 'radioTable')
  1111.             {
  1112.                 $arrAttributes['onclick'] = trim(($arrAttributes['onclick'] ?? '') . " Backend.autoSubmit('" $strTable "')");
  1113.             }
  1114.             else
  1115.             {
  1116.                 $arrAttributes['onchange'] = trim(($arrAttributes['onchange'] ?? '') . " Backend.autoSubmit('" $strTable "')");
  1117.             }
  1118.         }
  1119.         if (!empty($arrData['eval']['preserveTags']))
  1120.         {
  1121.             $arrAttributes['allowHtml'] = true;
  1122.         }
  1123.         if (!isset($arrAttributes['allowHtml']))
  1124.         {
  1125.             $rte $arrData['eval']['rte'] ?? '';
  1126.             $arrAttributes['allowHtml'] = 'ace|html' === $rte || === strpos($rte'tiny');
  1127.         }
  1128.         // Decode entities if HTML is allowed
  1129.         if ($arrAttributes['allowHtml'] || ($arrData['inputType'] ?? null) == 'fileTree')
  1130.         {
  1131.             $arrAttributes['decodeEntities'] = true;
  1132.         }
  1133.         // Add Ajax event
  1134.         if (($arrData['inputType'] ?? null) == 'checkbox' && ($arrData['eval']['submitOnChange'] ?? null) && \is_array($GLOBALS['TL_DCA'][$strTable]['subpalettes'] ?? null) && \array_key_exists($strField$GLOBALS['TL_DCA'][$strTable]['subpalettes']))
  1135.         {
  1136.             $arrAttributes['onclick'] = "AjaxRequest.toggleSubpalette(this, 'sub_" $strName "', '" $strField "')";
  1137.         }
  1138.         // Options callback
  1139.         if (\is_array($arrData['options_callback'] ?? null))
  1140.         {
  1141.             $arrCallback $arrData['options_callback'];
  1142.             $arrData['options'] = static::importStatic($arrCallback[0])->{$arrCallback[1]}($objDca);
  1143.         }
  1144.         elseif (\is_callable($arrData['options_callback'] ?? null))
  1145.         {
  1146.             $arrData['options'] = $arrData['options_callback']($objDca);
  1147.         }
  1148.         // Foreign key
  1149.         elseif (isset($arrData['foreignKey']))
  1150.         {
  1151.             $arrKey explode('.'$arrData['foreignKey'], 2);
  1152.             $objOptions Database::getInstance()->query("SELECT id, " $arrKey[1] . " AS value FROM " $arrKey[0] . " WHERE tstamp>0 ORDER BY value");
  1153.             $arrData['options'] = array();
  1154.             while ($objOptions->next())
  1155.             {
  1156.                 $arrData['options'][$objOptions->id] = $objOptions->value;
  1157.             }
  1158.         }
  1159.         // Add default option to single checkbox
  1160.         if (($arrData['inputType'] ?? null) == 'checkbox' && !isset($arrData['options']) && !isset($arrData['options_callback']) && !isset($arrData['foreignKey']))
  1161.         {
  1162.             if (TL_MODE == 'FE' && isset($arrAttributes['description']))
  1163.             {
  1164.                 $arrAttributes['options'][] = array('value'=>1'label'=>$arrAttributes['description']);
  1165.             }
  1166.             else
  1167.             {
  1168.                 $arrAttributes['options'][] = array('value'=>1'label'=>$arrAttributes['label']);
  1169.             }
  1170.         }
  1171.         // Add options
  1172.         if (\is_array($arrData['options'] ?? null))
  1173.         {
  1174.             $blnIsAssociative = ($arrData['eval']['isAssociative'] ?? null) || ArrayUtil::isAssoc($arrData['options'] ?? null);
  1175.             $blnUseReference = isset($arrData['reference']);
  1176.             if (($arrData['eval']['includeBlankOption'] ?? null) && !($arrData['eval']['multiple'] ?? null))
  1177.             {
  1178.                 $strLabel $arrData['eval']['blankOptionLabel'] ?? '-';
  1179.                 $arrAttributes['options'][] = array('value'=>'''label'=>$strLabel);
  1180.             }
  1181.             $unknown = (array) $arrAttributes['value'];
  1182.             foreach ($arrData['options'] as $k=>$v)
  1183.             {
  1184.                 if (!\is_array($v))
  1185.                 {
  1186.                     $value $blnIsAssociative $k $v;
  1187.                     if (($i array_search($value$unknown)) !== false)
  1188.                     {
  1189.                         unset($unknown[$i]);
  1190.                     }
  1191.                     $arrAttributes['options'][] = array('value'=>$value'label'=>($blnUseReference && isset($arrData['reference'][$v]) ? (($ref = (\is_array($arrData['reference'][$v]) ? $arrData['reference'][$v][0] : $arrData['reference'][$v])) ? $ref $v) : $v));
  1192.                     continue;
  1193.                 }
  1194.                 $key $blnUseReference && isset($arrData['reference'][$k]) ? (($ref = (\is_array($arrData['reference'][$k]) ? $arrData['reference'][$k][0] : $arrData['reference'][$k])) ? $ref $k) : $k;
  1195.                 $blnIsAssoc ArrayUtil::isAssoc($v);
  1196.                 foreach ($v as $kk=>$vv)
  1197.                 {
  1198.                     $value $blnIsAssoc $kk $vv;
  1199.                     if (($i array_search($value$unknown)) !== false)
  1200.                     {
  1201.                         unset($unknown[$i]);
  1202.                     }
  1203.                     $arrAttributes['options'][$key][] = array('value'=>$value'label'=>($blnUseReference && isset($arrData['reference'][$vv]) ? (($ref = (\is_array($arrData['reference'][$vv]) ? $arrData['reference'][$vv][0] : $arrData['reference'][$vv])) ? $ref $vv) : $vv));
  1204.                 }
  1205.             }
  1206.             $arrAttributes['unknownOption'] = array_filter($unknown);
  1207.         }
  1208.         if (\is_array($arrAttributes['sql'] ?? null) && !isset($arrAttributes['sql']['columnDefinition']))
  1209.         {
  1210.             if (!isset($arrAttributes['maxlength']) && isset($arrAttributes['sql']['length']))
  1211.             {
  1212.                 $arrAttributes['maxlength'] = $arrAttributes['sql']['length'];
  1213.             }
  1214.             if (!isset($arrAttributes['unique']) && isset($arrAttributes['sql']['customSchemaOptions']['unique']))
  1215.             {
  1216.                 $arrAttributes['unique'] = $arrAttributes['sql']['customSchemaOptions']['unique'];
  1217.             }
  1218.         }
  1219.         // Convert timestamps
  1220.         if ($varValue !== null && $varValue !== '' && \in_array($arrData['eval']['rgxp'] ?? null, array('date''time''datim')))
  1221.         {
  1222.             $objDate = new Date($varValueDate::getFormatFromRgxp($arrData['eval']['rgxp']));
  1223.             $arrAttributes['value'] = $objDate->{$arrData['eval']['rgxp']};
  1224.         }
  1225.         // Convert URL insert tags
  1226.         if ($varValue && 'url' === ($arrData['eval']['rgxp'] ?? null))
  1227.         {
  1228.             $arrAttributes['value'] = str_replace('|urlattr}}''}}'$varValue);
  1229.         }
  1230.         // Add the "rootNodes" array as attribute (see #3563)
  1231.         if (isset($arrData['rootNodes']) && !isset($arrData['eval']['rootNodes']))
  1232.         {
  1233.             $arrAttributes['rootNodes'] = $arrData['rootNodes'];
  1234.         }
  1235.         // HOOK: add custom logic
  1236.         if (isset($GLOBALS['TL_HOOKS']['getAttributesFromDca']) && \is_array($GLOBALS['TL_HOOKS']['getAttributesFromDca']))
  1237.         {
  1238.             foreach ($GLOBALS['TL_HOOKS']['getAttributesFromDca'] as $callback)
  1239.             {
  1240.                 $arrAttributes = static::importStatic($callback[0])->{$callback[1]}($arrAttributes$objDca);
  1241.             }
  1242.         }
  1243.         // Warn if someone uses the "encrypt" flag (see #8589)
  1244.         if (isset($arrAttributes['encrypt']))
  1245.         {
  1246.             trigger_deprecation('contao/core-bundle''4.0''Using the "encrypt" flag' . (!empty($strTable) && !empty($strField) ? ' on ' $strTable '.' $strField '') . ' has been deprecated and will no longer work in Contao 5.0. Use the load and save callbacks with a third-party library such as OpenSSL or phpseclib instead.');
  1247.         }
  1248.         return $arrAttributes;
  1249.     }
  1250.     /**
  1251.      * Return the empty value based on the SQL string
  1252.      *
  1253.      * @return string|integer|null The empty value
  1254.      */
  1255.     public function getEmptyValue()
  1256.     {
  1257.         if (!isset($GLOBALS['TL_DCA'][$this->strTable]['fields'][$this->strField]['sql']))
  1258.         {
  1259.             return '';
  1260.         }
  1261.         return static::getEmptyValueByFieldType($GLOBALS['TL_DCA'][$this->strTable]['fields'][$this->strField]['sql']);
  1262.     }
  1263.     /**
  1264.      * Return the empty value based on the SQL string
  1265.      *
  1266.      * @param string|array $sql The SQL string
  1267.      *
  1268.      * @return string|integer|null The empty value
  1269.      */
  1270.     public static function getEmptyValueByFieldType($sql)
  1271.     {
  1272.         if (empty($sql))
  1273.         {
  1274.             return '';
  1275.         }
  1276.         if (\is_array($sql))
  1277.         {
  1278.             if (isset($sql['columnDefinition']))
  1279.             {
  1280.                 $sql $sql['columnDefinition'];
  1281.             }
  1282.             else
  1283.             {
  1284.                 if (isset($sql['notnull']) && !$sql['notnull'])
  1285.                 {
  1286.                     return null;
  1287.                 }
  1288.                 if (\in_array($sql['type'], array(Types::BIGINTTypes::DECIMALTypes::INTEGERTypes::SMALLINTTypes::FLOAT)))
  1289.                 {
  1290.                     return 0;
  1291.                 }
  1292.                 if ($sql['type'] === Types::BOOLEAN)
  1293.                 {
  1294.                     return false;
  1295.                 }
  1296.                 return '';
  1297.             }
  1298.         }
  1299.         if (stripos($sql'NOT NULL') === false)
  1300.         {
  1301.             return null;
  1302.         }
  1303.         $type strtolower(preg_replace('/^([A-Za-z]+)[( ].*$/''$1'$sql));
  1304.         if (\in_array($type, array('int''integer''tinyint''smallint''mediumint''bigint''float''double''dec''decimal')))
  1305.         {
  1306.             return 0;
  1307.         }
  1308.         return '';
  1309.     }
  1310.     /**
  1311.      * Return either an empty string or null based on the SQL string
  1312.      *
  1313.      * @return string|int|null The empty value
  1314.      */
  1315.     public function getEmptyStringOrNull()
  1316.     {
  1317.         if (!isset($GLOBALS['TL_DCA'][$this->strTable]['fields'][$this->strField]['sql']))
  1318.         {
  1319.             return '';
  1320.         }
  1321.         return static::getEmptyStringOrNullByFieldType($GLOBALS['TL_DCA'][$this->strTable]['fields'][$this->strField]['sql']);
  1322.     }
  1323.     /**
  1324.      * Return either an empty string or null based on the SQL string
  1325.      *
  1326.      * @param string $sql The SQL string
  1327.      *
  1328.      * @return string|null The empty string or null
  1329.      */
  1330.     public static function getEmptyStringOrNullByFieldType($sql)
  1331.     {
  1332.         if (empty($sql))
  1333.         {
  1334.             return '';
  1335.         }
  1336.         return static::getEmptyValueByFieldType($sql) === null null '';
  1337.     }
  1338.     /**
  1339.      * Generate a submit button
  1340.      *
  1341.      * @return string The submit button markup
  1342.      *
  1343.      * @deprecated Deprecated since Contao 4.0, to be removed in Contao 5.0.
  1344.      */
  1345.     protected function addSubmit()
  1346.     {
  1347.         trigger_deprecation('contao/core-bundle''4.0''Using "Contao\Widget::addSubmit()" has been deprecated and will no longer work in Contao 5.0.');
  1348.         return '';
  1349.     }
  1350. }
  1351. class_alias(Widget::class, 'Widget');