diff --git a/modules/json_form_widget/json_form_widget.module b/modules/json_form_widget/json_form_widget.module index 444a8d4849..ccdef19e86 100644 --- a/modules/json_form_widget/json_form_widget.module +++ b/modules/json_form_widget/json_form_widget.module @@ -35,7 +35,7 @@ function json_form_widget_remove_one(array &$form, FormStateInterface $form_stat /** * Update count property by the given offset. * - * @param FormStateInterface $form_state + * @param \Drupal\Core\Form\FormStateInterface $form_state * Form state. * @param int $offset * Offset to change count by. diff --git a/modules/json_form_widget/src/ArrayHelper.php b/modules/json_form_widget/src/ArrayHelper.php index 09b13e19fe..e5866658dc 100644 --- a/modules/json_form_widget/src/ArrayHelper.php +++ b/modules/json_form_widget/src/ArrayHelper.php @@ -2,9 +2,10 @@ namespace Drupal\json_form_widget; -use Drupal\Core\Form\FormStateInterface; +use Drupal\Component\Utility\NestedArray; use Drupal\Core\DependencyInjection\ContainerInjectionInterface; use Drupal\Core\DependencyInjection\DependencySerializationTrait; +use Drupal\Core\Form\FormStateInterface; use Drupal\Core\StringTranslation\StringTranslationTrait; use Symfony\Component\DependencyInjection\ContainerInterface; @@ -30,9 +31,7 @@ class ArrayHelper implements ContainerInjectionInterface { public FieldTypeRouter $builder; /** - * Inherited. - * - * @{inheritdocs} + * {@inheritdoc} */ public static function create(ContainerInterface $container) { return new static( @@ -56,7 +55,7 @@ public function setBuilder(FieldTypeRouter $builder): void { } /** - * Update wrapper element of the triggering button after build. + * Shared AJAX callback function for all array buttons. * * @param array $form * Newly built form render array. @@ -66,7 +65,7 @@ public function setBuilder(FieldTypeRouter $builder): void { * @return array * Field wrapper render array. */ - public function addOrRemoveButtonCallback(array &$form, FormStateInterface $form_state): array { + public function arrayActionButtonCallback(array &$form, FormStateInterface $form_state): array { // Retrieve triggering button element. $button = $form_state->getTriggeringElement(); // Extract full heritage for the triggered button. @@ -96,37 +95,150 @@ public function addOrRemoveButtonCallback(array &$form, FormStateInterface $form * Handle form element for an array. */ public function handleArrayElement(array $definition, ?array $data, FormStateInterface $form_state, array $context): array { - // Extract field name from field definition and min items from field schema - // for later reference. - $field_name = $definition['name']; + // Extract field name from field definition and min items from field schema. $min_items = $definition['schema']->minItems ?? 0; - // Build context name. + $context_name = self::buildContextName($context); - // Determine number of form items to generate. $item_count = $this->getItemCount($context_name, count($data ?? []), $min_items, $form_state); + $is_required = in_array($definition['name'], $this->builder->getSchema()->required ?? []); + + // Build the parent fieldset. + $element = $this->buildArrayParentElement($definition, $is_required, $context_name); - // Determine if this field is required. - $required_fields = $this->builder->getSchema()->required ?? []; - $field_required = in_array($field_name, $required_fields); // Build the specified number of field item elements. - $field_properties = []; + $items = []; for ($i = 0; $i < $item_count; $i++) { - $property_required = $field_required && ($i < $min_items); - $field_properties[] = $this->buildArrayElement($definition, $data[$i] ?? NULL, $form_state, array_merge($context, [$i]), $property_required); + $item = $this->buildArrayItemElement($definition, $data[$i] ?? NULL, $form_state, array_merge($context, [$i])); + $item['#required'] = $is_required && ($i < $min_items); + $items[] = $item; } + $element[$definition['name']] = $items; + return $element; + } - // Build field element. - return [ - '#type' => 'fieldset', - '#title' => ($definition['schema']->title ?? $field_name), - '#description' => ($definition['schema']->description ?? ''), + /** + * Build the parent fieldset for an array. + * + * @param array $definition + * Field definition. + * @param bool $is_required + * Whether the field is required. + * @param string $context_name + * Field context name. + * + * @return array + * Render array for the array parent element. + */ + protected function buildArrayParentElement(array $definition, bool $is_required, string $context_name) { + $element = [ + '#type' => 'fieldset', + '#title' => ($definition['schema']->title ?? $definition['name']), + '#description' => ($definition['schema']->description ?? ''), '#description_display' => 'before', - '#prefix' => '
', - '#suffix' => '
', - '#tree' => TRUE, - 'actions' => $this->buildActions($item_count, $min_items, $field_name, $context_name), - $field_name => $field_properties, + '#prefix' => '
', + '#suffix' => '
', + '#tree' => TRUE, + '#required' => $is_required, + 'actions' => [ + '#type' => 'actions', + 'actions' => [ + 'add' => $this->buildAction($this->t('Add one'), 'addOne', $definition['name'], $context_name), + ], + ], ]; + return $element; + } + + /** + * Build a single element from an array. + * + * @param array $definition + * Field definition. + * @param mixed $data + * Field data. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * Form state. + * @param string[] $context + * Field context. + * + * @return array + * Render array for the array element. + */ + protected function buildArrayItemElement(array $definition, $data, FormStateInterface $form_state, array $context): array { + // Use the simple or complex method depending on whether items are objects. + if (isset($definition['schema']->items->properties)) { + $element = $this->buildComplexArrayElement($definition, $data, $form_state, $context); + } + else { + $element = $this->buildSimpleArrayElement($definition, $data, $context); + } + return $element; + } + + /** + * Returns single simple element from array. + * + * @param array $definition + * Field definition. + * @param mixed $data + * Field data. + * @param string[] $context + * Field context. + * + * @return array + * Render array for the simple array element. + */ + protected function buildSimpleArrayElement(array $definition, $data, array $context): array { + return [ + '#type' => 'fieldset', + '#attributes' => [ + 'data-parent' => $definition['name'], + 'class' => ['json-form-widget-array-item'], + ], + 'field' => array_filter([ + '#type' => 'textfield', + '#title' => $definition['schema']->items->title ?? NULL, + '#default_value' => $data, + ]), + 'actions' => $this->buildElementActions($definition['name'], self::buildContextName($context)), + ]; + } + + /** + * Flatten array element fieldset w/buttons for processing. + * + * @param array $element + * A form element. + */ + public static function flattenArrayElementFieldset(array &$element): void { + if (isset($element['field']) && $element['#type'] == 'fieldset') { + $element = ['#required' => ($element['#required'] ?? FALSE)] + $element['field']; + } + } + + /** + * Returns single complex element from array. + * + * @param array $definition + * Field definition. + * @param mixed $data + * Field data. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * Form state. + * @param string[] $context + * Field context. + * + * @return array + * Render array for the complex array element. + */ + protected function buildComplexArrayElement(array $definition, $data, FormStateInterface $form_state, array $context): array { + $subdefinition = [ + 'name' => $definition['name'], + 'schema' => $definition['schema']->items, + ]; + $element = $this->objectHelper->handleObjectElement($subdefinition, $data, $form_state, $context, $this->builder); + $element[$definition['name']]['actions'] = $this->buildElementActions($definition['name'], self::buildContextName($context)); + return $element; } /** @@ -200,85 +312,182 @@ public static function buildCountProperty(string $context_name): array { } /** - * Helper function to build form field actions. - */ - protected function buildActions(int $item_count, int $min_items, string $parent, string $context_name): array { - $actions = []; - - // Build add action. - $actions['add'] = $this->buildAction($this->t('Add one'), 'json_form_widget_add_one', $parent, $context_name); - // Build remove action if there are more than the minimum required elements - // in this field array. - if ($item_count > $min_items) { - $actions['remove'] = $this->buildAction($this->t('Remove one'), 'json_form_widget_remove_one', $parent, $context_name); - } - - return [ - '#type' => 'actions', - 'actions' => $actions, - ]; - } - - /** - * Helper function to get action. + * Helper function to build an action button. + * + * @param string $title + * Button title. + * @param string $method + * Button submit method; should be a static method from this class. + * @param string $parent + * The parent element for the action; usually the current field name. + * @param string $context_name + * The context name, output of ::buildContextName(). */ - protected function buildAction(string $title, string $function, string $parent, string $context_name): array { - return [ + protected function buildAction(string $title, string $method, string $parent, string $context_name): array { + $action = [ '#type' => 'submit', '#name' => $context_name, '#value' => $title, - '#submit' => [$function], + '#submit' => [self::class . '::' . $method], '#ajax' => [ - 'callback' => [$this, 'addOrRemoveButtonCallback'], - 'wrapper' => self::buildWrapperIdentifier($context_name), + 'callback' => [$this, 'arrayActionButtonCallback'], + 'wrapper' => self::buildWrapperIdentifier($parent), ], '#attributes' => [ 'data-parent' => $parent, - 'data-context' => $context_name, ], '#limit_validation_errors' => [], ]; + return $action; + } + + /** + * Build the remove/reorder actions for a single element. + * + * @param string $parent + * Parent element name. + * @param string $context_name + * Data context. + * + * @return array + * Actions render array. + */ + protected function buildElementActions(string $parent, string $context_name):array { + return [ + '#type' => 'actions', + 'remove' => $this->buildAction($this->t('Remove'), 'remove', $parent, $context_name), + 'move_up' => $this->buildAction($this->t('Move Up'), 'moveUp', $parent, $context_name), + 'move_down' => $this->buildAction($this->t('Move Down'), 'moveDown', $parent, $context_name), + ]; } /** - * Handle single element from array. + * Submit function for element "remove" button. * - * Chooses whether element is simple or complex. + * @param array $form + * Form render array. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * Form state. */ - protected function buildArrayElement(array $definition, $data, FormStateInterface $form_state, array $context, bool $required): array { - // If this element's definition has properties defined... - $element = isset($definition['schema']->items->properties) ? - // Attempt to build a complex element, otherwise... - $this->buildComplexArrayElement($definition, $data, $form_state, $context) : - // Build a simple element. - $this->buildSimpleArrayElement($definition, $data); + public static function remove(array &$form, FormStateInterface $form_state) { + $button_element = $form_state->getTriggeringElement(); + $parent = $button_element['#attributes']['data-parent']; + $parents = $button_element['#parents']; + $element_index = str_replace("{$parent}-", '', $button_element['#name']); + $count_property = self::buildCountProperty($parent); + $user_input = $form_state->getUserInput(); + + // Update the user input to remove the specific element. + $key_exists = NULL; + static::trimParents($parents, $element_index); + $input_values = &NestedArray::getValue($user_input, $parents, $key_exists); + if ($key_exists) { + unset($input_values[$element_index]); + // Re-index the array to maintain proper keys. + $input_values = \array_values($input_values); + } - // Set element requirement. - $element['#required'] = $required; + $form_state->setUserInput($user_input); - return $element; + // Modify stored item count. The form rebuilds before the alter, so it needs + // to be one more than the current item count to avoid removing twice. + $item_count = count($input_values); + $form_state->set($count_property, $item_count); + + $form_state->setRebuild(); } /** - * Returns single simple element from array. + * Submit function for element "move up" button. + * + * @param array $form + * Form render array. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * Form state. */ - protected function buildSimpleArrayElement(array $definition, $data): array { - return array_filter([ - '#type' => 'textfield', - '#title' => $definition['schema']->items->title ?? NULL, - '#default_value' => $data, - ]); + public static function moveUp(array &$form, FormStateInterface $form_state) { + return static::moveElement($form_state, -1); } /** - * Returns single complex element from array. + * Submit function for element "move down" button. + * + * @param array $form + * Form render array. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * Form state. */ - protected function buildComplexArrayElement(array $definition, $data, FormStateInterface $form_state, array $context): array { - $subdefinition = [ - 'name' => $definition['name'], - 'schema' => $definition['schema']->items, - ]; - return $this->objectHelper->handleObjectElement($subdefinition, $data, $form_state, $context, $this->builder); + public static function moveDown(array &$form, FormStateInterface $form_state) { + return static::moveElement($form_state, 1); + } + + /** + * Common function to move element within array by the given offset. + * + * @param \Drupal\Core\Form\FormStateInterface $form_state + * Form state. + * @param int $offset + * Offset to move the element by. + */ + protected static function moveElement(FormStateInterface $form_state, int $offset) { + $button_element = $form_state->getTriggeringElement(); + $parent = $button_element['#attributes']['data-parent']; + $parents = $button_element['#parents']; + $element_index = str_replace("{$parent}-", '', $button_element['#name']); + $user_input = $form_state->getUserInput(); + + // Update the user input to change the order. + $key_exists = NULL; + static::trimParents($parents, $element_index); + $input_values = &NestedArray::getValue($user_input, $parents, $key_exists); + if ($key_exists) { + $moved_element = array_splice($input_values, $element_index, 1); + array_splice($input_values, $element_index + $offset, 0, $moved_element); + // Re-index the array to maintain proper keys. + $input_values = \array_values($input_values); + } + + $form_state->setUserInput($user_input); + $form_state->setRebuild(); + } + + /** + * Submit function for array "add one" button. + * + * @param array $form + * Form render array. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * Form state. + */ + public static function addOne(array &$form, FormStateInterface $form_state) { + $button_element = $form_state->getTriggeringElement(); + $count_property = static::buildCountProperty($button_element['#name']); + // Modify stored item count. + $item_count = $form_state->get($count_property) ?? 0; + $item_count++; + $form_state->set($count_property, $item_count); + $form_state->setRebuild(); + } + + /** + * Utility function to trim the triggering element's parents array. + * + * Used to get the correct position in the user input array for modifications. + * + * @param array $parents + * Parents array. + * @param int $element_index + * Element index. + */ + public static function trimParents(array &$parents, int $element_index): void { + for ($i = count($parents) - 1; $i >= 0; $i--) { + if ($parents[$i] == $element_index) { + $ei_position = $i; + break; + } + } + $offset = 0 - (count($parents) - $ei_position); + \array_splice($parents, $offset); } } diff --git a/modules/json_form_widget/src/FormBuilder.php b/modules/json_form_widget/src/FormBuilder.php index 0eef2582f6..38f2771117 100644 --- a/modules/json_form_widget/src/FormBuilder.php +++ b/modules/json_form_widget/src/FormBuilder.php @@ -68,7 +68,7 @@ public function __construct( SchemaRetriever $schema_retriever, FieldTypeRouter $router, SchemaUiHandler $schema_ui_handler, - LoggerInterface $loggerChannel + LoggerInterface $loggerChannel, ) { $this->schemaRetriever = $schema_retriever; $this->router = $router; diff --git a/modules/json_form_widget/src/Plugin/Field/FieldWidget/JsonFormWidget.php b/modules/json_form_widget/src/Plugin/Field/FieldWidget/JsonFormWidget.php index 57a632e976..3c260f3a6f 100644 --- a/modules/json_form_widget/src/Plugin/Field/FieldWidget/JsonFormWidget.php +++ b/modules/json_form_widget/src/Plugin/Field/FieldWidget/JsonFormWidget.php @@ -3,14 +3,14 @@ namespace Drupal\json_form_widget\Plugin\Field\FieldWidget; use Drupal\Core\Entity\FieldableEntityInterface; +use Drupal\Core\Field\FieldDefinitionInterface; use Drupal\Core\Field\FieldItemListInterface; use Drupal\Core\Field\WidgetBase; use Drupal\Core\Form\FormStateInterface; use Drupal\json_form_widget\FormBuilder; +use Drupal\json_form_widget\ValueHandler; use Symfony\Component\DependencyInjection\ContainerInterface; -use Drupal\Core\Field\FieldDefinitionInterface; use Symfony\Component\HttpFoundation\RequestStack; -use Drupal\json_form_widget\ValueHandler; /** * Plugin implementation of the 'json_form_widget'. @@ -82,7 +82,7 @@ public function __construct( array $third_party_settings, FormBuilder $builder, ValueHandler $value_handler, - RequestStack $request_stack + RequestStack $request_stack, ) { parent::__construct($plugin_id, $plugin_definition, $field_definition, $settings, $third_party_settings); $this->builder = $builder; diff --git a/modules/json_form_widget/src/SchemaUiHandler.php b/modules/json_form_widget/src/SchemaUiHandler.php index 8d47b9a5eb..b8e67532f6 100644 --- a/modules/json_form_widget/src/SchemaUiHandler.php +++ b/modules/json_form_widget/src/SchemaUiHandler.php @@ -66,7 +66,7 @@ public static function create(ContainerInterface $container) { public function __construct( SchemaRetriever $schema_retriever, LoggerInterface $loggerChannel, - WidgetRouter $widget_router + WidgetRouter $widget_router, ) { $this->schemaRetriever = $schema_retriever; $this->schemaUi = FALSE; @@ -199,7 +199,13 @@ public function flattenArrays(mixed $spec, array $element) { unset($element[$spec->child][$key]); } } - $element[$spec->child][0]['#default_value'] = $default_value; + + if (isset($element[$spec->child][0]['field'])) { + $element[$spec->child][0]['field']['#default_value'] = $default_value; + } + else { + $element[$spec->child][0]['#default_value'] = $default_value; + } return $element; } @@ -210,6 +216,9 @@ private function formatArrayDefaultValue($item) { if (!empty($item['#default_value'])) { return [$item['#default_value'] => $item['#default_value']]; } + if (!empty($item['field']['#default_value'])) { + return [$item['field']['#default_value'] => $item['field']['#default_value']]; + } return []; } diff --git a/modules/json_form_widget/src/ValueHandler.php b/modules/json_form_widget/src/ValueHandler.php index 8902f50533..79f4dfd94a 100644 --- a/modules/json_form_widget/src/ValueHandler.php +++ b/modules/json_form_widget/src/ValueHandler.php @@ -148,9 +148,12 @@ public function handleArrayValues($formValues, $property, $schema) { */ private function flattenArraysInArrays($value) { $data = []; + if (isset($value['actions'])) { + unset($value['actions']); + } if (is_array($value)) { foreach ($value as $item) { - $data[] = $this->cleanSelectId($item); + $data[] = is_array($item) ? $this->flattenArraysInArrays($item) : $this->cleanSelectId($item); } } elseif (!empty($value)) { diff --git a/modules/json_form_widget/src/WidgetRouter.php b/modules/json_form_widget/src/WidgetRouter.php index 79047a42d1..00a0663558 100644 --- a/modules/json_form_widget/src/WidgetRouter.php +++ b/modules/json_form_widget/src/WidgetRouter.php @@ -147,6 +147,7 @@ public function handleListElement(mixed $spec, array $element) { * The dropdown element configured. */ public function getDropdownElement(mixed $element, mixed $spec, mixed $titleProperty = FALSE) { + ArrayHelper::flattenArrayElementFieldset($element); $element['#type'] = $this->getSelectType($spec); $element['#options'] = $this->getDropdownOptions($spec->source, $titleProperty); if ($element['#type'] === 'select_or_other_select') { diff --git a/modules/json_form_widget/tests/src/Unit/ArrayHelperTest.php b/modules/json_form_widget/tests/src/Unit/ArrayHelperTest.php index 7819dc8bd4..8634e4d302 100644 --- a/modules/json_form_widget/tests/src/Unit/ArrayHelperTest.php +++ b/modules/json_form_widget/tests/src/Unit/ArrayHelperTest.php @@ -2,20 +2,20 @@ namespace Drupal\Tests\json_form_widget\Unit; -use PHPUnit\Framework\TestCase; -use Drupal\json_form_widget\ArrayHelper; -use MockChain\Chain; use Drupal\Component\DependencyInjection\Container; use Drupal\Core\Form\FormState; use Drupal\Core\Logger\LoggerChannelFactory; use Drupal\Core\StringTranslation\TranslationManager; +use Drupal\json_form_widget\ArrayHelper; use Drupal\json_form_widget\FieldTypeRouter; use Drupal\json_form_widget\IntegerHelper; use Drupal\json_form_widget\ObjectHelper; use Drupal\json_form_widget\SchemaUiHandler; use Drupal\json_form_widget\StringHelper; use Drupal\metastore\SchemaRetriever; +use MockChain\Chain; use MockChain\Options; +use PHPUnit\Framework\TestCase; /** * Test class for ArrayHelper. @@ -79,7 +79,7 @@ public function testComplex() { $result = $array_helper->handleArrayElement($definition, [], $form_state, $context); $expected = $this->getExpectedComplexArrayElement(); unset($result['actions']); - unset($result['distribution'][0]['distribution']['schema']['schema']['fields']['actions']); + unset($result['distribution'][0]['distribution']['actions']); $this->assertEquals($expected, $result); } @@ -182,10 +182,10 @@ private function getExpectedObject() { "#required" => FALSE, ], ], - ] + ], ], ], - ] + ], ], ], "#required" => FALSE, @@ -207,6 +207,7 @@ private function getExpectedComplexArrayElement() { 'distribution' => [ 0 => $this->getExpectedObject(), ], + '#required' => FALSE, ]; } diff --git a/modules/json_form_widget/tests/src/Unit/JsonFormBuilderTest.php b/modules/json_form_widget/tests/src/Unit/JsonFormBuilderTest.php index 7ee6b13a54..a32f92be48 100644 --- a/modules/json_form_widget/tests/src/Unit/JsonFormBuilderTest.php +++ b/modules/json_form_widget/tests/src/Unit/JsonFormBuilderTest.php @@ -306,17 +306,29 @@ public function testSchema() { "#suffix" => '', "keyword" => [ 0 => [ - "#type" => "textfield", - "#title" => "Tag", + "#type" => "fieldset", "#required" => FALSE, + '#attributes' => [ + 'class' => ['json-form-widget-array-item'], + 'data-parent' => 'keyword', + ], + "field" => [ + '#type' => 'textfield', + '#title' => 'Tag', + ], ], ], + '#required' => FALSE, ], ]; $form_state = new FormState(); $form_state->set(ArrayHelper::buildCountProperty('keyword'), 1); $result = $form_builder->getJsonForm([], $form_state); - unset($result['keyword']['actions']); + // The actions are too complex to deal with in the $expected array, we just + // assert the count is correct then remove them. + $this->assertCount(1, $result['keyword']['keyword']); + $this->assertCount(1, $result['keyword']['keyword']); + unset($result['keyword']['actions'], $result['keyword']['keyword'][0]['actions']); $this->assertEquals($expected, $result); // Test array required. @@ -355,16 +367,26 @@ public function testSchema() { '#description_display' => 'before', "keyword" => [ 0 => [ - "#type" => "textfield", - "#title" => "Tag", + "#type" => "fieldset", "#required" => TRUE, + 'field' => [ + '#type' => 'textfield', + '#title' => 'Tag', + ], + '#attributes' => [ + 'class' => ['json-form-widget-array-item'], + 'data-parent' => 'keyword', + ], ], ], + '#required' => TRUE, ], ]; $form_state = new FormState(); $result = $form_builder->getJsonForm([], $form_state); - unset($result['keyword']['actions']); + $this->assertCount(1, $result['keyword']['keyword']); + $this->assertCount(1, $result['keyword']['keyword']); + unset($result['keyword']['actions'], $result['keyword']['keyword'][0]['actions']); $this->assertEquals($expected, $result); }