View source
<?php
declare (strict_types=1);
namespace Drupal\date_recur\Plugin\Field\FieldType;
use Drupal\Component\Utility\NestedArray;
use Drupal\Core\Datetime\DrupalDateTime;
use Drupal\Core\Field\FieldDefinitionInterface;
use Drupal\Core\Field\FieldStorageDefinitionInterface;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\StringTranslation\TranslatableMarkup;
use Drupal\Core\TypedData\DataDefinition;
use Drupal\Core\TypedData\ListDataDefinition;
use Drupal\date_recur\DateRecurHelper;
use Drupal\date_recur\DateRecurHelperInterface;
use Drupal\date_recur\DateRecurNonRecurringHelper;
use Drupal\date_recur\DateRecurRruleMap;
use Drupal\date_recur\Exception\DateRecurHelperArgumentException;
use Drupal\date_recur\Plugin\Field\DateRecurDateTimeComputed;
use Drupal\date_recur\Plugin\Field\DateRecurOccurrencesComputed;
use Drupal\datetime_range\Plugin\Field\FieldType\DateRangeItem;
class DateRecurItem extends DateRangeItem {
public const PART_SUPPORTS_ALL = '*';
public const FREQUENCY_SETTINGS_DISABLED = 'disabled';
public const FREQUENCY_SETTINGS_PARTS_ALL = 'all-parts';
public const FREQUENCY_SETTINGS_PARTS_PARTIAL = 'some-parts';
protected $helper;
public static function propertyDefinitions(FieldStorageDefinitionInterface $field_definition) : array {
$properties = parent::propertyDefinitions($field_definition);
$startDateProperty = $properties['start_date'];
$startDateProperty
->setClass(DateRecurDateTimeComputed::class);
$endDateProperty = $properties['end_date'];
$endDateProperty
->setClass(DateRecurDateTimeComputed::class);
$properties['rrule'] = DataDefinition::create('string')
->setLabel((string) new TranslatableMarkup('RRule'))
->setRequired(FALSE);
$rruleMaxLength = $field_definition
->getSetting('rrule_max_length');
assert(empty($rruleMaxLength) || is_numeric($rruleMaxLength) && $rruleMaxLength > 0);
if (!empty($rruleMaxLength)) {
$properties['rrule']
->addConstraint('Length', [
'max' => $rruleMaxLength,
]);
}
$properties['timezone'] = DataDefinition::create('string')
->setLabel((string) new TranslatableMarkup('Timezone'))
->setRequired(TRUE)
->addConstraint('DateRecurTimeZone');
$properties['infinite'] = DataDefinition::create('boolean')
->setLabel((string) new TranslatableMarkup('Whether the RRule is an infinite rule. Derived value from RRULE.'))
->setRequired(FALSE);
$properties['occurrences'] = ListDataDefinition::create('any')
->setLabel((string) new TranslatableMarkup('Occurrences'))
->setComputed(TRUE)
->setClass(DateRecurOccurrencesComputed::class);
return $properties;
}
public static function schema(FieldStorageDefinitionInterface $field_definition) : array {
$schema = parent::schema($field_definition);
$schema['columns']['rrule'] = [
'description' => 'The repeat rule.',
'type' => 'text',
];
$schema['columns']['timezone'] = [
'description' => 'The timezone.',
'type' => 'varchar',
'length' => 255,
];
$schema['columns']['infinite'] = [
'description' => 'Whether the RRule is an infinite rule. Derived value from RRULE.',
'type' => 'int',
'size' => 'tiny',
];
return $schema;
}
public static function defaultStorageSettings() : array {
return [
'rrule_max_length' => 256,
] + parent::defaultStorageSettings();
}
public static function defaultFieldSettings() : array {
return [
'precreate' => 'P2Y',
'parts' => [
'all' => TRUE,
'frequencies' => [],
],
] + parent::defaultFieldSettings();
}
public function storageSettingsForm(array &$form, FormStateInterface $form_state, $has_data) : array {
assert(is_bool($has_data));
$element = parent::storageSettingsForm($form, $form_state, $has_data);
$element['rrule_max_length'] = [
'#type' => 'number',
'#title' => $this
->t('Maximum character length of RRULE'),
'#description' => $this
->t('Define the maximum characters a RRULE can contain.'),
'#default_value' => $this
->getSetting('rrule_max_length'),
'#min' => 0,
];
return $element;
}
public function fieldSettingsForm(array $form, FormStateInterface $form_state) : array {
$elementParts = [
'settings',
];
$element = parent::fieldSettingsForm($form, $form_state);
$options = [];
foreach (range(1, 5) as $i) {
$options['P' . $i . 'Y'] = $this
->formatPlural($i, '@year year', '@year years', [
'@year' => $i,
]);
}
$element['precreate'] = [
'#type' => 'select',
'#title' => $this
->t('Precreate occurrences'),
'#description' => $this
->t('For infinitely repeating dates, precreate occurrences for this amount of time in the views cache table.'),
'#options' => $options,
'#default_value' => $this
->getSetting('precreate'),
];
$element['parts'] = [
'#type' => 'container',
];
$element['parts']['#after_build'][] = [
get_class($this),
'partsAfterBuild',
];
$allPartsSettings = $this
->getSetting('parts');
$element['parts']['all'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Allow all frequency and parts'),
'#default_value' => $allPartsSettings['all'] ?? TRUE,
];
$parents = array_merge($elementParts, [
'parts',
'all',
]);
$allPartsCheckboxName = $parents[0] . '[' . implode('][', array_slice($parents, 1)) . ']';
$frequencyLabels = DateRecurRruleMap::frequencyLabels();
$partLabels = DateRecurRruleMap::partLabels();
$partsCheckboxes = [];
foreach (DateRecurRruleMap::PARTS as $part) {
$partsCheckboxes[$part] = [
'#type' => 'checkbox',
'#title' => $partLabels[$part],
];
}
$settingsOptions = [
static::FREQUENCY_SETTINGS_DISABLED => $this
->t('Disabled'),
static::FREQUENCY_SETTINGS_PARTS_ALL => $this
->t('All parts'),
static::FREQUENCY_SETTINGS_PARTS_PARTIAL => $this
->t('Specify parts'),
];
$element['parts']['table'] = [
'#theme' => 'date_recur_settings_frequency_table',
'#type' => 'container',
'#states' => [
'visible' => [
':input[name="' . $allPartsCheckboxName . '"]' => [
'checked' => FALSE,
],
],
],
];
foreach (DateRecurRruleMap::FREQUENCIES as $frequency) {
$row = [];
$row['frequency']['#markup'] = $frequencyLabels[$frequency];
$parents = array_merge($elementParts, [
'parts',
'table',
$frequency,
'setting',
]);
$settingsCheckboxName = $parents[0] . '[' . implode('][', array_slice($parents, 1)) . ']';
$enabledParts = $allPartsSettings['frequencies'][$frequency] ?? [];
$defaultSetting = NULL;
if (count($enabledParts) === 0) {
$defaultSetting = static::FREQUENCY_SETTINGS_DISABLED;
}
elseif (in_array(static::PART_SUPPORTS_ALL, $enabledParts)) {
$defaultSetting = static::FREQUENCY_SETTINGS_PARTS_ALL;
}
elseif (count($enabledParts) > 0) {
$defaultSetting = static::FREQUENCY_SETTINGS_PARTS_PARTIAL;
}
$row['setting'] = [
'#type' => 'radios',
'#options' => $settingsOptions,
'#required' => TRUE,
'#default_value' => $defaultSetting,
];
$row['parts'] = $partsCheckboxes;
foreach ($row['parts'] as $part => &$partsCheckbox) {
$partsCheckbox['#states']['visible'][] = [
':input[name="' . $settingsCheckboxName . '"]' => [
'value' => static::FREQUENCY_SETTINGS_PARTS_PARTIAL,
],
];
$partsCheckbox['#default_value'] = in_array($part, $enabledParts, TRUE);
}
$element['parts']['table'][$frequency] = $row;
}
$list = [];
$partLabels = DateRecurRruleMap::partLabels();
foreach (DateRecurRruleMap::partDescriptions() as $part => $partDescription) {
$list[] = $this
->t('<strong>@label:</strong> @description', [
'@label' => $partLabels[$part],
'@description' => $partDescription,
]);
}
$element['parts']['help']['#markup'] = '<ul><li>' . implode('</li><li>', $list) . '</li></ul></ul>';
return $element;
}
public static function partsAfterBuild(array $element, FormStateInterface $form_state) : array {
$values = NestedArray::getValue($form_state
->getValues(), $element['#parents']);
NestedArray::unsetValue($form_state
->getValues(), $element['#parents']);
$parts = [];
$parts['all'] = !empty($values['all']);
$parts['frequencies'] = [];
foreach ($values['table'] as $frequency => $row) {
$enabledParts = array_keys(array_filter($row['parts']));
if ($row['setting'] === static::FREQUENCY_SETTINGS_PARTS_ALL) {
$enabledParts[] = static::PART_SUPPORTS_ALL;
}
elseif ($row['setting'] === static::FREQUENCY_SETTINGS_DISABLED) {
$enabledParts = [];
}
sort($enabledParts);
$parts['frequencies'][$frequency] = $enabledParts;
}
$form_state
->setValue($element['#parents'], $parts);
return $element;
}
public function getDateStorageFormat() : string {
return $this
->getSetting('datetime_type') == static::DATETIME_TYPE_DATE ? static::DATE_STORAGE_FORMAT : static::DATETIME_STORAGE_FORMAT;
}
public function preSave() : void {
parent::preSave();
try {
$isInfinite = $this
->getHelper()
->isInfinite();
} catch (DateRecurHelperArgumentException $e) {
$isInfinite = FALSE;
}
$this
->get('infinite')
->setValue($isInfinite);
}
public function setValue($values, $notify = TRUE) : void {
$values['infinite'] = !empty($values['infinite']);
$this
->resetHelper();
parent::setValue($values, $notify);
}
public function onChange($property_name, $notify = TRUE) {
if (in_array($property_name, [
'value',
'end_value',
'rrule',
'timezone',
])) {
$this
->resetHelper();
}
parent::onChange($property_name, $notify);
}
public function isRecurring() : bool {
return !empty($this->rrule);
}
public function getHelper() : DateRecurHelperInterface {
if (isset($this->helper)) {
return $this->helper;
}
try {
$timeZoneString = $this->timezone;
$timeZone = new \DateTimeZone(is_string($timeZoneString) ? $timeZoneString : '');
} catch (\Exception $exception) {
throw new DateRecurHelperArgumentException('Invalid time zone');
}
$startDate = NULL;
$startDateEnd = NULL;
if ($this->start_date instanceof DrupalDateTime) {
$startDate = $this->start_date
->getPhpDateTime();
$startDate
->setTimezone($timeZone);
}
else {
throw new DateRecurHelperArgumentException('Start date is required.');
}
if ($this->end_date instanceof DrupalDateTime) {
$startDateEnd = $this->end_date
->getPhpDateTime();
$startDateEnd
->setTimezone($timeZone);
}
$this->helper = $this
->isRecurring() ? DateRecurHelper::create((string) $this->rrule, $startDate, $startDateEnd) : DateRecurNonRecurringHelper::createInstance('', $startDate, $startDateEnd);
return $this->helper;
}
public function isEmpty() : bool {
$start_value = $this
->get('value')
->getValue();
$end_value = $this
->get('end_value')
->getValue();
return $start_value === NULL || $start_value === '' || ($end_value === NULL || $end_value === '') || empty($this
->get('timezone')
->getValue());
}
public static function generateSampleValue(FieldDefinitionInterface $field_definition) : array {
$values = parent::generateSampleValue($field_definition);
$timeZoneList = timezone_identifiers_list();
$values['timezone'] = $timeZoneList[array_rand($timeZoneList)];
$values['rrule'] = 'FREQ=DAILY;COUNT=' . rand(2, 10);
$values['infinite'] = FALSE;
return $values;
}
public function resetHelper() : void {
$this->helper = NULL;
}
}