{"version":3,"file":"PdsFormElement.chunk-Bk2LVulk.js","sources":["../../../../../packages/web-components/src/lib/components/pds-form-element/PdsFormElement.ts"],"sourcesContent":["import '@principal/design-system-icons-web/alert-circle';\nimport { html, isServer, nothing } from 'lit';\nimport { property, state } from 'lit/decorators.js';\nimport 'element-internals-polyfill';\nimport { required } from '../../decorators/required';\nimport { PdsElement } from '../PdsElement';\n\n/**\n * The base class for form elements\n * All field properties are reflected due to this bug: https://github.com/lit/lit/issues/3912\n * @slot label-after Optional: Use this slot if markup should be inserted after the label text, e.g. a toolip.\n */\nexport abstract class PdsFormElement extends PdsElement {\n // This adds an index signature for TypeScript so we can\n // index into the Lit element object with a string key\n // It allows us to do this[propertyName] checks\n [key: string]: any;\n\n /**\n * @internal\n *\n * Tells the browser to treat the element like a form field\n * Ties the element to the HTML form it is in\n *\n */\n static formAssociated = true;\n\n /**\n * @internal\n *\n * An instance of element internals\n * Contains properties and methods that allows the element\n * to participate fully in the HTML form it's in\n */\n internals: Omit<\n ElementInternals,\n 'ariaBrailleLabel' | 'ariaBrailleRoleDescription'\n >;\n\n /**\n * @internal\n *\n * Stores the value for the `value` getter and setter\n */\n internalValue: string;\n\n /**\n * @internal\n *\n * Stores the intial value of the field so that it can be reset\n */\n defaultValue: string;\n\n /**\n * @internal\n *\n * Stores the value for the `errorMessage` getter and setter\n */\n internalErrorMessage: string;\n\n /**\n * @internal\n *\n * The underlying HTML form field\n * This should be implemented with `@query`\n * in the implementing class.\n */\n abstract field: HTMLElement;\n\n constructor() {\n super();\n\n /**\n * @internal\n *\n * Call the attachInternals() method on the element\n * to get access to extra methods and properties\n * for form fields, like setFormValue() and setValidity().\n */\n this.internals = this.attachInternals();\n }\n\n protected override firstUpdated() {\n super.firstUpdated();\n this.setInternalValidity();\n this.randomId = this.getRandomId();\n\n this.defaultValue = this.value || this.getAttribute('value') || '';\n\n // TODOv4: [ValidationTimeout] - Check to see if this setTimeout is still needed\n // This validation logic needs to happen late because it messes\n // with the render content and leads to hydration issues (and issues with child elements)\n // This may be something we can remove if we can find a better way to handle this\n // The validation is only for development - so end users should never see the render content change\n setTimeout(() => {\n if (!this.verifyLabel()) {\n this.hasLabel = false;\n }\n }, 1000);\n }\n\n protected customSetterSetAttribute(\n attributeName: string,\n newValue: boolean | string | number,\n ): void {\n if (\n newValue &&\n newValue !== 'undefined' &&\n // Don't run this in tests\n (!isServer || process.env.NODE_ENV !== 'test') &&\n newValue.toString() !== this[attributeName]?.toString()\n ) {\n this.setAttribute(attributeName, newValue.toString());\n } else {\n this.removeAttribute(attributeName);\n }\n }\n\n set value(newValue) {\n const oldValue = this.value;\n\n this.customSetterSetAttribute('value', newValue);\n\n // store the value so it can be retrieved by the getter\n this.internalValue = newValue;\n\n // sets the current value of the control\n this.internals.setFormValue(newValue);\n\n // update the actual field\n this.updateField();\n\n // rerender the component\n this.requestUpdate('value', oldValue);\n }\n\n /**\n * The value of the form field.\n */\n @property()\n get value() {\n return this.internalValue;\n }\n\n /**\n * The label of the form field.\n * Must be plain text.\n * If label requires additional markup (e.g., superscript),\n * then use the label slot instead.\n */\n @property({ reflect: true })\n label?: string;\n\n /**\n * Makes the label screen-reader-only and visually hidden.\n * Hiding the label is **strongly discouraged** and should only\n * be used when descriptive text exists elsewhere on the page.\n */\n @property({ type: Boolean })\n hideLabel: boolean = false;\n\n /**\n * The name of the form field. This is a **required** property.\n */\n @required\n @property({ reflect: true })\n name: string;\n\n /**\n * The id of the form field\n *\n * Overrides the auto-generated id\n */\n @property()\n fieldId?: string;\n\n /**\n *\n * Explains why the form element is disabled\n */\n @property()\n disabledContext?: string;\n\n set required(newRequired: boolean) {\n const oldRequiredValue = this.required;\n\n this.customSetterSetAttribute('required', newRequired);\n\n // ariaRequired needs to be a string\n this.internals.ariaRequired = newRequired ? 'true' : 'false';\n\n // needed to rerender the component\n this.requestUpdate('required', oldRequiredValue);\n }\n\n /**\n * Indicates that the form field is required.\n */\n @property({ type: Boolean })\n get required(): boolean {\n // default to false\n if (typeof this.internals.ariaRequired === 'undefined') {\n return false;\n }\n\n return this.internals.ariaRequired === 'true';\n }\n\n set disabled(newDisabled: boolean) {\n const oldDisabledValue = this.disabled;\n\n this.customSetterSetAttribute('disabled', newDisabled);\n\n // ariaDisabled needs to be a string\n this.internals.ariaDisabled = newDisabled ? 'true' : 'false';\n\n // needed to rerender the component\n this.requestUpdate('disabled', oldDisabledValue);\n }\n\n /**\n * Indicates that the form field is disabled.\n */\n @property({ type: Boolean })\n get disabled(): boolean {\n // default to false\n if (typeof this.internals.ariaDisabled === 'undefined') {\n return false;\n }\n return this.internals.ariaDisabled === 'true';\n }\n\n set readonly(newReadonly: boolean) {\n const oldReadonlyValue = this.readonly;\n\n this.customSetterSetAttribute('readonly', newReadonly);\n\n // ariaReadOnly needs to be a string\n this.internals.ariaReadOnly = newReadonly ? 'true' : 'false';\n\n // needed to rerender the component\n this.requestUpdate('readonly', oldReadonlyValue);\n }\n\n /**\n * Indicates that the form field is readonly.\n */\n @property({ type: Boolean })\n get readonly(): boolean {\n // default to false\n if (typeof this.internals.ariaReadOnly === 'undefined') {\n return false;\n }\n\n return this.internals.ariaReadOnly === 'true';\n }\n\n /**\n * The additional context used to assist the\n * user in filling out the form field.\n */\n @property()\n helpText?: string;\n\n set errorMessage(newErrorMessage) {\n const oldErrorMessageValue = this.errorMessage;\n\n this.customSetterSetAttribute('errormessage', newErrorMessage);\n\n this.internalErrorMessage = newErrorMessage;\n\n this.setInternalValidity();\n\n // needed to rerender the component\n this.requestUpdate('errorMessage', oldErrorMessageValue);\n }\n\n /**\n * The error message to display when the form\n * field is in an invalid state.\n */\n @property()\n get errorMessage() {\n return this.internalErrorMessage;\n }\n\n /**\n * @internal\n */\n @state()\n randomId: string;\n\n /**\n * Reset the field's value to it's value attribute.\n * This gets called when `form.reset()` is invoked\n * (typically by clicking a type=\"reset\" button).\n */\n formResetCallback() {\n this.value = this.defaultValue;\n }\n\n /**\n * Calls `setValidity` on the element internals\n * with the appropriate options depending on whether\n * the element has an `errorMessage` or not.\n */\n setInternalValidity() {\n if (this.field && this.internalErrorMessage) {\n this.internals.setValidity(\n { customError: true },\n this.internalErrorMessage,\n this.field,\n );\n } else {\n // calling setValidity with an empty object\n // indicates that the element meets contraint\n // validation rules (i.e., is in a valid state)\n this.internals.setValidity({});\n }\n }\n\n /**\n * update the actual field's value\n */\n protected updateField() {}\n\n /**\n * Determines whether the element includes `helpText`\n */\n hasHelpText() {\n return !!this.helpText || !!this.slotNotEmpty('helpText');\n }\n\n /**\n * Gets the appropriate `aria-describedby` value\n * based on the existence of `helpText` and `errorMessage`.\n *\n * @returns space-seperated strings of ids\n */\n getAriaDescribedBy() {\n return [\n this.hasHelpText() ? `${this.name}-help-text` : '',\n this.errorMessage ? `${this.name}-error-message` : '',\n ]\n .filter(Boolean)\n .join(' ');\n }\n\n /**\n * Gets the \"fieldId\" value\n *\n * @returns the field id\n */\n getFieldId() {\n return this.fieldId || `${this.name}-${this.randomId}-field`;\n }\n\n /**\n * @internal\n *\n * This boolean is used to determine if the form element has a label\n * It will be validated in the ValidationTimeout in firstUpdated\n */\n @state()\n hasLabel: boolean = true;\n\n /**\n * Determines whether the element has a label as an attribute or slot\n * Console logs an error if the element does not have a label\n */\n verifyLabel() {\n const labelExists = !!this.label || !!this.slotNotEmpty('label');\n\n if (!labelExists) {\n console.error(\n '\"label\" is required as a property or as a slot but is undefined',\n this,\n );\n }\n\n return labelExists;\n }\n\n /**\n * Creates an HTML template for the label\n * that is tied to the form field via the `for`\n * attribute and adds the required indicator,\n * if applicable.\n *\n * @returns an HTML template for the label\n */\n protected labelTemplate() {\n return html``;\n }\n\n /**\n * Creates an HTML template for help text,\n * if the element has help text.\n * Adds an `id` that is included in the form field's\n * aria-describedby attribute.\n *\n * @returns an HTML template for help text\n */\n protected helpTextTemplate() {\n return this.hasHelpText()\n ? html`