Password confirm field does not inherit #attributes attribute

Created on 12 September 2019, over 5 years ago
Updated 14 February 2023, about 2 years ago

Use Case: password_confirm form field type isn't inheriting the attributes

When a custom form is to be developed that has password and confirm password fields, that need to follow a pattern.

Based on the above requirements password_confirm is the correct form field type to use.

However since password_confirm field type does not inherit #attributes, the pass1 and pass2 fields will not contain the attributes set on the password_confirm field.

Proposed resolution: Allow #attributes attribute for password_confirm field

Change required
A small change in PasswordConfirm.php should solve the problem. Function processPasswordConfirm(&$element, FormStateInterface $form_state, &$complete_form) , should have following changed lines

'#attributes' => array_merge_recursive($element['#attributes'], ['class' => ['password-field', 'js-password-field']]),

'#attributes' => array_merge_recursive($element['#attributes'], ['class' => ['password-confirm', 'js-password-confirm']]),

Function should look as below

/**
   * Expand a password_confirm field into two text boxes.
   */
  public static function processPasswordConfirm(&$element, FormStateInterface $form_state, &$complete_form) {
    $element['pass1'] = [
      '#type' => 'password',
      '#title' => t('Password'),
      '#value' => empty($element['#value']) ? NULL : $element['#value']['pass1'],
      '#required' => $element['#required'],
      '#attributes' => array_merge_recursive(
        $element['#attributes'],
        ['class' => ['password-field', 'js-password-field']]
      ),
      '#error_no_message' => TRUE,
    ];
    $element['pass2'] = [
      '#type' => 'password',
      '#title' => t('Confirm password'),
      '#value' => empty($element['#value']) ? NULL : $element['#value']['pass2'],
      '#required' => $element['#required'],
      '#attributes' => array_merge_recursive(
        $element['#attributes'],
        ['class' => ['password-confirm', 'js-password-confirm']]
      ),
      '#error_no_message' => TRUE,
    ];
    $element['#element_validate'] = [[get_called_class(), 'validatePasswordConfirm']];
    $element['#tree'] = TRUE;

    if (isset($element['#size'])) {
      $element['pass1']['#size'] = $element['pass2']['#size'] = $element['#size'];
    }

    return $element;
  }

Remaining tasks

API changes

Forms API

๐Ÿ› Bug report
Status

Needs review

Version

10.1 โœจ

Component
Formย  โ†’

Last updated about 7 hours ago

Created by

๐Ÿ‡ง๐Ÿ‡ชBelgium tim-diels Belgium ๐Ÿ‡ง๐Ÿ‡ช

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

Merge Requests

Comments & Activities

