Radios and checkboxes not rendered correctly when nested inside other render arrays

Created on 30 October 2021, over 3 years ago
Updated 16 February 2025, about 2 months ago

Problem/Motivation

When a render array contains a form field with '#type' of "radios" or "checkboxes", and that render array is added to the markup for the block being rendered the title is rendered but not the radio buttons or checkboxes themselves (from '#options').

I can work around this by replacing '#options' with '#children' => [['#type' => 'radio', ...], ['#type' => 'radio', ...], ...], which succeeds in getting the radio buttons rendered, but according to the documentation for Radios or Checkboxes, I should not have to do that.

Steps to reproduce

  1. Create a custom template
  2. Add ['#type' => 'radios', '#title' => 'Whatever', '#options' => [1 => 'Larry', 2 => 'Moe', 3 => 'Curly']] to part of what the custom template is fed.
  3. Observe that the title is rendered, but the radio buttons are not.

Proposed resolution

Render the radio buttons.

πŸ› Bug report
Status

Active

Version

11.0 πŸ”₯

Component

render system

Created by

πŸ‡ΊπŸ‡ΈUnited States bkline Virginia

Live updates comments and jobs are added and updated live.
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 daggerhart

    Ran into this while working on Radios in a table.

    I looked into the form processing a little bit, and confirmed #12 πŸ› Radios and checkboxes not rendered correctly when nested inside other render arrays Active . \Drupal\Core\Render\Element\Radios::processRadios was not being run on radios elements within a table.

    My temp solution was "whatever it takes" to get the radios working and I ended up with this method:

    
    /**
     * Takes a normal render array for a radios element and makes it work within
     * a rendered table element. This solves a core Drupal bug where Radios are
     * not rendered at all within a table.
     *
     * @link https://www.drupal.org/project/drupal/issues/3246825
     *
     * @param array $radios
     *   Normal radios render array.
     *
     * @return array
     *   Fixed render array with child Radio (singular) elements.
     */
    public function fixNestedRadios(array &$form, array $render_parents, string $render_name, array $radios): array {
      // First item in the parents array is the "tree name".
      $tree_name = reset($render_parents);
    
    
      // This container contains a hidden field that registers the $tree_name with
      // the $form_state. This trick allows our custom-rendered radios to be found
      // in $form_state->getValue($tree_name);
      if ($tree_name && !isset($form[$tree_name])) {
        $form[$tree_name] = [
          '#type' => 'hidden',
        ];
      }
    
      // Build the <input> element names and ids we'll need.
      $parent_names = $render_parents;
      $parent_name = array_shift($parent_names);
      if ($parent_name && $parent_names) {
        $parent_name .= '[' . implode('][', $parent_names) . ']';
      }
    
      $child_name = $parent_name ?
        $parent_name . "[$render_name]" :
        $render_name;
    
      $radios['#id'] = $radios['#id'] ?? Html::getUniqueId($child_name);
      $radios['#title_display'] = $radios['#title_display'] ?? 'visible';
      $radios['#description_display'] = $radios['#description_display'] ?? 'visible';
      $radios['#default_value'] = $radios['#default_value'] ?? FALSE;
      $radios['#attributes'] = $radios['#attributes'] ?? [];
      $radios['#parents'] = $render_parents;
    
      // Render each of the radios options as a single radio element. Neither
      // $form nor $form_state are actually used in this process, just required.
      $form_state = new FormState();
      $radios = Element\Radios::processRadios($radios, $form_state, $form);
    
      foreach (Element::children($radios) as $index) {
        // Radios::processRadios() doesn't set the #value field for the child radio
        // elements, but later the Radio::preRenderRadio() method will expect it. We
        // can set these values from the $radios #default_value if needed.
        // - '#return_value' is the "value='123'" attribute for the form element.
        // - '#value' is the over-all value of the radios group of elements.
        $radios[$index]['#value'] = $radios[$index]['#value'] ?? $radios['#default_value'];
    
        // Some other part of the rendering process isn't working, and this field
        // rendered as an <input> ends up not having a "name" attribute.
        $radios[$index]['#name'] = $child_name;
      }
    
      return $radios;
    }
    

    Maybe these things I had to do to make it work can be a clue for someone more familiar with the how forms -> render arrays -> forms work.

Production build 0.71.5 2024