eefd7abf83
Reorganize components into category folders (forms/canvas/overlays/etc.); add the media-editor headless family (timeline, curve-editor, waveform, crop, color picker, etc.); apply perf fixes (O(1) collection lookups, plain-object drag state, gesture-leak teardown, shallowRef color state, rect caching) and replace source `any` with proper types.
80 lines
2.4 KiB
Vue
80 lines
2.4 KiB
Vue
<script lang="ts">
|
|
import type { PrimitiveProps } from '../../internal/primitive';
|
|
|
|
/**
|
|
* The scrollable list container (`role="listbox"`) that wraps the items. It
|
|
* receives focus, owns the keyboard handlers (navigation, enter, type-ahead),
|
|
* and exposes the collection of items to the root.
|
|
*/
|
|
export interface ListboxContentProps extends PrimitiveProps {}
|
|
</script>
|
|
|
|
<script setup lang="ts">
|
|
import { Primitive } from '../../internal/primitive';
|
|
import { ref } from 'vue';
|
|
import { useCollectionInjector } from '../../utilities/collection';
|
|
import { useForwardExpose } from '@robonen/vue';
|
|
import { useListboxRootContext } from './context';
|
|
|
|
// Module-scoped to avoid re-allocating on every keydown.
|
|
const NAV_KEYS = new Set(['ArrowUp', 'ArrowDown', 'ArrowLeft', 'ArrowRight', 'Home', 'End', 'PageUp', 'PageDown']);
|
|
|
|
const { as = 'div' } = defineProps<ListboxContentProps>();
|
|
|
|
const ctx = useListboxRootContext();
|
|
const { forwardRef } = useForwardExpose();
|
|
const { CollectionSlot } = useCollectionInjector();
|
|
|
|
const isClickFocus = ref(false);
|
|
|
|
function onMouseDown(event: MouseEvent): void {
|
|
if (event.button !== 0) return;
|
|
isClickFocus.value = true;
|
|
setTimeout(() => {
|
|
isClickFocus.value = false;
|
|
}, 10);
|
|
}
|
|
|
|
function onFocus(event: FocusEvent): void {
|
|
if (isClickFocus.value) return;
|
|
ctx.onEnter(event);
|
|
}
|
|
|
|
function onKeyDown(event: KeyboardEvent): void {
|
|
const { key } = event;
|
|
const o = ctx.orientation.value;
|
|
if ((o === 'vertical' && (key === 'ArrowLeft' || key === 'ArrowRight'))
|
|
|| (o === 'horizontal' && (key === 'ArrowUp' || key === 'ArrowDown'))) return;
|
|
|
|
if (key === 'Enter') return ctx.onKeydownEnter(event);
|
|
|
|
if (NAV_KEYS.has(key)) {
|
|
event.preventDefault();
|
|
if (ctx.focusable.value) ctx.onKeydownNavigation(event);
|
|
return;
|
|
}
|
|
ctx.onKeydownTypeAhead(event);
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<CollectionSlot>
|
|
<Primitive
|
|
:ref="forwardRef"
|
|
:as="as"
|
|
role="listbox"
|
|
:tabindex="ctx.focusable.value ? (ctx.highlightedElement.value ? '-1' : '0') : '-1'"
|
|
:aria-orientation="ctx.orientation.value"
|
|
:aria-multiselectable="ctx.multiple.value ? true : undefined"
|
|
:data-orientation="ctx.orientation.value"
|
|
@mousedown="onMouseDown"
|
|
@focus="onFocus"
|
|
@keydown="onKeyDown"
|
|
@compositionstart="ctx.onCompositionStart"
|
|
@compositionend="ctx.onCompositionEnd"
|
|
>
|
|
<slot />
|
|
</Primitive>
|
|
</CollectionSlot>
|
|
</template>
|