From 32056846ffb5814ffc6810d1b783ece87bae63b3 Mon Sep 17 00:00:00 2001 From: Elizabeth Mitchell Date: Thu, 17 Aug 2023 16:04:24 -0700 Subject: [PATCH] fix(button): add value to form when submitting Fixes #4526 PiperOrigin-RevId: 557962836 --- button/internal/button.ts | 23 ++++++++++++++++++++++ iconbutton/internal/icon-button.ts | 23 ++++++++++++++++++++++ internal/controller/form-submitter.ts | 18 ++++++++++++++++- internal/controller/form-submitter_test.ts | 23 +++++++++++++++++++++- 4 files changed, 85 insertions(+), 2 deletions(-) diff --git a/button/internal/button.ts b/button/internal/button.ts index 130781ebdba..c976b6c66aa 100644 --- a/button/internal/button.ts +++ b/button/internal/button.ts @@ -67,6 +67,29 @@ export abstract class Button extends LitElement implements FormSubmitter { @property() type: FormSubmitterType = 'submit'; + @property() value = ''; + + get name() { + return this.getAttribute('name') ?? ''; + } + set name(name: string) { + this.setAttribute('name', name); + } + + /** + * The associated form element with which this element's value will submit. + */ + get form() { + return this[internals].form; + } + + /** + * The labels this element is associated with. + */ + get labels() { + return this[internals].labels; + } + @query('.button') private readonly buttonElement!: HTMLElement|null; @queryAssignedElements({slot: 'icon', flatten: true}) diff --git a/iconbutton/internal/icon-button.ts b/iconbutton/internal/icon-button.ts index a4a91725ca4..9bba1f16f9c 100644 --- a/iconbutton/internal/icon-button.ts +++ b/iconbutton/internal/icon-button.ts @@ -77,6 +77,29 @@ export class IconButton extends LitElement implements FormSubmitter { @property() type: FormSubmitterType = 'submit'; + @property() value = ''; + + get name() { + return this.getAttribute('name') ?? ''; + } + set name(name: string) { + this.setAttribute('name', name); + } + + /** + * The associated form element with which this element's value will submit. + */ + get form() { + return this[internals].form; + } + + /** + * The labels this element is associated with. + */ + get labels() { + return this[internals].labels; + } + @state() private flipIcon = isRtl(this, this.flipIconInRtl); /** @private */ diff --git a/internal/controller/form-submitter.ts b/internal/controller/form-submitter.ts index a2ff01bc3da..02cd4a460d6 100644 --- a/internal/controller/form-submitter.ts +++ b/internal/controller/form-submitter.ts @@ -34,6 +34,20 @@ export interface FormSubmitter extends ReactiveElement, WithInternals { * - button: The element does nothing. */ type: FormSubmitterType; + + /** + * The HTML name to use in form submission. When combined with a `value`, the + * submitting button's name/value will be added to the form. + * + * Names must reflect to a `name` attribute for form integration. + */ + name: string; + + /** + * The value of the button. When combined with a `name`, the submitting + * button's name/value will be added to the form. + */ + value: string; } type FormSubmitterConstructor = @@ -71,7 +85,8 @@ export function setupFormSubmitter(ctor: FormSubmitterConstructor) { (ctor as unknown as typeof ReactiveElement).addInitializer(instance => { const submitter = instance as FormSubmitter; submitter.addEventListener('click', async event => { - const {type, [internals]: {form}} = submitter; + const {type, [internals]: elementInternals} = submitter; + const {form} = elementInternals; if (!form || type === 'button') { return; } @@ -102,6 +117,7 @@ export function setupFormSubmitter(ctor: FormSubmitterConstructor) { }); }, {capture: true, once: true}); + elementInternals.setFormValue(submitter.value); form.requestSubmit(); }); }); diff --git a/internal/controller/form-submitter_test.ts b/internal/controller/form-submitter_test.ts index d5d85e52b4f..25f813ab9d3 100644 --- a/internal/controller/form-submitter_test.ts +++ b/internal/controller/form-submitter_test.ts @@ -7,7 +7,7 @@ // import 'jasmine'; (google3-only) import {html, LitElement} from 'lit'; -import {customElement} from 'lit/decorators.js'; +import {customElement, property} from 'lit/decorators.js'; import {Environment} from '../../testing/environment.js'; import {Harness} from '../../testing/harness.js'; @@ -30,6 +30,8 @@ class FormSubmitterButton extends LitElement { static formAssociated = true; type: FormSubmitterType = 'submit'; + @property({reflect: true}) name = ''; + value = ''; [internals] = this.attachInternals(); } @@ -116,4 +118,23 @@ describe('setupFormSubmitter()', () => { .withContext('event.submitter') .toBe(harness.element); }); + + it('should add name/value to form data when present', async () => { + const {harness, form} = await setupTest(); + form.addEventListener('submit', event => { + event.preventDefault(); + }); + + harness.element.name = 'foo'; + harness.element.value = 'bar'; + + await harness.clickWithMouse(); + + const formData = Array.from(new FormData(form)); + expect(formData.length).withContext('formData.length').toBe(1); + + const [formName, formValue] = formData[0]; + expect(formName).toBe('foo'); + expect(formValue).toBe('bar'); + }); });