A work-in-progress collection of tips for working with Drupal code.
Find a file
2022-02-08 09:59:18 +00:00
README.rst chore: remove unneeded separator from namespace 2022-02-08 09:59:18 +00:00

..
  TODO:
  - Cron jobs
  - Forms
  - Logging
  - Querying the database
  - Queues and workers

Drupal tips
###########

A collection of tips and tricks for working with `Drupal <https://www.drupal.org>`_ code. Ideas and PRs welcome.

Inspired by https://github.com/PovilasKorop/laravel-tips.

.. contents::
  :depth: 2

Entity storage
==============

Loading a single entity by ID
-----------------------------

.. code:: php

  \Drupal::entityTypeManager()->getStorage('node')->load(1);

The entity type manager can also be loaded using the ``entity_type.manager`` service name. For example:

.. code:: php

  \Drupal::service('entity_type.manager')->getStorage('node')->load(1);

Loading multiple entities by ID
-------------------------------

.. code:: php

  \Drupal::entityTypeManager()->getStorage('node')->loadMultiple([1, 2]);

Loading entities by properties
------------------------------

.. code:: php

  use Drupal\node\NodeInterface;

  // Load all published `event` nodes.
  \Drupal::entityTypeManager()->getStorage('node')->loadByProperties([
    'status' => NodeInterface::PUBLISHED,
    'type' => 'event',
  ]);

  // Load all published `talk` nodes.
  \Drupal::entityTypeManager()->getStorage('node')->loadByProperties([
    'status' => NodeInterface::PUBLISHED,
    'type' => 'talk',
  ]);

Querying for entities
---------------------

Returns an instance of ``Drupal\Core\Entity\Query\QueryInterface`` for the specified entity type.

.. code:: php

  // Load all node IDs.
  \Drupal::entityTypeManager()->getStorage('node')->getQuery()->execute();

  // Load node IDs that match the specified conditions.
  \Drupal::entityTypeManager()->getStorage('node')->getQuery()
    ->condition('type', 'event')
    ->condition('title', '%Online%', 'LIKE')
    ->range(0, 10)
    ->addTag('node_access')
    ->execute();

Services
========

Using a class name as a service name
------------------------------------

Before:

.. code:: yaml

  # my_module.services.yml

  services:
    my_module.example_service:
      class: Drupal\my_module\Service\ExampleService

After:

.. code:: yaml

  # my_module.services.yml

  services:
    Drupal\my_module\Service\ExampleService: []

Automatically inject dependencies with autowiring
-------------------------------------------------

Before:

.. code:: yaml

  # my_module.services.yml

  services:
    Drupal\my_module\Service\ExampleService:
      arguments: ['@entity_type.manager']

After:

.. code:: yaml

  # my_module.services.yml

  services:
    Drupal\my_module\Service\ExampleService:
      autowire: true

Controllers
===========

Controllers as services
-----------------------

.. code-block:: yaml

  # my_module.services.yml

  services:
    Drupal\my_module\Controller\ExampleController: []

Single-action controllers
-------------------------

Before:

.. code-block:: yaml

  # my_module.routing.yml

  my_module.example:
    path: '/example'
    defaults:
      _controller: 'Drupal\my_module\Controller\ExampleController::handle'
    requirements:
      _permission: 'access content'

.. code-block:: php

  // modules/my_module/src/Controller/ExampleController.php

  class ExampleController {

    public function handle() {
      // ...
    }

  }

After:

.. code-block:: yaml

  # my_module.routing.yml

  my_module.example:
    path: '/example'
    defaults:
      _controller: 'Drupal\my_module\Controller\ExampleController'
    requirements:
      _permission: 'access content'

.. code-block:: php

  // modules/my_module/src/Controller/ExampleController.php

  class ExampleController {

    public function __invoke() {
      // ...
    }

  }

Automated testing
=================

* `Workshop notes <https://github.com/opdavies/workshop-drupal-automated-testing>`_
* `Workshop code <https://github.com/opdavies/workshop-drupal-automated-testing-code>`_