feat(primitives): media-editor components, category reorg, perf + type cleanup
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.
This commit is contained in:
@@ -0,0 +1,79 @@
|
||||
<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>
|
||||
Reference in New Issue
Block a user