FieldCollectionItem.php in Field collection 8.3
Same filename and directory in other branches
Namespace
Drupal\field_collection\EntityFile
src/Entity/FieldCollectionItem.phpView source
<?php
namespace Drupal\field_collection\Entity;
use Drupal\Core\Database\Database;
use Drupal\Core\Entity\ContentEntityBase;
use Drupal\Core\Entity\Sql\SqlContentEntityStorage;
use Drupal\Core\Field\BaseFieldDefinition;
use Drupal\Core\Entity\EntityTypeInterface;
use Drupal\field\Entity\FieldStorageConfig;
use Drupal\field_collection\FieldCollectionItemInterface;
/**
* Defines the field collection item entity class.
*
* @ContentEntityType(
* id = "field_collection_item",
* label = @Translation("Field Collection Item"),
* bundle_label = @Translation("Field Name"),
* handlers = {
* "storage" = "Drupal\Core\Entity\Sql\SqlContentEntityStorage",
* "view_builder" = "Drupal\Core\Entity\EntityViewBuilder",
* "access" = "Drupal\field_collection\FieldCollectionItemAccessControlHandler",
* "form" = {
* "default" = "Drupal\field_collection\FieldCollectionItemForm",
* "edit" = "Drupal\field_collection\FieldCollectionItemForm",
* "delete" = "Drupal\field_collection\Form\FieldCollectionItemDeleteForm"
* },
* "views_data" = "Drupal\views\EntityViewsData",
* },
* base_table = "field_collection_item",
* revision_table = "field_collection_item_revision",
* fieldable = TRUE,
* translatable = FALSE,
* render_cache = FALSE,
* entity_keys = {
* "id" = "item_id",
* "revision" = "revision_id",
* "bundle" = "field_name",
* "label" = "field_name",
* "uuid" = "uuid"
* },
* bundle_keys = {
* "bundle" = "field_name"
* },
* bundle_entity_type = "field_collection",
* field_ui_base_route = "entity.field_collection.edit_form",
* permission_granularity = "bundle",
* links = {
* "canonical" = "/field_collection_item/{field_collection_item}",
* "delete-form" = "/field_collection_item/{field_collection_item}",
* "edit-form" = "/field_collection_item/{field_collection_item}/edit"
* }
* )
*/
class FieldCollectionItem extends ContentEntityBase implements FieldCollectionItemInterface {
// TODO: Should references to $this->host_type (a base field) use a getter?
/**
* The id of the host entity.
*
* TODO: Possibly convert it to a FieldInterface.
*/
protected $host_id;
/**
* The revision id of the host entity.
*
* @var int
*/
protected $host_revision_id;
/**
* Implements Drupal\Core\Entity\EntityInterface::id().
*/
public function id() {
return $this->item_id->value;
}
/**
* Overrides \Drupal\Core\Entity\label().
*/
public function label() {
$field_label = $this
->getHost()
->getFieldDefinition($this
->bundle())
->label();
if (empty($field_label)) {
return parent::label();
}
else {
return t('@label @delta of @host', [
'@label' => $field_label,
'@delta' => $this
->getDelta(),
'@host' => $this
->getHost()
->label(),
]);
}
}
/**
* {@inheritdoc}
*/
public static function baseFieldDefinitions(EntityTypeInterface $entity_type) {
$fields['item_id'] = BaseFieldDefinition::create('integer')
->setLabel(t('Field collection item ID'))
->setDescription(t('The field collection item ID.'))
->setReadOnly(TRUE)
->setSetting('unsigned', TRUE);
$fields['host_type'] = BaseFieldDefinition::create('string')
->setLabel(t('Host\'s entity type'))
->setDescription(t('Type of entity for the field collection item\'s host.'))
->setReadOnly(TRUE);
$fields['uuid'] = BaseFieldDefinition::create('uuid')
->setLabel(t('UUID'))
->setDescription(t('The field collection item UUID.'))
->setReadOnly(TRUE);
$fields['revision_id'] = BaseFieldDefinition::create('integer')
->setLabel(t('Revision ID'))
->setDescription(t('The field collection item revision ID.'))
->setReadOnly(TRUE);
$fields['field_name'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('Type'))
->setDescription(t('The field collection item field.'))
->setSetting('target_type', 'field_collection')
->setReadOnly(TRUE);
return $fields;
}
/**
* Save the field collection item.
*
* By default, always save the host entity, so modules are able to react
* upon changes to the content of the host and any 'last updated' dates of
* entities get updated.
*
* For creating an item a host entity has to be specified via setHostEntity()
* before this function is invoked. For the link between the entities to be
* fully established, the host entity object has to be updated to include a
* reference on this field collection item during saving. So do not skip
* saving the host for creating items.
*
* @param $skip_host_save
* (internal) If TRUE is passed, the host entity is not saved automatically
* and therefore no link is created between the host and the item or
* revision updates might be skipped. Use with care.
*/
public function save($skip_host_save = FALSE) {
/* TODO: Need this.
// Make sure we have a host entity during creation.
if (!empty($this->is_new) && !(isset($this->hostEntityId) || isset($this->hostEntity) || isset($this->hostEntityRevisionId))) {
throw new Exception("Unable to create a field collection item without a given host entity.");
}
*/
// Only save directly if we are told to skip saving the host entity. Else,
// we always save via the host as saving the host might trigger saving
// field collection items anyway (e.g. if a new revision is created).
if ($skip_host_save) {
return parent::save();
}
else {
$host_entity = $this
->getHost();
if (!$host_entity) {
throw new \Exception('Unable to save a field collection item without a valid reference to a host entity');
}
/* TODO: Need this.
// If this is creating a new revision, also do so for the host entity.
if (!empty($this->revision) || !empty($this->is_new_revision)) {
$host_entity->revision = TRUE;
if (!empty($this->default_revision)) {
entity_revision_set_default($this->hostEntityType, $host_entity);
}
}
*/
// Set the host entity reference, so the item will be saved with the host.
// @see field_collection_field_presave()
$delta = $this
->getDelta();
$value = $host_entity->{$this
->bundle()}
->getValue();
if (isset($delta)) {
$value[$delta] = [
'entity' => $this,
];
}
else {
$value[] = [
'entity' => $this,
];
}
$host_entity->{$this
->bundle()}
->setValue($value);
return $host_entity
->save();
}
}
/**
* {@inheritdoc}
*/
public function delete() {
if (empty($this->field_collection_deleting) && $this
->getHost()) {
$this
->deleteHostEntityReference();
}
parent::delete();
}
/**
* Deletes the host entity's reference of the field collection item.
*/
protected function deleteHostEntityReference() {
$delta = $this
->getDelta();
if ($this
->id() && isset($delta) && NULL !== $this
->getHost() && isset($this
->getHost()->{$this
->bundle()}[$delta])) {
$host = $this
->getHost();
unset($host->{$this
->bundle()}[$delta]);
// Do not save when the host entity is being deleted. See
// \Drupal\field_collection\Plugin\Field\FieldType\FieldCollection::delete().
if (empty($this->field_collection_deleting)) {
$host
->save();
}
}
}
/**
* Overrides \Drupal\Core\Entity\Entity::getRevisionId().
*/
public function getRevisionId() {
return $this->revision_id->value;
}
/**
* Overrides \Drupal\Core\Entity\Entity::uri().
*/
public function uri() {
$ret = [
'path' => 'field-collection-item/' . $this
->id(),
'options' => [
'entity_type' => $this->entityType,
'entity' => $this,
],
];
return $ret;
}
/**
* {@inheritdoc}
*/
public function getDelta() {
if (($host = $this
->getHost()) && isset($host->{$this
->bundle()})) {
foreach ($host->{$this
->bundle()} as $delta => $item) {
if (isset($item->target_id) && $item->target_id == $this
->id()) {
return $delta;
}
elseif (isset($item->entity) && $item->entity === $this) {
return $delta;
}
}
}
}
/**
* {@inheritdoc}
*/
public function getHost($reset = FALSE) {
$host_type = $this->host_type->value;
$entity_info = $this
->entityTypeManager()
->getDefinition($host_type, TRUE);
if ($id = $this
->getHostId()) {
$storage = $this
->entityTypeManager()
->getStorage($host_type);
if ($reset) {
$storage
->resetCache([
$id,
]);
}
$host_entity = $storage
->load($id);
if ($entity_info
->isRevisionable() && ($rev_id = $this
->getHostRevisionId()) && $rev_id != $host_entity
->getRevisionId()) {
$host_entity = $storage
->loadRevision($rev_id);
}
return $host_entity;
}
else {
return NULL;
}
}
/**
* {@inheritdoc}
*/
public function getHostId() {
if (!isset($this->host_id)) {
$entity_info = $this
->entityTypeManager()
->getDefinition($this->host_type->value, TRUE);
$host_id_results = \Drupal::entityQuery($entity_info
->id())
->condition($this
->bundle(), $this
->id())
->execute();
$this->host_id = reset($host_id_results);
}
return $this->host_id;
}
/**
* {@inheritdoc}
*/
public function getHostRevisionId() {
$host_type = $this->host_type->value;
$entity_info = $this
->entityTypeManager()
->getDefinition($host_type, TRUE);
if (!isset($this->host_revision_id) && $entity_info
->isRevisionable()) {
/** @var SqlContentEntityStorage $storage */
$storage = $this
->entityTypeManager()
->getStorage($host_type);
// generate revision table name of the current field
$field_storage = FieldStorageConfig::loadByName($host_type, $this
->bundle());
$table = $storage
->getTableMapping()
->getDedicatedRevisionTableName($field_storage);
// fetch entity + revision id of the related host
$query = \Drupal::database()
->select($table, 'base');
$query
->addExpression('base.entity_id', 'entity_id');
$query
->addExpression('base.revision_id', 'revision_id');
$query
->condition('base.' . $this
->bundle() . '_target_id', $this
->id());
$query
->condition('base.' . $this
->bundle() . '_revision_id', $this
->getRevisionId());
$query
->range(0, 1);
$result = $query
->execute()
->fetch();
$this->host_revision_id = $result->revision_id;
if ($this->host_revision_id) {
$this->host_id = $result->entity_id;
}
}
return $this->host_revision_id;
}
/**
* {@inheritdoc}
*/
public function setHostEntity($entity, $create_link = TRUE) {
if ($this
->isNew()) {
$this->host_type = $entity
->getEntityTypeId();
$this->host_id = $entity
->id();
$this->host_entity = $entity;
// If the host entity is not saved yet, set the id to FALSE. So
// fetchHostDetails() does not try to load the host entity details.
if (!isset($this->host_id)) {
$this->host_id = FALSE;
}
/*
// We are create a new field collection for a non-default entity, thus
// set archived to TRUE.
if (!entity_revision_is_default($entity_type, $entity)) {
$this->hostEntityId = FALSE;
$this->archived = TRUE;
}
*/
// Add the field collection item to its host.
if ($create_link) {
if (_field_collection_field_item_list_full($entity->{$this
->bundle()})) {
drupal_set_message(t('Field is already full.'), 'error');
}
else {
$bundle = $this
->bundle();
$entity->{$bundle}
->appendItem([
'entity' => $this,
]);
$entity
->save();
}
}
}
else {
throw new \Exception(t('The host entity may be set only during creation of a field collection item.'));
}
}
/**
* {@inheritdoc}
*/
public function isEmpty() {
$is_empty = TRUE;
foreach ($this
->getIterator() as $field) {
// Only check configured fields, skip base fields like uuid.
if (!$field
->isEmpty() && 'Drupal\\field\\Entity\\FieldConfig' == get_class($field
->getFieldDefinition())) {
$is_empty = FALSE;
}
}
// TODO: Allow other modules a chance to alter the value before returning?
//drupal_alter('field_collection_is_empty', $is_empty, $this);
return $is_empty;
}
}
Classes
Name | Description |
---|---|
FieldCollectionItem | Defines the field collection item entity class. |