Clarify SDC documentation by toning down Twig blocks promotion

Created on 30 October 2024, 5 months ago

Problem/Motivation

When helping people with SDC, doing my consultancy job, on Drupal slack, or in contrib issues, I see a lot of confusion and struggle with the components slots, especially related to the embed and block mechanisms.

The current SDC documentation needs to be updated to:

  • tone down the promotion of Twig embed and Twig block
  • preparing the people of using less the templates to templates calls and more the Render API (manually or through display buidling)

Twig blocks are problematic:

  • Originally, they come from Jinja templating language, as a template inheritance mechanism. Something we never want to do in/with a component template.
  • Then, Twig introduced embed tag to use blocks for nesting. It is very Twig specific, and this is the reason why blocks may initially look interesting for SDC authors.

Slots declared as blocks are verbose and confusing.

For example, instead of {{ branding }} and {{ copyright }}:

  • Radix do that in navbar:{% block branding %}{{ branding }}{% endblock %}
  • umami do that in disclaimer : {% block copyright %}{% endblock %}

Some component authors are so confused that they overlap slots and props. For example, bootstrap's card:

{% block card_header_block %}
  {{ card_header }}
{% endblock %}

Slots declared as blocks are complicated to call from another template.

For example, node--event.html.twig template which is passing some formatted fields to umami disclaimer's slots :

{% embed 'umami:disclaimer' only %}
  {% block disclaimer %}
     {{ content.field_date }}
  {% endblock %}
  {% block copyright %}
     {{ content.field_location }}
  {% endblock %}
{% endembed %}

Instead of the simpler and safer:

{{ include(
  'umami:disclaimer', {
    disclaimer: content.field_date,
    copyright: content.field_location
  },
  with_context = false,
) }}

Testing if a slot is empty seems complicated with a block.

Today, on #components Slack channel, brayn7 β†’ came with this snippet which was not working:

Removing the blocks and embed was a straightforward fix:

Twig block is a "template to template" only mechanism.

It is triggered only when a component is called from a presenter template (node.html.twig, field.html.twig...), not when its is called from the Render API. This create a gap between usage of SDC components, which will grow with the development of new display builders tools (SDC Display, UI Patterns 2, Experience Builder...).

Indeed, SDC API has currently a workaround to inject missing blocks with the Render API but this may not stay forever:

$template .= "  {% block $slot_name %}" . PHP_EOL
    . "    {{ $slot_name }}" . PHP_EOL
    . "  {% endblock %}" . PHP_EOL;
}

Proper variables manipulation ({% for item in slot %}, {% if slot %}, {{ slot }}...) regardless of where the component is called, without the need of such a hack.

Proposed resolution

Update Using your new single-directory component β†’

Keep the mentions of block and embed but move it to the page bottom and update any parts which can make readers believing that "slots == Twig blocks". For examples:

Use the Twig include() function if there are no slots or arbitrary HTML properties and the data all follows a defined structure.

Use a Twig {% embed %} tag if you need to populate slots.

Also, don't promote the use of include tag at the same level of include function:

It is recommended to use the include function instead as it provides the same features with a bit more flexibility:

  • The include function is semantically more "correct" (including a template outputs its rendered contents in the current scope; a tag should not display anything)
  • The include function is more "composable"
  • The include function does not impose any specific order for arguments thanks to named arguments.

Source: https://twig.symfony.com/doc/3.x/tags/include.html

Render element is already explained in this page, but it can be promoted a bit. It is a good opportunity to explain the difference between presenter templates:

  • unavoidable because of the lack of display building: breadcrumbs.html.twig, status-message.html.twig, menu.html.twig, pager.html.twig..
  • the ones conflicting with the display building: node.html.twig, field.html.twig, views.html.twig, block.html.twig...

Also, "Component data validation" section can be moved in "Creating a single-directory component" page.

Update What are Props and Slots in Drupal SDC Theming? β†’

Remove any mention of block and embed. The goal of this page is to teach about slots and props, not to list all ways of calling components from templates.

A bit of rewording and an annotated component screenshot like this can also help the reader:

Update API for Single-Directory Components β†’

Replace the example with block and embed by one with include. Or simply remove the example.

Also, a schema can be a good way to show how the different parts of the API are interacting.

Conclusion

SDC in Core is now 16 months old, and the community is still struggling with component templating and usage. I believe updating the confusing parts of the documentation will help the adoption and the blooming of the ecosystem.

🌱 Plan
Status

Active

Version

11.0 πŸ”₯

Component

single-directory components

Created by

