// Copyright: Ankitects Pty Ltd and contributors // License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html import { getRange, getSelection } from "../lib/cross-browser"; import { on } from "../lib/events"; import { keyboardEventIsPrintableKey } from "../lib/keys"; import { HandlerList } from "./handler-list"; const nbsp = "\xa0"; export type SetupInputHandlerAction = (element: HTMLElement) => { destroy(): void }; export interface InputEventParams { event: InputEvent; } export interface InsertTextParams { event: InputEvent; text: Text; } export interface InputHandlerAPI { readonly beforeInput: HandlerList; readonly insertText: HandlerList; } /** * An interface that allows Svelte components to attach event listeners via triggers. * They will be attached to the component(s) that install the manager. * Prevents that too many event listeners are attached and allows for some * coordination between them. */ function useInputHandler(): [InputHandlerAPI, SetupInputHandlerAction] { const beforeInput = new HandlerList(); const insertText = new HandlerList(); async function onBeforeInput(this: Element, event: InputEvent): Promise { const selection = getSelection(this)!; const range = getRange(selection); await beforeInput.dispatch({ event }); if (!range || event.inputType !== "insertText" || insertText.length === 0) { return; } event.preventDefault(); const content = !event.data || event.data === " " ? nbsp : event.data; const text = new Text(content); range.deleteContents(); range.insertNode(text); range.selectNode(text); range.collapse(false); await insertText.dispatch({ event, text }); range.commonAncestorContainer.normalize(); } function clearInsertText(): void { insertText.clear(); } function onKeydown(event: KeyboardEvent): void { /* using arrow keys should cancel */ if (!keyboardEventIsPrintableKey(event)) { clearInsertText(); } } function onInput(this: HTMLElement, event: InputEvent): void { // prevent unwanted
from being left behind when clearing field contents if ( !event.data && this.children.length === 1 && this.children.item(0) instanceof HTMLDivElement && /^\n?$/.test(this.innerText) ) { this.innerHTML = ""; } } function setupHandler(element: HTMLElement): { destroy(): void } { const beforeInputOff = on(element, "beforeinput", onBeforeInput); const inputOff = on(element, "input" as "beforeinput", onInput); const blurOff = on(element, "blur", clearInsertText); const pointerDownOff = on(element, "pointerdown", clearInsertText); const keyDownOff = on(element, "keydown", onKeydown); return { destroy() { beforeInputOff(); inputOff(); blurOff(); pointerDownOff(); keyDownOff(); }, }; } return [ { beforeInput, insertText, }, setupHandler, ]; } export default useInputHandler;