annotate core/lib/Drupal/Core/Field/FieldDefinition.php @ 5:12f9dff5fda9 tip

Update to Drupal core 8.7.1
author Chris Cannam
date Thu, 09 May 2019 15:34:47 +0100
parents
children
rev   line source
Chris@5 1 <?php
Chris@5 2
Chris@5 3 namespace Drupal\Core\Field;
Chris@5 4
Chris@5 5 use Drupal\Core\Cache\UnchangingCacheableDependencyTrait;
Chris@5 6 use Drupal\Core\Entity\FieldableEntityInterface;
Chris@5 7 use Drupal\Core\Field\TypedData\FieldItemDataDefinition;
Chris@5 8 use Drupal\Core\TypedData\ListDataDefinition;
Chris@5 9
Chris@5 10 /**
Chris@5 11 * A class for defining entity field definitions.
Chris@5 12 *
Chris@5 13 * A field definition in the context of a bundle field is different from a base
Chris@5 14 * field in that it may exist only for one or more bundles of an entity type. A
Chris@5 15 * bundle field definition may also override the definition of an existing base
Chris@5 16 * field definition on a per bundle basis. The bundle field definition is used
Chris@5 17 * for code driven overrides, while the
Chris@5 18 * \Drupal\Core\Field\Entity\BaseFieldOverride uses config to override the base
Chris@5 19 * field definition.
Chris@5 20 *
Chris@5 21 * Bundle fields can be defined in code using hook_entity_bundle_field_info() or
Chris@5 22 * via the
Chris@5 23 * \Drupal\Core\Entity\FieldableEntityInterface::bundleFieldDefinitions() method
Chris@5 24 * when defining an entity type. All bundle fields require an associated storage
Chris@5 25 * definition. A storage definition may have automatically been defined when
Chris@5 26 * overriding a base field or it may be manually provided via
Chris@5 27 * hook_entity_field_storage_info().
Chris@5 28 *
Chris@5 29 * @see \Drupal\Core\Entity\FieldableEntityInterface::bundleFieldDefinitions()
Chris@5 30 * @see \Drupal\Core\Field\FieldDefinitionInterface
Chris@5 31 * @see \Drupal\Core\Field\FieldStorageDefinitionInterface
Chris@5 32 * @see hook_entity_bundle_field_info()
Chris@5 33 * @see hook_entity_field_storage_info()
Chris@5 34 */
Chris@5 35 class FieldDefinition extends ListDataDefinition implements FieldDefinitionInterface {
Chris@5 36
Chris@5 37 use UnchangingCacheableDependencyTrait;
Chris@5 38 use FieldInputValueNormalizerTrait;
Chris@5 39
Chris@5 40 /**
Chris@5 41 * The associated field storage definition.
Chris@5 42 *
Chris@5 43 * @var \Drupal\Core\Field\FieldStorageDefinitionInterface
Chris@5 44 */
Chris@5 45 protected $fieldStorageDefinition;
Chris@5 46
Chris@5 47 /**
Chris@5 48 * Creates a new field definition.
Chris@5 49 *
Chris@5 50 * @param \Drupal\Core\Field\FieldStorageDefinitionInterface $storageDefinition
Chris@5 51 * The associated field storage definition.
Chris@5 52 *
Chris@5 53 * @return static
Chris@5 54 */
Chris@5 55 public static function createFromFieldStorageDefinition(FieldStorageDefinitionInterface $storageDefinition) {
Chris@5 56 $field_definition = new static();
Chris@5 57 $field_definition->setFieldStorageDefinition($storageDefinition);
Chris@5 58 return $field_definition;
Chris@5 59 }
Chris@5 60
Chris@5 61 /**
Chris@5 62 * {@inheritdoc}
Chris@5 63 */
Chris@5 64 public function getName() {
Chris@5 65 return $this->getFieldStorageDefinition()->getName();
Chris@5 66 }
Chris@5 67
Chris@5 68 /**
Chris@5 69 * {@inheritdoc}
Chris@5 70 */
Chris@5 71 public function getType() {
Chris@5 72 return $this->getFieldStorageDefinition()->getType();
Chris@5 73 }
Chris@5 74
Chris@5 75 /**
Chris@5 76 * {@inheritdoc}
Chris@5 77 */
Chris@5 78 public function getTargetEntityTypeId() {
Chris@5 79 return $this->getFieldStorageDefinition()->getTargetEntityTypeId();
Chris@5 80 }
Chris@5 81
Chris@5 82 /**
Chris@5 83 * Set the target bundle.
Chris@5 84 *
Chris@5 85 * @param string $bundle
Chris@5 86 * The target bundle.
Chris@5 87 *
Chris@5 88 * @return $this
Chris@5 89 */
Chris@5 90 public function setTargetBundle($bundle) {
Chris@5 91 $this->definition['bundle'] = $bundle;
Chris@5 92 return $this;
Chris@5 93 }
Chris@5 94
Chris@5 95 /**
Chris@5 96 * {@inheritdoc}
Chris@5 97 */
Chris@5 98 public function getTargetBundle() {
Chris@5 99 return $this->definition['bundle'];
Chris@5 100 }
Chris@5 101
Chris@5 102 /**
Chris@5 103 * Sets whether the display for the field can be configured.
Chris@5 104 *
Chris@5 105 * @param string $display_context
Chris@5 106 * The display context. Either 'view' or 'form'.
Chris@5 107 * @param bool $configurable
Chris@5 108 * Whether the display options can be configured (e.g., via the "Manage
Chris@5 109 * display" / "Manage form display" UI screens). If TRUE, the options
Chris@5 110 * specified via getDisplayOptions() act as defaults.
Chris@5 111 *
Chris@5 112 * @return $this
Chris@5 113 */
Chris@5 114 public function setDisplayConfigurable($display_context, $configurable) {
Chris@5 115 // If no explicit display options have been specified, default to 'hidden'.
Chris@5 116 if (empty($this->definition['display'][$display_context])) {
Chris@5 117 $this->definition['display'][$display_context]['options'] = ['region' => 'hidden'];
Chris@5 118 }
Chris@5 119 $this->definition['display'][$display_context]['configurable'] = $configurable;
Chris@5 120 return $this;
Chris@5 121 }
Chris@5 122
Chris@5 123 /**
Chris@5 124 * {@inheritdoc}
Chris@5 125 */
Chris@5 126 public function isDisplayConfigurable($display_context) {
Chris@5 127 return isset($this->definition['display'][$display_context]['configurable']) ? $this->definition['display'][$display_context]['configurable'] : FALSE;
Chris@5 128 }
Chris@5 129
Chris@5 130 /**
Chris@5 131 * Sets the display options for the field in forms or rendered entities.
Chris@5 132 *
Chris@5 133 * This enables generic rendering of the field with widgets / formatters,
Chris@5 134 * including automated support for "In place editing", and with optional
Chris@5 135 * configurability in the "Manage display" / "Manage form display" UI screens.
Chris@5 136 *
Chris@5 137 * Unless this method is called, the field remains invisible (or requires
Chris@5 138 * ad-hoc rendering logic).
Chris@5 139 *
Chris@5 140 * @param string $display_context
Chris@5 141 * The display context. Either 'view' or 'form'.
Chris@5 142 * @param array $options
Chris@5 143 * An array of display options. Refer to
Chris@5 144 * \Drupal\Core\Field\FieldDefinitionInterface::getDisplayOptions() for
Chris@5 145 * a list of supported keys. The options should include at least a 'weight',
Chris@5 146 * or specify 'type' = 'hidden'. The 'default_widget' / 'default_formatter'
Chris@5 147 * for the field type will be used if no 'type' is specified.
Chris@5 148 *
Chris@5 149 * @return $this
Chris@5 150 */
Chris@5 151 public function setDisplayOptions($display_context, array $options) {
Chris@5 152 $this->definition['display'][$display_context]['options'] = $options;
Chris@5 153 return $this;
Chris@5 154 }
Chris@5 155
Chris@5 156 /**
Chris@5 157 * {@inheritdoc}
Chris@5 158 */
Chris@5 159 public function getDisplayOptions($display_context) {
Chris@5 160 return isset($this->definition['display'][$display_context]['options']) ? $this->definition['display'][$display_context]['options'] : NULL;
Chris@5 161 }
Chris@5 162
Chris@5 163 /**
Chris@5 164 * {@inheritdoc}
Chris@5 165 */
Chris@5 166 public function getDefaultValueLiteral() {
Chris@5 167 return isset($this->definition['default_value']) ? $this->definition['default_value'] : [];
Chris@5 168 }
Chris@5 169
Chris@5 170 /**
Chris@5 171 * Set the default value callback for the field.
Chris@5 172 *
Chris@5 173 * @param string $callback
Chris@5 174 * The default value callback.
Chris@5 175 *
Chris@5 176 * @return $this
Chris@5 177 */
Chris@5 178 public function setDefaultValueCallback($callback) {
Chris@5 179 if (isset($callback) && !is_string($callback)) {
Chris@5 180 throw new \InvalidArgumentException('Default value callback must be a string, like "function_name" or "ClassName::methodName"');
Chris@5 181 }
Chris@5 182 $this->definition['default_value_callback'] = $callback;
Chris@5 183 return $this;
Chris@5 184 }
Chris@5 185
Chris@5 186 /**
Chris@5 187 * {@inheritdoc}
Chris@5 188 */
Chris@5 189 public function getDefaultValueCallback() {
Chris@5 190 return isset($this->definition['default_value_callback']) ? $this->definition['default_value_callback'] : NULL;
Chris@5 191 }
Chris@5 192
Chris@5 193 /**
Chris@5 194 * Set a default value for the field.
Chris@5 195 *
Chris@5 196 * @param mixed $value
Chris@5 197 * The default value.
Chris@5 198 *
Chris@5 199 * @return $this
Chris@5 200 */
Chris@5 201 public function setDefaultValue($value) {
Chris@5 202 $this->definition['default_value'] = $this->normalizeValue($value, $this->getFieldStorageDefinition()->getMainPropertyName());
Chris@5 203 return $this;
Chris@5 204 }
Chris@5 205
Chris@5 206 /**
Chris@5 207 * {@inheritdoc}
Chris@5 208 */
Chris@5 209 public function getDefaultValue(FieldableEntityInterface $entity) {
Chris@5 210 // Allow custom default values function.
Chris@5 211 if ($callback = $this->getDefaultValueCallback()) {
Chris@5 212 $value = call_user_func($callback, $entity, $this);
Chris@5 213 }
Chris@5 214 else {
Chris@5 215 $value = $this->getDefaultValueLiteral();
Chris@5 216 }
Chris@5 217 $value = $this->normalizeValue($value, $this->getFieldStorageDefinition()->getMainPropertyName());
Chris@5 218 // Allow the field type to process default values.
Chris@5 219 $field_item_list_class = $this->getClass();
Chris@5 220 return $field_item_list_class::processDefaultValue($value, $entity, $this);
Chris@5 221 }
Chris@5 222
Chris@5 223 /**
Chris@5 224 * Sets whether the field is translatable.
Chris@5 225 *
Chris@5 226 * @param bool $translatable
Chris@5 227 * Whether the field is translatable.
Chris@5 228 *
Chris@5 229 * @return $this
Chris@5 230 */
Chris@5 231 public function setTranslatable($translatable) {
Chris@5 232 $this->definition['translatable'] = $translatable;
Chris@5 233 return $this;
Chris@5 234 }
Chris@5 235
Chris@5 236 /**
Chris@5 237 * {@inheritdoc}
Chris@5 238 */
Chris@5 239 public function isTranslatable() {
Chris@5 240 return !empty($this->definition['translatable']) && $this->getFieldStorageDefinition()->isTranslatable();
Chris@5 241 }
Chris@5 242
Chris@5 243 /**
Chris@5 244 * Set the field storage definition.
Chris@5 245 *
Chris@5 246 * @param \Drupal\Core\Field\FieldStorageDefinitionInterface $storageDefinition
Chris@5 247 * The field storage definition associated with this field definition.
Chris@5 248 *
Chris@5 249 * @return $this
Chris@5 250 */
Chris@5 251 public function setFieldStorageDefinition(FieldStorageDefinitionInterface $storageDefinition) {
Chris@5 252 $this->fieldStorageDefinition = $storageDefinition;
Chris@5 253 $this->itemDefinition = FieldItemDataDefinition::create($this);
Chris@5 254 // Create a definition for the items, and initialize it with the default
Chris@5 255 // settings for the field type.
Chris@5 256 $field_type_manager = \Drupal::service('plugin.manager.field.field_type');
Chris@5 257 $default_settings = $field_type_manager->getDefaultFieldSettings($storageDefinition->getType());
Chris@5 258 $this->itemDefinition->setSettings($default_settings);
Chris@5 259 return $this;
Chris@5 260 }
Chris@5 261
Chris@5 262 /**
Chris@5 263 * {@inheritdoc}
Chris@5 264 */
Chris@5 265 public function getFieldStorageDefinition() {
Chris@5 266 return $this->fieldStorageDefinition;
Chris@5 267 }
Chris@5 268
Chris@5 269 /**
Chris@5 270 * {@inheritdoc}
Chris@5 271 */
Chris@5 272 public function getConfig($bundle) {
Chris@5 273 // @todo provide a FieldDefinitionOverride config entity in
Chris@5 274 // https://www.drupal.org/project/drupal/issues/2935978.
Chris@5 275 throw new \Exception('Field definitions do not currently have an override config entity.');
Chris@5 276 }
Chris@5 277
Chris@5 278 /**
Chris@5 279 * {@inheritdoc}
Chris@5 280 */
Chris@5 281 public function getUniqueIdentifier() {
Chris@5 282 return $this->getTargetEntityTypeId() . '-' . $this->getTargetBundle() . '-' . $this->getName();
Chris@5 283 }
Chris@5 284
Chris@5 285 /**
Chris@5 286 * {@inheritdoc}
Chris@5 287 */
Chris@5 288 public function getSetting($setting_name) {
Chris@5 289 if (array_key_exists($setting_name, $this->itemDefinition->getSettings())) {
Chris@5 290 return $this->itemDefinition->getSetting($setting_name);
Chris@5 291 }
Chris@5 292 else {
Chris@5 293 return $this->getFieldStorageDefinition()->getSetting($setting_name);
Chris@5 294 }
Chris@5 295 }
Chris@5 296
Chris@5 297 /**
Chris@5 298 * {@inheritdoc}
Chris@5 299 */
Chris@5 300 public function getSettings() {
Chris@5 301 return $this->getItemDefinition()->getSettings() + $this->getFieldStorageDefinition()->getSettings();
Chris@5 302 }
Chris@5 303
Chris@5 304 /**
Chris@5 305 * {@inheritdoc}
Chris@5 306 */
Chris@5 307 public function setSetting($setting_name, $value) {
Chris@5 308 $this->getItemDefinition()->setSetting($setting_name, $value);
Chris@5 309 return $this;
Chris@5 310 }
Chris@5 311
Chris@5 312 /**
Chris@5 313 * {@inheritdoc}
Chris@5 314 */
Chris@5 315 public function setSettings(array $settings) {
Chris@5 316 // Assign settings individually, in order to keep the current values
Chris@5 317 // of settings not specified in $settings.
Chris@5 318 foreach ($settings as $setting_name => $setting) {
Chris@5 319 $this->getItemDefinition()->setSetting($setting_name, $setting);
Chris@5 320 }
Chris@5 321 return $this;
Chris@5 322 }
Chris@5 323
Chris@5 324 }