const smForm = document.createElement('template') smForm.innerHTML = `
` customElements.define('sm-form', class extends HTMLElement { constructor() { super() this.attachShadow({ mode: 'open' }).append(smForm.content.cloneNode(true)) this.form = this.shadowRoot.querySelector('form') this.formElements this.requiredElements this.submitButton this.resetButton this.allRequiredValid = false this.debounce = this.debounce.bind(this) this.handleInput = this.handleInput.bind(this) this.handleKeydown = this.handleKeydown.bind(this) this.reset = this.reset.bind(this) } debounce(callback, wait) { let timeoutId = null; return (...args) => { window.clearTimeout(timeoutId); timeoutId = window.setTimeout(() => { callback.apply(null, args); }, wait); }; } handleInput(e) { this.allRequiredValid = this.requiredElements.every(elem => elem.isValid) if (!this.submitButton) return; if (this.allRequiredValid) { this.submitButton.disabled = false; } else { this.submitButton.disabled = true; } } handleKeydown(e) { if (e.key === 'Enter' && e.target.tagName !== 'SM-TEXTAREA' ) { if (this.allRequiredValid) { this.submitButton.click() } else { this.requiredElements.find(elem => !elem.isValid).vibrate() } } } reset() { this.formElements.forEach(elem => elem.reset()) } connectedCallback() { const slot = this.shadowRoot.querySelector('slot') slot.addEventListener('slotchange', e => { this.formElements = [...this.querySelectorAll('sm-input, sm-textarea, sm-checkbox, tags-input, file-input, sm-switch, sm-radio')] this.requiredElements = this.formElements.filter(elem => elem.hasAttribute('required')) this.submitButton = e.target.assignedElements().find(elem => elem.getAttribute('variant') === 'primary' || elem.getAttribute('type') === 'submit'); this.resetButton = e.target.assignedElements().find(elem => elem.getAttribute('type') === 'reset'); if (this.resetButton) { this.resetButton.addEventListener('click', this.reset) } }) this.addEventListener('input', this.debounce(this.handleInput, 100)) this.addEventListener('keydown', this.debounce(this.handleKeydown, 100)) } disconnectedCallback() { this.removeEventListener('input', this.debounce(this.handleInput, 100)) this.removeEventListener('keydown', this.debounce(this.handleKeydown, 100)) } })