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:
2026-06-15 16:54:29 +07:00
parent 661a55719e
commit eefd7abf83
1029 changed files with 65815 additions and 9449 deletions
@@ -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>