From eae2558f5fbbad949979b6472763bf0fc651b9a4 Mon Sep 17 00:00:00 2001 From: Serhii Kulykov Date: Fri, 29 Nov 2024 18:01:02 +0200 Subject: [PATCH] refactor: extract message-input logic into reusable mixin (#8249) --- packages/message-input/package.json | 1 + .../src/vaadin-message-input-mixin.d.ts | 47 +++++ .../src/vaadin-message-input-mixin.js | 180 ++++++++++++++++++ .../src/vaadin-message-input.d.ts | 38 +--- .../message-input/src/vaadin-message-input.js | 173 +---------------- 5 files changed, 234 insertions(+), 205 deletions(-) create mode 100644 packages/message-input/src/vaadin-message-input-mixin.d.ts create mode 100644 packages/message-input/src/vaadin-message-input-mixin.js diff --git a/packages/message-input/package.json b/packages/message-input/package.json index 9cf3903bb0..ec9d2ae3cb 100644 --- a/packages/message-input/package.json +++ b/packages/message-input/package.json @@ -35,6 +35,7 @@ "polymer" ], "dependencies": { + "@open-wc/dedupe-mixin": "^1.3.0", "@polymer/polymer": "^3.0.0", "@vaadin/button": "24.6.0-beta1", "@vaadin/component-base": "24.6.0-beta1", diff --git a/packages/message-input/src/vaadin-message-input-mixin.d.ts b/packages/message-input/src/vaadin-message-input-mixin.d.ts new file mode 100644 index 0000000000..6beb169474 --- /dev/null +++ b/packages/message-input/src/vaadin-message-input-mixin.d.ts @@ -0,0 +1,47 @@ +/** + * @license + * Copyright (c) 2021 - 2024 Vaadin Ltd. + * This program is available under Apache License Version 2.0, available at https://vaadin.com/license/ + */ +import type { Constructor } from '@open-wc/dedupe-mixin'; +import type { ControllerMixinClass } from '@vaadin/component-base/src/controller-mixin.js'; + +export interface MessageInputI18n { + send: string; + message: string; +} + +export declare function MessageInputMixin>( + base: T, +): Constructor & Constructor & T; + +export declare class MessageInputMixinClass { + /** + * Current content of the text input field + */ + value: string | null | undefined; + + /** + * The object used to localize this component. + * For changing the default localization, change the entire + * `i18n` object. + * + * The object has the following JSON structure and default values: + * + * ``` + * { + * // Used as the button label + * send: 'Send', + * + * // Used as the input field's placeholder and aria-label + * message: 'Message' + * } + * ``` + */ + i18n: MessageInputI18n; + + /** + * Set to true to disable this element. + */ + disabled: boolean; +} diff --git a/packages/message-input/src/vaadin-message-input-mixin.js b/packages/message-input/src/vaadin-message-input-mixin.js new file mode 100644 index 0000000000..b3e14c536b --- /dev/null +++ b/packages/message-input/src/vaadin-message-input-mixin.js @@ -0,0 +1,180 @@ +/** + * @license + * Copyright (c) 2021 - 2024 Vaadin Ltd. + * This program is available under Apache License Version 2.0, available at https://vaadin.com/license/ + */ +import { ControllerMixin } from '@vaadin/component-base/src/controller-mixin.js'; +import { SlotController } from '@vaadin/component-base/src/slot-controller.js'; +import { TooltipController } from '@vaadin/component-base/src/tooltip-controller.js'; + +/** + * @polymerMixin + * @mixes ControllerMixin + */ +export const MessageInputMixin = (superClass) => + class MessageInputMixinClass extends ControllerMixin(superClass) { + static get properties() { + return { + /** + * Current content of the text input field + */ + value: { + type: String, + value: '', + }, + + /** + * The object used to localize this component. + * For changing the default localization, change the entire + * `i18n` object. + * + * The object has the following JSON structure and default values: + * + * ``` + * { + * // Used as the button label + * send: 'Send', + * + * // Used as the input field's placeholder and aria-label + * message: 'Message' + * } + * ``` + * + * @type {!MessageInputI18n} + * @default {English} + */ + i18n: { + type: Object, + value: () => ({ + send: 'Send', + message: 'Message', + }), + }, + + /** + * Set to true to disable this element. + * @type {boolean} + */ + disabled: { + type: Boolean, + value: false, + reflectToAttribute: true, + }, + + /** @private */ + _button: { + type: Object, + }, + + /** @private */ + _textArea: { + type: Object, + }, + }; + } + + static get observers() { + return [ + '__buttonPropsChanged(_button, disabled, i18n)', + '__textAreaPropsChanged(_textArea, disabled, i18n, value)', + ]; + } + + /** @protected */ + ready() { + super.ready(); + + this._buttonController = new SlotController(this, 'button', 'vaadin-button', { + initializer: (btn) => { + btn.setAttribute('theme', 'primary contained'); + + btn.addEventListener('click', () => { + this.__submit(); + }); + + this._button = btn; + }, + }); + this.addController(this._buttonController); + + this._textAreaController = new SlotController(this, 'textarea', 'vaadin-text-area', { + initializer: (textarea) => { + textarea.addEventListener('value-changed', (event) => { + this.value = event.detail.value; + }); + + textarea.addEventListener('keydown', (event) => { + if (event.key === 'Enter' && !event.shiftKey) { + event.preventDefault(); + event.stopImmediatePropagation(); + this.__submit(); + } + }); + + const input = textarea.inputElement; + input.removeAttribute('aria-labelledby'); + + // Set initial height to one row + input.setAttribute('rows', 1); + input.style.minHeight = '0'; + + this._textArea = textarea; + }, + }); + this.addController(this._textAreaController); + + this._tooltipController = new TooltipController(this); + this.addController(this._tooltipController); + } + + focus() { + if (this._textArea) { + this._textArea.focus(); + } + } + + /** @private */ + __buttonPropsChanged(button, disabled, i18n) { + if (button) { + button.disabled = disabled; + button.textContent = i18n.send; + } + } + + /** @private */ + __textAreaPropsChanged(textArea, disabled, i18n, value) { + if (textArea) { + textArea.disabled = disabled; + textArea.value = value; + + const message = i18n.message; + textArea.placeholder = message; + + if (message) { + textArea.inputElement.setAttribute('aria-label', message); + } else { + textArea.inputElement.removeAttribute('aria-label'); + } + } + } + + /** + * Submits the current value as an custom event named 'submit'. + * It also clears the text input and refocuses it for sending another message. + * In UI, can be triggered by pressing the submit button or pressing enter key when field is focused. + * It does not submit anything if text is empty. + */ + __submit() { + if (this.value !== '') { + this.dispatchEvent(new CustomEvent('submit', { detail: { value: this.value } })); + this.value = ''; + } + this._textArea.focus(); + } + + /** + * Fired when a new message is submitted with ``, either + * by clicking the "send" button, or pressing the Enter key. + * @event submit + */ + }; diff --git a/packages/message-input/src/vaadin-message-input.d.ts b/packages/message-input/src/vaadin-message-input.d.ts index bfe46ac087..63da5c7a64 100644 --- a/packages/message-input/src/vaadin-message-input.d.ts +++ b/packages/message-input/src/vaadin-message-input.d.ts @@ -3,14 +3,11 @@ * Copyright (c) 2021 - 2024 Vaadin Ltd. * This program is available under Apache License Version 2.0, available at https://vaadin.com/license/ */ -import { ControllerMixin } from '@vaadin/component-base/src/controller-mixin.js'; import { ElementMixin } from '@vaadin/component-base/src/element-mixin.js'; import { ThemableMixin } from '@vaadin/vaadin-themable-mixin/vaadin-themable-mixin.js'; +import { MessageInputMixin } from './vaadin-message-input-mixin.js'; -export interface MessageInputI18n { - send: string; - message: string; -} +export { MessageInputI18n } from './vaadin-message-input-mixin.js'; /** * Fired when a new message is submitted with ``, either @@ -36,36 +33,7 @@ export type MessageInputEventMap = HTMLElementEventMap & MessageInputCustomEvent * * ``` */ -declare class MessageInput extends ThemableMixin(ElementMixin(ControllerMixin(HTMLElement))) { - /** - * Current content of the text input field - */ - value: string | null | undefined; - - /** - * The object used to localize this component. - * For changing the default localization, change the entire - * `i18n` object. - * - * The object has the following JSON structure and default values: - * - * ``` - * { - * // Used as the button label - * send: 'Send', - * - * // Used as the input field's placeholder and aria-label - * message: 'Message' - * } - * ``` - */ - i18n: MessageInputI18n; - - /** - * Set to true to disable this element. - */ - disabled: boolean; - +declare class MessageInput extends MessageInputMixin(ThemableMixin(ElementMixin(HTMLElement))) { addEventListener( type: K, listener: (this: MessageInput, ev: MessageInputEventMap[K]) => void, diff --git a/packages/message-input/src/vaadin-message-input.js b/packages/message-input/src/vaadin-message-input.js index a6e934bc5a..3396f8327b 100644 --- a/packages/message-input/src/vaadin-message-input.js +++ b/packages/message-input/src/vaadin-message-input.js @@ -6,12 +6,10 @@ import '@vaadin/button/src/vaadin-button.js'; import '@vaadin/text-area/src/vaadin-text-area.js'; import { html, PolymerElement } from '@polymer/polymer/polymer-element.js'; -import { ControllerMixin } from '@vaadin/component-base/src/controller-mixin.js'; import { defineCustomElement } from '@vaadin/component-base/src/define.js'; import { ElementMixin } from '@vaadin/component-base/src/element-mixin.js'; -import { SlotController } from '@vaadin/component-base/src/slot-controller.js'; -import { TooltipController } from '@vaadin/component-base/src/tooltip-controller.js'; import { ThemableMixin } from '@vaadin/vaadin-themable-mixin/vaadin-themable-mixin.js'; +import { MessageInputMixin } from './vaadin-message-input-mixin.js'; /** * `` is a Web Component for sending messages. @@ -27,71 +25,11 @@ import { ThemableMixin } from '@vaadin/vaadin-themable-mixin/vaadin-themable-mix * * @customElement * @extends HTMLElement - * @mixes ControllerMixin + * @mixes MessageInputMixin * @mixes ThemableMixin * @mixes ElementMixin */ -class MessageInput extends ElementMixin(ThemableMixin(ControllerMixin(PolymerElement))) { - static get properties() { - return { - /** - * Current content of the text input field - */ - value: { - type: String, - value: '', - }, - - /** - * The object used to localize this component. - * For changing the default localization, change the entire - * `i18n` object. - * - * The object has the following JSON structure and default values: - * - * ``` - * { - * // Used as the button label - * send: 'Send', - * - * // Used as the input field's placeholder and aria-label - * message: 'Message' - * } - * ``` - * - * @type {!MessageInputI18n} - * @default {English} - */ - i18n: { - type: Object, - value: () => ({ - send: 'Send', - message: 'Message', - }), - }, - - /** - * Set to true to disable this element. - * @type {boolean} - */ - disabled: { - type: Boolean, - value: false, - reflectToAttribute: true, - }, - - /** @private */ - _button: { - type: Object, - }, - - /** @private */ - _textArea: { - type: Object, - }, - }; - } - +class MessageInput extends MessageInputMixin(ElementMixin(ThemableMixin(PolymerElement))) { static get template() { return html`