Use the builder pattern to make it easier to create render arrays

Created on 7 August 2014, over 10 years ago
Updated 15 February 2023, about 2 years ago

Problem/Motivation

Since Drupal 8 we've done some work to remove 'magic array keys'. But we still have render arrays, which every time you use them cause you to need to look-up the allowed keys. This is really bad DX. It's not clear what the allowed keys are, or where to find them. For backend developers who don't have years of experience with the nuts and bolts of Drupal's rendering system, this is more of a learning cliff than a learning curve, and it's very discouraging.

Proposed resolution

Provide builder classes for each element type to make defining render arrays discoverable from IDEs.

before

$variables['table'] = [
  '#type' => 'table',
  '#header' => $headers,
  '#rows' => $rows,
  '#attributes' => [
    'id' => $table_id,
  ],
  '#tabledrag' => [
    [
      'action' => 'order',
      'relationship' => 'sibling',
      'group' => $weight_class,
    ],
  ],
];

after

$variables['table'] = TableElement::create()
  ->setHeader($headers)
  ->setRows($rows)
  ->setAttributes([
    'id' => $table_id,
  ])
  ->setTableDrag([
    [
      'action' => 'order',
      'relationship' => 'sibling',
      'group' => $weight_class,
    ],
  ])->toRenderArray();

This sort of interface is a lot clearer, and borrows heavily from both Url helper and FieldDefinition stuff.

In this issue, we should only introduce a base class for these builders, plus one or two concrete implementations that we use in core, to prove they work. Then, in follow-up child issues, we can introduce more builders for the various core elements, piecemeal.

Remaining tasks

Refactor the patch in #106 to modernize it -- i.e., adding type hints -- and remove most of the builders it implements (punt to follow-ups). Keep one or two useful builders, and change core to use them where possible. Add unit or kernel test coverage. Then review and commit.

Manual testing

  • To check DropbuttonBuilder: go to /admin/content, check that the dropbutton display correctly
  • To check ImageBuilder: go to http://drupal.local/en/admin/help/contextual, make sure the icon displays in the help text

User interface changes

None

API changes

Creates a new optional way to create render elements.

Feature request
Status

Needs work

Version

10.1

Component
Theme 

Last updated about 5 hours ago

Created by

🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10

Live updates comments and jobs are added and updated live.
  • Needs tests

    The change is currently missing an automated test that fails when run with the original code, and succeeds when the bug has been fixed.

  • Needs change record

    A change record needs to be drafted before an issue is committed. Note: Change records used to be called change notifications.

  • Needs manual testing

    The change/bugfix cannot be fully demonstrated by automated testing, and thus requires manual testing in a variety of environments.

Sign in to follow issues

Comments & Activities

Not all content is available!

It's likely this issue predates Contrib.social: some issue and comment data are missing.

  • 🇺🇸United States smustgrave

    Was tagged for tests and change record that I believe still need to happen.

  • 🇷🇺Russia Chi

    PHP has evolved a lot since this issue was created. Type hints, property promotion, named arguments, etc. So instead of builders we can create value objects. It'll give same DX in modern IDEs.

    Builder

    $variables['table'] = TableElement::create()
      ->setHeader($headers)
      ->setRows($rows)
      ->setAttributes(['id' => $table_id])
      ->setTableDrag([
        [
          'action' => 'order',
          'relationship' => 'sibling',
          'group' => $weight_class,
        ],
      ],
    )->toRenderArray();
    

    Constructor

    $variables['table'] = new TableElement(
      header: $headers,
      rows: $rows,
      attributes: ['id' => $table_id],
      tableDrag: [
        [
          'action' => 'order',
          'relationship' => 'sibling',
          'group' => $weight_class,
        ],
      ],
    )->toRenderArray();
    
  • 🇷🇺Russia Chi

    Overall, this will only mitigate the problem. Dropping render arrays is the only way to fully resolve the issue. I’ve filed a ticket in the ideas queue.

  • 🇬🇧United Kingdom oily Greater London
  • 🇬🇧United Kingdom oily Greater London

    Added change record based on the current issue description.

  • 🇬🇧United Kingdom oily Greater London

    @chi Re: #143 Good point. But this seems like a major step/ leap forward. I hope this issue be completed as-is as a first step?

  • 🇷🇺Russia Chi

    @oily The are dozens different render elements in Drupal core. The MR currently covers just a few ones.

  • 🇨🇦Canada Charlie ChX Negyesi 🍁Canada

    if people want to go ahead with this -- and I am not saying they either should or they should not, I absolutely have no opinion, I do not event want have an opinion -- then, if someone wanted the opinion of an old ghost then #142 is the way because then it's possible to add a reflection based fromRenderable somewhere (ElementInfoManager would be my guess) so that form alter does not need to deal with arrays either. Otherwise, people would need to learn both syntax to be able to debug code.

Production build 0.71.5 2024