Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions cli/templates.mjs
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ const ui = computed(() => tv({ extend: tv(theme), ...(appConfig.ui?.${camelName}
</script>
<template>
<Primitive :as="as" :class="ui.root({ class: [props.ui?.root, props.class] })">
<Primitive :as="as" data-slot="root" :class="ui.root({ class: [props.ui?.root, props.class] })">
<slot />
</Primitive>
</template>
Expand Down Expand Up @@ -105,7 +105,7 @@ const ui = computed(() => tv({ extend: tv(theme), ...(appConfig.ui?.${camelName}
</script>
<template>
<${upperName}Root v-bind="rootProps" :class="ui.root({ class: [props.ui?.root, props.class] })" />
<${upperName}Root v-bind="rootProps" data-slot="root" :class="ui.root({ class: [props.ui?.root, props.class] })" />
</template>
`
}
Expand Down
17 changes: 9 additions & 8 deletions src/runtime/components/Accordion.vue
Original file line number Diff line number Diff line change
Expand Up @@ -93,34 +93,35 @@ const ui = computed(() => tv({ extend: tv(theme), ...(appConfig.ui?.accordion ||
</script>

<template>
<AccordionRoot v-bind="rootProps" :type="type" :class="ui.root({ class: [props.ui?.root, props.class] })">
<AccordionRoot v-bind="rootProps" :type="type" data-slot="root" :class="ui.root({ class: [props.ui?.root, props.class] })">
<AccordionItem
v-for="(item, index) in props.items"
v-slot="{ open }"
:key="index"
:value="item.value || String(index)"
:disabled="item.disabled"
data-slot="item"
:class="ui.item({ class: [props.ui?.item, item.ui?.item, item.class] })"
>
<AccordionHeader as="div" :class="ui.header({ class: [props.ui?.header, item.ui?.header] })">
<AccordionTrigger :class="ui.trigger({ class: [props.ui?.trigger, item.ui?.trigger], disabled: item.disabled })">
<AccordionHeader as="div" data-slot="header" :class="ui.header({ class: [props.ui?.header, item.ui?.header] })">
<AccordionTrigger data-slot="trigger" :class="ui.trigger({ class: [props.ui?.trigger, item.ui?.trigger], disabled: item.disabled })">
<slot name="leading" :item="item" :index="index" :open="open" :ui="ui">
<UIcon v-if="item.icon" :name="item.icon" :class="ui.leadingIcon({ class: [props.ui?.leadingIcon, item?.ui?.leadingIcon] })" />
<UIcon v-if="item.icon" :name="item.icon" data-slot="leadingIcon" :class="ui.leadingIcon({ class: [props.ui?.leadingIcon, item?.ui?.leadingIcon] })" />
</slot>

<span v-if="get(item, props.labelKey as string) || !!slots.default" :class="ui.label({ class: [props.ui?.label, item.ui?.label] })">
<span v-if="get(item, props.labelKey as string) || !!slots.default" data-slot="label" :class="ui.label({ class: [props.ui?.label, item.ui?.label] })">
<slot :item="item" :index="index" :open="open">{{ get(item, props.labelKey as string) }}</slot>
</span>

<slot name="trailing" :item="item" :index="index" :open="open" :ui="ui">
<UIcon :name="item.trailingIcon || trailingIcon || appConfig.ui.icons.chevronDown" :class="ui.trailingIcon({ class: [props.ui?.trailingIcon, item.ui?.trailingIcon] })" />
<UIcon :name="item.trailingIcon || trailingIcon || appConfig.ui.icons.chevronDown" data-slot="trailingIcon" :class="ui.trailingIcon({ class: [props.ui?.trailingIcon, item.ui?.trailingIcon] })" />
</slot>
</AccordionTrigger>
</AccordionHeader>

<AccordionContent v-if="item.content || !!slots.content || (item.slot && !!slots[item.slot as keyof AccordionSlots<T>]) || !!slots.body || (item.slot && !!slots[`${item.slot}-body` as keyof AccordionSlots<T>])" :class="ui.content({ class: [props.ui?.content, item.ui?.content] })">
<AccordionContent v-if="item.content || !!slots.content || (item.slot && !!slots[item.slot as keyof AccordionSlots<T>]) || !!slots.body || (item.slot && !!slots[`${item.slot}-body` as keyof AccordionSlots<T>])" data-slot="content" :class="ui.content({ class: [props.ui?.content, item.ui?.content] })">
<slot :name="((item.slot || 'content') as keyof AccordionSlots<T>)" :item="(item as Extract<T, { slot: string; }>)" :index="index" :open="open" :ui="ui">
<div :class="ui.body({ class: [props.ui?.body, item.ui?.body] })">
<div data-slot="body" :class="ui.body({ class: [props.ui?.body, item.ui?.body] })">
<slot :name="((item.slot ? `${item.slot}-body`: 'body') as keyof AccordionSlots<T>)" :item="(item as Extract<T, { slot: string; }>)" :index="index" :open="open" :ui="ui">
{{ item.content }}
</slot>
Expand Down
17 changes: 9 additions & 8 deletions src/runtime/components/Alert.vue
Original file line number Diff line number Diff line change
Expand Up @@ -97,32 +97,32 @@ const ui = computed(() => tv({ extend: tv(theme), ...(appConfig.ui?.alert || {})
</script>

<template>
<Primitive :as="as" :data-orientation="orientation" :class="ui.root({ class: [props.ui?.root, props.class] })">
<Primitive :as="as" :data-orientation="orientation" data-slot="root" :class="ui.root({ class: [props.ui?.root, props.class] })">
<slot name="leading" :ui="ui">
<UAvatar v-if="avatar" :size="((props.ui?.avatarSize || ui.avatarSize()) as AvatarProps['size'])" v-bind="avatar" :class="ui.avatar({ class: props.ui?.avatar })" />
<UIcon v-else-if="icon" :name="icon" :class="ui.icon({ class: props.ui?.icon })" />
<UAvatar v-if="avatar" :size="((props.ui?.avatarSize || ui.avatarSize()) as AvatarProps['size'])" v-bind="avatar" data-slot="avatar" :class="ui.avatar({ class: props.ui?.avatar })" />
<UIcon v-else-if="icon" :name="icon" data-slot="icon" :class="ui.icon({ class: props.ui?.icon })" />
</slot>

<div :class="ui.wrapper({ class: props.ui?.wrapper })">
<div v-if="title || !!slots.title" :class="ui.title({ class: props.ui?.title })">
<div data-slot="wrapper" :class="ui.wrapper({ class: props.ui?.wrapper })">
<div v-if="title || !!slots.title" data-slot="title" :class="ui.title({ class: props.ui?.title })">
<slot name="title">
{{ title }}
</slot>
</div>
<div v-if="description || !!slots.description" :class="ui.description({ class: props.ui?.description })">
<div v-if="description || !!slots.description" data-slot="description" :class="ui.description({ class: props.ui?.description })">
<slot name="description">
{{ description }}
</slot>
</div>

<div v-if="orientation === 'vertical' && (actions?.length || !!slots.actions)" :class="ui.actions({ class: props.ui?.actions })">
<div v-if="orientation === 'vertical' && (actions?.length || !!slots.actions)" data-slot="actions" :class="ui.actions({ class: props.ui?.actions })">
<slot name="actions">
<UButton v-for="(action, index) in actions" :key="index" size="xs" v-bind="action" />
</slot>
</div>
</div>

<div v-if="(orientation === 'horizontal' && (actions?.length || !!slots.actions)) || close" :class="ui.actions({ class: props.ui?.actions, orientation: 'horizontal' })">
<div v-if="(orientation === 'horizontal' && (actions?.length || !!slots.actions)) || close" data-slot="actions" :class="ui.actions({ class: props.ui?.actions, orientation: 'horizontal' })">
<template v-if="orientation === 'horizontal' && (actions?.length || !!slots.actions)">
<slot name="actions">
<UButton v-for="(action, index) in actions" :key="index" size="xs" v-bind="action" />
Expand All @@ -137,6 +137,7 @@ const ui = computed(() => tv({ extend: tv(theme), ...(appConfig.ui?.alert || {})
variant="link"
:aria-label="t('alert.close')"
v-bind="(typeof close === 'object' ? close as Partial<ButtonProps> : {})"
data-slot="close"
:class="ui.close({ class: props.ui?.close })"
@click="emits('update:open', false)"
/>
Expand Down
25 changes: 16 additions & 9 deletions src/runtime/components/AuthForm.vue
Original file line number Diff line number Diff line change
Expand Up @@ -186,31 +186,31 @@ defineExpose({
</script>

<template>
<Primitive :as="as" :class="ui.root({ class: [props.ui?.root, props.class] })">
<div v-if="(icon || !!slots.leading) || (title || !!slots.title) || (description || !!slots.description) || !!slots.header" :class="ui.header({ class: props.ui?.header })">
<Primitive :as="as" data-slot="root" :class="ui.root({ class: [props.ui?.root, props.class] })">
<div v-if="(icon || !!slots.leading) || (title || !!slots.title) || (description || !!slots.description) || !!slots.header" data-slot="header" :class="ui.header({ class: props.ui?.header })">
<slot name="header">
<div v-if="icon || !!slots.leading" :class="ui.leading({ class: props.ui?.leading })">
<div v-if="icon || !!slots.leading" data-slot="leading" :class="ui.leading({ class: props.ui?.leading })">
<slot name="leading" :ui="ui">
<UIcon v-if="icon" :name="icon" :class="ui.leadingIcon({ class: props.ui?.leadingIcon })" />
<UIcon v-if="icon" :name="icon" data-slot="leadingIcon" :class="ui.leadingIcon({ class: props.ui?.leadingIcon })" />
</slot>
</div>

<div v-if="title || !!slots.title" :class="ui.title({ class: props.ui?.title })">
<div v-if="title || !!slots.title" data-slot="title" :class="ui.title({ class: props.ui?.title })">
<slot name="title">
{{ title }}
</slot>
</div>

<div v-if="description || !!slots.description" :class="ui.description({ class: props.ui?.description })">
<div v-if="description || !!slots.description" data-slot="description" :class="ui.description({ class: props.ui?.description })">
<slot name="description">
{{ description }}
</slot>
</div>
</slot>
</div>

<div :class="ui.body({ class: props.ui?.body })">
<div v-if="providers?.length || !!slots.providers" :class="ui.providers({ class: props.ui?.providers })">
<div data-slot="body" :class="ui.body({ class: props.ui?.body })">
<div v-if="providers?.length || !!slots.providers" data-slot="providers" :class="ui.providers({ class: props.ui?.providers })">
<slot name="providers">
<UButton
v-for="(provider, index) in providers"
Expand All @@ -226,6 +226,7 @@ defineExpose({
<USeparator
v-if="providers?.length && fields?.length"
v-bind="typeof separator === 'object' ? separator : { label: separator }"
data-slot="separator"
:class="ui.separator({ class: props.ui?.separator })"
/>

Expand All @@ -238,6 +239,7 @@ defineExpose({
:validate-on="validateOn"
:disabled="disabled"
:loading-auto="loadingAuto"
data-slot="form"
:class="ui.form({ class: props.ui?.form })"
v-bind="$attrs"
@submit="onSubmit"
Expand All @@ -251,19 +253,22 @@ defineExpose({
<UCheckbox
v-if="field.type === 'checkbox'"
v-model="state[field.name]"
data-slot="checkbox"
:class="ui.checkbox({ class: props.ui?.checkbox })"
v-bind="(omitFieldProps(field))"
/>
<USelectMenu
v-else-if="field.type === 'select'"
v-model="state[field.name]"
data-slot="select"
:class="ui.select({ class: props.ui?.select })"
v-bind="(omitFieldProps(field) as AuthFormSelectField)"
/>
<UPinInput
v-else-if="field.type === 'otp'"
:id="field.name"
v-model="state[field.name]"
data-slot="otp"
:class="ui.otp({ class: props.ui?.otp })"
v-bind="(Object.assign({}, omitFieldProps(field), typeof (field as AuthFormOtpField).otp === 'object' ? (field as AuthFormOtpField).otp : {}) as any)"
otp
Expand All @@ -272,6 +277,7 @@ defineExpose({
v-else-if="field.type === 'password'"
ref="passwordRef"
v-model="state[field.name]"
data-slot="password"
:class="ui.password({ class: props.ui?.password })"
v-bind="(omitFieldProps(field) as AuthFormInputField<'password'>)"
:type="passwordVisibility ? 'text' : 'password'"
Expand All @@ -292,6 +298,7 @@ defineExpose({
<UInput
v-else
v-model="state[field.name]"
data-slot="input"
:class="ui.input({ class: props.ui?.input })"
v-bind="(omitFieldProps(field) as AuthFormInputField)"
/>
Expand Down Expand Up @@ -329,7 +336,7 @@ defineExpose({
</UForm>
</div>

<div v-if="!!slots.footer" :class="ui.footer({ class: props.ui?.footer })">
<div v-if="!!slots.footer" data-slot="footer" :class="ui.footer({ class: props.ui?.footer })">
<slot name="footer" />
</div>
</Primitive>
Expand Down
6 changes: 4 additions & 2 deletions src/runtime/components/Avatar.vue
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,7 @@ function onError() {
:is="props.chip ? UChip : Primitive"
:as="as.root"
v-bind="props.chip ? (typeof props.chip === 'object' ? { inset: true, ...props.chip } : { inset: true }) : {}"
data-slot="root"
:class="ui.root({ class: [props.ui?.root, props.class] })"
:style="props.style"
>
Expand All @@ -109,14 +110,15 @@ function onError() {
:width="sizePx"
:height="sizePx"
v-bind="$attrs"
data-slot="image"
:class="ui.image({ class: props.ui?.image })"
@error="onError"
/>

<Slot v-else v-bind="$attrs">
<slot>
<UIcon v-if="icon" :name="icon" :class="ui.icon({ class: props.ui?.icon })" />
<span v-else :class="ui.fallback({ class: props.ui?.fallback })">{{ fallback || '&nbsp;' }}</span>
<UIcon v-if="icon" :name="icon" data-slot="icon" :class="ui.icon({ class: props.ui?.icon })" />
<span v-else data-slot="fallback" :class="ui.fallback({ class: props.ui?.fallback })">{{ fallback || '&nbsp;' }}</span>
</slot>
</Slot>
</component>
Expand Down
6 changes: 3 additions & 3 deletions src/runtime/components/AvatarGroup.vue
Original file line number Diff line number Diff line change
Expand Up @@ -93,8 +93,8 @@ provide(avatarGroupInjectionKey, computed(() => ({
</script>

<template>
<Primitive :as="as" :class="ui.root({ class: [props.ui?.root, props.class] })">
<UAvatar v-if="hiddenCount > 0" :text="`+${hiddenCount}`" :class="ui.base({ class: props.ui?.base })" />
<component :is="avatar" v-for="(avatar, count) in visibleAvatars" :key="count" :class="ui.base({ class: props.ui?.base })" />
<Primitive :as="as" data-slot="root" :class="ui.root({ class: [props.ui?.root, props.class] })">
<UAvatar v-if="hiddenCount > 0" :text="`+${hiddenCount}`" data-slot="base" :class="ui.base({ class: props.ui?.base })" />
<component :is="avatar" v-for="(avatar, count) in visibleAvatars" :key="count" data-slot="base" :class="ui.base({ class: props.ui?.base })" />
</Primitive>
</template>
10 changes: 5 additions & 5 deletions src/runtime/components/Badge.vue
Original file line number Diff line number Diff line change
Expand Up @@ -68,20 +68,20 @@ const ui = computed(() => tv({ extend: tv(theme), ...(appConfig.ui?.badge || {})
</script>

<template>
<Primitive :as="as" :class="ui.base({ class: [props.ui?.base, props.class] })">
<Primitive :as="as" data-slot="base" :class="ui.base({ class: [props.ui?.base, props.class] })">
<slot name="leading" :ui="ui">
<UIcon v-if="isLeading && leadingIconName" :name="leadingIconName" :class="ui.leadingIcon({ class: props.ui?.leadingIcon })" />
<UAvatar v-else-if="!!avatar" :size="((props.ui?.leadingAvatarSize || ui.leadingAvatarSize()) as AvatarProps['size'])" v-bind="avatar" :class="ui.leadingAvatar({ class: props.ui?.leadingAvatar })" />
<UIcon v-if="isLeading && leadingIconName" :name="leadingIconName" data-slot="leadingIcon" :class="ui.leadingIcon({ class: props.ui?.leadingIcon })" />
<UAvatar v-else-if="!!avatar" :size="((props.ui?.leadingAvatarSize || ui.leadingAvatarSize()) as AvatarProps['size'])" v-bind="avatar" data-slot="leadingAvatar" :class="ui.leadingAvatar({ class: props.ui?.leadingAvatar })" />
</slot>

<slot :ui="ui">
<span v-if="label !== undefined && label !== null" :class="ui.label({ class: props.ui?.label })">
<span v-if="label !== undefined && label !== null" data-slot="label" :class="ui.label({ class: props.ui?.label })">
{{ label }}
</span>
</slot>

<slot name="trailing" :ui="ui">
<UIcon v-if="isTrailing && trailingIconName" :name="trailingIconName" :class="ui.trailingIcon({ class: props.ui?.trailingIcon })" />
<UIcon v-if="isTrailing && trailingIconName" :name="trailingIconName" data-slot="trailingIcon" :class="ui.trailingIcon({ class: props.ui?.trailingIcon })" />
</slot>
</Primitive>
</template>
17 changes: 9 additions & 8 deletions src/runtime/components/Banner.vue
Original file line number Diff line number Diff line change
Expand Up @@ -119,7 +119,7 @@ function onClose() {
</script>

<template>
<Primitive :as="as" class="banner" :class="ui.root({ class: [props.ui?.root, props.class] })">
<Primitive :as="as" class="banner" data-slot="root" :class="ui.root({ class: [props.ui?.root, props.class] })">
<ULink
v-if="to"
:aria-label="title"
Expand All @@ -131,28 +131,28 @@ function onClose() {
<span class="absolute inset-0 " aria-hidden="true" />
</ULink>

<UContainer :class="ui.container({ class: props.ui?.container })">
<div :class="ui.left({ class: props.ui?.left })" />
<UContainer data-slot="container" :class="ui.container({ class: props.ui?.container })">
<div data-slot="left" :class="ui.left({ class: props.ui?.left })" />

<div :class="ui.center({ class: props.ui?.center })">
<div data-slot="center" :class="ui.center({ class: props.ui?.center })">
<slot name="leading" :ui="ui">
<UIcon v-if="icon" :name="icon" :class="ui.icon({ class: props.ui?.icon })" />
<UIcon v-if="icon" :name="icon" data-slot="icon" :class="ui.icon({ class: props.ui?.icon })" />
</slot>

<div v-if="title || !!slots.title" :class="ui.title({ class: props.ui?.title })">
<div v-if="title || !!slots.title" data-slot="title" :class="ui.title({ class: props.ui?.title })">
<slot name="title">
{{ title }}
</slot>
</div>

<div v-if="actions?.length || !!slots.actions" :class="ui.actions({ class: props.ui?.actions })">
<div v-if="actions?.length || !!slots.actions" data-slot="actions" :class="ui.actions({ class: props.ui?.actions })">
<slot name="actions">
<UButton v-for="(action, index) in actions" :key="index" color="neutral" size="xs" v-bind="action" />
</slot>
</div>
</div>

<div :class="ui.right({ class: props.ui?.right })">
<div data-slot="right" :class="ui.right({ class: props.ui?.right })">
<slot name="close" :ui="ui">
<UButton
v-if="close"
Expand All @@ -162,6 +162,7 @@ function onClose() {
variant="ghost"
:aria-label="t('banner.close')"
v-bind="(typeof close === 'object' ? close as Partial<ButtonProps> : {})"
data-slot="close"
:class="ui.close({ class: props.ui?.close })"
@click="onClose"
/>
Expand Down
Loading
Loading