File size: 3,117 Bytes
bc20498
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
import { useEscapeKeydown } from '../../internal/actions/index.js';
import { addEventListener, addMeltEventListener, makeElement, createElHelpers, effect, executeCallbacks, isContentEditable, isHTMLInputElement, kbd, noop, omit, } from '../../internal/helpers/index.js';
import { writable } from 'svelte/store';
import { createListbox } from '../listbox/create.js';
// prettier-ignore
export const INTERACTION_KEYS = [kbd.ARROW_LEFT, kbd.ESCAPE, kbd.ARROW_RIGHT, kbd.SHIFT, kbd.CAPS_LOCK, kbd.CONTROL, kbd.ALT, kbd.META, kbd.ENTER, kbd.F1, kbd.F2, kbd.F3, kbd.F4, kbd.F5, kbd.F6, kbd.F7, kbd.F8, kbd.F9, kbd.F10, kbd.F11, kbd.F12];
const { name } = createElHelpers('combobox');
/**
 * Creates an ARIA-1.2-compliant combobox.
 *
 * @TODO multi-select using `tags-input` builder?
 */
export function createCombobox(props) {
    const listbox = createListbox({ ...props, builder: 'combobox', typeahead: false });
    const inputValue = writable('');
    const touchedInput = writable(false);
    /* -------- */
    /* ELEMENTS */
    /* -------- */
    /** Action and attributes for the text input. */
    const input = makeElement(name('input'), {
        stores: [listbox.elements.trigger, inputValue],
        returned: ([$trigger, $inputValue]) => {
            return {
                ...omit($trigger, 'action'),
                role: 'combobox',
                value: $inputValue,
                autocomplete: 'off',
            };
        },
        action: (node) => {
            const unsubscribe = executeCallbacks(addMeltEventListener(node, 'input', (e) => {
                if (!isHTMLInputElement(e.target) && !isContentEditable(e.target))
                    return;
                touchedInput.set(true);
            }), 
            // This shouldn't be cancelled ever, so we don't use addMeltEventListener.
            addEventListener(node, 'input', (e) => {
                if (isHTMLInputElement(e.target)) {
                    inputValue.set(e.target.value);
                }
                if (isContentEditable(e.target)) {
                    inputValue.set(e.target.innerText);
                }
            }));
            let unsubEscapeKeydown = noop;
            const escape = useEscapeKeydown(node, {
                handler: () => {
                    listbox.helpers.closeMenu();
                },
            });
            if (escape && escape.destroy) {
                unsubEscapeKeydown = escape.destroy;
            }
            const { destroy } = listbox.elements.trigger(node);
            return {
                destroy() {
                    destroy?.();
                    unsubscribe();
                    unsubEscapeKeydown();
                },
            };
        },
    });
    effect(listbox.states.open, ($open) => {
        if (!$open) {
            touchedInput.set(false);
        }
    });
    return {
        ...listbox,
        elements: {
            ...omit(listbox.elements, 'trigger'),
            input,
        },
        states: {
            ...listbox.states,
            touchedInput,
            inputValue,
        },
    };
}