NodeTypeForm.php 8.94 KB
Newer Older
1 2 3 4
<?php

namespace Drupal\node;

5
use Drupal\Core\Entity\BundleEntityFormBase;
6
use Drupal\Core\Entity\EntityManagerInterface;
7
use Drupal\Core\Entity\EntityTypeInterface;
8
use Drupal\Core\Form\FormStateInterface;
9
use Drupal\language\Entity\ContentLanguageSettings;
10
use Symfony\Component\DependencyInjection\ContainerInterface;
11 12

/**
13
 * Form handler for node type forms.
14
 */
15
class NodeTypeForm extends BundleEntityFormBase {
16

17 18 19 20 21 22 23 24 25 26 27
  /**
   * The entity manager.
   *
   * @var \Drupal\Core\Entity\EntityManagerInterface
   */
  protected $entityManager;

  /**
   * Constructs the NodeTypeForm object.
   *
   * @param \Drupal\Core\Entity\EntityManagerInterface $entity_manager
28
   *   The entity manager.
29 30 31 32 33 34 35 36 37 38 39 40 41 42
   */
  public function __construct(EntityManagerInterface $entity_manager) {
    $this->entityManager = $entity_manager;
  }

  /**
   * {@inheritdoc}
   */
  public static function create(ContainerInterface $container) {
    return new static(
      $container->get('entity.manager')
    );
  }

43 44 45
  /**
   * {@inheritdoc}
   */
46
  public function form(array $form, FormStateInterface $form_state) {
47 48 49 50
    $form = parent::form($form, $form_state);

    $type = $this->entity;
    if ($this->operation == 'add') {
51
      $form['#title'] = $this->t('Add content type');
52 53 54 55 56
      $fields = $this->entityManager->getBaseFieldDefinitions('node');
      // Create a node with a fake bundle using the type's UUID so that we can
      // get the default values for workflow settings.
      // @todo Make it possible to get default values without an entity.
      //   https://www.drupal.org/node/2318187
57
      $node = $this->entityManager->getStorage('node')->create(['type' => $type->uuid()]);
58
    }
59
    else {
60
      $form['#title'] = $this->t('Edit %label content type', ['%label' => $type->label()]);
61 62
      $fields = $this->entityManager->getFieldDefinitions('node', $type->id());
      // Create a node to get the current values for workflow settings fields.
63
      $node = $this->entityManager->getStorage('node')->create(['type' => $type->id()]);
64 65
    }

66
    $form['name'] = [
67 68
      '#title' => t('Name'),
      '#type' => 'textfield',
69
      '#default_value' => $type->label(),
70
      '#description' => t('The human-readable name of this content type. This text will be displayed as part of the list on the <em>Add content</em> page. This name must be unique.'),
71 72
      '#required' => TRUE,
      '#size' => 30,
73
    ];
74

75
    $form['type'] = [
76 77
      '#type' => 'machine_name',
      '#default_value' => $type->id(),
78
      '#maxlength' => EntityTypeInterface::BUNDLE_MAX_LENGTH,
79
      '#disabled' => $type->isLocked(),
80
      '#machine_name' => [
81
        'exists' => ['Drupal\node\Entity\NodeType', 'load'],
82 83 84
        'source' => ['name'],
      ],
      '#description' => t('A unique machine-readable name for this content type. It must only contain lowercase letters, numbers, and underscores. This name will be used for constructing the URL of the %node-add page, in which underscores will be converted into hyphens.', [
85
        '%node-add' => t('Add content'),
86 87
      ]),
    ];
88

89
    $form['description'] = [
90 91
      '#title' => t('Description'),
      '#type' => 'textarea',
92
      '#default_value' => $type->getDescription(),
93
      '#description' => t('This text will be displayed on the <em>Add new content</em> page.'),
94
    ];
95

96
    $form['additional_settings'] = [
97
      '#type' => 'vertical_tabs',
98 99 100 101
      '#attached' => [
        'library' => ['node/drupal.content_types'],
      ],
    ];
102

103
    $form['submission'] = [
104 105 106
      '#type' => 'details',
      '#title' => t('Submission form settings'),
      '#group' => 'additional_settings',
107
      '#open' => TRUE,
108 109
    ];
    $form['submission']['title_label'] = [
110 111
      '#title' => t('Title field label'),
      '#type' => 'textfield',
112
      '#default_value' => $fields['title']->getLabel(),
113
      '#required' => TRUE,
114 115
    ];
    $form['submission']['preview_mode'] = [
116 117
      '#type' => 'radios',
      '#title' => t('Preview before submitting'),
118
      '#default_value' => $type->getPreviewMode(),
119
      '#options' => [
120 121 122
        DRUPAL_DISABLED => t('Disabled'),
        DRUPAL_OPTIONAL => t('Optional'),
        DRUPAL_REQUIRED => t('Required'),
123 124 125
      ],
    ];
    $form['submission']['help']  = [
126 127
      '#type' => 'textarea',
      '#title' => t('Explanation or submission guidelines'),
128
      '#default_value' => $type->getHelp(),
129
      '#description' => t('This text will be displayed at the top of the page when creating or editing content of this type.'),
130 131
    ];
    $form['workflow'] = [
132 133 134
      '#type' => 'details',
      '#title' => t('Publishing options'),
      '#group' => 'additional_settings',
135 136
    ];
    $workflow_options = [
137 138 139
      'status' => $node->status->value,
      'promote' => $node->promote->value,
      'sticky' => $node->sticky->value,
140
      'revision' => $type->isNewRevision(),
141
    ];
142 143 144
    // Prepare workflow options to be used for 'checkboxes' form element.
    $keys = array_keys(array_filter($workflow_options));
    $workflow_options = array_combine($keys, $keys);
145
    $form['workflow']['options'] = [
146
      '#type' => 'checkboxes',
147
      '#title' => t('Default options'),
148
      '#default_value' => $workflow_options,
149
      '#options' => [
150 151 152 153
        'status' => t('Published'),
        'promote' => t('Promoted to front page'),
        'sticky' => t('Sticky at top of lists'),
        'revision' => t('Create new revision'),
154
      ],
155
      '#description' => t('Users with the <em>Administer content</em> permission will be able to override these options.'),
156
    ];
157
    if ($this->moduleHandler->moduleExists('language')) {
158
      $form['language'] = [
159 160 161
        '#type' => 'details',
        '#title' => t('Language settings'),
        '#group' => 'additional_settings',
162
      ];
163

164
      $language_configuration = ContentLanguageSettings::loadByEntityTypeBundle('node', $type->id());
165
      $form['language']['language_configuration'] = [
166
        '#type' => 'language_configuration',
167
        '#entity_information' => [
168 169
          'entity_type' => 'node',
          'bundle' => $type->id(),
170
        ],
171
        '#default_value' => $language_configuration,
172
      ];
173
    }
174
    $form['display'] = [
175 176 177
      '#type' => 'details',
      '#title' => t('Display settings'),
      '#group' => 'additional_settings',
178 179
    ];
    $form['display']['display_submitted'] = [
180
      '#type' => 'checkbox',
181
      '#title' => t('Display author and date information'),
182
      '#default_value' => $type->displaySubmitted(),
183
      '#description' => t('Author username and publish date will be displayed.'),
184
    ];
185 186

    return $this->protectBundleIdElement($form);
187 188 189 190 191
  }

