880 lines
15 KiB
PHP
880 lines
15 KiB
PHP
<?php
|
|
|
|
namespace Kirby\Form;
|
|
|
|
use Closure;
|
|
use Exception;
|
|
use Kirby\Cms\App;
|
|
use Kirby\Cms\HasSiblings;
|
|
use Kirby\Cms\ModelWithContent;
|
|
use Kirby\Data\Data;
|
|
use Kirby\Toolkit\I18n;
|
|
use Kirby\Toolkit\Str;
|
|
use Throwable;
|
|
|
|
/**
|
|
* Abstract field class to be used instead
|
|
* of functional field components for more
|
|
* control.
|
|
*
|
|
* @package Kirby Form
|
|
* @author Bastian Allgeier <bastian@getkirby.com>
|
|
* @link https://getkirby.com
|
|
* @copyright Bastian Allgeier
|
|
* @license https://getkirby.com/license
|
|
*/
|
|
abstract class FieldClass
|
|
{
|
|
use HasSiblings;
|
|
|
|
/**
|
|
* @var string|null
|
|
*/
|
|
protected $after;
|
|
|
|
/**
|
|
* @var bool
|
|
*/
|
|
protected $autofocus;
|
|
|
|
/**
|
|
* @var string|null
|
|
*/
|
|
protected $before;
|
|
|
|
/**
|
|
* @var mixed
|
|
*/
|
|
protected $default;
|
|
|
|
/**
|
|
* @var bool
|
|
*/
|
|
protected $disabled;
|
|
|
|
/**
|
|
* @var string|null
|
|
*/
|
|
protected $help;
|
|
|
|
/**
|
|
* @var string|null
|
|
*/
|
|
protected $icon;
|
|
|
|
/**
|
|
* @var string|null
|
|
*/
|
|
protected $label;
|
|
|
|
/**
|
|
* @var \Kirby\Cms\ModelWithContent
|
|
*/
|
|
protected $model;
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
protected $name;
|
|
|
|
/**
|
|
* @var array
|
|
*/
|
|
protected $params;
|
|
|
|
/**
|
|
* @var string|null
|
|
*/
|
|
protected $placeholder;
|
|
|
|
/**
|
|
* @var bool
|
|
*/
|
|
protected $required;
|
|
|
|
/**
|
|
* @var \Kirby\Form\Fields
|
|
*/
|
|
protected $siblings;
|
|
|
|
/**
|
|
* @var bool
|
|
*/
|
|
protected $translate;
|
|
|
|
/**
|
|
* @var mixed
|
|
*/
|
|
protected $value;
|
|
|
|
/**
|
|
* @var array|null
|
|
*/
|
|
protected $when;
|
|
|
|
/**
|
|
* @var string|null
|
|
*/
|
|
protected $width;
|
|
|
|
/**
|
|
* @param string $param
|
|
* @param array $args
|
|
* @return mixed
|
|
*/
|
|
public function __call(string $param, array $args)
|
|
{
|
|
if (isset($this->$param) === true) {
|
|
return $this->$param;
|
|
}
|
|
|
|
return $this->params[$param] ?? null;
|
|
}
|
|
|
|
/**
|
|
* @param array $params
|
|
*/
|
|
public function __construct(array $params = [])
|
|
{
|
|
$this->params = $params;
|
|
|
|
$this->setAfter($params['after'] ?? null);
|
|
$this->setAutofocus($params['autofocus'] ?? false);
|
|
$this->setBefore($params['before'] ?? null);
|
|
$this->setDefault($params['default'] ?? null);
|
|
$this->setDisabled($params['disabled'] ?? false);
|
|
$this->setHelp($params['help'] ?? null);
|
|
$this->setIcon($params['icon'] ?? null);
|
|
$this->setLabel($params['label'] ?? null);
|
|
$this->setModel($params['model'] ?? App::instance()->site());
|
|
$this->setName($params['name'] ?? null);
|
|
$this->setPlaceholder($params['placeholder'] ?? null);
|
|
$this->setRequired($params['required'] ?? false);
|
|
$this->setSiblings($params['siblings'] ?? null);
|
|
$this->setTranslate($params['translate'] ?? true);
|
|
$this->setWhen($params['when'] ?? null);
|
|
$this->setWidth($params['width'] ?? null);
|
|
|
|
if (array_key_exists('value', $params) === true) {
|
|
$this->fill($params['value']);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return string|null
|
|
*/
|
|
public function after(): string|null
|
|
{
|
|
return $this->stringTemplate($this->after);
|
|
}
|
|
|
|
/**
|
|
* @return array
|
|
*/
|
|
public function api(): array
|
|
{
|
|
return $this->routes();
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function autofocus(): bool
|
|
{
|
|
return $this->autofocus;
|
|
}
|
|
|
|
/**
|
|
* @return string|null
|
|
*/
|
|
public function before(): string|null
|
|
{
|
|
return $this->stringTemplate($this->before);
|
|
}
|
|
|
|
/**
|
|
* @deprecated 3.5.0
|
|
* @todo remove when the general field class setup has been refactored
|
|
*
|
|
* Returns the field data
|
|
* in a format to be stored
|
|
* in Kirby's content fields
|
|
*
|
|
* @param bool $default
|
|
* @return mixed
|
|
*/
|
|
public function data(bool $default = false)
|
|
{
|
|
return $this->store($this->value($default));
|
|
}
|
|
|
|
/**
|
|
* Returns the default value for the field,
|
|
* which will be used when a page/file/user is created
|
|
*
|
|
* @return mixed
|
|
*/
|
|
public function default()
|
|
{
|
|
if (is_string($this->default) === false) {
|
|
return $this->default;
|
|
}
|
|
|
|
return $this->stringTemplate($this->default);
|
|
}
|
|
|
|
/**
|
|
* If `true`, the field is no longer editable and will not be saved
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function disabled(): bool
|
|
{
|
|
return $this->disabled;
|
|
}
|
|
|
|
/**
|
|
* Runs all validations and returns an array of
|
|
* error messages
|
|
*
|
|
* @return array
|
|
*/
|
|
public function errors(): array
|
|
{
|
|
return $this->validate();
|
|
}
|
|
|
|
/**
|
|
* Setter for the value
|
|
*
|
|
* @param mixed $value
|
|
* @return void
|
|
*/
|
|
public function fill($value = null)
|
|
{
|
|
$this->value = $value;
|
|
}
|
|
|
|
/**
|
|
* Optional help text below the field
|
|
*
|
|
* @return string|null
|
|
*/
|
|
public function help(): string|null
|
|
{
|
|
if (empty($this->help) === false) {
|
|
$help = $this->stringTemplate($this->help);
|
|
$help = $this->kirby()->kirbytext($help);
|
|
return $help;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param string|array|null $param
|
|
* @return string|null
|
|
*/
|
|
protected function i18n($param = null): string|null
|
|
{
|
|
return empty($param) === false ? I18n::translate($param, $param) : null;
|
|
}
|
|
|
|
/**
|
|
* Optional icon that will be shown at the end of the field
|
|
*
|
|
* @return string|null
|
|
*/
|
|
public function icon(): string|null
|
|
{
|
|
return $this->icon;
|
|
}
|
|
|
|
/**
|
|
* @return string
|
|
*/
|
|
public function id(): string
|
|
{
|
|
return $this->name();
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function isDisabled(): bool
|
|
{
|
|
return $this->disabled;
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function isEmpty(): bool
|
|
{
|
|
return $this->isEmptyValue($this->value());
|
|
}
|
|
|
|
/**
|
|
* @param mixed $value
|
|
* @return bool
|
|
*/
|
|
public function isEmptyValue($value = null): bool
|
|
{
|
|
return in_array($value, [null, '', []], true);
|
|
}
|
|
|
|
/**
|
|
* Checks if the field is invalid
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function isInvalid(): bool
|
|
{
|
|
return $this->isValid() === false;
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function isRequired(): bool
|
|
{
|
|
return $this->required;
|
|
}
|
|
|
|
/**
|
|
* @return bool
|
|
*/
|
|
public function isSaveable(): bool
|
|
{
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Checks if the field is valid
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function isValid(): bool
|
|
{
|
|
return empty($this->errors()) === true;
|
|
}
|
|
|
|
/**
|
|
* Returns the Kirby instance
|
|
*
|
|
* @return \Kirby\Cms\App
|
|
*/
|
|
public function kirby()
|
|
{
|
|
return $this->model->kirby();
|
|
}
|
|
|
|
/**
|
|
* The field label can be set as string or associative array with translations
|
|
*
|
|
* @return string
|
|
*/
|
|
public function label(): string
|
|
{
|
|
return $this->stringTemplate($this->label ?? Str::ucfirst($this->name()));
|
|
}
|
|
|
|
/**
|
|
* Returns the parent model
|
|
*
|
|
* @return mixed
|
|
*/
|
|
public function model()
|
|
{
|
|
return $this->model;
|
|
}
|
|
|
|
/**
|
|
* Returns the field name
|
|
*
|
|
* @return string
|
|
*/
|
|
public function name(): string
|
|
{
|
|
return $this->name ?? $this->type();
|
|
}
|
|
|
|
/**
|
|
* Checks if the field needs a value before being saved;
|
|
* this is the case if all of the following requirements are met:
|
|
* - The field is saveable
|
|
* - The field is required
|
|
* - The field is currently empty
|
|
* - The field is not currently inactive because of a `when` rule
|
|
*
|
|
* @return bool
|
|
*/
|
|
protected function needsValue(): bool
|
|
{
|
|
// check simple conditions first
|
|
if (
|
|
$this->isSaveable() === false ||
|
|
$this->isRequired() === false ||
|
|
$this->isEmpty() === false
|
|
) {
|
|
return false;
|
|
}
|
|
|
|
// check the data of the relevant fields if there is a `when` option
|
|
if (empty($this->when) === false && is_array($this->when) === true) {
|
|
$formFields = $this->siblings();
|
|
|
|
if ($formFields !== null) {
|
|
foreach ($this->when as $field => $value) {
|
|
$field = $formFields->get($field);
|
|
$inputValue = $field !== null ? $field->value() : '';
|
|
|
|
// if the input data doesn't match the requested `when` value,
|
|
// that means that this field is not required and can be saved
|
|
// (*all* `when` conditions must be met for this field to be required)
|
|
if ($inputValue !== $value) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// either there was no `when` condition or all conditions matched
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Returns all original params for the field
|
|
*
|
|
* @return array
|
|
*/
|
|
public function params(): array
|
|
{
|
|
return $this->params;
|
|
}
|
|
|
|
/**
|
|
* Optional placeholder value that will be shown when the field is empty
|
|
*
|
|
* @return string|null
|
|
*/
|
|
public function placeholder(): string|null
|
|
{
|
|
return $this->stringTemplate($this->placeholder);
|
|
}
|
|
|
|
/**
|
|
* Define the props that will be sent to
|
|
* the Vue component
|
|
*
|
|
* @return array
|
|
*/
|
|
public function props(): array
|
|
{
|
|
return [
|
|
'after' => $this->after(),
|
|
'autofocus' => $this->autofocus(),
|
|
'before' => $this->before(),
|
|
'default' => $this->default(),
|
|
'disabled' => $this->isDisabled(),
|
|
'help' => $this->help(),
|
|
'icon' => $this->icon(),
|
|
'label' => $this->label(),
|
|
'name' => $this->name(),
|
|
'placeholder' => $this->placeholder(),
|
|
'required' => $this->isRequired(),
|
|
'saveable' => $this->isSaveable(),
|
|
'translate' => $this->translate(),
|
|
'type' => $this->type(),
|
|
'when' => $this->when(),
|
|
'width' => $this->width(),
|
|
];
|
|
}
|
|
|
|
/**
|
|
* If `true`, the field has to be filled in correctly to be saved.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function required(): bool
|
|
{
|
|
return $this->required;
|
|
}
|
|
|
|
/**
|
|
* Routes for the field API
|
|
*
|
|
* @return array
|
|
*/
|
|
public function routes(): array
|
|
{
|
|
return [];
|
|
}
|
|
|
|
/**
|
|
* @deprecated 3.5.0
|
|
* @todo remove when the general field class setup has been refactored
|
|
* @return bool
|
|
*/
|
|
public function save()
|
|
{
|
|
return $this->isSaveable();
|
|
}
|
|
|
|
/**
|
|
* @param array|string|null $after
|
|
* @return void
|
|
*/
|
|
protected function setAfter($after = null)
|
|
{
|
|
$this->after = $this->i18n($after);
|
|
}
|
|
|
|
/**
|
|
* @param bool $autofocus
|
|
* @return void
|
|
*/
|
|
protected function setAutofocus(bool $autofocus = false)
|
|
{
|
|
$this->autofocus = $autofocus;
|
|
}
|
|
|
|
/**
|
|
* @param array|string|null $before
|
|
* @return void
|
|
*/
|
|
protected function setBefore($before = null)
|
|
{
|
|
$this->before = $this->i18n($before);
|
|
}
|
|
|
|
/**
|
|
* @param mixed $default
|
|
* @return void
|
|
*/
|
|
protected function setDefault($default = null)
|
|
{
|
|
$this->default = $default;
|
|
}
|
|
|
|
/**
|
|
* @param bool $disabled
|
|
* @return void
|
|
*/
|
|
protected function setDisabled(bool $disabled = false)
|
|
{
|
|
$this->disabled = $disabled;
|
|
}
|
|
|
|
/**
|
|
* @param array|string|null $help
|
|
* @return void
|
|
*/
|
|
protected function setHelp($help = null)
|
|
{
|
|
$this->help = $this->i18n($help);
|
|
}
|
|
|
|
/**
|
|
* @param string|null $icon
|
|
* @return void
|
|
*/
|
|
protected function setIcon(string|null $icon = null)
|
|
{
|
|
$this->icon = $icon;
|
|
}
|
|
|
|
/**
|
|
* @param array|string|null $label
|
|
* @return void
|
|
*/
|
|
protected function setLabel($label = null)
|
|
{
|
|
$this->label = $this->i18n($label);
|
|
}
|
|
|
|
/**
|
|
* @param \Kirby\Cms\ModelWithContent $model
|
|
* @return void
|
|
*/
|
|
protected function setModel(ModelWithContent $model)
|
|
{
|
|
$this->model = $model;
|
|
}
|
|
|
|
/**
|
|
* @param string|null $name
|
|
* @return void
|
|
*/
|
|
protected function setName(string $name = null)
|
|
{
|
|
$this->name = $name;
|
|
}
|
|
|
|
/**
|
|
* @param array|string|null $placeholder
|
|
* @return void
|
|
*/
|
|
protected function setPlaceholder($placeholder = null)
|
|
{
|
|
$this->placeholder = $this->i18n($placeholder);
|
|
}
|
|
|
|
/**
|
|
* @param bool $required
|
|
* @return void
|
|
*/
|
|
protected function setRequired(bool $required = false)
|
|
{
|
|
$this->required = $required;
|
|
}
|
|
|
|
/**
|
|
* @param \Kirby\Form\Fields|null $siblings
|
|
* @return void
|
|
*/
|
|
protected function setSiblings(?Fields $siblings = null)
|
|
{
|
|
$this->siblings = $siblings ?? new Fields([$this]);
|
|
}
|
|
|
|
/**
|
|
* @param bool $translate
|
|
* @return void
|
|
*/
|
|
protected function setTranslate(bool $translate = true)
|
|
{
|
|
$this->translate = $translate;
|
|
}
|
|
|
|
/**
|
|
* Setter for the when condition
|
|
*
|
|
* @param mixed $when
|
|
* @return void
|
|
*/
|
|
protected function setWhen($when = null)
|
|
{
|
|
$this->when = $when;
|
|
}
|
|
|
|
/**
|
|
* Setter for the field width
|
|
*
|
|
* @param string|null $width
|
|
* @return void
|
|
*/
|
|
protected function setWidth(string $width = null)
|
|
{
|
|
$this->width = $width;
|
|
}
|
|
|
|
/**
|
|
* Returns all sibling fields
|
|
*
|
|
* @return \Kirby\Form\Fields
|
|
*/
|
|
protected function siblingsCollection()
|
|
{
|
|
return $this->siblings;
|
|
}
|
|
|
|
/**
|
|
* Parses a string template in the given value
|
|
*
|
|
* @param string|null $string
|
|
* @return string|null
|
|
*/
|
|
protected function stringTemplate(string|null $string = null): string|null
|
|
{
|
|
if ($string !== null) {
|
|
return $this->model->toString($string);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Converts the given value to a value
|
|
* that can be stored in the text file
|
|
*
|
|
* @param mixed $value
|
|
* @return mixed
|
|
*/
|
|
public function store($value)
|
|
{
|
|
return $value;
|
|
}
|
|
|
|
/**
|
|
* Should the field be translatable?
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function translate(): bool
|
|
{
|
|
return $this->translate;
|
|
}
|
|
|
|
/**
|
|
* Converts the field to a plain array
|
|
*
|
|
* @return array
|
|
*/
|
|
public function toArray(): array
|
|
{
|
|
$props = $this->props();
|
|
$props['signature'] = md5(json_encode($props));
|
|
|
|
ksort($props);
|
|
|
|
return array_filter($props, fn ($item) => $item !== null);
|
|
}
|
|
|
|
/**
|
|
* Returns the field type
|
|
*
|
|
* @return string
|
|
*/
|
|
public function type(): string
|
|
{
|
|
return lcfirst(basename(str_replace(['\\', 'Field'], ['/', ''], static::class)));
|
|
}
|
|
|
|
/**
|
|
* Runs the validations defined for the field
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function validate(): array
|
|
{
|
|
$validations = $this->validations();
|
|
$value = $this->value();
|
|
$errors = [];
|
|
|
|
// validate required values
|
|
if ($this->needsValue() === true) {
|
|
$errors['required'] = I18n::translate('error.validation.required');
|
|
}
|
|
|
|
foreach ($validations as $key => $validation) {
|
|
if (is_int($key) === true) {
|
|
// predefined validation
|
|
try {
|
|
Validations::$validation($this, $value);
|
|
} catch (Exception $e) {
|
|
$errors[$validation] = $e->getMessage();
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if ($validation instanceof Closure) {
|
|
try {
|
|
$validation->call($this, $value);
|
|
} catch (Exception $e) {
|
|
$errors[$key] = $e->getMessage();
|
|
}
|
|
}
|
|
}
|
|
|
|
return $errors;
|
|
}
|
|
|
|
/**
|
|
* Defines all validation rules
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function validations(): array
|
|
{
|
|
return [];
|
|
}
|
|
|
|
/**
|
|
* Returns the value of the field if saveable
|
|
* otherwise it returns null
|
|
*
|
|
* @return mixed
|
|
*/
|
|
public function value(bool $default = false)
|
|
{
|
|
if ($this->isSaveable() === false) {
|
|
return null;
|
|
}
|
|
|
|
if ($default === true && $this->isEmpty() === true) {
|
|
return $this->default();
|
|
}
|
|
|
|
return $this->value;
|
|
}
|
|
|
|
/**
|
|
* @param mixed $value
|
|
* @return array
|
|
*/
|
|
protected function valueFromJson($value): array
|
|
{
|
|
try {
|
|
return Data::decode($value, 'json');
|
|
} catch (Throwable) {
|
|
return [];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param mixed $value
|
|
* @return array
|
|
*/
|
|
protected function valueFromYaml($value): array
|
|
{
|
|
return Data::decode($value, 'yaml');
|
|
}
|
|
|
|
/**
|
|
* @param array|null $value
|
|
* @param bool $pretty
|
|
* @return string
|
|
*/
|
|
protected function valueToJson(array $value = null, bool $pretty = false): string
|
|
{
|
|
$constants = JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE;
|
|
|
|
if ($pretty === true) {
|
|
$constants |= JSON_PRETTY_PRINT;
|
|
}
|
|
|
|
return json_encode($value, $constants);
|
|
}
|
|
|
|
/**
|
|
* @param array|null $value
|
|
* @return string
|
|
*/
|
|
protected function valueToYaml(array $value = null): string
|
|
{
|
|
return Data::encode($value, 'yaml');
|
|
}
|
|
|
|
/**
|
|
* Conditions when the field will be shown
|
|
*
|
|
* @return array|null
|
|
*/
|
|
public function when(): array|null
|
|
{
|
|
return $this->when;
|
|
}
|
|
|
|
/**
|
|
* Returns the width of the field in
|
|
* the Panel grid
|
|
*
|
|
* @return string
|
|
*/
|
|
public function width(): string
|
|
{
|
|
return $this->width ?? '1/1';
|
|
}
|
|
}
|