X-Git-Url: http://www.aleph1.co.uk/gitweb/?a=blobdiff_plain;f=web%2Fcore%2Fmodules%2Ffield_ui%2Fsrc%2FForm%2FFieldStorageAddForm.php;fp=web%2Fcore%2Fmodules%2Ffield_ui%2Fsrc%2FForm%2FFieldStorageAddForm.php;h=e1f278796e577dbdff111cc6cbbb375d03a68ce8;hb=a2bd1bf0c2c1f1a17d188f4dc0726a45494cefae;hp=0000000000000000000000000000000000000000;hpb=57c063afa3f66b07c4bbddc2d6129a96d90f0aad;p=yaffs-website diff --git a/web/core/modules/field_ui/src/Form/FieldStorageAddForm.php b/web/core/modules/field_ui/src/Form/FieldStorageAddForm.php new file mode 100644 index 000000000..e1f278796 --- /dev/null +++ b/web/core/modules/field_ui/src/Form/FieldStorageAddForm.php @@ -0,0 +1,543 @@ +entityManager = $entity_manager; + $this->fieldTypePluginManager = $field_type_plugin_manager; + $this->configFactory = $config_factory; + } + + /** + * {@inheritdoc} + */ + public function getFormId() { + return 'field_ui_field_storage_add_form'; + } + + /** + * {@inheritdoc} + */ + public static function create(ContainerInterface $container) { + return new static( + $container->get('entity.manager'), + $container->get('plugin.manager.field.field_type'), + $container->get('config.factory') + ); + } + + /** + * {@inheritdoc} + */ + public function buildForm(array $form, FormStateInterface $form_state, $entity_type_id = NULL, $bundle = NULL) { + if (!$form_state->get('entity_type_id')) { + $form_state->set('entity_type_id', $entity_type_id); + } + if (!$form_state->get('bundle')) { + $form_state->set('bundle', $bundle); + } + + $this->entityTypeId = $form_state->get('entity_type_id'); + $this->bundle = $form_state->get('bundle'); + + // Gather valid field types. + $field_type_options = []; + foreach ($this->fieldTypePluginManager->getGroupedDefinitions($this->fieldTypePluginManager->getUiDefinitions()) as $category => $field_types) { + foreach ($field_types as $name => $field_type) { + $field_type_options[$category][$name] = $field_type['label']; + } + } + + $form['add'] = [ + '#type' => 'container', + '#attributes' => ['class' => ['form--inline', 'clearfix']], + ]; + + $form['add']['new_storage_type'] = [ + '#type' => 'select', + '#title' => $this->t('Add a new field'), + '#options' => $field_type_options, + '#empty_option' => $this->t('- Select a field type -'), + ]; + + // Re-use existing field. + if ($existing_field_storage_options = $this->getExistingFieldStorageOptions()) { + $form['add']['separator'] = [ + '#type' => 'item', + '#markup' => $this->t('or'), + ]; + $form['add']['existing_storage_name'] = [ + '#type' => 'select', + '#title' => $this->t('Re-use an existing field'), + '#options' => $existing_field_storage_options, + '#empty_option' => $this->t('- Select an existing field -'), + ]; + + $form['#attached']['drupalSettings']['existingFieldLabels'] = $this->getExistingFieldLabels(array_keys($existing_field_storage_options)); + } + else { + // Provide a placeholder form element to simplify the validation code. + $form['add']['existing_storage_name'] = [ + '#type' => 'value', + '#value' => FALSE, + ]; + } + + // Field label and field_name. + $form['new_storage_wrapper'] = [ + '#type' => 'container', + '#states' => [ + '!visible' => [ + ':input[name="new_storage_type"]' => ['value' => ''], + ], + ], + ]; + $form['new_storage_wrapper']['label'] = [ + '#type' => 'textfield', + '#title' => $this->t('Label'), + '#size' => 15, + ]; + + $field_prefix = $this->config('field_ui.settings')->get('field_prefix'); + $form['new_storage_wrapper']['field_name'] = [ + '#type' => 'machine_name', + // This field should stay LTR even for RTL languages. + '#field_prefix' => '' . $field_prefix, + '#field_suffix' => '‎', + '#size' => 15, + '#description' => $this->t('A unique machine-readable name containing letters, numbers, and underscores.'), + // Calculate characters depending on the length of the field prefix + // setting. Maximum length is 32. + '#maxlength' => FieldStorageConfig::NAME_MAX_LENGTH - strlen($field_prefix), + '#machine_name' => [ + 'source' => ['new_storage_wrapper', 'label'], + 'exists' => [$this, 'fieldNameExists'], + ], + '#required' => FALSE, + ]; + + // Provide a separate label element for the "Re-use existing field" case + // and place it outside the $form['add'] wrapper because those elements + // are displayed inline. + if ($existing_field_storage_options) { + $form['existing_storage_label'] = [ + '#type' => 'textfield', + '#title' => $this->t('Label'), + '#size' => 15, + '#states' => [ + '!visible' => [ + ':input[name="existing_storage_name"]' => ['value' => ''], + ], + ], + ]; + } + + // Place the 'translatable' property as an explicit value so that contrib + // modules can form_alter() the value for newly created fields. By default + // we create field storage as translatable so it will be possible to enable + // translation at field level. + $form['translatable'] = [ + '#type' => 'value', + '#value' => TRUE, + ]; + + $form['actions'] = ['#type' => 'actions']; + $form['actions']['submit'] = [ + '#type' => 'submit', + '#value' => $this->t('Save and continue'), + '#button_type' => 'primary', + ]; + + $form['#attached']['library'][] = 'field_ui/drupal.field_ui'; + + return $form; + } + + /** + * {@inheritdoc} + */ + public function validateForm(array &$form, FormStateInterface $form_state) { + // Missing field type. + if (!$form_state->getValue('new_storage_type') && !$form_state->getValue('existing_storage_name')) { + $form_state->setErrorByName('new_storage_type', $this->t('You need to select a field type or an existing field.')); + } + // Both field type and existing field option selected. This is prevented in + // the UI with JavaScript but we also need a proper server-side validation. + elseif ($form_state->getValue('new_storage_type') && $form_state->getValue('existing_storage_name')) { + $form_state->setErrorByName('new_storage_type', $this->t('Adding a new field and re-using an existing field at the same time is not allowed.')); + return; + } + + $this->validateAddNew($form, $form_state); + $this->validateAddExisting($form, $form_state); + } + + /** + * Validates the 'add new field' case. + * + * @param array $form + * An associative array containing the structure of the form. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * The current state of the form. + * + * @see \Drupal\field_ui\Form\FieldStorageAddForm::validateForm() + */ + protected function validateAddNew(array $form, FormStateInterface $form_state) { + // Validate if any information was provided in the 'add new field' case. + if ($form_state->getValue('new_storage_type')) { + // Missing label. + if (!$form_state->getValue('label')) { + $form_state->setErrorByName('label', $this->t('Add new field: you need to provide a label.')); + } + + // Missing field name. + if (!$form_state->getValue('field_name')) { + $form_state->setErrorByName('field_name', $this->t('Add new field: you need to provide a machine name for the field.')); + } + // Field name validation. + else { + $field_name = $form_state->getValue('field_name'); + + // Add the field prefix. + $field_name = $this->configFactory->get('field_ui.settings')->get('field_prefix') . $field_name; + $form_state->setValueForElement($form['new_storage_wrapper']['field_name'], $field_name); + } + } + } + + /** + * Validates the 're-use existing field' case. + * + * @param array $form + * An associative array containing the structure of the form. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * The current state of the form. + * + * @see \Drupal\field_ui\Form\FieldStorageAddForm::validateForm() + */ + protected function validateAddExisting(array $form, FormStateInterface $form_state) { + if ($form_state->getValue('existing_storage_name')) { + // Missing label. + if (!$form_state->getValue('existing_storage_label')) { + $form_state->setErrorByName('existing_storage_label', $this->t('Re-use existing field: you need to provide a label.')); + } + } + } + + /** + * {@inheritdoc} + */ + public function submitForm(array &$form, FormStateInterface $form_state) { + $error = FALSE; + $values = $form_state->getValues(); + $destinations = []; + $entity_type = $this->entityManager->getDefinition($this->entityTypeId); + + // Create new field. + if ($values['new_storage_type']) { + $field_storage_values = [ + 'field_name' => $values['field_name'], + 'entity_type' => $this->entityTypeId, + 'type' => $values['new_storage_type'], + 'translatable' => $values['translatable'], + ]; + $field_values = [ + 'field_name' => $values['field_name'], + 'entity_type' => $this->entityTypeId, + 'bundle' => $this->bundle, + 'label' => $values['label'], + // Field translatability should be explicitly enabled by the users. + 'translatable' => FALSE, + ]; + $widget_id = $formatter_id = NULL; + + // Check if we're dealing with a preconfigured field. + if (strpos($field_storage_values['type'], 'field_ui:') !== FALSE) { + list(, $field_type, $option_key) = explode(':', $field_storage_values['type'], 3); + $field_storage_values['type'] = $field_type; + + $field_type_class = $this->fieldTypePluginManager->getDefinition($field_type)['class']; + $field_options = $field_type_class::getPreconfiguredOptions()[$option_key]; + + // Merge in preconfigured field storage options. + if (isset($field_options['field_storage_config'])) { + foreach (['cardinality', 'settings'] as $key) { + if (isset($field_options['field_storage_config'][$key])) { + $field_storage_values[$key] = $field_options['field_storage_config'][$key]; + } + } + } + + // Merge in preconfigured field options. + if (isset($field_options['field_config'])) { + foreach (['required', 'settings'] as $key) { + if (isset($field_options['field_config'][$key])) { + $field_values[$key] = $field_options['field_config'][$key]; + } + } + } + + $widget_id = isset($field_options['entity_form_display']['type']) ? $field_options['entity_form_display']['type'] : NULL; + $formatter_id = isset($field_options['entity_view_display']['type']) ? $field_options['entity_view_display']['type'] : NULL; + } + + // Create the field storage and field. + try { + $this->entityManager->getStorage('field_storage_config')->create($field_storage_values)->save(); + $field = $this->entityManager->getStorage('field_config')->create($field_values); + $field->save(); + + $this->configureEntityFormDisplay($values['field_name'], $widget_id); + $this->configureEntityViewDisplay($values['field_name'], $formatter_id); + + // Always show the field settings step, as the cardinality needs to be + // configured for new fields. + $route_parameters = [ + 'field_config' => $field->id(), + ] + FieldUI::getRouteBundleParameter($entity_type, $this->bundle); + $destinations[] = ['route_name' => "entity.field_config.{$this->entityTypeId}_storage_edit_form", 'route_parameters' => $route_parameters]; + $destinations[] = ['route_name' => "entity.field_config.{$this->entityTypeId}_field_edit_form", 'route_parameters' => $route_parameters]; + $destinations[] = ['route_name' => "entity.{$this->entityTypeId}.field_ui_fields", 'route_parameters' => $route_parameters]; + + // Store new field information for any additional submit handlers. + $form_state->set(['fields_added', '_add_new_field'], $values['field_name']); + } + catch (\Exception $e) { + $error = TRUE; + drupal_set_message($this->t('There was a problem creating field %label: @message', ['%label' => $values['label'], '@message' => $e->getMessage()]), 'error'); + } + } + + // Re-use existing field. + if ($values['existing_storage_name']) { + $field_name = $values['existing_storage_name']; + + try { + $field = $this->entityManager->getStorage('field_config')->create([ + 'field_name' => $field_name, + 'entity_type' => $this->entityTypeId, + 'bundle' => $this->bundle, + 'label' => $values['existing_storage_label'], + ]); + $field->save(); + + $this->configureEntityFormDisplay($field_name); + $this->configureEntityViewDisplay($field_name); + + $route_parameters = [ + 'field_config' => $field->id(), + ] + FieldUI::getRouteBundleParameter($entity_type, $this->bundle); + $destinations[] = ['route_name' => "entity.field_config.{$this->entityTypeId}_field_edit_form", 'route_parameters' => $route_parameters]; + $destinations[] = ['route_name' => "entity.{$this->entityTypeId}.field_ui_fields", 'route_parameters' => $route_parameters]; + + // Store new field information for any additional submit handlers. + $form_state->set(['fields_added', '_add_existing_field'], $field_name); + } + catch (\Exception $e) { + $error = TRUE; + drupal_set_message($this->t('There was a problem creating field %label: @message', ['%label' => $values['label'], '@message' => $e->getMessage()]), 'error'); + } + } + + if ($destinations) { + $destination = $this->getDestinationArray(); + $destinations[] = $destination['destination']; + $form_state->setRedirectUrl(FieldUI::getNextDestination($destinations, $form_state)); + } + elseif (!$error) { + drupal_set_message($this->t('Your settings have been saved.')); + } + } + + /** + * Configures the field for the default form mode. + * + * @param string $field_name + * The field name. + * @param string|null $widget_id + * (optional) The plugin ID of the widget. Defaults to NULL. + */ + protected function configureEntityFormDisplay($field_name, $widget_id = NULL) { + // Make sure the field is displayed in the 'default' form mode (using + // default widget and settings). It stays hidden for other form modes + // until it is explicitly configured. + $options = $widget_id ? ['type' => $widget_id] : []; + entity_get_form_display($this->entityTypeId, $this->bundle, 'default') + ->setComponent($field_name, $options) + ->save(); + } + + /** + * Configures the field for the default view mode. + * + * @param string $field_name + * The field name. + * @param string|null $formatter_id + * (optional) The plugin ID of the formatter. Defaults to NULL. + */ + protected function configureEntityViewDisplay($field_name, $formatter_id = NULL) { + // Make sure the field is displayed in the 'default' view mode (using + // default formatter and settings). It stays hidden for other view + // modes until it is explicitly configured. + $options = $formatter_id ? ['type' => $formatter_id] : []; + entity_get_display($this->entityTypeId, $this->bundle, 'default') + ->setComponent($field_name, $options) + ->save(); + } + + /** + * Returns an array of existing field storages that can be added to a bundle. + * + * @return array + * An array of existing field storages keyed by name. + */ + protected function getExistingFieldStorageOptions() { + $options = []; + // Load the field_storages and build the list of options. + $field_types = $this->fieldTypePluginManager->getDefinitions(); + foreach ($this->entityManager->getFieldStorageDefinitions($this->entityTypeId) as $field_name => $field_storage) { + // Do not show: + // - non-configurable field storages, + // - locked field storages, + // - field storages that should not be added via user interface, + // - field storages that already have a field in the bundle. + $field_type = $field_storage->getType(); + if ($field_storage instanceof FieldStorageConfigInterface + && !$field_storage->isLocked() + && empty($field_types[$field_type]['no_ui']) + && !in_array($this->bundle, $field_storage->getBundles(), TRUE)) { + $options[$field_name] = $this->t('@type: @field', [ + '@type' => $field_types[$field_type]['label'], + '@field' => $field_name, + ]); + } + } + asort($options); + + return $options; + } + + /** + * Gets the human-readable labels for the given field storage names. + * + * Since not all field storages are required to have a field, we can only + * provide the field labels on a best-effort basis (e.g. the label of a field + * storage without any field attached to a bundle will be the field name). + * + * @param array $field_names + * An array of field names. + * + * @return array + * An array of field labels keyed by field name. + */ + protected function getExistingFieldLabels(array $field_names) { + // Get all the fields corresponding to the given field storage names and + // this entity type. + $field_ids = $this->entityManager->getStorage('field_config')->getQuery() + ->condition('entity_type', $this->entityTypeId) + ->condition('field_name', $field_names) + ->execute(); + $fields = $this->entityManager->getStorage('field_config')->loadMultiple($field_ids); + + // Go through all the fields and use the label of the first encounter. + $labels = []; + foreach ($fields as $field) { + if (!isset($labels[$field->getName()])) { + $labels[$field->getName()] = $field->label(); + } + } + + // For field storages without any fields attached to a bundle, the default + // label is the field name. + $labels += array_combine($field_names, $field_names); + + return $labels; + } + + /** + * Checks if a field machine name is taken. + * + * @param string $value + * The machine name, not prefixed. + * @param array $element + * An array containing the structure of the 'field_name' element. + * @param \Drupal\Core\Form\FormStateInterface $form_state + * The current state of the form. + * + * @return bool + * Whether or not the field machine name is taken. + */ + public function fieldNameExists($value, $element, FormStateInterface $form_state) { + // Don't validate the case when an existing field has been selected. + if ($form_state->getValue('existing_storage_name')) { + return FALSE; + } + + // Add the field prefix. + $field_name = $this->configFactory->get('field_ui.settings')->get('field_prefix') . $value; + + $field_storage_definitions = $this->entityManager->getFieldStorageDefinitions($this->entityTypeId); + return isset($field_storage_definitions[$field_name]); + } + +}