  /**
   * {@inheritdoc}
   */
192
  protected function actions(array $form, FormStateInterface $form_state) {
193 194 195 196 197 198 199 200 201
    $actions = parent::actions($form, $form_state);
    $actions['submit']['#value'] = t('Save content type');
    $actions['delete']['#value'] = t('Delete content type');
    return $actions;
  }

  /**
   * {@inheritdoc}
   */
202 203
  public function validateForm(array &$form, FormStateInterface $form_state) {
    parent::validateForm($form, $form_state);
204

205
    $id = trim($form_state->getValue('type'));
206 207
    // '0' is invalid, since elsewhere we check it using empty().
    if ($id == '0') {
208
      $form_state->setErrorByName('type', $this->t("Invalid machine-readable name. Enter a name other than %invalid.", ['%invalid' => $id]));
209 210 211 212 213 214
    }
  }

  /**
   * {@inheritdoc}
   */
215
  public function save(array $form, FormStateInterface $form_state) {
216
    $type = $this->entity;
217
    $type->setNewRevision($form_state->getValue(['options', 'revision']));
218 219
    $type->set('type', trim($type->id()));
    $type->set('name', trim($type->label()));
220 221 222

    $status = $type->save();

223
    $t_args = ['%name' => $type->label()];
224 225 226 227 228

    if ($status == SAVED_UPDATED) {
      drupal_set_message(t('The content type %name has been updated.', $t_args));
    }
    elseif ($status == SAVED_NEW) {
229
      node_add_body_field($type);
230
      drupal_set_message(t('The content type %name has been added.', $t_args));
231
      $context = array_merge($t_args, ['link' => $type->link($this->t('View'), 'collection')]);
232
      $this->logger('node')->notice('Added content type %name.', $context);
233 234
    }

235 236 237
    $fields = $this->entityManager->getFieldDefinitions('node', $type->id());
    // Update title field definition.
    $title_field = $fields['title'];
238 239 240
    $title_label = $form_state->getValue('title_label');
    if ($title_field->getLabel() != $title_label) {
      $title_field->getConfig($type->id())->setLabel($title_label)->save();
241 242 243 244
    }
    // Update workflow options.
    // @todo Make it possible to get default values without an entity.
    //   https://www.drupal.org/node/2318187
245 246
    $node = $this->entityManager->getStorage('node')->create(['type' => $type->id()]);
    foreach (['status', 'promote', 'sticky'] as $field_name) {
247
      $value = (bool) $form_state->getValue(['options', $field_name]);
248 249 250 251 252 253
      if ($node->$field_name->value != $value) {
        $fields[$field_name]->getConfig($type->id())->setDefaultValue($value)->save();
      }
    }

    $this->entityManager->clearCachedFieldDefinitions();
254
    $form_state->setRedirectUrl($type->urlInfo('collection'));
255 256 257
  }

}