mirror of
https://github.com/ArthurDanjou/ui.git
synced 2026-01-28 02:40:35 +01:00
feat(CommandPalette): add custom slots
This commit is contained in:
@@ -22,10 +22,6 @@ const { data: users, status } = await useFetch('https://jsonplaceholder.typicode
|
|||||||
const loading = ref(false)
|
const loading = ref(false)
|
||||||
|
|
||||||
const groups = computed(() => [{
|
const groups = computed(() => [{
|
||||||
id: 'users',
|
|
||||||
label: searchTerm.value ? `Users matching “${searchTerm.value}”...` : 'Users',
|
|
||||||
items: users.value || []
|
|
||||||
}, {
|
|
||||||
id: 'actions',
|
id: 'actions',
|
||||||
items: [{
|
items: [{
|
||||||
label: 'Add new file',
|
label: 'Add new file',
|
||||||
@@ -74,6 +70,12 @@ const groups = computed(() => [{
|
|||||||
toast.add({ title: 'Label added!' })
|
toast.add({ title: 'Label added!' })
|
||||||
},
|
},
|
||||||
kbds: ['meta', 'L']
|
kbds: ['meta', 'L']
|
||||||
|
}, {
|
||||||
|
label: 'Set Wallpaper',
|
||||||
|
suffix: 'Choose from beautiful wallpaper collection.',
|
||||||
|
icon: 'i-lucide-image',
|
||||||
|
interface: 'wallpaper',
|
||||||
|
placeholder: 'Search wallpapers...'
|
||||||
}, {
|
}, {
|
||||||
label: 'More actions',
|
label: 'More actions',
|
||||||
placeholder: 'Search actions...',
|
placeholder: 'Search actions...',
|
||||||
@@ -140,6 +142,116 @@ const labels = [{
|
|||||||
}]
|
}]
|
||||||
const label = ref()
|
const label = ref()
|
||||||
|
|
||||||
|
const wallpapers = [
|
||||||
|
{
|
||||||
|
id: 1,
|
||||||
|
name: 'red_distortion_1',
|
||||||
|
gradient: 'from-red-500 via-orange-500 to-pink-500',
|
||||||
|
category: 'Abstract',
|
||||||
|
featured: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 2,
|
||||||
|
name: 'blue_distortion_1',
|
||||||
|
gradient: 'from-blue-600 via-purple-600 to-indigo-600',
|
||||||
|
category: 'Abstract',
|
||||||
|
featured: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 3,
|
||||||
|
name: 'mono_dark_distortion_1',
|
||||||
|
gradient: 'from-gray-900 via-gray-700 to-gray-800',
|
||||||
|
category: 'Monochrome',
|
||||||
|
featured: false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 4,
|
||||||
|
name: 'chromatic_dark_1',
|
||||||
|
gradient: 'from-emerald-600 via-teal-600 to-cyan-600',
|
||||||
|
category: 'Chromatic',
|
||||||
|
featured: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 5,
|
||||||
|
name: 'red_distortion_2',
|
||||||
|
gradient: 'from-rose-600 via-red-600 to-orange-600',
|
||||||
|
category: 'Abstract',
|
||||||
|
featured: false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 6,
|
||||||
|
name: 'purple_cosmic_1',
|
||||||
|
gradient: 'from-violet-700 via-purple-700 to-fuchsia-700',
|
||||||
|
category: 'Cosmic',
|
||||||
|
featured: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 7,
|
||||||
|
name: 'golden_sunset_1',
|
||||||
|
gradient: 'from-yellow-500 via-orange-500 to-red-500',
|
||||||
|
category: 'Nature',
|
||||||
|
featured: false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 8,
|
||||||
|
name: 'ocean_deep_1',
|
||||||
|
gradient: 'from-blue-800 via-blue-900 to-indigo-900',
|
||||||
|
category: 'Nature',
|
||||||
|
featured: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 9,
|
||||||
|
name: 'mono_light_distortion_1',
|
||||||
|
gradient: 'from-gray-200 via-gray-300 to-gray-400',
|
||||||
|
category: 'Monochrome',
|
||||||
|
featured: false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 10,
|
||||||
|
name: 'green_matrix_1',
|
||||||
|
gradient: 'from-green-800 via-emerald-700 to-teal-700',
|
||||||
|
category: 'Chromatic',
|
||||||
|
featured: false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 11,
|
||||||
|
name: 'pink_dreams_1',
|
||||||
|
gradient: 'from-pink-500 via-rose-500 to-purple-500',
|
||||||
|
category: 'Abstract',
|
||||||
|
featured: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 12,
|
||||||
|
name: 'midnight_blue_1',
|
||||||
|
gradient: 'from-slate-900 via-blue-900 to-indigo-900',
|
||||||
|
category: 'Nature',
|
||||||
|
featured: false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
const filteredWallpapers = computed(() => {
|
||||||
|
let filtered = wallpapers
|
||||||
|
|
||||||
|
// Filter by search term
|
||||||
|
if (searchTerm.value.trim()) {
|
||||||
|
const search = searchTerm.value.toLowerCase()
|
||||||
|
filtered = filtered.filter(w =>
|
||||||
|
w.name.toLowerCase().includes(search)
|
||||||
|
|| w.category.toLowerCase().includes(search)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return filtered
|
||||||
|
})
|
||||||
|
|
||||||
|
function setWallpaper(wallpaper: any) {
|
||||||
|
toast.add({
|
||||||
|
title: `Wallpaper set to ${wallpaper.name}!`,
|
||||||
|
description: 'Your desktop wallpaper has been updated.',
|
||||||
|
icon: 'i-lucide-image'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
// function onSelect(item: typeof groups.value[number]['items'][number]) {
|
// function onSelect(item: typeof groups.value[number]['items'][number]) {
|
||||||
function onSelect(item: any) {
|
function onSelect(item: any) {
|
||||||
console.log('Selected', item)
|
console.log('Selected', item)
|
||||||
@@ -166,7 +278,44 @@ defineShortcuts({
|
|||||||
multiple
|
multiple
|
||||||
class="sm:max-h-80"
|
class="sm:max-h-80"
|
||||||
@update:model-value="onSelect"
|
@update:model-value="onSelect"
|
||||||
/>
|
>
|
||||||
|
<template #interface="{ interfaceName }">
|
||||||
|
<div v-if="interfaceName === 'wallpaper'" class="flex flex-col h-full w-full">
|
||||||
|
<div class="flex-1 overflow-y-auto p-6">
|
||||||
|
<div class="grid grid-cols-4 gap-4">
|
||||||
|
<div
|
||||||
|
v-for="wallpaper in filteredWallpapers"
|
||||||
|
:key="wallpaper.id"
|
||||||
|
class="group relative cursor-pointer"
|
||||||
|
@click="setWallpaper(wallpaper)"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="aspect-video rounded-lg bg-gradient-to-br shadow-lg ring-1 ring-black/5"
|
||||||
|
:class="wallpaper.gradient"
|
||||||
|
/>
|
||||||
|
<div class="mt-2 px-1">
|
||||||
|
<div class="flex items-center gap-2">
|
||||||
|
<h3 class="text-sm font-medium text-highlighted truncate">
|
||||||
|
{{ wallpaper.name }}
|
||||||
|
</h3>
|
||||||
|
<UChip
|
||||||
|
v-if="wallpaper.featured"
|
||||||
|
label="★"
|
||||||
|
size="xs"
|
||||||
|
color="primary"
|
||||||
|
class="shrink-0"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<p class="text-xs text-dimmed">
|
||||||
|
{{ wallpaper.category }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</UCommandPalette>
|
||||||
</DefineTemplate>
|
</DefineTemplate>
|
||||||
|
|
||||||
<div class="flex-1 flex flex-col gap-12 w-full max-w-lg">
|
<div class="flex-1 flex flex-col gap-12 w-full max-w-lg">
|
||||||
|
|||||||
@@ -31,6 +31,11 @@ export interface CommandPaletteItem extends Omit<LinkProps, 'type' | 'raw' | 'cu
|
|||||||
*/
|
*/
|
||||||
placeholder?: string
|
placeholder?: string
|
||||||
children?: CommandPaletteItem[]
|
children?: CommandPaletteItem[]
|
||||||
|
/**
|
||||||
|
* Custom interface slot to display instead of children items.
|
||||||
|
* When defined, clicking this item will show the custom interface.
|
||||||
|
*/
|
||||||
|
interface?: string
|
||||||
onSelect?(e?: Event): void
|
onSelect?(e?: Event): void
|
||||||
class?: any
|
class?: any
|
||||||
ui?: Pick<CommandPalette['slots'], 'item' | 'itemLeadingIcon' | 'itemLeadingAvatarSize' | 'itemLeadingAvatar' | 'itemLeadingChipSize' | 'itemLeadingChip' | 'itemLabel' | 'itemLabelPrefix' | 'itemLabelBase' | 'itemLabelSuffix' | 'itemTrailing' | 'itemTrailingKbds' | 'itemTrailingKbdsSize' | 'itemTrailingHighlightedIcon' | 'itemTrailingIcon'>
|
ui?: Pick<CommandPalette['slots'], 'item' | 'itemLeadingIcon' | 'itemLeadingAvatarSize' | 'itemLeadingAvatar' | 'itemLeadingChipSize' | 'itemLeadingChip' | 'itemLabel' | 'itemLabelPrefix' | 'itemLabelBase' | 'itemLabelSuffix' | 'itemTrailing' | 'itemTrailingKbds' | 'itemTrailingKbdsSize' | 'itemTrailingHighlightedIcon' | 'itemTrailingIcon'>
|
||||||
@@ -149,6 +154,7 @@ export type CommandPaletteSlots<G extends CommandPaletteGroup<T> = CommandPalett
|
|||||||
'empty'(props: { searchTerm?: string }): any
|
'empty'(props: { searchTerm?: string }): any
|
||||||
'back'(props: { ui: { [K in keyof Required<CommandPalette['slots']>]: (props?: Record<string, any>) => string } }): any
|
'back'(props: { ui: { [K in keyof Required<CommandPalette['slots']>]: (props?: Record<string, any>) => string } }): any
|
||||||
'close'(props: { ui: { [K in keyof Required<CommandPalette['slots']>]: (props?: Record<string, any>) => string } }): any
|
'close'(props: { ui: { [K in keyof Required<CommandPalette['slots']>]: (props?: Record<string, any>) => string } }): any
|
||||||
|
'interface'(props: { interfaceName?: string, current: any, searchTerm: string, navigateBack: () => void }): any
|
||||||
'item': SlotProps<T>
|
'item': SlotProps<T>
|
||||||
'item-leading': SlotProps<T>
|
'item-leading': SlotProps<T>
|
||||||
'item-label': SlotProps<T>
|
'item-label': SlotProps<T>
|
||||||
@@ -208,12 +214,17 @@ const fuse = computed(() => defu({}, props.fuse, {
|
|||||||
matchAllWhenSearchEmpty: true
|
matchAllWhenSearchEmpty: true
|
||||||
}))
|
}))
|
||||||
|
|
||||||
const history = ref<(CommandPaletteGroup & { placeholder?: string })[]>([])
|
const history = ref<(CommandPaletteGroup & { placeholder?: string, interface?: string })[]>([])
|
||||||
|
|
||||||
const placeholder = computed(() => history.value[history.value.length - 1]?.placeholder || props.placeholder || t('commandPalette.placeholder'))
|
const placeholder = computed(() => history.value[history.value.length - 1]?.placeholder || props.placeholder || t('commandPalette.placeholder'))
|
||||||
|
|
||||||
const groups = computed(() => history.value?.length ? [history.value[history.value.length - 1] as G] : props.groups)
|
const groups = computed(() => history.value?.length ? [history.value[history.value.length - 1] as G] : props.groups)
|
||||||
|
|
||||||
|
const _currentInterface = computed(() => {
|
||||||
|
const current = history.value[history.value.length - 1]
|
||||||
|
return current?.interface ? current : null
|
||||||
|
})
|
||||||
|
|
||||||
const items = computed(() => groups.value?.filter((group) => {
|
const items = computed(() => groups.value?.filter((group) => {
|
||||||
if (!group.id) {
|
if (!group.id) {
|
||||||
console.warn(`[@nuxt/ui] CommandPalette group is missing an \`id\` property`)
|
console.warn(`[@nuxt/ui] CommandPalette group is missing an \`id\` property`)
|
||||||
@@ -280,7 +291,7 @@ const filteredGroups = computed(() => {
|
|||||||
const listboxRootRef = useTemplateRef('listboxRootRef')
|
const listboxRootRef = useTemplateRef('listboxRootRef')
|
||||||
|
|
||||||
function navigate(item: T) {
|
function navigate(item: T) {
|
||||||
if (!item.children?.length) {
|
if (!item.children?.length && !item.interface) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -289,7 +300,8 @@ function navigate(item: T) {
|
|||||||
label: item.label,
|
label: item.label,
|
||||||
slot: item.slot,
|
slot: item.slot,
|
||||||
placeholder: item.placeholder,
|
placeholder: item.placeholder,
|
||||||
items: item.children
|
interface: item.interface,
|
||||||
|
items: item.children || []
|
||||||
} as any)
|
} as any)
|
||||||
|
|
||||||
searchTerm.value = ''
|
searchTerm.value = ''
|
||||||
@@ -316,7 +328,7 @@ function onBackspace() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function onSelect(e: Event, item: T) {
|
function onSelect(e: Event, item: T) {
|
||||||
if (item.children?.length) {
|
if (item.children?.length || item.interface) {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
|
|
||||||
navigate(item)
|
navigate(item)
|
||||||
@@ -371,7 +383,17 @@ function onSelect(e: Event, item: T) {
|
|||||||
</ListboxFilter>
|
</ListboxFilter>
|
||||||
|
|
||||||
<ListboxContent :class="ui.content({ class: props.ui?.content })">
|
<ListboxContent :class="ui.content({ class: props.ui?.content })">
|
||||||
<div v-if="filteredGroups?.length" role="presentation" :class="ui.viewport({ class: props.ui?.viewport })">
|
<div v-if="_currentInterface" :class="ui.viewport({ class: props.ui?.viewport })">
|
||||||
|
<slot
|
||||||
|
name="interface"
|
||||||
|
:interface-name="_currentInterface.interface"
|
||||||
|
:current="_currentInterface"
|
||||||
|
:search-term="searchTerm"
|
||||||
|
:navigate-back="navigateBack"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else-if="filteredGroups?.length" role="presentation" :class="ui.viewport({ class: props.ui?.viewport })">
|
||||||
<ListboxGroup v-for="group in filteredGroups" :key="`group-${group.id}`" :class="ui.group({ class: props.ui?.group })">
|
<ListboxGroup v-for="group in filteredGroups" :key="`group-${group.id}`" :class="ui.group({ class: props.ui?.group })">
|
||||||
<ListboxGroupLabel v-if="get(group, props.labelKey as string)" :class="ui.label({ class: props.ui?.label })">
|
<ListboxGroupLabel v-if="get(group, props.labelKey as string)" :class="ui.label({ class: props.ui?.label })">
|
||||||
{{ get(group, props.labelKey as string) }}
|
{{ get(group, props.labelKey as string) }}
|
||||||
@@ -415,7 +437,7 @@ function onSelect(e: Event, item: T) {
|
|||||||
<span :class="ui.itemTrailing({ class: [props.ui?.itemTrailing, item.ui?.itemTrailing] })">
|
<span :class="ui.itemTrailing({ class: [props.ui?.itemTrailing, item.ui?.itemTrailing] })">
|
||||||
<slot :name="((item.slot ? `${item.slot}-trailing` : group.slot ? `${group.slot}-trailing` : `item-trailing`) as keyof CommandPaletteSlots<G, T>)" :item="(item as any)" :index="index">
|
<slot :name="((item.slot ? `${item.slot}-trailing` : group.slot ? `${group.slot}-trailing` : `item-trailing`) as keyof CommandPaletteSlots<G, T>)" :item="(item as any)" :index="index">
|
||||||
<UIcon
|
<UIcon
|
||||||
v-if="item.children && item.children.length > 0"
|
v-if="(item.children && item.children.length > 0) || item.interface"
|
||||||
:name="trailingIcon || appConfig.ui.icons.chevronRight"
|
:name="trailingIcon || appConfig.ui.icons.chevronRight"
|
||||||
:class="ui.itemTrailingIcon({ class: [props.ui?.itemTrailingIcon, item.ui?.itemTrailingIcon] })"
|
:class="ui.itemTrailingIcon({ class: [props.ui?.itemTrailingIcon, item.ui?.itemTrailingIcon] })"
|
||||||
/>
|
/>
|
||||||
|
|||||||
Reference in New Issue
Block a user