Entity types

Drupal 7 - entities were generic stdClass objects.

Drupal 8 - entities are now specifically typed objects, with each entity type defining a class that will be used for instances of the given entity.

Requirements
Entity classes must be placed in the Entity sub-namespace of the module that provides the entity type, e.g. \Drupal\[module_name]\Entity. This means entity class PHP files may be found in a module's src/Entity directory.

The docblock for the class must contain an EntityType annotation that defines the metadata for entities of that type. These include things like the entity type's label, controllers, tables, etc. For a documented list of all the available metadata properties, refer to the \Drupal\Core\Entity\Annotation\EntityType class.

Naming

Entity type names should be prefixed with the module name if the entity type and module name aren't the same. Prefixing the entity type's class itself is not required since it lives within the namespace of the defining module, provided it is meaningful enough on its own. For example, the entity type for taxonomy terms is named taxonomy_term and the class name is Drupal\taxonomy\Entity\Term.

Interfaces

Drupal 8 recommends you type hint functions and methods with interfaces instead of classes. For example, the generic entity storage hooks type hint with EntityInterface as in hook_entity_insert(EntityInterface $entity) and the node specific storage hooks type hint with NodeInterface as in hook_node_insert(NodeInterface $node).

Entity fields / properties are often very short, storage-centric and not very self-descriptive. Additionally, content entities don't use defined properties for their fields (including base fields like the node title) at all.

Therefore, the recommended approach is to provide an Interface with documented method names. A few rules to follow when doing so:

Methods usually have a get/set/is or similar prefix: getSomething(), setSomething($value), isSomething()

Only add methods for things that other code is supposed to change. The last changed date of nodes ($node->changed) isn't supposed to be changed, so there is $node->getChangedTime() but no $node->setChangedTime() method.

Use self-descriptive method names, for example, the method to access $node->status is called $node->isPublished().

Discoverability

To find out which entity types a module provides, refer to the classes in the Entity sub-namespace of that module that have an @EntityType annotation, which also contains the name in the id annotation key.

When trying to find where a given entity type is defined, the first thing to look for is the prefix of the entity type. If a module doesn't follow that naming convention, then it can be found by searching for id = "$type". If the class or interface is known instead of the entity type, then the namespace of that indicates where it is coming from.

Example

core/modules/node/src/Entity/Node.php:

namespace Drupal\node\Entity;

 

use Drupal\Core\Entity\ContentEntityBase;

use Drupal\Core\Entity\EntityChangedTrait;

use Drupal\Core\Entity\EntityStorageInterface;

use Drupal\Core\Entity\EntityTypeInterface;

use Drupal\Core\Field\BaseFieldDefinition;

use Drupal\Core\Session\AccountInterface;

use Drupal\node\NodeInterface;

use Drupal\user\UserInterface;

 

/**

 * Defines the node entity class.

 *

 * @ContentEntityType(

 *   id = "node",

 *   label = @Translation("Content"),

 *   bundle_label = @Translation("Content type"),

 *   handlers = {

 *     "storage" = "Drupal\node\NodeStorage",

 *     "storage_schema" = "Drupal\node\NodeStorageSchema",

 *     "view_builder" = "Drupal\node\NodeViewBuilder",

 *     "access" = "Drupal\node\NodeAccessControlHandler",

 *     "views_data" = "Drupal\node\NodeViewsData",

 *     "form" = {

 *       "default" = "Drupal\node\NodeForm",

 *       "delete" = "Drupal\node\Form\NodeDeleteForm",

 *       "edit" = "Drupal\node\NodeForm"

 *     },

 *     "route_provider" = {

 *       "html" = "Drupal\node\Entity\NodeRouteProvider",

 *     },

 *     "list_builder" = "Drupal\node\NodeListBuilder",

 *     "translation" = "Drupal\node\NodeTranslationHandler"

 *   },

 *   base_table = "node",

 *   data_table = "node_field_data",

 *   revision_table = "node_revision",

 *   revision_data_table = "node_field_revision",

 *   translatable = TRUE,

 *   list_cache_contexts = { "user.node_grants:view" },

 *   entity_keys = {

 *     "id" = "nid",

 *     "revision" = "vid",

 *     "bundle" = "type",

 *     "label" = "title",

 *     "langcode" = "langcode",

 *     "uuid" = "uuid",

 *     "status" = "status",

 *     "uid" = "uid",

 *   },

 *   bundle_entity_type = "node_type",

 *   field_ui_base_route = "entity.node_type.edit_form",

 *   common_reference_target = TRUE,

 *   permission_granularity = "bundle",

 *   links = {

 *     "canonical" = "/node/{node}",

 *     "delete-form" = "/node/{node}/delete",

 *     "edit-form" = "/node/{node}/edit",

 *     "version-history" = "/node/{node}/revisions",

 *     "revision" = "/node/{node}/revisions/{node_revision}/view",

 *   }

 * )

 */

class Node extends ContentEntityBase implements NodeInterface {

  // ...

}

To get a complete picture of the entities in Drupal 8 we can review the following diagram. This represent the entity classes. To view open in a new tab:
 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值