Not all content is available!

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

  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand danielveza Brisbane, AU

    I quite like the suggested approach at #33 of adding a dedicated key for the password_confirm attributes.

    I've added a new patch that supports this, they now no longer inhertit the elements base #attributes, there is a new array key that can be passed called #pass2_attributes. Happy for feedback on the name. I called it this because the password confirmation form element is called #pass2, so this seemed the most consistent.

  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand danielveza Brisbane, AU
  • Status changed to Needs work about 2 years ago
  • ๐Ÿ‡บ๐Ÿ‡ธUnited States smustgrave

    Reviewing #35 and also like the idea of using a 2nd variable.

    Can the new solution approach be added to the issue summary.

    For the remaining tasks will those have to happen after this is merged? Should they be follow ups?

  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand danielveza Brisbane, AU
    +++ b/core/lib/Drupal/Core/Render/Element/PasswordConfirm.php
    @@ -68,28 +68,38 @@ public static function valueCallback(&$element, $input, FormStateInterface $form
    +      '#attributes' => is_array($element['#attributes']) ?
    +      array_merge_recursive($element['#attributes'], $pass1_attributes) :
    +      $pass1_attributes,
    

    #pass1 inherits the attributes from the element. I'll update the IS and write a CR

  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand danielveza Brisbane, AU
  • Status changed to Needs review about 2 years ago
  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand danielveza Brisbane, AU

    Updated the existing CR - Ready for review

  • Status changed to Needs work about 2 years ago
  • ๐Ÿ‡ฆ๐Ÿ‡บAustralia darvanen Sydney, Australia

    Fabulous, thanks @DanielVeza, that just leaves drafting/adding some documentation.

    For anyone wishing to help with that, I'd say the @code example at the top of \Drupal\Core\Render\Element\PasswordConfirm is a good place to put a simple example, perhaps even a copy of the one in the change record.

  • ๐Ÿ‡ช๐Ÿ‡ธSpain eduardo morales alberti Spain, ๐Ÿ‡ช๐Ÿ‡บ
  • ๐Ÿ‡ช๐Ÿ‡ธSpain eduardo morales alberti Spain, ๐Ÿ‡ช๐Ÿ‡บ

    Not the same issue, but could be related because are altering the same field.

  • Pipeline finished with Failed
    about 1 year ago
    Total: 512s
    #125753
  • Pipeline finished with Failed
    about 1 year ago
    Total: 484s
    #125762
  • ๐Ÿ‡ฆ๐Ÿ‡บAustralia dpi Perth, Australia

    Lets try to implement this a little more simpler by working with what we have, and not implementing brand new element properties.

    This is a similar methodology to what you can do with checkboxes and/or radios to access the sub elements:

    # Where `$form['account']['pass']` is the `password_confirm` element:
    $form['account']['pass']['pass1']['#attributes']['class'] = ['foobar'];
    

    We just need to update \Drupal\Core\Render\Element\PasswordConfirm::processPasswordConfirm to merge attributes from the passed render array, rather than override:

    app/core/lib/Drupal/Core/Render/Element/PasswordConfirm.php:76

    Old/current:

    '#attributes' => [

    Would change to:

    '#attributes' => $element["pass1"]["#attributes"] ?? [] + [

  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand danielveza Brisbane, AU

    Agree with the changes that @dpi suggested. Updating the title and the MR to reflect what the new solution will be.

  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand quietone

    Closed some issues that I think are duplicates, no credit to transfer. Also hiding files.

  • First commit to issue fork.
  • Pipeline finished with Failed
    26 days ago
    Total: 114s
    #445232
  • ๐Ÿ‡ฆ๐Ÿ‡บAustralia geoffreyr

    The latest version of the MR now has a dedicated form and test case for Password Confirm widgets, and the hook to modify the User Registration form has been removed.

  • Pipeline finished with Failed
    21 days ago
    Total: 421453s
    #445249
  • ๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand danielveza Brisbane, AU

    Popped a review with some thoughts.

    I think we need to decide the scope of this. Do we want to:

    Allow specific attributes to be passed to the pass2 element.
    OR
    Allow the pass2 element to inherit the properties passed to the password_confirm element.

    I'd vote for the 2nd, and maybe open a follow up for the first.

    Whats others thoughts?

  • ๐Ÿ‡ฆ๐Ÿ‡บAustralia darvanen Sydney, Australia

    I would love pass2 to inherit the attributes of pass1 with an override capacityโ€ฆ but I suspect that will break a great many sites.
    Iโ€™m not sure getting the pattern right in this instance is worth the pain weโ€™re likely to cause for a small heap of intermediate site builders.

  • Pipeline finished with Failed
    18 days ago
    Total: 236s
    #452167
  • Pipeline finished with Failed
    18 days ago
    Total: 447s
    #452168
  • ๐Ÿ‡ฆ๐Ÿ‡บAustralia geoffreyr

    One change I added in the latest version of the MR is to give the attribute merging its own dedicated method PasswordConfirm::combineAttributes to merge the attribute arrays. This allows us to reuse the same merging code across both pass1 and pass2, but also to normalise the autocomplete attribute for both.

    The reason why I think we have to normalise autocomplete is because the Nightwatch tests appeared to pick up a case via Axe where pass1 on the user edit form was invalid. After some investigation, I found that pass1 was rendering with the attribute autocomplete="off new-password", which appears to be invalid. To this end, I've tried to pick up the "off" value within either the array or string values of this attribute, and if it's found, reduce it to just autocomplete="off".

    A good example of where this might be set is in \Drupal\user\AccountForm::form. This obviously makes handling attributes more complicated, and I'm wondering if this too is the right approach.

  • Pipeline finished with Success
    18 days ago
    Total: 698s
    #452245
Production build 0.71.5 2024