-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathform-native.js
148 lines (135 loc) · 4.41 KB
/
form-native.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
import { css, html, LitElement } from 'lit';
import { findFormElements, getFormElementData, isCustomFormElement, isNativeFormElement } from './form-helper.js';
import { FormMixin } from './form-mixin.js';
import { getUniqueId } from '../../helpers/uniqueId.js';
/**
* A component that can be used to build sections containing interactive controls that are validated and submitted as a group.
* These interactive controls are submitted using a native HTML form submission.
* @slot - The native and custom form elements that participate in validation and submission
* @fires submit - Dispatched when the form is submitted. Cancelling this event will prevent form submission.
*/
class FormNative extends FormMixin(LitElement) {
static get properties() {
return {
/**
* The URL that processes the form submission.
* @type {string}
*/
action: { type: String },
/**
* If the value of the method attribute is post, enctype is the MIME type of the form submission.
* @type {'application/x-www-form-urlencoded'|'multipart/form-data'|'text/plain'}
*/
enctype: { type: String },
/**
* The HTTP method to submit the form with.
* @type {'get'|'post'}
*/
method: { type: String },
/**
* Indicates where to display the response after submitting the form.
* @type {'_self '|'_blank'|'_parent'|'_top'}
*/
target: { type: String },
};
}
static get styles() {
return css`
:host {
display: block;
}
:host([hidden]) {
display: none;
}
`;
}
constructor() {
super();
this.action = '';
this.enctype = 'application/x-www-form-urlencoded';
this.method = 'get';
this.target = '_self';
}
render() {
const errors = [...this._errors]
.filter(([, eleErrors]) => eleErrors.length > 0)
.map(([ele, eleErrors]) => ({ href: `#${ele.id}`, message: eleErrors[0], onClick: () => ele.focus() }));
return html`
<d2l-form-error-summary .errors=${errors}></d2l-form-error-summary>
<slot></slot>
`;
}
shouldUpdate(changedProperties) {
if (!super.shouldUpdate(changedProperties)) {
return false;
}
const ignoredProps = new Set(['action', 'enctype', 'method', 'target']);
return [...changedProperties].filter(([prop]) => !ignoredProps.has(prop)).length > 0;
}
async requestSubmit(submitter) {
const errors = await this.validate();
if (errors.size > 0) {
return;
}
this._dirty = false;
const form = document.createElement('form');
form.addEventListener('formdata', this._onFormData);
form.id = getUniqueId();
form.action = this.action;
form.enctype = this.enctype;
form.method = this.method;
form.target = this.target;
this.appendChild(form);
let customFormData = {};
const formElements = findFormElements(this);
for (const ele of formElements) {
const eleData = getFormElementData(ele, submitter);
const isCustom = isCustomFormElement(ele);
if (isCustom || ele === submitter) {
customFormData = { ...customFormData, ...eleData };
}
if (!isCustom && isNativeFormElement(ele)) {
ele.setAttribute('form', form.id);
}
}
for (const entry of Object.entries(customFormData)) {
const input = document.createElement('input');
input.type = 'hidden';
input.name = entry[0];
input.value = entry[1];
form.appendChild(input);
}
const submit = this.dispatchEvent(new CustomEvent('submit', { bubbles: true, cancelable: true }));
/** Dispatched after the entry list representing the form's data is constructed. This happens when the form is submitted just prior to submission. The form data can be obtained from the `detail`'s `formData` property. */
this.dispatchEvent(new CustomEvent('formdata', { detail: { formData: new FormData(form) } }));
if (submit) {
form.submit();
}
form.remove();
}
async submit() {
return this.requestSubmit(null);
}
async validate() {
let errors = [];
const errorMap = new Map();
const formElements = findFormElements(this);
for (const ele of formElements) {
const eleErrors = await this._validateFormElement(ele, true);
if (eleErrors.length > 0) {
errors = [...errors, ...eleErrors];
errorMap.set(ele, eleErrors);
}
}
this._errors = errorMap;
if (this.shadowRoot && errorMap.size > 0) {
const errorSummary = this.shadowRoot.querySelector('d2l-form-error-summary');
this.updateComplete.then(() => errorSummary.focus());
}
return errorMap;
}
_onFormData(e) {
e.stopPropagation();
}
}
customElements.define('d2l-form-native', FormNative);