vendor/contao/core-bundle/src/Resources/contao/forms/Form.php line 250

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. /**
  11.  * Provide methods to handle front end forms.
  12.  *
  13.  * @property integer $id
  14.  * @property string  $title
  15.  * @property string  $formID
  16.  * @property string  $method
  17.  * @property boolean $allowTags
  18.  * @property string  $attributes
  19.  * @property boolean $novalidate
  20.  * @property integer $jumpTo
  21.  * @property boolean $sendViaEmail
  22.  * @property boolean $skipEmpty
  23.  * @property string  $format
  24.  * @property string  $recipient
  25.  * @property string  $subject
  26.  * @property boolean $storeValues
  27.  * @property string  $targetTable
  28.  * @property string  $customTpl
  29.  */
  30. class Form extends Hybrid
  31. {
  32.     /**
  33.      * Model
  34.      * @var FormModel
  35.      */
  36.     protected $objModel;
  37.     /**
  38.      * Key
  39.      * @var string
  40.      */
  41.     protected $strKey 'form';
  42.     /**
  43.      * Table
  44.      * @var string
  45.      */
  46.     protected $strTable 'tl_form';
  47.     /**
  48.      * Template
  49.      * @var string
  50.      */
  51.     protected $strTemplate 'form_wrapper';
  52.     /**
  53.      * Remove name attributes in the back end so the form is not validated
  54.      *
  55.      * @return string
  56.      */
  57.     public function generate()
  58.     {
  59.         $request System::getContainer()->get('request_stack')->getCurrentRequest();
  60.         if ($request && System::getContainer()->get('contao.routing.scope_matcher')->isBackendRequest($request))
  61.         {
  62.             $objTemplate = new BackendTemplate('be_wildcard');
  63.             $objTemplate->wildcard '### ' $GLOBALS['TL_LANG']['CTE']['form'][0] . ' ###';
  64.             $objTemplate->id $this->id;
  65.             $objTemplate->link $this->title;
  66.             $objTemplate->href StringUtil::specialcharsUrl(System::getContainer()->get('router')->generate('contao_backend', array('do'=>'form''table'=>'tl_form_field''id'=>$this->id)));
  67.             return $objTemplate->parse();
  68.         }
  69.         if ($this->customTpl)
  70.         {
  71.             $request System::getContainer()->get('request_stack')->getCurrentRequest();
  72.             // Use the custom template unless it is a back end request
  73.             if (!$request || !System::getContainer()->get('contao.routing.scope_matcher')->isBackendRequest($request))
  74.             {
  75.                 $this->strTemplate $this->customTpl;
  76.             }
  77.         }
  78.         return parent::generate();
  79.     }
  80.     /**
  81.      * Generate the form
  82.      */
  83.     protected function compile()
  84.     {
  85.         $hasUpload false;
  86.         $doNotSubmit false;
  87.         $arrSubmitted = array();
  88.         $this->loadDataContainer('tl_form_field');
  89.         $formId $this->formID 'auto_' $this->formID 'auto_form_' $this->id;
  90.         $this->Template->fields '';
  91.         $this->Template->hidden '';
  92.         $this->Template->formSubmit $formId;
  93.         $this->Template->method = ($this->method == 'GET') ? 'get' 'post';
  94.         $this->Template->requestToken System::getContainer()->get('contao.csrf.token_manager')->getDefaultTokenValue();
  95.         $this->initializeSession($formId);
  96.         $arrLabels = array();
  97.         // Get all form fields
  98.         $arrFields = array();
  99.         $objFields FormFieldModel::findPublishedByPid($this->id);
  100.         if ($objFields !== null)
  101.         {
  102.             while ($objFields->next())
  103.             {
  104.                 // Ignore the name of form fields which do not use a name (see #1268)
  105.                 if ($objFields->name && isset($GLOBALS['TL_DCA']['tl_form_field']['palettes'][$objFields->type]) && preg_match('/[,;]name[,;]/'$GLOBALS['TL_DCA']['tl_form_field']['palettes'][$objFields->type]))
  106.                 {
  107.                     $arrFields[$objFields->name] = $objFields->current();
  108.                 }
  109.                 else
  110.                 {
  111.                     $arrFields[] = $objFields->current();
  112.                 }
  113.             }
  114.             System::getContainer()->get('contao.cache.entity_tags')->tagWith($objFields);
  115.         }
  116.         // HOOK: compile form fields
  117.         if (isset($GLOBALS['TL_HOOKS']['compileFormFields']) && \is_array($GLOBALS['TL_HOOKS']['compileFormFields']))
  118.         {
  119.             foreach ($GLOBALS['TL_HOOKS']['compileFormFields'] as $callback)
  120.             {
  121.                 $this->import($callback[0]);
  122.                 $arrFields $this->{$callback[0]}->{$callback[1]}($arrFields$formId$this);
  123.             }
  124.         }
  125.         // Process the fields
  126.         if (!empty($arrFields) && \is_array($arrFields))
  127.         {
  128.             $row 0;
  129.             $max_row \count($arrFields);
  130.             foreach ($arrFields as $objField)
  131.             {
  132.                 /** @var FormFieldModel $objField */
  133.                 $strClass $GLOBALS['TL_FFL'][$objField->type] ?? null;
  134.                 // Continue if the class is not defined
  135.                 if (!class_exists($strClass))
  136.                 {
  137.                     continue;
  138.                 }
  139.                 $arrData $objField->row();
  140.                 $arrData['decodeEntities'] = true;
  141.                 $arrData['allowHtml'] = $this->allowTags;
  142.                 $arrData['rowClass'] = 'row_' $row . (($row == 0) ? ' row_first' : (($row == ($max_row 1)) ? ' row_last' '')) . ((($row 2) == 0) ? ' even' ' odd');
  143.                 // Increase the row count if it's a password field
  144.                 if ($objField->type == 'password')
  145.                 {
  146.                     ++$row;
  147.                     ++$max_row;
  148.                     $arrData['rowClassConfirm'] = 'row_' $row . (($row == ($max_row 1)) ? ' row_last' '') . ((($row 2) == 0) ? ' even' ' odd');
  149.                 }
  150.                 // Submit buttons do not use the name attribute
  151.                 if ($objField->type == 'submit')
  152.                 {
  153.                     $arrData['name'] = '';
  154.                 }
  155.                 // Unset the default value depending on the field type (see #4722)
  156.                 if (!empty($arrData['value']) && !\in_array('value'StringUtil::trimsplit('[,;]'$GLOBALS['TL_DCA']['tl_form_field']['palettes'][$objField->type] ?? '')))
  157.                 {
  158.                     $arrData['value'] = '';
  159.                 }
  160.                 /** @var Widget $objWidget */
  161.                 $objWidget = new $strClass($arrData);
  162.                 $objWidget->required $objField->mandatory true false;
  163.                 // HOOK: load form field callback
  164.                 if (isset($GLOBALS['TL_HOOKS']['loadFormField']) && \is_array($GLOBALS['TL_HOOKS']['loadFormField']))
  165.                 {
  166.                     foreach ($GLOBALS['TL_HOOKS']['loadFormField'] as $callback)
  167.                     {
  168.                         $this->import($callback[0]);
  169.                         $objWidget $this->{$callback[0]}->{$callback[1]}($objWidget$formId$this->arrData$this);
  170.                     }
  171.                 }
  172.                 // Validate the input
  173.                 if (Input::post('FORM_SUBMIT') == $formId)
  174.                 {
  175.                     $objWidget->validate();
  176.                     // HOOK: validate form field callback
  177.                     if (isset($GLOBALS['TL_HOOKS']['validateFormField']) && \is_array($GLOBALS['TL_HOOKS']['validateFormField']))
  178.                     {
  179.                         foreach ($GLOBALS['TL_HOOKS']['validateFormField'] as $callback)
  180.                         {
  181.                             $this->import($callback[0]);
  182.                             $objWidget $this->{$callback[0]}->{$callback[1]}($objWidget$formId$this->arrData$this);
  183.                         }
  184.                     }
  185.                     if ($objWidget->hasErrors())
  186.                     {
  187.                         $doNotSubmit true;
  188.                     }
  189.                     // Store current value in the session
  190.                     elseif ($objWidget->submitInput())
  191.                     {
  192.                         $arrSubmitted[$objField->name] = $objWidget->value;
  193.                         $_SESSION['FORM_DATA'][$objField->name] = $objWidget->value;
  194.                         unset($_POST[$objField->name]); // see #5474
  195.                     }
  196.                 }
  197.                 if ($objWidget instanceof UploadableWidgetInterface)
  198.                 {
  199.                     $hasUpload true;
  200.                 }
  201.                 if ($objWidget instanceof FormHidden)
  202.                 {
  203.                     $this->Template->hidden .= $objWidget->parse();
  204.                     --$max_row;
  205.                     continue;
  206.                 }
  207.                 if ($objWidget->name && $objWidget->label)
  208.                 {
  209.                     $arrLabels[$objWidget->name] = System::getContainer()->get('contao.insert_tag.parser')->replaceInline($objWidget->label); // see #4268
  210.                 }
  211.                 $this->Template->fields .= $objWidget->parse();
  212.                 ++$row;
  213.             }
  214.         }
  215.         // Process the form data
  216.         if (!$doNotSubmit && Input::post('FORM_SUBMIT') == $formId)
  217.         {
  218.             $this->processFormData($arrSubmitted$arrLabels$arrFields);
  219.         }
  220.         // Remove any uploads, if form did not validate (#1185)
  221.         if ($doNotSubmit && $hasUpload && !empty($_SESSION['FILES']))
  222.         {
  223.             foreach ($_SESSION['FILES'] as $field => $upload)
  224.             {
  225.                 if (empty($arrFields[$field]))
  226.                 {
  227.                     continue;
  228.                 }
  229.                 if (!empty($upload['uuid']) && null !== ($file FilesModel::findById($upload['uuid'])))
  230.                 {
  231.                     $file->delete();
  232.                 }
  233.                 if (is_file($upload['tmp_name']))
  234.                 {
  235.                     unlink($upload['tmp_name']);
  236.                 }
  237.                 unset($_SESSION['FILES'][$field]);
  238.             }
  239.         }
  240.         // Add a warning to the page title
  241.         if ($doNotSubmit && !Environment::get('isAjaxRequest'))
  242.         {
  243.             /** @var PageModel $objPage */
  244.             global $objPage;
  245.             $title $objPage->pageTitle ?: $objPage->title;
  246.             $objPage->pageTitle $GLOBALS['TL_LANG']['ERR']['form'] . ' - ' $title;
  247.         }
  248.         $strAttributes '';
  249.         $arrAttributes StringUtil::deserialize($this->attributestrue);
  250.         if (!empty($arrAttributes[0]))
  251.         {
  252.             $strAttributes .= ' id="' $arrAttributes[0] . '"';
  253.         }
  254.         if (!empty($arrAttributes[1]))
  255.         {
  256.             $strAttributes .= ' class="' $arrAttributes[1] . '"';
  257.         }
  258.         $this->Template->hasError $doNotSubmit;
  259.         $this->Template->attributes $strAttributes;
  260.         $this->Template->enctype $hasUpload 'multipart/form-data' 'application/x-www-form-urlencoded';
  261.         $this->Template->maxFileSize $hasUpload $this->objModel->getMaxUploadFileSize() : false;
  262.         $this->Template->novalidate $this->novalidate ' novalidate' '';
  263.         // Get the target URL
  264.         if ($this->method == 'GET' && ($objTarget $this->objModel->getRelated('jumpTo')) instanceof PageModel)
  265.         {
  266.             /** @var PageModel $objTarget */
  267.             $this->Template->action $objTarget->getFrontendUrl();
  268.         }
  269.     }
  270.     /**
  271.      * Process form data, store it in the session and redirect to the jumpTo page
  272.      *
  273.      * @param array $arrSubmitted
  274.      * @param array $arrLabels
  275.      * @param array $arrFields
  276.      */
  277.     protected function processFormData($arrSubmitted$arrLabels$arrFields)
  278.     {
  279.         // HOOK: prepare form data callback
  280.         if (isset($GLOBALS['TL_HOOKS']['prepareFormData']) && \is_array($GLOBALS['TL_HOOKS']['prepareFormData']))
  281.         {
  282.             foreach ($GLOBALS['TL_HOOKS']['prepareFormData'] as $callback)
  283.             {
  284.                 $this->import($callback[0]);
  285.                 $this->{$callback[0]}->{$callback[1]}($arrSubmitted$arrLabels$arrFields$this);
  286.             }
  287.         }
  288.         // Send form data via e-mail
  289.         if ($this->sendViaEmail)
  290.         {
  291.             $keys = array();
  292.             $values = array();
  293.             $fields = array();
  294.             $message '';
  295.             foreach ($arrSubmitted as $k=>$v)
  296.             {
  297.                 if ($k == 'cc')
  298.                 {
  299.                     continue;
  300.                 }
  301.                 $v StringUtil::deserialize($v);
  302.                 // Skip empty fields
  303.                 if ($this->skipEmpty && !\is_array($v) && !\strlen($v))
  304.                 {
  305.                     continue;
  306.                 }
  307.                 // Add field to message
  308.                 $message .= ($arrLabels[$k] ?? ucfirst($k)) . ': ' . (\is_array($v) ? implode(', '$v) : $v) . "\n";
  309.                 // Prepare XML file
  310.                 if ($this->format == 'xml')
  311.                 {
  312.                     $fields[] = array
  313.                     (
  314.                         'name' => $k,
  315.                         'values' => (\is_array($v) ? $v : array($v))
  316.                     );
  317.                 }
  318.                 // Prepare CSV file
  319.                 if ($this->format == 'csv' || $this->format == 'csv_excel')
  320.                 {
  321.                     $keys[] = $k;
  322.                     $values[] = (\is_array($v) ? implode(','$v) : $v);
  323.                 }
  324.             }
  325.             $recipients StringUtil::splitCsv($this->recipient);
  326.             // Format recipients
  327.             foreach ($recipients as $k=>$v)
  328.             {
  329.                 $recipients[$k] = str_replace(array('['']''"'), array('<''>'''), $v);
  330.             }
  331.             $email = new Email();
  332.             // Get subject and message
  333.             if ($this->format == 'email')
  334.             {
  335.                 $message $arrSubmitted['message'] ?? '';
  336.                 $email->subject $arrSubmitted['subject'] ?? '';
  337.             }
  338.             // Set the admin e-mail as "from" address
  339.             $email->from $GLOBALS['TL_ADMIN_EMAIL'] ?? null;
  340.             $email->fromName $GLOBALS['TL_ADMIN_NAME'] ?? null;
  341.             // Get the "reply to" address
  342.             if (!empty($arrSubmitted['email']))
  343.             {
  344.                 $replyTo $arrSubmitted['email'];
  345.                 // Add the name
  346.                 if (!empty($arrSubmitted['name']))
  347.                 {
  348.                     $replyTo '"' $arrSubmitted['name'] . '" <' $replyTo '>';
  349.                 }
  350.                 elseif (!empty($arrSubmitted['firstname']) && !empty($arrSubmitted['lastname']))
  351.                 {
  352.                     $replyTo '"' $arrSubmitted['firstname'] . ' ' $arrSubmitted['lastname'] . '" <' $replyTo '>';
  353.                 }
  354.                 $email->replyTo($replyTo);
  355.             }
  356.             // Fallback to default subject
  357.             if (!$email->subject)
  358.             {
  359.                 $email->subject html_entity_decode(System::getContainer()->get('contao.insert_tag.parser')->replaceInline($this->subject), ENT_QUOTES'UTF-8');
  360.             }
  361.             // Send copy to sender
  362.             if (!empty($arrSubmitted['cc']) && !empty($arrSubmitted['email']))
  363.             {
  364.                 $email->sendCc($arrSubmitted['email']);
  365.                 unset($_SESSION['FORM_DATA']['cc']);
  366.             }
  367.             // Attach XML file
  368.             if ($this->format == 'xml')
  369.             {
  370.                 // Encode the values (see #6053)
  371.                 array_walk_recursive($fields, static function (&$value) { $value htmlspecialchars($valueENT_QUOTES|ENT_SUBSTITUTE|ENT_XML1); });
  372.                 $objTemplate = new FrontendTemplate('form_xml');
  373.                 $objTemplate->fields $fields;
  374.                 $objTemplate->charset System::getContainer()->getParameter('kernel.charset');
  375.                 $email->attachFileFromString($objTemplate->parse(), 'form.xml''application/xml');
  376.             }
  377.             // Attach CSV file
  378.             if ($this->format == 'csv')
  379.             {
  380.                 $email->attachFileFromString(StringUtil::decodeEntities('"' implode('";"'$keys) . '"' "\n" '"' implode('";"'$values) . '"'), 'form.csv''text/comma-separated-values');
  381.             }
  382.             elseif ($this->format == 'csv_excel')
  383.             {
  384.                 $email->attachFileFromString(mb_convert_encoding("\u{FEFF}sep=;\n" StringUtil::decodeEntities('"' implode('";"'$keys) . '"' "\n" '"' implode('";"'$values) . '"'), 'UTF-16LE''UTF-8'), 'form.csv''text/comma-separated-values');
  385.             }
  386.             $uploaded '';
  387.             // Attach uploaded files
  388.             if (!empty($_SESSION['FILES']))
  389.             {
  390.                 foreach ($_SESSION['FILES'] as $file)
  391.                 {
  392.                     // Add a link to the uploaded file
  393.                     if ($file['uploaded'] ?? null)
  394.                     {
  395.                         $uploaded .= "\n" Environment::get('base') . StringUtil::stripRootDir(\dirname($file['tmp_name'])) . '/' rawurlencode($file['name']);
  396.                         continue;
  397.                     }
  398.                     $email->attachFileFromString(file_get_contents($file['tmp_name']), $file['name'], $file['type']);
  399.                 }
  400.             }
  401.             $uploaded trim($uploaded) ? "\n\n---\n" $uploaded '';
  402.             $email->text StringUtil::decodeEntities(trim($message)) . $uploaded "\n\n";
  403.             // Set the transport
  404.             if (!empty($this->mailerTransport))
  405.             {
  406.                 $email->addHeader('X-Transport'$this->mailerTransport);
  407.             }
  408.             // Send the e-mail
  409.             $email->sendTo($recipients);
  410.         }
  411.         // Store the values in the database
  412.         if ($this->storeValues && $this->targetTable)
  413.         {
  414.             $arrSet = array();
  415.             // Add the timestamp
  416.             if ($this->Database->fieldExists('tstamp'$this->targetTable))
  417.             {
  418.                 $arrSet['tstamp'] = time();
  419.             }
  420.             // Fields
  421.             foreach ($arrSubmitted as $k=>$v)
  422.             {
  423.                 if ($k != 'cc' && $k != 'id')
  424.                 {
  425.                     $arrSet[$k] = $v;
  426.                     // Convert date formats into timestamps (see #6827)
  427.                     if ($arrSet[$k] && \in_array($arrFields[$k]->rgxp, array('date''time''datim')))
  428.                     {
  429.                         $objDate = new Date($arrSet[$k], Date::getFormatFromRgxp($arrFields[$k]->rgxp));
  430.                         $arrSet[$k] = $objDate->tstamp;
  431.                     }
  432.                 }
  433.             }
  434.             // Files
  435.             if (!empty($_SESSION['FILES']))
  436.             {
  437.                 foreach ($_SESSION['FILES'] as $k=>$v)
  438.                 {
  439.                     if ($v['uploaded'] ?? null)
  440.                     {
  441.                         $arrSet[$k] = StringUtil::stripRootDir($v['tmp_name']);
  442.                     }
  443.                 }
  444.             }
  445.             // HOOK: store form data callback
  446.             if (isset($GLOBALS['TL_HOOKS']['storeFormData']) && \is_array($GLOBALS['TL_HOOKS']['storeFormData']))
  447.             {
  448.                 foreach ($GLOBALS['TL_HOOKS']['storeFormData'] as $callback)
  449.                 {
  450.                     $this->import($callback[0]);
  451.                     $arrSet $this->{$callback[0]}->{$callback[1]}($arrSet$this);
  452.                 }
  453.             }
  454.             // Load DataContainer of target table before trying to determine empty value (see #3499)
  455.             Controller::loadDataContainer($this->targetTable);
  456.             // Set the correct empty value (see #6284, #6373)
  457.             foreach ($arrSet as $k=>$v)
  458.             {
  459.                 if ($v === '')
  460.                 {
  461.                     $arrSet[$k] = Widget::getEmptyValueByFieldType($GLOBALS['TL_DCA'][$this->targetTable]['fields'][$k]['sql'] ?? array());
  462.                 }
  463.             }
  464.             // Do not use Models here (backwards compatibility)
  465.             $this->Database->prepare("INSERT INTO " $this->targetTable " %s")->set($arrSet)->execute();
  466.         }
  467.         // Store all values in the session
  468.         foreach (array_keys($_POST) as $key)
  469.         {
  470.             $_SESSION['FORM_DATA'][$key] = $this->allowTags Input::postHtml($keytrue) : Input::post($keytrue);
  471.         }
  472.         // Store the submission time to invalidate the session later on
  473.         $_SESSION['FORM_DATA']['SUBMITTED_AT'] = time();
  474.         $arrFiles $_SESSION['FILES'] ?? null;
  475.         // HOOK: process form data callback
  476.         if (isset($GLOBALS['TL_HOOKS']['processFormData']) && \is_array($GLOBALS['TL_HOOKS']['processFormData']))
  477.         {
  478.             foreach ($GLOBALS['TL_HOOKS']['processFormData'] as $callback)
  479.             {
  480.                 $this->import($callback[0]);
  481.                 $this->{$callback[0]}->{$callback[1]}($arrSubmitted$this->arrData$arrFiles$arrLabels$this);
  482.             }
  483.         }
  484.         $_SESSION['FILES'] = array(); // DO NOT CHANGE
  485.         // Add a log entry
  486.         if (System::getContainer()->get('contao.security.token_checker')->hasFrontendUser())
  487.         {
  488.             $this->import(FrontendUser::class, 'User');
  489.             System::getContainer()->get('monolog.logger.contao.forms')->info('Form "' $this->title '" has been submitted by "' $this->User->username '".');
  490.         }
  491.         else
  492.         {
  493.             System::getContainer()->get('monolog.logger.contao.forms')->info('Form "' $this->title '" has been submitted by a guest.');
  494.         }
  495.         // Check whether there is a jumpTo page
  496.         if (($objJumpTo $this->objModel->getRelated('jumpTo')) instanceof PageModel)
  497.         {
  498.             $this->jumpToOrReload($objJumpTo->row());
  499.         }
  500.         $this->reload();
  501.     }
  502.     /**
  503.      * Get the maximum file size that is allowed for file uploads
  504.      *
  505.      * @return integer
  506.      *
  507.      * @deprecated Deprecated since Contao 4.0, to be removed in Contao 5.0.
  508.      *             Use $this->objModel->getMaxUploadFileSize() instead.
  509.      */
  510.     protected function getMaxFileSize()
  511.     {
  512.         trigger_deprecation('contao/core-bundle''4.0''Using "Contao\Form::getMaxFileSize()" has been deprecated and will no longer work in Contao 5.0. Use "$this->objModel->getMaxUploadFileSize()" instead.');
  513.         return $this->objModel->getMaxUploadFileSize();
  514.     }
  515.     /**
  516.      * Initialize the form in the current session
  517.      *
  518.      * @param string $formId
  519.      */
  520.     protected function initializeSession($formId)
  521.     {
  522.         if (Input::post('FORM_SUBMIT') != $formId)
  523.         {
  524.             return;
  525.         }
  526.         $arrMessageBox = array('TL_ERROR''TL_CONFIRM''TL_INFO');
  527.         $_SESSION['FORM_DATA'] = \is_array($_SESSION['FORM_DATA'] ?? null) ? $_SESSION['FORM_DATA'] : array();
  528.         foreach ($arrMessageBox as $tl)
  529.         {
  530.             if (\is_array($_SESSION[$formId][$tl] ?? null))
  531.             {
  532.                 $_SESSION[$formId][$tl] = array_unique($_SESSION[$formId][$tl]);
  533.                 foreach ($_SESSION[$formId][$tl] as $message)
  534.                 {
  535.                     $objTemplate = new FrontendTemplate('form_message');
  536.                     $objTemplate->message $message;
  537.                     $objTemplate->class strtolower($tl);
  538.                     $this->Template->fields .= $objTemplate->parse() . "\n";
  539.                 }
  540.                 $_SESSION[$formId][$tl] = array();
  541.             }
  542.         }
  543.     }
  544. }
  545. class_alias(Form::class, 'Form');