πŸ‡«πŸ‡·France pdureau Paris

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

  • Issue created by @pdureau
  • πŸ‡³πŸ‡ΏNew Zealand quietone
  • πŸ‡«πŸ‡·France pdureau Paris

    Also, le'ts remove this misleading section: Does SDC handle contextual links, attributes, title suffix, etc? β†’

    Using a PHP Namespace as a prop type is not JSON schema valid and trigger a weird behaviour, so it must be avoided.

    An empty array or an empty object is JSON Schema valid, but a bad practice. Let's not promote it.

  • πŸ‡«πŸ‡·France pdureau Paris

    Other idea for documentation update: help people to differentiate props and slots in presenter templates:

    In node.html.twig:

    • node variable has the data to send to props
    • content has the data to send to slots

    In user.html.twig:

    1. user variable has the data to send to props
    2. content has the data to send to slots

    ...

  • πŸ‡ΊπŸ‡ΈUnited States dalemoore

    I'm very much in favor of these updates! Your help on Drupal Slack a while back related to using node/user/paragraph rather than content for props was super helpful but that isn't anywhere in the docs IIRC.

    Would you also say that doing this:

    attributes:
          type: Drupal\Core\Template\Attribute
          title: 'Attributes'
          description: 'HTML attributes to apply to the component.'

    in the component.yml is incorrect? Because I see that being done in Umami components, and in Experience Builder components. In my own components, I am noticing some bleeding in from the parent template when then doing adding the attributes prop and doing something like:

    <msu-header{{ attributes }}>....</msu-header>

    Where if I include that component in my region--header.html.twig, it inherits the attributes from the parent div. If I try to use the 'only' keyword, like this:

    {% embed 'msuext:msu-header' only %}
    ... content here ...
    {% endembed %}

    then I cannot include slots/blocks that aren't explicitly included in the Twig template.

  • πŸ‡«πŸ‡·France pdureau Paris

    Hi @dalemoore,

    Would you also say that doing this:

    attributes:
          type: Drupal\Core\Template\Attribute
          title: 'Attributes'
          description: 'HTML attributes to apply to the component.'
    

    in the component.yml is incorrect?

    It is not incorrect in a SDC POV but:

    • useless because the attributes variable is automatically injected to the component anyway, so no need to add it to the component definition
    • ugly because using PHP namespaces (like Drupal\Core\Template\Attribute) as types is not compliant with JSON Schema and SDC has implemented a quirk to mange them. So let's avoid this

    If I try to use the 'only' keyword, like this then I cannot include slots/blocks that aren't explicitly included in the Twig template as blocks.

    I ddin't know this about embed, that's one more reason to promote Twig include() function (function, not tag !) (with with_context = false !) instead of the messy embed tag.

    If the attributes prop isn't right, then I would ask: how do we add a class, ID, or other attribute to our components from presenter templates? Having to manually add a class prop (or other attribute) onto every SDC seem to be the only way, but and is burdensome.

    I would not recommend adding a class prop. attributes props is perfect for this and it is not what is causing issue here. attributes also allows to attach other attributes, and is expected by any mechanism adding semantic annotation, SEO tags, styles utilities from design systems, accessibility attributes... It is very valuable.

    If including an attributes prop, I wonder if it would be better to name it component_attributes instead of just attributes to prevent the leaking of attributes from the presenter template. We already have title_attributes, content_attributes, etc. for other things.

    Radix theme did this mistake: πŸ› Inconsistent `attributes` handling Active Now the have have both COMPONENT_attributes and attributes in some templates, which is confusing and may lead to other issues.

    Using the standard attributes prop must be enough and work well.

  • πŸ‡ΊπŸ‡ΈUnited States dalemoore

    Excellent, thank you @pdureau! I know you have extensive experience working on components, SDC, and design systems. Thanks for sharing your insights on here, hopefully we can get these into the docs so others like me won't be confused and we can move towards some best practices.

    I'll remove the attributes prop from my components. Also, do you have examples of how you're using SDCs? I downloaded UI Patterns, but there are no SDCs in there as examples. Or if you can direct me toward how you mean with using include rather than embed? I don't get how I can do slots with include and again, the docs only say use embed with slots.

  • πŸ‡«πŸ‡·France pdureau Paris

    I downloaded UI Patterns, but there are no SDCs in there as examples

    We don't ship SDC component in UI Patterns 2 by design. UI Patterns 2 is only a "bridge" between SDC and other Drupal API.

    However, we are currently adding an extensive documentation to UI Patterns 2, expected before Christmas, and I hope it will give you the information you need.

    Also, do you have examples of how you're using SDCs?

    UI Suite team (the team doing UI Patterns, which I am part of) are also maintaining SDC themes:

    Beware, there are 2 additions from UI Patterns 2 in those components compared to regular SDC components:

    Beside that, you can get inspiration from them.

    Also, the tool we are using to check our SDC good practices, https://www.drupal.org/project/sdc_devel β†’ , is already usable, but the rules set needs some tweaking (wording, scope, levels...) which will happen around Christmas.

    I don't get how I can do slots with include and again, the docs only say use embed with slots.

    That's why the doc needs to be updated. The issue description has some examples of using include with slots.

  • πŸ‡«πŸ‡·France pdureau Paris
  • πŸ‡ΊπŸ‡ΈUnited States dalemoore

    Thanks @pdureau! Very helpful info. Hopefully we can get some more eyes on this issue to get some more feedback from others who have been/are interested in starting to use SDCs.

  • πŸ‡ΊπŸ‡ΈUnited States dalemoore

    While continuing my quest to slowly implement SDCs, I think it would also be beneficial in the docs to include actual real-world examples of using SDCs as @pdureau has done above. On Using your new single-directory component β†’ , all but one ( the embed example β†’ ) use hard-coded values in the props and slots. I think most of us are going to be wanting examples of how to get data from fields into those, and folks that are more frontend (πŸ™‹πŸ»β€β™‚οΈ) are going to be struggling to figure out how to do that. Nowhere on the page does it mention how to do that (the only ways I know of are to install Devel/Kint, and just recently learned of Xdebug and have been trying to get it working in VScode).

    So far, the only ways I know of to get data out of Drupal and into these SDCs are:

    • code the variable in an .html.twig file (e.g., content['#block_content'].field_name.0.value in a block.html.twig to get a prop value w/o the render bits)
    • some kind of render array magic, I guess through preprocess functions in your .theme? How else would you use this method β†’ ?
    • utilize a contrib module that handles displaying the SDCS, such as SDC Display β†’ , UI Patterns β†’ , and the upcoming Experience Builder β†’

    Are there other ways I'm not aware of? If so, they should be documented. If someone can help me answer these questions, I will try my first time at editing docs on here over the Christmas break. :)

  • πŸ‡«πŸ‡·France pdureau Paris

    some kind of render array magic, I guess through preprocess functions in your .theme? How else would you use this method?

    Most of the time, as the return value of configurable plugins related to display building:

    Do we add this information to the SDc documentation too?

  • πŸ‡«πŸ‡·France pdureau Paris
  • πŸ‡ΊπŸ‡ΈUnited States dalemoore

    @pdureau Yes I think that would be helpful to indicate for newcomers or non-backend folks how they meant to use for Using your component through a render array β†’ , if that isn't meant to be done in a theme. AFAIK everything that was supposed to go with an SDC is supposed to go in the theme in the same folder as the component, or at least done in the theme. But I don't think it's best practice to do any of that in a theme?

  • πŸ‡«πŸ‡·France pdureau Paris

    Step by step, we are gathering information to update this doc. That's nice

  • πŸ‡«πŸ‡·France pdureau Paris

    AFAIK everything that was supposed to go with an SDC is supposed to go in the theme in the same folder as the component, or at least done in the theme. But I don't think it's best practice to do any of that in a theme?

    The component are stored in a theme (or a module), but can be used outside of this theme (or module). Most of the time, component usage is stored in config entities related to display (block.block, entity view displays, views...). Sometimes in content (fields using ckeditor5 data or layout builder data...). Sometimes in code (custom plugins...)

  • πŸ‡ͺπŸ‡ΈSpain idiaz.roncero Madrid

    +1 to this. I also fell on the misleading slots===blocks (or "blocks is the recommended and only implementation for slots") comparison, and other colleagues as well.

    It will be much useful to emphasize that you can implement slots by using includes and embeds, and the tradeoffs and benefits of each approach.

  • πŸ‡«πŸ‡·France pdureau Paris

    We may do a documentation session with @mherchel during DrupalCon Atlanta

    I am very excited about that. There are many sides of SDC documentation which need to be clarified or completed after 2 years of practice.

  • πŸ‡«πŸ‡·France pdureau Paris

    Data primitives mapping between the 3 formats we are manipulating:

    Maybe adding such a table will be useful in the SDC documentation.

  • πŸ‡«πŸ‡·France pdureau Paris

    Suggestion from @mlncn about replaces property:

    looking to improve the documentation slightly, the general advice should be to copy the entire component subdirectory into your themes components directory and add the replaces line. It's not like it can inherit anything it's a true replacement, yes?

    Indeed, the replacement component is still a full component with its Twig template and its YAML definition.

  • πŸ‡«πŸ‡·France nod_ Lille

    @pdureau, please go ahead with updating the documentation per specified in the issue summary, we can always refine later

  • πŸ‡«πŸ‡·France pdureau Paris
  • πŸ‡«πŸ‡·France nod_ Lille

    updating credits

  • πŸ‡«πŸ‡·France pdureau Paris

    From Enrique Lacoma:

    Overriding components in a module, in the doc https://www.drupal.org/docs/develop/theming-drupal/using-single-director... β†’ says you can't override a component in a module, but I was able to do it, is the documentation correct or doing the override inside a module can create any issue? thanks

  • πŸ‡«πŸ‡·France pdureau Paris

    So we are good. I am currently editing the documentation.

  • πŸ‡«πŸ‡·France pdureau Paris

    Update Quickstart β†’

    Add an example without block and with include() as the primary example.

    Waiting πŸ“Œ Review and improve SDC documentation Active to be done to not override their current work.

    Keep the example with block and embed, but add the print node:

    {% block chip_content %}
      {{ chip_content }}
    {% endblock %}
    

    Waiting πŸ“Œ Review and improve SDC documentation Active to be done to not override their current work.

    Update Using your new single-directory component β†’

    Keep the mentions of block and embed but move it to the page bottom and update any parts which can make readers believing that "slots == Twig blocks".

    DONE. IN REVIEW.

    Also, don't promote the use of include tag at the same level of include function:

    DONE. IN REVIEW.

    Render element is already explained in this page, but it can be promoted a bit.

    DONE. IN REVIEW.

    It is a good opportunity to explain the difference between presenter templates:

    • unavoidable because of the lack of display building: breadcrumbs.html.twig, status-message.html.twig, menu.html.twig, pager.html.twig..
    • the ones conflicting with the display building: node.html.twig, field.html.twig, views.html.twig, block.html.twig...

    NOT DONE YET

    Also, "Component data validation" section can be moved in "Creating a single-directory component" page.

    DONE. IN REVIEW.

    Update Creating a single-directory component β†’

    The replacement component is still a full component with its Twig template and its YAML definition.

    TODO

    Data primitives mapping between the 3 formats we are manipulating:

    TODO

    Tell something about custom & contrib Twig extensions.

    TOOD

    Update What are Props and Slots in Drupal SDC Theming? β†’

    Remove any mention of block and embed. The goal of this page is to teach about slots and props, not to list all ways of calling components from templates.

    TODO

    A bit of rewording and an annotated component screenshot like this can also help the reader:

    TODO

    Update API for Single-Directory Components β†’

    Replace the example with block and embed by one with include. Or simply remove the example.

    TODO

    Also, a schema can be a good way to show how the different parts of the API are interacting.

    TODO

    Update Frequently Asked Questions β†’

    says you can't override a component in a module, but I was able to do it,

    TODO

  • πŸ‡ΊπŸ‡ΈUnited States mlncn Minneapolis, MN, USA
  • πŸ‡«πŸ‡·France pdureau Paris
  • πŸ‡ΊπŸ‡ΈUnited States ultimike Florida, USA

    Wow - this is all super-helpful. Huge kudos to @pdureau for taking this on.

    As someone new to SDC, this discussion has been especially helpful in filling in some knowledge gaps for me.

    -mike

  • πŸ‡§πŸ‡ͺBelgium wim leers Ghent πŸ‡§πŸ‡ͺπŸ‡ͺπŸ‡Ί

    Per @pdureau at #3493070-8: Enum values do not have translatable labels β†’ , this will also document meta:enum to provide human-readable labels that also are translatable.

    For example:

        "string_pattern": {
          "type": "string",
          "description": "A string following a regular expression",
          "pattern": "^ba.$",
          "examples": ["bar", "baz", "bat"],
          "meta:enum": {
            "baa": "the sounds of sheeps",
            "bad": "German bathroom",
            "bag": "holding device",
            "bah": "humbug!",
            "bam": "a loud sound",
            "ban": "don't do this",
            "bap": "a British soft bread roll",
            "bas": "from ancient Egyptian religion, an aspect of the soul",
            "bat": "…out of hell",
            "bay": ", sitting by the dock of the"
          },
    

    β€” https://github.com/adobe/jsonschema2md/blob/f3b5773eb610130891503c1cf71b...

Production build 0.71.5 2024