257 lines
9.4 KiB
PHP
257 lines
9.4 KiB
PHP
|
<?php // phpcs:ignore SlevomatCodingStandard.TypeHints.DeclareStrictTypes.DeclareStrictTypesMissing
|
||
|
|
||
|
namespace MailPoet\Form\Util;
|
||
|
|
||
|
if (!defined('ABSPATH')) exit;
|
||
|
|
||
|
|
||
|
use MailPoet\Entities\FormEntity;
|
||
|
use MailPoetVendor\Sabberworm\CSS\Parser as CSSParser;
|
||
|
|
||
|
class Styles {
|
||
|
public function prefixStyles($stylesheet, $prefix = '') {
|
||
|
if (!$stylesheet) return;
|
||
|
$styles = new CSSParser($stylesheet);
|
||
|
$styles = $styles->parse();
|
||
|
$formattedStyles = [];
|
||
|
foreach ($styles->getAllDeclarationBlocks() as $styleDeclaration) {
|
||
|
$selectors = array_map(function($selector) use ($prefix) {
|
||
|
$stringSelector = is_string($selector) ? $selector : $selector->__toString();
|
||
|
return sprintf('%s %s', $prefix, $stringSelector);
|
||
|
}, $styleDeclaration->getSelectors());
|
||
|
$selectors = implode(', ', $selectors);
|
||
|
$rules = array_map(function($rule) {
|
||
|
return $rule->__toString();
|
||
|
}, $styleDeclaration->getRules());
|
||
|
$rules = sprintf('{ %s }', implode(' ', $rules));
|
||
|
$formattedStyles[] = sprintf('%s %s', $selectors, $rules);
|
||
|
}
|
||
|
return implode(PHP_EOL, $formattedStyles);
|
||
|
}
|
||
|
|
||
|
public function renderFormSettingsStyles(FormEntity $form, string $selector, string $displayType): string {
|
||
|
if (!is_array($form->getSettings())) return '';
|
||
|
$formSettings = $form->getSettings();
|
||
|
// Wrapper styles
|
||
|
$styles = [];
|
||
|
|
||
|
if (isset($formSettings['border_size']) && isset($formSettings['border_color'])) {
|
||
|
$styles[] = 'border: ' . $formSettings['border_size'] . 'px solid ' . $formSettings['border_color'];
|
||
|
}
|
||
|
|
||
|
if (isset($formSettings['border_radius'])) {
|
||
|
$styles[] = 'border-radius: ' . $formSettings['border_radius'] . 'px';
|
||
|
}
|
||
|
|
||
|
$backgrounds = [];
|
||
|
$mobileBackgrounds = [];
|
||
|
if (isset($formSettings['background_image_url']) && $formSettings['background_image_url']) {
|
||
|
$backgroundPosition = 'center';
|
||
|
$backgroundRepeat = 'no-repeat';
|
||
|
$backgroundSize = 'cover';
|
||
|
if (isset($formSettings['background_image_display']) && $formSettings['background_image_display'] === 'fit') {
|
||
|
$backgroundPosition = 'center top';
|
||
|
$backgroundSize = 'contain';
|
||
|
}
|
||
|
if (isset($formSettings['background_image_display']) && $formSettings['background_image_display'] === 'tile') {
|
||
|
$backgroundRepeat = 'repeat';
|
||
|
$backgroundSize = 'auto';
|
||
|
}
|
||
|
$backgrounds[] = "url(" . trim($formSettings['background_image_url']) . ") $backgroundPosition / $backgroundSize $backgroundRepeat";
|
||
|
}
|
||
|
|
||
|
if (!empty($formSettings['gradient'])) {
|
||
|
$backgrounds[] = trim($formSettings['gradient']);
|
||
|
$mobileBackgrounds[] = trim($formSettings['gradient']);
|
||
|
}
|
||
|
|
||
|
if (!empty($formSettings['backgroundColor'])) {
|
||
|
$backgrounds[] = trim($formSettings['backgroundColor']);
|
||
|
$mobileBackgrounds[] = trim($formSettings['backgroundColor']);
|
||
|
}
|
||
|
|
||
|
if ($backgrounds) {
|
||
|
$styles[] = 'background: ' . join(', ', $backgrounds);
|
||
|
}
|
||
|
|
||
|
if (isset($formSettings['fontColor'])) {
|
||
|
$styles[] = 'color: ' . trim($formSettings['fontColor']);
|
||
|
}
|
||
|
|
||
|
if (isset($formSettings['alignment'])) {
|
||
|
$styles[] = 'text-align: ' . $formSettings['alignment'];
|
||
|
}
|
||
|
$formWrapperStyles = $selector . '{' . join(';', $styles) . ';}';
|
||
|
|
||
|
// Media styles for mobile
|
||
|
$media = $this->getMobileStyles($selector, $displayType, $mobileBackgrounds);
|
||
|
|
||
|
// Form element styles
|
||
|
$formStyles = [];
|
||
|
if (isset($formSettings['form_padding'])) {
|
||
|
if (
|
||
|
in_array(
|
||
|
$displayType,
|
||
|
[FormEntity::DISPLAY_TYPE_POPUP, FormEntity::DISPLAY_TYPE_FIXED_BAR, FormEntity::DISPLAY_TYPE_SLIDE_IN]
|
||
|
)
|
||
|
) {
|
||
|
$padding = $formSettings['form_padding'];
|
||
|
$media .= " @media (min-width: 500px) {{$selector} {padding: {$padding}px;}} ";
|
||
|
} else {
|
||
|
$formStyles[] = 'padding: ' . $formSettings['form_padding'] . 'px';
|
||
|
}
|
||
|
}
|
||
|
$formElementStyles = '';
|
||
|
if ($formStyles) {
|
||
|
$formElementStyles = $selector . ' form.mailpoet_form {' . join(';', $formStyles) . ';}';
|
||
|
}
|
||
|
|
||
|
// Width styles
|
||
|
$widthStyles = $this->renderWidthStyles($formSettings, $selector, $displayType);
|
||
|
|
||
|
$typeSpecificStyles = $this->getFormTypeSpecificStyles($selector, $displayType);
|
||
|
|
||
|
$messagesStyles = $this->renderFormMessageStyles($form, $selector);
|
||
|
|
||
|
$additionalStyles = $selector . ' .mailpoet_paragraph.last {margin-bottom: 0} ';
|
||
|
$media .= " @media (min-width: 500px) {{$selector} .last .mailpoet_paragraph:last-child {margin-bottom: 0}} ";
|
||
|
$media .= " @media (max-width: 500px) {{$selector} .mailpoet_form_column:last-child .mailpoet_paragraph:last-child {margin-bottom: 0}} ";
|
||
|
|
||
|
return $formWrapperStyles
|
||
|
. $formElementStyles
|
||
|
. $widthStyles
|
||
|
. $messagesStyles
|
||
|
. $typeSpecificStyles
|
||
|
. $additionalStyles
|
||
|
. $media;
|
||
|
}
|
||
|
|
||
|
private function renderWidthStyles(array $formSettings, string $selector, string $displayType): string {
|
||
|
$styles = [];
|
||
|
|
||
|
if (isset($formSettings['form_placement'][$displayType]['styles']['width'])) {
|
||
|
$width = $this->getWidthValue($formSettings['form_placement'][$displayType]['styles']['width']);
|
||
|
}
|
||
|
|
||
|
if ($displayType === FormEntity::DISPLAY_TYPE_POPUP) {
|
||
|
if (isset($width)) {
|
||
|
$styles[] = "width: $width";
|
||
|
$styles[] = "max-width: 100vw";
|
||
|
} else { // BC compatibilty
|
||
|
$styles[] = 'width: 560px';
|
||
|
$styles[] = 'max-width: 560px';
|
||
|
}
|
||
|
} elseif ($displayType === FormEntity::DISPLAY_TYPE_SLIDE_IN) {
|
||
|
if (isset($width)) {
|
||
|
$styles[] = "width: $width";
|
||
|
$styles[] = "max-width: 100vw";
|
||
|
} else { // BC compatibilty
|
||
|
$styles[] = 'max-width: 600px';
|
||
|
$styles[] = 'min-width: 350px';
|
||
|
}
|
||
|
} elseif ($displayType === FormEntity::DISPLAY_TYPE_FIXED_BAR) {
|
||
|
if (isset($width)) {
|
||
|
$styles[] = "width: $width";
|
||
|
$styles[] = "max-width: 100%";
|
||
|
} else { // BC compatibilty
|
||
|
$styles[] = 'max-width: 960px';
|
||
|
}
|
||
|
} elseif ($displayType === FormEntity::DISPLAY_TYPE_BELOW_POST) {
|
||
|
if (isset($width)) {
|
||
|
$styles[] = "width: $width";
|
||
|
}
|
||
|
} elseif ($displayType === FormEntity::DISPLAY_TYPE_OTHERS) {
|
||
|
if (isset($width)) {
|
||
|
$styles[] = "width: $width";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$widthSelector = $selector;
|
||
|
$widthSelector .= $displayType === FormEntity::DISPLAY_TYPE_FIXED_BAR ? ' form.mailpoet_form' : '';
|
||
|
|
||
|
if (!$styles) {
|
||
|
return '';
|
||
|
}
|
||
|
return $widthSelector . '{' . join(';', $styles) . ';}';
|
||
|
}
|
||
|
|
||
|
private function getWidthValue(array $width) {
|
||
|
return $width['value'] . ($width['unit'] === 'percent' ? '%' : 'px');
|
||
|
}
|
||
|
|
||
|
public function renderFormMessageStyles(FormEntity $form, string $selector): string {
|
||
|
$formSettings = $form->getSettings();
|
||
|
if (!is_array($formSettings)) {
|
||
|
return '';
|
||
|
}
|
||
|
return $this->renderMessagesStyles($formSettings, $selector);
|
||
|
}
|
||
|
|
||
|
private function renderMessagesStyles(array $formSettings, string $selector): string {
|
||
|
$styles = "$selector .mailpoet_message {margin: 0; padding: 0 20px;}";
|
||
|
if (isset($formSettings['success_validation_color']) && $formSettings['success_validation_color']) {
|
||
|
$success = $formSettings['success_validation_color'];
|
||
|
$styles .= "
|
||
|
$selector .mailpoet_validate_success {color: $success}
|
||
|
$selector input.parsley-success {color: $success}
|
||
|
$selector select.parsley-success {color: $success}
|
||
|
$selector textarea.parsley-success {color: $success}
|
||
|
";
|
||
|
}
|
||
|
if (isset($formSettings['error_validation_color']) && $formSettings['error_validation_color']) {
|
||
|
$error = $formSettings['error_validation_color'];
|
||
|
$styles .= "
|
||
|
$selector .mailpoet_validate_error {color: $error}
|
||
|
$selector input.parsley-error {color: $error}
|
||
|
$selector select.parsley-error {color: $error}
|
||
|
$selector textarea.textarea.parsley-error {color: $error}
|
||
|
$selector .parsley-errors-list {color: $error}
|
||
|
$selector .parsley-required {color: $error}
|
||
|
$selector .parsley-custom-error-message {color: $error}
|
||
|
";
|
||
|
}
|
||
|
return $styles;
|
||
|
}
|
||
|
|
||
|
private function getFormTypeSpecificStyles(string $selector, string $displayType): string {
|
||
|
$styles = [];
|
||
|
if ($displayType === FormEntity::DISPLAY_TYPE_SLIDE_IN) {
|
||
|
$styles[] = $selector . '.mailpoet_form_slide_in { border-bottom-left-radius: 0; border-bottom-right-radius: 0; }';
|
||
|
$styles[] = $selector . '.mailpoet_form_position_right { border-top-right-radius: 0; }';
|
||
|
$styles[] = $selector . '.mailpoet_form_position_left { border-top-left-radius: 0; }';
|
||
|
}
|
||
|
return join('', $styles);
|
||
|
}
|
||
|
|
||
|
private function getMobileStyles(string $selector, string $displayType, array $mobileBackgrounds): string {
|
||
|
$wrapperStyles = [];
|
||
|
if ($mobileBackgrounds) {
|
||
|
$wrapperStyles[] = 'background: ' . join(', ', $mobileBackgrounds) . ';';
|
||
|
} else {
|
||
|
$wrapperStyles[] = 'background-image: none;';
|
||
|
}
|
||
|
if (
|
||
|
in_array(
|
||
|
$displayType,
|
||
|
[FormEntity::DISPLAY_TYPE_POPUP, FormEntity::DISPLAY_TYPE_FIXED_BAR, FormEntity::DISPLAY_TYPE_SLIDE_IN]
|
||
|
)
|
||
|
) {
|
||
|
$wrapperStyles = array_merge($wrapperStyles, [
|
||
|
'animation: none;',
|
||
|
'border: none;',
|
||
|
'border-radius: 0;',
|
||
|
'bottom: 0;',
|
||
|
'left: 0;',
|
||
|
'max-height: 40%;',
|
||
|
'padding: 20px;',
|
||
|
'right: 0;',
|
||
|
'top: auto;',
|
||
|
'transform: none;',
|
||
|
'width: 100%;',
|
||
|
'min-width: 100%;',
|
||
|
]);
|
||
|
}
|
||
|
return "@media (max-width: 500px) {{$selector} {" . join('', $wrapperStyles) . "}}";
|
||
|
}
|
||
|
}
|