feat(module): define neutral utilities (#3629)

Co-authored-by: Sébastien Chopin <atinux@gmail.com>
This commit is contained in:
Benjamin Canac
2025-04-21 15:20:53 +02:00
committed by GitHub
parent 2b315fd855
commit d49e0dadee
238 changed files with 14319 additions and 14046 deletions

View File

@@ -1,7 +1,7 @@
@import "tailwindcss" theme(static) source("../../../.."); @import "tailwindcss" theme(static) source("../../../..");
@import "@nuxt/ui-pro"; @import "@nuxt/ui-pro";
@source "../../../content"; @source "../../../content/**/*";
@source "../../../node_modules/.c12"; @source "../../../node_modules/.c12";
@theme static { @theme static {

View File

@@ -23,7 +23,7 @@ onMounted(() => {
@reference "../assets/css/main.css"; @reference "../assets/css/main.css";
.carbon :deep(#carbonads) { .carbon :deep(#carbonads) {
@apply relative border border-(--ui-border) rounded-md hover:bg-(--ui-bg-elevated)/50 w-full transition-colors min-h-[220px] p-2; @apply relative border border-default rounded-md hover:bg-elevated/50 w-full transition-colors min-h-[220px] p-2;
.carbon-img { .carbon-img {
@apply flex justify-center w-full; @apply flex justify-center w-full;
@@ -34,16 +34,16 @@ onMounted(() => {
} }
.carbon-text { .carbon-text {
@apply text-sm text-(--ui-text-muted) transition-colors text-center text-pretty flex pt-2; @apply text-sm text-muted transition-colors text-center text-pretty flex pt-2;
} }
.carbon-poweredby { .carbon-poweredby {
@apply block text-xs text-center text-(--ui-text-muted) pt-2; @apply block text-xs text-center text-muted pt-2;
} }
&:hover { &:hover {
.carbon-text { .carbon-text {
@apply text-(--ui-text); @apply text-default;
} }
} }
} }

View File

@@ -22,8 +22,8 @@ const links = [{
<UFooter> <UFooter>
<template #left> <template #left>
<NuxtLink to="https://github.com/nuxt/ui" target="_blank" class="text-sm text-(--ui-text-muted)"> <NuxtLink to="https://github.com/nuxt/ui" target="_blank" class="text-sm text-muted">
Published under <span class="text-(--ui-text-highlighted)">MIT License</span> Published under <span class="text-highlighted">MIT License</span>
</NuxtLink> </NuxtLink>
</template> </template>

View File

@@ -19,8 +19,8 @@ watch(framework, () => {
:content="false" :content="false"
color="neutral" color="neutral"
:ui="{ :ui="{
indicator: 'bg-(--ui-bg)', indicator: 'bg-default',
trigger: 'px-1 data-[state=active]:text-(--ui-text-highlighted)' trigger: 'px-1 data-[state=active]:text-highlighted'
}" }"
size="xs" size="xs"
@update:model-value="(framework = $event as string)" @update:model-value="(framework = $event as string)"

View File

@@ -41,7 +41,7 @@ const mobileLinks = computed(() => [
<template> <template>
<UHeader :ui="{ left: 'min-w-0' }" :menu="{ shouldScaleBackground: true }"> <UHeader :ui="{ left: 'min-w-0' }" :menu="{ shouldScaleBackground: true }">
<template #left> <template #left>
<NuxtLink to="/" class="flex items-end gap-2 font-bold text-xl text-(--ui-text-highlighted) min-w-0 focus-visible:outline-(--ui-primary) shrink-0" aria-label="Nuxt UI"> <NuxtLink to="/" class="flex items-end gap-2 font-bold text-xl text-highlighted min-w-0 focus-visible:outline-primary shrink-0" aria-label="Nuxt UI">
<Logo v-if="route.path === '/'" class="w-auto h-6 shrink-0" /> <Logo v-if="route.path === '/'" class="w-auto h-6 shrink-0" />
<LogoPro v-else-if="route.path.startsWith('/pro')" class="w-auto h-6 shrink-0" /> <LogoPro v-else-if="route.path.startsWith('/pro')" class="w-auto h-6 shrink-0" />
<template v-else> <template v-else>
@@ -63,7 +63,7 @@ const mobileLinks = computed(() => [
trailing-icon="i-lucide-chevron-down" trailing-icon="i-lucide-chevron-down"
size="xs" size="xs"
class="-mb-[6px] font-semibold rounded-full truncate" class="-mb-[6px] font-semibold rounded-full truncate"
:class="[open && 'bg-(--ui-primary)/15 ']" :class="[open && 'bg-primary/15 ']"
:ui="{ :ui="{
trailingIcon: ['transition-transform duration-200', open ? 'rotate-180' : undefined].filter(Boolean).join(' ') trailingIcon: ['transition-transform duration-200', open ? 'rotate-180' : undefined].filter(Boolean).join(' ')
}" }"
@@ -108,7 +108,7 @@ const mobileLinks = computed(() => [
<span class="inline-flex items-center gap-0.5"> <span class="inline-flex items-center gap-0.5">
{{ link.title }} {{ link.title }}
<sup v-if="link.module === 'ui-pro'" class="text-[8px] font-medium text-(--ui-primary)">PRO</sup> <sup v-if="link.module === 'ui-pro'" class="text-[8px] font-medium text-primary">PRO</sup>
</span> </span>
</template> </template>
</UContentNavigation> </UContentNavigation>

View File

@@ -19,8 +19,8 @@ watch(module, () => {
:content="false" :content="false"
color="neutral" color="neutral"
:ui="{ :ui="{
indicator: 'bg-(--ui-bg)', indicator: 'bg-default',
trigger: 'px-1 data-[state=active]:text-(--ui-text-highlighted)' trigger: 'px-1 data-[state=active]:text-highlighted'
}" }"
size="xs" size="xs"
@update:model-value="(module = $event as string)" @update:model-value="(module = $event as string)"

View File

@@ -329,15 +329,15 @@ const { data: ast } = await useAsyncData(`component-code-${name}-${hash({ props:
<template> <template>
<div class="my-5"> <div class="my-5">
<div class="relative"> <div class="relative">
<div v-if="options.length" class="flex flex-wrap items-center gap-2.5 border border-(--ui-border-muted) border-b-0 relative rounded-t-md px-4 py-2.5 overflow-x-auto"> <div v-if="options.length" class="flex flex-wrap items-center gap-2.5 border border-muted border-b-0 relative rounded-t-md px-4 py-2.5 overflow-x-auto">
<template v-for="option in options" :key="option.name"> <template v-for="option in options" :key="option.name">
<UFormField <UFormField
:label="option.label" :label="option.label"
size="sm" size="sm"
class="inline-flex ring ring-(--ui-border-accented) rounded-sm" class="inline-flex ring ring-accented rounded-sm"
:ui="{ :ui="{
wrapper: 'bg-(--ui-bg-elevated)/50 rounded-l-sm flex border-r border-(--ui-border-accented)', wrapper: 'bg-elevated/50 rounded-l-sm flex border-r border-accented',
label: 'text-(--ui-text-muted) px-2 py-1.5', label: 'text-muted px-2 py-1.5',
container: 'mt-0' container: 'mt-0'
}" }"
> >
@@ -377,7 +377,7 @@ const { data: ast } = await useAsyncData(`component-code-${name}-${hash({ props:
</template> </template>
</div> </div>
<div v-if="component" class="flex justify-center border border-b-0 border-(--ui-border-muted) relative p-4 z-[1]" :class="[!options.length && 'rounded-t-md', props.class, { 'overflow-hidden': props.overflowHidden }]"> <div v-if="component" class="flex justify-center border border-b-0 border-muted relative p-4 z-[1]" :class="[!options.length && 'rounded-t-md', props.class, { 'overflow-hidden': props.overflowHidden }]">
<component :is="component" v-bind="{ ...componentProps, ...componentEvents }"> <component :is="component" v-bind="{ ...componentProps, ...componentEvents }">
<template v-for="slot in Object.keys(slots || {})" :key="slot" #[slot]> <template v-for="slot in Object.keys(slots || {})" :key="slot" #[slot]>
<slot :name="slot" mdc-unwrap="p"> <slot :name="slot" mdc-unwrap="p">

View File

@@ -150,8 +150,8 @@ const urlSearchParams = computed(() => {
<template> <template>
<div ref="el" class="my-5"> <div ref="el" class="my-5">
<template v-if="preview"> <template v-if="preview">
<div class="border border-(--ui-border-muted) relative z-[1]" :class="[{ 'border-b-0 rounded-t-md': props.source, 'rounded-md': !props.source, 'overflow-hidden': props.overflowHidden }]"> <div class="border border-muted relative z-[1]" :class="[{ 'border-b-0 rounded-t-md': props.source, 'rounded-md': !props.source, 'overflow-hidden': props.overflowHidden }]">
<div v-if="props.options?.length || !!slots.options" class="flex gap-4 p-4 border-b border-(--ui-border-muted)"> <div v-if="props.options?.length || !!slots.options" class="flex gap-4 p-4 border-b border-muted">
<slot name="options" /> <slot name="options" />
<UFormField <UFormField
@@ -160,10 +160,10 @@ const urlSearchParams = computed(() => {
:label="option.label" :label="option.label"
:name="option.name" :name="option.name"
size="sm" size="sm"
class="inline-flex ring ring-(--ui-border-accented) rounded-sm" class="inline-flex ring ring-accented rounded-sm"
:ui="{ :ui="{
wrapper: 'bg-(--ui-bg-elevated)/50 rounded-l-sm flex border-r border-(--ui-border-accented)', wrapper: 'bg-elevated/50 rounded-l-sm flex border-r border-accented',
label: 'text-(--ui-text-muted) px-2 py-1.5', label: 'text-muted px-2 py-1.5',
container: 'mt-0' container: 'mt-0'
}" }"
> >

View File

@@ -112,7 +112,7 @@ const metaProps: ComputedRef<ComponentMeta['props']> = computed(() => {
<ProseTd> <ProseTd>
<HighlightInlineType v-if="prop.type" :type="prop.type" /> <HighlightInlineType v-if="prop.type" :type="prop.type" />
<MDC v-if="prop.description" :value="prop.description" class="text-(--ui-text-toned) mt-1" :cache-key="`${kebabCase(route.path)}-${prop.name}-description`" /> <MDC v-if="prop.description" :value="prop.description" class="text-toned mt-1" :cache-key="`${kebabCase(route.path)}-${prop.name}-description`" />
<ComponentPropsLinks v-if="prop.tags?.length" :prop="prop" /> <ComponentPropsLinks v-if="prop.tags?.length" :prop="prop" />
<ComponentPropsSchema v-if="prop.schema" :prop="prop" :ignore="ignore" /> <ComponentPropsSchema v-if="prop.schema" :prop="prop" :ignore="ignore" />

View File

@@ -43,7 +43,7 @@ const schemaProps = computed(() => {
<ProseLi v-for="schemaProp in schemaProps" :key="schemaProp.name"> <ProseLi v-for="schemaProp in schemaProps" :key="schemaProp.name">
<HighlightInlineType :type="`${schemaProp.name}${schemaProp.required === false ? '?' : ''}: ${schemaProp.type}`" /> <HighlightInlineType :type="`${schemaProp.name}${schemaProp.required === false ? '?' : ''}: ${schemaProp.type}`" />
<MDC v-if="schemaProp.description" :value="schemaProp.description" class="text-(--ui-text-muted) my-1" :cache-key="`${kebabCase(route.path)}-${prop.name}-${schemaProp.name}-description`" /> <MDC v-if="schemaProp.description" :value="schemaProp.description" class="text-muted my-1" :cache-key="`${kebabCase(route.path)}-${prop.name}-${schemaProp.name}-description`" />
</ProseLi> </ProseLi>
</ProseUl> </ProseUl>
</ProseCollapsible> </ProseCollapsible>

View File

@@ -36,7 +36,7 @@ const meta = await fetchComponentMeta(name as any)
<ProseTd> <ProseTd>
<HighlightInlineType v-if="slot.type" :type="slot.type" /> <HighlightInlineType v-if="slot.type" :type="slot.type" />
<MDC v-if="slot.description" :value="slot.description" class="text-(--ui-text-toned) mt-1" :cache-key="`${kebabCase(route.path)}-${slot.name}-description`" /> <MDC v-if="slot.description" :value="slot.description" class="text-toned mt-1" :cache-key="`${kebabCase(route.path)}-${slot.name}-description`" />
</ProseTd> </ProseTd>
</ProseTr> </ProseTr>
</ProseTbody> </ProseTbody>

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="relative overflow-hidden rounded-sm border border-dashed border-(--ui-border-accented) opacity-75 px-4 flex items-center justify-center"> <div class="relative overflow-hidden rounded-sm border border-dashed border-accented opacity-75 px-4 flex items-center justify-center">
<svg class="absolute inset-0 h-full w-full stroke-(--ui-border-inverted)/10" fill="none"> <svg class="absolute inset-0 h-full w-full stroke-inverted/10" fill="none">
<defs> <defs>
<pattern <pattern
id="pattern-5c1e4f0e-62d5-498b-8ff0-cf77bb448c8e" id="pattern-5c1e4f0e-62d5-498b-8ff0-cf77bb448c8e"

View File

@@ -20,7 +20,7 @@ const items: AccordionItem[] = [
<template> <template>
<UAccordion :items="items"> <UAccordion :items="items">
<template #content="{ item }"> <template #content="{ item }">
<p class="pb-3.5 text-sm text-(--ui-text-muted)"> <p class="pb-3.5 text-sm text-muted">
This is the {{ item.label }} panel. This is the {{ item.label }} panel.
</p> </p>
</template> </template>

View File

@@ -24,7 +24,7 @@ const items = [
<template> <template>
<UAccordion :items="items"> <UAccordion :items="items">
<template #colors="{ item }"> <template #colors="{ item }">
<p class="text-sm pb-3.5 text-(--ui-primary)"> <p class="text-sm pb-3.5 text-primary">
{{ item.content }} {{ item.content }}
</p> </p>
</template> </template>

View File

@@ -3,7 +3,7 @@
<ULink <ULink
to="https://github.com/benjamincanac" to="https://github.com/benjamincanac"
target="_blank" target="_blank"
class="hover:ring-(--ui-primary) transition" class="hover:ring-primary transition"
raw raw
> >
<UAvatar <UAvatar
@@ -15,7 +15,7 @@
<ULink <ULink
to="https://github.com/romhml" to="https://github.com/romhml"
target="_blank" target="_blank"
class="hover:ring-(--ui-primary) transition" class="hover:ring-primary transition"
raw raw
> >
<UAvatar <UAvatar
@@ -27,7 +27,7 @@
<ULink <ULink
to="https://github.com/noook" to="https://github.com/noook"
target="_blank" target="_blank"
class="hover:ring-(--ui-primary) transition" class="hover:ring-primary transition"
raw raw
> >
<UAvatar <UAvatar

View File

@@ -20,7 +20,7 @@ const items: BreadcrumbItem[] = [
<template> <template>
<UBreadcrumb :items="items"> <UBreadcrumb :items="items">
<template #separator> <template #separator>
<span class="mx-2 text-(--ui-text-muted)">/</span> <span class="mx-2 text-muted">/</span>
</template> </template>
</UBreadcrumb> </UBreadcrumb>
</template> </template>

View File

@@ -35,7 +35,7 @@ const items = computed<ContextMenuItem[]>(() => [{
<template> <template>
<UContextMenu :items="items" :ui="{ content: 'w-48' }"> <UContextMenu :items="items" :ui="{ content: 'w-48' }">
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Right click here Right click here
</div> </div>
</UContextMenu> </UContextMenu>

View File

@@ -28,7 +28,7 @@ const items: ContextMenuItem[][] = [
<template> <template>
<UContextMenu :items="items" :ui="{ content: 'w-48' }"> <UContextMenu :items="items" :ui="{ content: 'w-48' }">
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Right click here Right click here
</div> </div>
</UContextMenu> </UContextMenu>

View File

@@ -19,7 +19,7 @@ const items = [
<template> <template>
<UContextMenu :items="items" :ui="{ content: 'w-48' }"> <UContextMenu :items="items" :ui="{ content: 'w-48' }">
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Right click here Right click here
</div> </div>
@@ -28,7 +28,7 @@ const items = [
</template> </template>
<template #refresh-trailing> <template #refresh-trailing>
<UIcon v-if="loading" name="i-lucide-refresh-cw" class="shrink-0 size-5 text-(--ui-primary) animate-spin" /> <UIcon v-if="loading" name="i-lucide-refresh-cw" class="shrink-0 size-5 text-primary animate-spin" />
</template> </template>
</UContextMenu> </UContextMenu>
</template> </template>

View File

@@ -7,7 +7,7 @@ const open = ref(false)
<UButton label="Open" color="neutral" variant="subtle" trailing-icon="i-lucide-chevron-up" /> <UButton label="Open" color="neutral" variant="subtle" trailing-icon="i-lucide-chevron-up" />
<template #header> <template #header>
<h2 class="text-(--ui-text-highlighted) font-semibold"> <h2 class="text-highlighted font-semibold">
Drawer non-dismissible Drawer non-dismissible
</h2> </h2>

View File

@@ -21,7 +21,7 @@ const items = [
<UButton label="Open" color="neutral" variant="outline" icon="i-lucide-menu" /> <UButton label="Open" color="neutral" variant="outline" icon="i-lucide-menu" />
<template #profile-trailing> <template #profile-trailing>
<UIcon name="i-lucide-badge-check" class="shrink-0 size-5 text-(--ui-primary)" /> <UIcon name="i-lucide-badge-check" class="shrink-0 size-5 text-primary" />
</template> </template>
</UDropdownMenu> </UDropdownMenu>
</template> </template>

View File

@@ -36,7 +36,7 @@ const { data: users, status } = await useFetch('https://jsonplaceholder.typicode
<template #item-label="{ item }"> <template #item-label="{ item }">
{{ item.label }} {{ item.label }}
<span class="text-(--ui-text-muted)"> <span class="text-muted">
{{ item.email }} {{ item.email }}
</span> </span>
</template> </template>

View File

@@ -15,7 +15,7 @@ const domain = ref(domains[0])
}" }"
> >
<template #leading> <template #leading>
<p class="text-sm text-(--ui-text-muted)"> <p class="text-sm text-muted">
https:// https://
</p> </p>
</template> </template>

View File

@@ -13,7 +13,7 @@ const maxLength = 15
<template #trailing> <template #trailing>
<div <div
id="character-count" id="character-count"
class="text-xs text-(--ui-text-muted) tabular-nums" class="text-xs text-muted tabular-nums"
aria-live="polite" aria-live="polite"
role="status" role="status"
> >

View File

@@ -4,8 +4,8 @@ const value = ref('')
<template> <template>
<UInput v-model="value" placeholder="" :ui="{ base: 'peer' }"> <UInput v-model="value" placeholder="" :ui="{ base: 'peer' }">
<label class="pointer-events-none absolute left-0 -top-2.5 text-(--ui-text-highlighted) text-xs font-medium px-1.5 transition-all peer-focus:-top-2.5 peer-focus:text-(--ui-text-highlighted) peer-focus:text-xs peer-focus:font-medium peer-placeholder-shown:text-sm peer-placeholder-shown:text-(--ui-text-dimmed) peer-placeholder-shown:top-1.5 peer-placeholder-shown:font-normal"> <label class="pointer-events-none absolute left-0 -top-2.5 text-highlighted text-xs font-medium px-1.5 transition-all peer-focus:-top-2.5 peer-focus:text-highlighted peer-focus:text-xs peer-focus:font-medium peer-placeholder-shown:text-sm peer-placeholder-shown:text-dimmed peer-placeholder-shown:top-1.5 peer-placeholder-shown:font-normal">
<span class="inline-flex bg-(--ui-bg) px-1">Email address</span> <span class="inline-flex bg-default px-1">Email address</span>
</label> </label>
</UInput> </UInput>
</template> </template>

View File

@@ -77,7 +77,7 @@ const text = computed(() => {
v-for="(req, index) in strength" v-for="(req, index) in strength"
:key="index" :key="index"
class="flex items-center gap-0.5" class="flex items-center gap-0.5"
:class="req.met ? 'text-(--ui-success)' : 'text-(--ui-text-muted)'" :class="req.met ? 'text-success' : 'text-muted'"
> >
<UIcon :name="req.met ? 'i-lucide-circle-check' : 'i-lucide-circle-x'" class="size-4 shrink-0" /> <UIcon :name="req.met ? 'i-lucide-circle-check' : 'i-lucide-circle-x'" class="size-4 shrink-0" />

View File

@@ -76,11 +76,11 @@ const items = [
</li> </li>
<li v-for="child in item.children" :key="child.label"> <li v-for="child in item.children" :key="child.label">
<ULink class="text-sm text-left rounded-md p-3 transition-colors hover:bg-(--ui-bg-elevated)/50"> <ULink class="text-sm text-left rounded-md p-3 transition-colors hover:bg-elevated/50">
<p class="font-medium text-(--ui-text-highlighted)"> <p class="font-medium text-highlighted">
{{ child.label }} {{ child.label }}
</p> </p>
<p class="text-(--ui-text-muted) line-clamp-2"> <p class="text-muted line-clamp-2">
{{ child.description }} {{ child.description }}
</p> </p>
</ULink> </ULink>

View File

@@ -8,7 +8,7 @@ const open = ref(false)
<template #content> <template #content>
<div class="flex items-center gap-4 mb-4"> <div class="flex items-center gap-4 mb-4">
<h2 class="text-(--ui-text-highlighted) font-semibold"> <h2 class="text-highlighted font-semibold">
Popover non-dismissible Popover non-dismissible
</h2> </h2>

View File

@@ -36,7 +36,7 @@ const { data: users, status } = await useFetch('https://jsonplaceholder.typicode
<template #item-label="{ item }"> <template #item-label="{ item }">
{{ item.label }} {{ item.label }}
<span class="text-(--ui-text-muted)"> <span class="text-muted">
{{ item.email }} {{ item.email }}
</span> </span>
</template> </template>

View File

@@ -100,7 +100,7 @@ const columnFilters = ref([{
<template> <template>
<div class="flex flex-col flex-1 w-full"> <div class="flex flex-col flex-1 w-full">
<div class="flex px-4 py-3.5 border-b border-(--ui-border-accented)"> <div class="flex px-4 py-3.5 border-b border-accented">
<UInput <UInput
:model-value="(table?.tableApi?.getColumn('email')?.getFilterValue() as string)" :model-value="(table?.tableApi?.getColumn('email')?.getFilterValue() as string)"
class="max-w-sm" class="max-w-sm"

View File

@@ -131,7 +131,7 @@ function getHeader(column: Column<Payment>, label: string) {
'variant': 'ghost', 'variant': 'ghost',
label, label,
'icon': isSorted ? (isSorted === 'asc' ? 'i-lucide-arrow-up-narrow-wide' : 'i-lucide-arrow-down-wide-narrow') : 'i-lucide-arrow-up-down', 'icon': isSorted ? (isSorted === 'asc' ? 'i-lucide-arrow-up-narrow-wide' : 'i-lucide-arrow-down-wide-narrow') : 'i-lucide-arrow-up-down',
'class': '-mx-2.5 data-[state=open]:bg-(--ui-bg-elevated)', 'class': '-mx-2.5 data-[state=open]:bg-elevated',
'aria-label': `Sort by ${isSorted === 'asc' ? 'descending' : 'ascending'}` 'aria-label': `Sort by ${isSorted === 'asc' ? 'descending' : 'ascending'}`
})) }))
} }

View File

@@ -100,7 +100,7 @@ const columnVisibility = ref({
<template> <template>
<div class="flex flex-col flex-1 w-full"> <div class="flex flex-col flex-1 w-full">
<div class="flex justify-end px-4 py-3.5 border-b border-(--ui-border-accented)"> <div class="flex justify-end px-4 py-3.5 border-b border-accented">
<UDropdownMenu <UDropdownMenu
:items="table?.tableApi?.getAllColumns().filter(column => column.getCanHide()).map(column => ({ :items="table?.tableApi?.getAllColumns().filter(column => column.getCanHide()).map(column => ({
label: upperFirst(column.id), label: upperFirst(column.id),

View File

@@ -265,7 +265,7 @@ function randomize() {
</script> </script>
<template> <template>
<div class="flex-1 divide-y divide-(--ui-border-accented) w-full"> <div class="flex-1 divide-y divide-accented w-full">
<div class="flex items-center gap-2 px-4 py-3.5 overflow-x-auto"> <div class="flex items-center gap-2 px-4 py-3.5 overflow-x-auto">
<UInput <UInput
:model-value="(table?.tableApi?.getColumn('email')?.getFilterValue() as string)" :model-value="(table?.tableApi?.getColumn('email')?.getFilterValue() as string)"
@@ -313,7 +313,7 @@ function randomize() {
</template> </template>
</UTable> </UTable>
<div class="px-4 py-3.5 text-sm text-(--ui-text-muted)"> <div class="px-4 py-3.5 text-sm text-muted">
{{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of {{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of
{{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected. {{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected.
</div> </div>

View File

@@ -36,7 +36,7 @@ const columns: TableColumn<User>[] = [{
size: 'lg' size: 'lg'
}), }),
h('div', undefined, [ h('div', undefined, [
h('p', { class: 'font-medium text-(--ui-text-highlighted)' }, row.original.name), h('p', { class: 'font-medium text-highlighted' }, row.original.name),
h('p', { class: '' }, `@${row.original.username}`) h('p', { class: '' }, `@${row.original.username}`)
]) ])
]) ])

View File

@@ -95,7 +95,7 @@ const globalFilter = ref('45')
<template> <template>
<div class="flex flex-col flex-1 w-full"> <div class="flex flex-col flex-1 w-full">
<div class="flex px-4 py-3.5 border-b border-(--ui-border-accented)"> <div class="flex px-4 py-3.5 border-b border-accented">
<UInput <UInput
v-model="globalFilter" v-model="globalFilter"
class="max-w-sm" class="max-w-sm"

View File

@@ -162,7 +162,7 @@ const pagination = ref({
class="flex-1" class="flex-1"
/> />
<div class="flex justify-center border-t border-(--ui-border) pt-4"> <div class="flex justify-center border-t border-default pt-4">
<UPagination <UPagination
:default-page="(table?.tableApi?.getState().pagination.pageIndex || 0) + 1" :default-page="(table?.tableApi?.getState().pagination.pageIndex || 0) + 1"
:items-per-page="table?.tableApi?.getState().pagination.pageSize" :items-per-page="table?.tableApi?.getState().pagination.pageSize"

View File

@@ -112,7 +112,7 @@ const expanded = ref({ 1: true })
v-model:expanded="expanded" v-model:expanded="expanded"
:data="data" :data="data"
:columns="columns" :columns="columns"
:ui="{ tr: 'data-[expanded=true]:bg-(--ui-bg-elevated)/50' }" :ui="{ tr: 'data-[expanded=true]:bg-elevated/50' }"
class="flex-1" class="flex-1"
> >
<template #expanded="{ row }"> <template #expanded="{ row }">

View File

@@ -122,7 +122,7 @@ function onSelect(row: TableRow<Payment>, e?: Event) {
@select="onSelect" @select="onSelect"
/> />
<div class="px-4 py-3.5 border-t border-[var(--ui-border-accented)] text-sm text-[var(--ui-text-muted)]"> <div class="px-4 py-3.5 border-t border-accented text-sm text-muted">
{{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of {{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of
{{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected. {{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected.
</div> </div>

View File

@@ -113,7 +113,7 @@ const rowSelection = ref({ 1: true })
:columns="columns" :columns="columns"
/> />
<div class="px-4 py-3.5 border-t border-(--ui-border-accented) text-sm text-(--ui-text-muted)"> <div class="px-4 py-3.5 border-t border-accented text-sm text-muted">
{{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of {{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of
{{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected. {{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected.
</div> </div>

View File

@@ -97,7 +97,7 @@ function getDropdownActions(user: User): DropdownMenuItem[][] {
<div class="flex items-center gap-3"> <div class="flex items-center gap-3">
<UAvatar :src="`https://i.pravatar.cc/120?img=${row.original.id}`" size="lg" :alt="`${row.original.name} avatar`" /> <UAvatar :src="`https://i.pravatar.cc/120?img=${row.original.id}`" size="lg" :alt="`${row.original.name} avatar`" />
<div> <div>
<p class="font-medium text-(--ui-text-highlighted)"> <p class="font-medium text-highlighted">
{{ row.original.name }} {{ row.original.name }}
</p> </p>
<p> <p>

View File

@@ -28,7 +28,7 @@ const state = reactive({
<template> <template>
<UTabs :items="items" variant="link" class="gap-4 w-full" :ui="{ trigger: 'flex-1' }"> <UTabs :items="items" variant="link" class="gap-4 w-full" :ui="{ trigger: 'flex-1' }">
<template #account="{ item }"> <template #account="{ item }">
<p class="text-(--ui-text-muted) mb-4"> <p class="text-muted mb-4">
{{ item.description }} {{ item.description }}
</p> </p>
@@ -45,7 +45,7 @@ const state = reactive({
</template> </template>
<template #password="{ item }"> <template #password="{ item }">
<p class="text-(--ui-text-muted) mb-4"> <p class="text-muted mb-4">
{{ item.description }} {{ item.description }}
</p> </p>

View File

@@ -7,10 +7,10 @@ const appConfig = useAppConfig()
<UFormField <UFormField
label="toaster.duration" label="toaster.duration"
size="sm" size="sm"
class="inline-flex ring ring-(--ui-border-accented) rounded-sm" class="inline-flex ring ring-accented rounded-sm"
:ui="{ :ui="{
wrapper: 'bg-(--ui-bg-elevated)/50 rounded-l-sm flex border-r border-(--ui-border-accented)', wrapper: 'bg-elevated/50 rounded-l-sm flex border-r border-accented',
label: 'text-(--ui-text-muted) px-2 py-1.5', label: 'text-muted px-2 py-1.5',
container: 'mt-0' container: 'mt-0'
}" }"
> >

View File

@@ -7,10 +7,10 @@ const appConfig = useAppConfig()
<UFormField <UFormField
label="toaster.expand" label="toaster.expand"
size="sm" size="sm"
class="inline-flex ring ring-(--ui-border-accented) rounded-sm" class="inline-flex ring ring-accented rounded-sm"
:ui="{ :ui="{
wrapper: 'bg-(--ui-bg-elevated)/50 rounded-l-sm flex border-r border-(--ui-border-accented)', wrapper: 'bg-elevated/50 rounded-l-sm flex border-r border-accented',
label: 'text-(--ui-text-muted) px-2 py-1.5', label: 'text-muted px-2 py-1.5',
container: 'mt-0' container: 'mt-0'
}" }"
> >

View File

@@ -10,10 +10,10 @@ const appConfig = useAppConfig()
<UFormField <UFormField
label="toaster.position" label="toaster.position"
size="sm" size="sm"
class="inline-flex ring ring-(--ui-border-accented) rounded-sm" class="inline-flex ring ring-accented rounded-sm"
:ui="{ :ui="{
wrapper: 'bg-(--ui-bg-elevated)/50 rounded-l-sm flex border-r border-(--ui-border-accented)', wrapper: 'bg-elevated/50 rounded-l-sm flex border-r border-accented',
label: 'text-(--ui-text-muted) px-2 py-1.5', label: 'text-muted px-2 py-1.5',
container: 'mt-0' container: 'mt-0'
}" }"
> >

View File

@@ -20,7 +20,7 @@ const { width } = useElementSize(el)
<template> <template>
<div <div
class="isolate rounded-full relative circle w-full aspect-[1/1] p-8 sm:p-12 md:p-14 lg:p-10 xl:p-16 before:absolute before:inset-px before:bg-(--ui-bg) before:rounded-full z-(--level)" class="isolate rounded-full relative circle w-full aspect-[1/1] p-8 sm:p-12 md:p-14 lg:p-10 xl:p-16 before:absolute before:inset-px before:bg-default before:rounded-full z-(--level)"
:class="{ 'animation-paused': paused }" :class="{ 'animation-paused': paused }"
:style="{ :style="{
'--duration': `${((level + 1) * 8)}s`, '--duration': `${((level + 1) * 8)}s`,
@@ -65,7 +65,7 @@ const { width } = useElementSize(el)
:src="`https://ipx.nuxt.com/s_56x56/gh_avatar/${contributor.username}`" :src="`https://ipx.nuxt.com/s_56x56/gh_avatar/${contributor.username}`"
:srcset="`https://ipx.nuxt.com/s_112x112/gh_avatar/${contributor.username} 2x`" :srcset="`https://ipx.nuxt.com/s_112x112/gh_avatar/${contributor.username} 2x`"
:alt="contributor.username" :alt="contributor.username"
class="ring-2 ring-(--ui-border) lg:hover:ring-(--ui-border-inverted) transition rounded-full size-7" class="ring-2 ring-default lg:hover:ring-inverted transition rounded-full size-7"
loading="lazy" loading="lazy"
> >
</NuxtLink> </NuxtLink>

View File

@@ -69,7 +69,7 @@ function setBlackAsPrimary(value: boolean) {
:variant="open ? 'soft' : 'ghost'" :variant="open ? 'soft' : 'ghost'"
square square
aria-label="Color picker" aria-label="Color picker"
:ui="{ leadingIcon: 'text-(--ui-primary)' }" :ui="{ leadingIcon: 'text-primary' }"
/> />
</template> </template>

View File

@@ -18,8 +18,8 @@ const slots = defineSlots<{
variant="outline" variant="outline"
:icon="icon" :icon="icon"
:label="label" :label="label"
class="capitalize ring-(--ui-border) rounded-sm text-[11px]" class="capitalize ring-default rounded-sm text-[11px]"
:class="[selected ? 'bg-(--ui-bg-elevated)' : 'hover:bg-(--ui-bg-elevated)/50']" :class="[selected ? 'bg-elevated' : 'hover:bg-elevated/50']"
> >
<template v-if="chip || !!slots.leading" #leading> <template v-if="chip || !!slots.leading" #leading>
<slot name="leading"> <slot name="leading">

View File

@@ -22,7 +22,7 @@ const navigation = inject<Ref<ContentNavigationItem[]>>('navigation')
<span class="inline-flex items-center gap-0.5"> <span class="inline-flex items-center gap-0.5">
{{ link.title }} {{ link.title }}
<sup v-if="link.module === 'ui-pro'" class="text-[8px] font-medium text-(--ui-primary)">PRO</sup> <sup v-if="link.module === 'ui-pro'" class="text-[8px] font-medium text-primary">PRO</sup>
</span> </span>
</template> </template>
</UContentNavigation> </UContentNavigation>

View File

@@ -130,7 +130,7 @@ const communityLinks = computed(() => [{
</template> </template>
<template #title> <template #title>
{{ page.title }}<sup v-if="page.module === 'ui-pro'" class="ml-1 text-xs align-super font-medium text-(--ui-primary)">PRO</sup> {{ page.title }}<sup v-if="page.module === 'ui-pro'" class="ml-1 text-xs align-super font-medium text-primary">PRO</sup>
</template> </template>
<template #description> <template #description>

View File

@@ -82,7 +82,7 @@ onMounted(() => {
:ui="{ title: 'text-balance', container: 'relative' }" :ui="{ title: 'text-balance', container: 'relative' }"
> >
<template #top> <template #top>
<div class="absolute z-[-1] rounded-full bg-(--ui-primary) blur-[300px] size-60 sm:size-80 transform -translate-x-1/2 left-1/2 -translate-y-80" /> <div class="absolute z-[-1] rounded-full bg-primary blur-[300px] size-60 sm:size-80 transform -translate-x-1/2 left-1/2 -translate-y-80" />
</template> </template>
<template #headline> <template #headline>
@@ -97,7 +97,7 @@ onMounted(() => {
/> />
</template> </template>
<template #title> <template #title>
Build beautiful UI with <span class="text-(--ui-primary)">{{ components!.length }}+</span> powerful components Build beautiful UI with <span class="text-primary">{{ components!.length }}+</span> powerful components
</template> </template>
<template #links> <template #links>
@@ -121,22 +121,22 @@ onMounted(() => {
<LazyStarsBg /> <LazyStarsBg />
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
</UPageHero> </UPageHero>
<div v-for="category in categories" :key="category.id"> <div v-for="category in categories" :key="category.id">
<div data-track-sticky class="group mb-4 sm:mb-6 lg:mb-8 sticky top-[calc(var(--ui-header-height)-1px)] bg-(--ui-bg)/75 backdrop-blur z-[1]"> <div data-track-sticky class="group mb-4 sm:mb-6 lg:mb-8 sticky top-[calc(var(--ui-header-height)-1px)] bg-default/75 backdrop-blur z-[1]">
<div class="relative border-y border-(--ui-border) py-4 sm:not-group-[[data-stuck]]:py-6 lg:not-group-[[data-stuck]]:py-8 transition-all duration-300"> <div class="relative border-y border-default py-4 sm:not-group-[[data-stuck]]:py-6 lg:not-group-[[data-stuck]]:py-8 transition-all duration-300">
<UContainer> <UContainer>
<h2 class="relative text-pretty font-bold text-(--ui-text-highlighted) text-base sm:not-group-[[data-stuck]]:text-xl lg:not-group-[[data-stuck]]:text-2xl transition-all duration-300 "> <h2 class="relative text-pretty font-bold text-highlighted text-base sm:not-group-[[data-stuck]]:text-xl lg:not-group-[[data-stuck]]:text-2xl transition-all duration-300 ">
<a :href="`#${category.id}`" class="group lg:not-group-[[data-stuck]]:ps-2 lg:not-group-[[data-stuck]]:-ms-2"> <a :href="`#${category.id}`" class="group lg:not-group-[[data-stuck]]:ps-2 lg:not-group-[[data-stuck]]:-ms-2">
<span class="absolute -ms-8 top-1 opacity-0 group-hover:opacity-100 group-focus:opacity-100 p-1 bg-(--ui-bg-elevated) hover:text-(--ui-primary) rounded-md hidden lg:not-group-[[data-stuck]]:flex text-(--ui-text-muted) transition"> <span class="absolute -ms-8 top-1 opacity-0 group-hover:opacity-100 group-focus:opacity-100 p-1 bg-elevated hover:text-primary rounded-md hidden lg:not-group-[[data-stuck]]:flex text-muted transition">
<UIcon name="i-lucide-hash" class="size-4 shrink-0" /> <UIcon name="i-lucide-hash" class="size-4 shrink-0" />
</span> </span>
{{ category.title }} {{ category.title }}
</a> </a>
</h2> </h2>
<p class="text-pretty text-(--ui-text-muted) text-sm sm:not-group-[[data-stuck]]:text-base lg:not-group-[[data-stuck]]:text-lg mt-1 sm:not-group-[[data-stuck]]:mt-2 line-clamp-1 transition-all duration-300"> <p class="text-pretty text-muted text-sm sm:not-group-[[data-stuck]]:text-base lg:not-group-[[data-stuck]]:text-lg mt-1 sm:not-group-[[data-stuck]]:mt-2 line-clamp-1 transition-all duration-300">
{{ category.description }} {{ category.description }}
</p> </p>
</UContainer> </UContainer>
@@ -157,11 +157,11 @@ onMounted(() => {
<template #title> <template #title>
<div class="flex items-center gap-0.5"> <div class="flex items-center gap-0.5">
<span>{{ component.title }}</span> <span>{{ component.title }}</span>
<sup v-if="component.module === 'ui-pro'" class="text-[8px] font-medium text-(--ui-primary)">PRO</sup> <sup v-if="component.module === 'ui-pro'" class="text-[8px] font-medium text-primary">PRO</sup>
</div> </div>
</template> </template>
<div class="rounded-md border border-(--ui-border-muted) overflow-hidden aspect-[16/9]"> <div class="rounded-md border border-muted overflow-hidden aspect-[16/9]">
<UColorModeImage <UColorModeImage
:light="`${component.path.replace('/components/', '/components/light/')}.png`" :light="`${component.path.replace('/components/', '/components/light/')}.png`"
:dark="`${component.path.replace('/components/', '/components/dark/')}.png`" :dark="`${component.path.replace('/components/', '/components/dark/')}.png`"

View File

@@ -27,7 +27,7 @@ features1:
description: Start with essential components, or unlock Pro for complete blocks and templates. description: Start with essential components, or unlock Pro for complete blocks and templates.
icon: i-lucide-files icon: i-lucide-files
cta1: cta1:
title: Everything you need in a [single file]{class="text-(--ui-primary)"}. title: Everything you need in a [single file]{class="text-primary"}.
description: Design and development in perfect sync with our [Free](https://www.figma.com/community/file/1288455405058138934/nuxt-ui-v3-official-design-kit-free) and Pro files. Developers can implement designs faster, while designers work with production-ready components. description: Design and development in perfect sync with our [Free](https://www.figma.com/community/file/1288455405058138934/nuxt-ui-v3-official-design-kit-free) and Pro files. Developers can implement designs faster, while designers work with production-ready components.
section1: section1:
title: Customize in a few clicks to fit your needs title: Customize in a few clicks to fit your needs
@@ -181,7 +181,7 @@ pricing:
# discount: $119 # discount: $119
billing_period: one-time payment billing_period: one-time payment
billing_cycle: plus local taxes billing_cycle: plus local taxes
class: bg-(--ui-bg-elevated)/50 class: bg-elevated/50
features: features:
- '**1 Designer**' - '**1 Designer**'
- Nuxt UI & Nuxt UI Pro Components - Nuxt UI & Nuxt UI Pro Components
@@ -203,7 +203,7 @@ pricing:
# discount: $279 # discount: $279
billing_period: one-time payment billing_period: one-time payment
billing_cycle: plus local taxes billing_cycle: plus local taxes
class: bg-(--ui-bg-elevated)/50 class: bg-elevated/50
features: features:
- '**Up to 20 Designers**' - '**Up to 20 Designers**'
- Nuxt UI & Nuxt UI Pro Components - Nuxt UI & Nuxt UI Pro Components

View File

@@ -57,7 +57,7 @@ onMounted(async () => {
<template> <template>
<div class="relative"> <div class="relative">
<div id="cursor1" class="absolute z-10 pointer-events-none" :style="{ opacity: 0 }"> <div id="cursor1" class="absolute z-10 pointer-events-none" :style="{ opacity: 0 }">
<svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 24 24" class="absolute top-0 left-0 drop-shadow-[0_1px_2px_rgb(0,0,0,0.25)] text-white dark:text-(--ui-bg)"> <svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 24 24" class="absolute top-0 left-0 drop-shadow-[0_1px_2px_rgb(0,0,0,0.25)] text-inverted">
<path <path
fill="var(--ui-info)" fill="var(--ui-info)"
stroke="currentColor" stroke="currentColor"
@@ -72,7 +72,7 @@ onMounted(async () => {
</UBadge> </UBadge>
</div> </div>
<div id="cursor2" class="absolute z-10 pointer-events-none" :style="{ opacity: 0 }"> <div id="cursor2" class="absolute z-10 pointer-events-none" :style="{ opacity: 0 }">
<svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 24 24" class="absolute top-0 left-0 drop-shadow-[0_1px_2px_rgb(0,0,0,0.25)] text-white dark:text-(--ui-bg)"> <svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 24 24" class="absolute top-0 left-0 drop-shadow-[0_1px_2px_rgb(0,0,0,0.25)] text-inverted">
<path <path
fill="var(--ui-success)" fill="var(--ui-success)"
stroke="currentColor" stroke="currentColor"
@@ -99,7 +99,7 @@ onMounted(async () => {
<template #description> <template #description>
<MDC :value="page.hero.description" unwrap="p" cache-key="figma-hero-description" /> <MDC :value="page.hero.description" unwrap="p" cache-key="figma-hero-description" />
</template> </template>
<!-- <img src="/pro/figma/nuxt-ui-figma.png" alt="Screnshot of the Nuxt UI Figma design kit" class="w-full h-auto border border-(--ui-border) border-b-0"> --> <!-- <img src="/pro/figma/nuxt-ui-figma.png" alt="Screnshot of the Nuxt UI Figma design kit" class="w-full h-auto border border-default border-b-0"> -->
<div class="relative"> <div class="relative">
<video <video
ref="video" ref="video"
@@ -126,10 +126,10 @@ onMounted(async () => {
</div> </div>
</div> </div>
<Motion as-child :initial="{ height: 0 }" :animate="{ height: 'auto' }" :transition="{ delay: 0.2, duration: 1 }"> <Motion as-child :initial="{ height: 0 }" :animate="{ height: 'auto' }" :transition="{ delay: 0.2, duration: 1 }">
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
</Motion> </Motion>
</UPageHero> </UPageHero>
<UPageSection v-bind="page.features1" :ui="{ container: 'py-16 sm:py-16 lg:py-16', features: 'mt-0' }" class="border-y border-(--ui-border)" /> <UPageSection v-bind="page.features1" :ui="{ container: 'py-16 sm:py-16 lg:py-16', features: 'mt-0' }" class="border-y border-default" />
<UPageCTA <UPageCTA
v-if="page.cta1" v-if="page.cta1"
variant="naked" variant="naked"
@@ -138,7 +138,7 @@ onMounted(async () => {
wrapper: 'grid grid-cols-1 lg:grid-cols-2', wrapper: 'grid grid-cols-1 lg:grid-cols-2',
description: 'lg:mt-0' }" description: 'lg:mt-0' }"
orientation="horizontal" orientation="horizontal"
class="rounded-none bg-gradient-to-b from-(--ui-bg-muted) to-(--ui-bg)" class="rounded-none bg-gradient-to-b from-elevated/50 to-default"
> >
<template #title> <template #title>
<MDC :value="page.cta1.title" unwrap="p" cache-key="figma-cta-1-title" /> <MDC :value="page.cta1.title" unwrap="p" cache-key="figma-cta-1-title" />
@@ -192,8 +192,8 @@ onMounted(async () => {
<template #description> <template #description>
<MDC :value="page.section4.description" unwrap="p" cache-key="figma-section-4-description" /> <MDC :value="page.section4.description" unwrap="p" cache-key="figma-section-4-description" />
</template> </template>
<div aria-hidden="true" class="absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<ul class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 items-start justify-center border border-(--ui-border) border-b-0 sm:divide-x divide-y lg:divide-y-0 divide-(--ui-border)"> <ul class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 items-start justify-center border border-default border-b-0 sm:divide-x divide-y lg:divide-y-0 divide-default">
<li v-for="(step, index) in page?.section4.steps" :key="step.title" class="flex flex-col gap-y-4 justify-start group h-full p-4"> <li v-for="(step, index) in page?.section4.steps" :key="step.title" class="flex flex-col gap-y-4 justify-start group h-full p-4">
<NuxtImg <NuxtImg
v-if="step.image" v-if="step.image"
@@ -205,14 +205,14 @@ onMounted(async () => {
<h2 class="font-semibold inline-flex items-center gap-x-1"> <h2 class="font-semibold inline-flex items-center gap-x-1">
<UBadge :label="index + 1" size="sm" color="neutral" variant="subtle" class="rounded-full tabular-nums" /> {{ step.title }} <UBadge :label="index + 1" size="sm" color="neutral" variant="subtle" class="rounded-full tabular-nums" /> {{ step.title }}
</h2> </h2>
<p class="text-(--ui-text-muted) text-sm"> <p class="text-muted text-sm">
{{ step.description }} {{ step.description }}
</p> </p>
</div> </div>
</li> </li>
</ul> </ul>
</UPageSection> </UPageSection>
<UPageSection v-bind="page.features2" :ui="{ container: 'py-16 sm:py-16 lg:py-16', features: 'mt-0' }" class="border-y border-(--ui-border)" /> <UPageSection v-bind="page.features2" :ui="{ container: 'py-16 sm:py-16 lg:py-16', features: 'mt-0' }" class="border-y border-default" />
<UPageSection <UPageSection
v-if="page.pricing" v-if="page.pricing"
:title="page.pricing.title" :title="page.pricing.title"
@@ -226,7 +226,7 @@ onMounted(async () => {
wrapper: 'sm:pl-8' wrapper: 'sm:pl-8'
}" }"
> >
<div aria-hidden="true" class="absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<UPricingPlans compact class="-space-x-px"> <UPricingPlans compact class="-space-x-px">
<UPricingPlan <UPricingPlan
v-for="(plan, index) in page.pricing.plans" v-for="(plan, index) in page.pricing.plans"
@@ -246,8 +246,8 @@ onMounted(async () => {
> >
<template #features> <template #features>
<li v-for="(feature, i) in plan.features" :key="i" class="flex items-center gap-2 min-w-0"> <li v-for="(feature, i) in plan.features" :key="i" class="flex items-center gap-2 min-w-0">
<UIcon name="i-lucide-circle-check" class="size-5 shrink-0 text-(--ui-primary)" /> <UIcon name="i-lucide-circle-check" class="size-5 shrink-0 text-primary" />
<MDC :value="feature" unwrap="p" tag="span" class="text-sm truncate text-(--ui-text-accented)" :cache-key="`figma-pricing-plan-${index}-feature-${i}`" /> <MDC :value="feature" unwrap="p" tag="span" class="text-sm truncate text-accented" :cache-key="`figma-pricing-plan-${index}-feature-${i}`" />
</li> </li>
</template> </template>
<template #button> <template #button>
@@ -278,7 +278,7 @@ onMounted(async () => {
</UPageMarquee> </UPageMarquee>
</UPageCTA> </UPageCTA>
<UPageSection v-bind="page.faq" :ui="{ container: 'relative' }"> <UPageSection v-bind="page.faq" :ui="{ container: 'relative' }">
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<UPageAccordion <UPageAccordion
multiple multiple
:items="(page.faq.items as any[])" :items="(page.faq.items as any[])"

View File

@@ -48,7 +48,7 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
}" }"
> >
<template #title> <template #title>
The Intuitive <br> <span class="text-(--ui-primary)">Vue UI Library</span> The Intuitive <br> <span class="text-primary">Vue UI Library</span>
</template> </template>
<template #description> <template #description>
{{ page.hero.description }} {{ page.hero.description }}
@@ -81,14 +81,14 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
pause-on-hover pause-on-hover
:overlay="false" :overlay="false"
:ui="{ :ui="{
root: '[--gap:--spacing(4)] [--duration:40s] border-(--ui-border) absolute w-full left-0 border-y lg:border-x lg:border-y-0 lg:w-[calc(50%-6px)] 2xl:w-[320px] lg:flex-col', root: '[--gap:--spacing(4)] [--duration:40s] border-default absolute w-full left-0 border-y lg:border-x lg:border-y-0 lg:w-[calc(50%-6px)] 2xl:w-[320px] lg:flex-col',
content: 'lg:w-auto lg:flex-col lg:animate-[marquee-vertical_var(--duration)_linear_infinite] lg:rtl:animate-[marquee-vertical-rtl_var(--duration)_linear_infinite] lg:h-[fit-content]' content: 'lg:w-auto lg:flex-col lg:animate-[marquee-vertical_var(--duration)_linear_infinite] lg:rtl:animate-[marquee-vertical-rtl_var(--duration)_linear_infinite] lg:h-[fit-content]'
}" }"
> >
<ULink <ULink
v-for="component of components?.slice(0, 10)" v-for="component of components?.slice(0, 10)"
:key="component.path" :key="component.path"
class="relative group/link aspect-video border-(--ui-border) w-[290px] xl:w-[330px] 2xl:w-[320px] 2xl:p-2 2xl:border-y" class="relative group/link aspect-video border-default w-[290px] xl:w-[330px] 2xl:w-[320px] 2xl:p-2 2xl:border-y"
:to="component.path" :to="component.path"
> >
<UColorModeImage <UColorModeImage
@@ -98,7 +98,7 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
width="290" width="290"
height="163" height="163"
format="webp" format="webp"
class="hover:scale-105 lg:hover:scale-110 transition-transform aspect-video w-full border-x lg:border-x-0 lg:border-y border-(--ui-border) 2xl:border-y-0" class="hover:scale-105 lg:hover:scale-110 transition-transform aspect-video w-full border-x lg:border-x-0 lg:border-y border-default 2xl:border-y-0"
loading="lazy" loading="lazy"
/> />
<UBadge color="neutral" variant="outline" size="md" :label="component.title" class="hidden lg:block absolute mx-auto top-4 left-6 xl:left-4 group-hover/link:opacity-100 opacity-0 transition-all duration-300 pointer-events-none -translate-y-2 group-hover/link:translate-y-0" /> <UBadge color="neutral" variant="outline" size="md" :label="component.title" class="hidden lg:block absolute mx-auto top-4 left-6 xl:left-4 group-hover/link:opacity-100 opacity-0 transition-all duration-300 pointer-events-none -translate-y-2 group-hover/link:translate-y-0" />
@@ -110,14 +110,14 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
reverse reverse
:overlay="false" :overlay="false"
:ui="{ :ui="{
root: '[--gap:--spacing(4)] [--duration:40s] border-(--ui-border) absolute w-full mt-[180px] left-0 border-y lg:mt-auto lg:left-auto lg:border-y-0 lg:border-x lg:w-[calc(50%-6px)] 2xl:w-[320px] lg:right-0 lg:flex-col', root: '[--gap:--spacing(4)] [--duration:40s] border-default absolute w-full mt-[180px] left-0 border-y lg:mt-auto lg:left-auto lg:border-y-0 lg:border-x lg:w-[calc(50%-6px)] 2xl:w-[320px] lg:right-0 lg:flex-col',
content: 'lg:w-auto lg:flex-col lg:animate-[marquee-vertical_var(--duration)_linear_infinite] lg:rtl:animate-[marquee-vertical-rtl_var(--duration)_linear_infinite] lg:h-[fit-content] lg:[animation-direction:reverse]' content: 'lg:w-auto lg:flex-col lg:animate-[marquee-vertical_var(--duration)_linear_infinite] lg:rtl:animate-[marquee-vertical-rtl_var(--duration)_linear_infinite] lg:h-[fit-content] lg:[animation-direction:reverse]'
}" }"
> >
<ULink <ULink
v-for="component of components?.slice(10, 20)" v-for="component of components?.slice(10, 20)"
:key="component.path" :key="component.path"
class="relative group/link aspect-video border-(--ui-border) w-[290px] xl:w-[330px] 2xl:w-[320px] 2xl:p-2 2xl:border-y" class="relative group/link aspect-video border-default w-[290px] xl:w-[330px] 2xl:w-[320px] 2xl:p-2 2xl:border-y"
:to="component.path" :to="component.path"
> >
<UColorModeImage <UColorModeImage
@@ -127,7 +127,7 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
width="290" width="290"
height="163" height="163"
format="webp" format="webp"
class="hover:scale-105 lg:hover:scale-110 transition-transform aspect-video w-full border-x lg:border-x-0 lg:border-y border-(--ui-border) 2xl:border-y-0" class="hover:scale-105 lg:hover:scale-110 transition-transform aspect-video w-full border-x lg:border-x-0 lg:border-y border-default 2xl:border-y-0"
loading="lazy" loading="lazy"
/> />
<UBadge color="neutral" variant="outline" size="md" :label="component.title" class="hidden lg:block absolute mx-auto top-4 left-6 xl:left-4 group-hover/link:opacity-100 opacity-0 transition-all duration-300 pointer-events-none -translate-y-2 group-hover/link:translate-y-0" /> <UBadge color="neutral" variant="outline" size="md" :label="component.title" class="hidden lg:block absolute mx-auto top-4 left-6 xl:left-4 group-hover/link:opacity-100 opacity-0 transition-all duration-300 pointer-events-none -translate-y-2 group-hover/link:translate-y-0" />
@@ -168,11 +168,11 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
<UIcon :name="feature.icon" class="size-5 shrink-0" /> <UIcon :name="feature.icon" class="size-5 shrink-0" />
</div> </div>
<div class="flex flex-col"> <div class="flex flex-col">
<h2 class="font-medium text-(--ui-text-highlighted) inline-flex items-center gap-x-1"> <h2 class="font-medium text-highlighted inline-flex items-center gap-x-1">
{{ feature.title }} {{ feature.title }}
<UIcon v-if="feature.to" name="i-lucide-arrow-right" class="size-4 shrink-0 opacity-0 group-hover:opacity-100 transition-all duration-200 -translate-x-1 group-hover:translate-x-0" /> <UIcon v-if="feature.to" name="i-lucide-arrow-right" class="size-4 shrink-0 opacity-0 group-hover:opacity-100 transition-all duration-200 -translate-x-1 group-hover:translate-x-0" />
</h2> </h2>
<p class="text-sm text-(--ui-text-muted)"> <p class="text-sm text-muted">
{{ feature.description }} {{ feature.description }}
</p> </p>
</div> </div>
@@ -215,33 +215,33 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
:links="page.community.links" :links="page.community.links"
orientation="horizontal" orientation="horizontal"
:ui="{ features: 'flex items-center gap-4 lg:gap-8' }" :ui="{ features: 'flex items-center gap-4 lg:gap-8' }"
class="border-b border-(--ui-border)" class="border-b border-default"
> >
<template #features> <template #features>
<li> <li>
<NuxtLink to="https://npm.chart.dev/@nuxt/ui" target="_blank" class="min-w-0"> <NuxtLink to="https://npm.chart.dev/@nuxt/ui" target="_blank" class="min-w-0">
<p class="text-4xl font-semibold text-(--ui-text-highlighted) truncate"> <p class="text-4xl font-semibold text-highlighted truncate">
{{ format(module?.stats?.downloads ?? 0) }}+ {{ format(module?.stats?.downloads ?? 0) }}+
</p> </p>
<p class="text-(--ui-text-muted) text-sm truncate">monthly downloads</p> <p class="text-muted text-sm truncate">monthly downloads</p>
</NuxtLink> </NuxtLink>
</li> </li>
<li> <li>
<NuxtLink to="https://github.com/nuxt/ui" target="_blank" class="min-w-0"> <NuxtLink to="https://github.com/nuxt/ui" target="_blank" class="min-w-0">
<p class="text-4xl font-semibold text-(--ui-text-highlighted) truncate"> <p class="text-4xl font-semibold text-highlighted truncate">
{{ format(module?.stats?.stars ?? 0) }}+ {{ format(module?.stats?.stars ?? 0) }}+
</p> </p>
<p class="text-(--ui-text-muted) text-sm truncate">GitHub stars</p> <p class="text-muted text-sm truncate">GitHub stars</p>
</NuxtLink> </NuxtLink>
</li> </li>
<li> <li>
<NuxtLink to="https://github.com/nuxt/ui/graphs/contributors" target="_blank" class="min-w-0"> <NuxtLink to="https://github.com/nuxt/ui/graphs/contributors" target="_blank" class="min-w-0">
<p class="text-4xl font-semibold text-(--ui-text-highlighted) truncate"> <p class="text-4xl font-semibold text-highlighted truncate">
175+ 175+
</p> </p>
<p class="text-(--ui-text-muted) text-sm truncate">Contributors</p> <p class="text-muted text-sm truncate">Contributors</p>
</NuxtLink> </NuxtLink>
</li> </li>
</template> </template>
@@ -253,10 +253,10 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
<UPageSection :ui="{ container: 'relative !pb-0 overflow-hidden' }"> <UPageSection :ui="{ container: 'relative !pb-0 overflow-hidden' }">
<template #title> <template #title>
Build faster with Nuxt UI <span class="text-(--ui-primary)">Pro</span>. Build faster with Nuxt UI <span class="text-primary">Pro</span>.
</template> </template>
<template #description> <template #description>
A collection of premium Vue components, composables and utils built on top of Nuxt UI. <br> Focused on structure and layout, these <span class="text-(--ui-text)">responsive components</span> are designed to be the perfect <span class="text-(--ui-text)">building blocks for your next idea</span>. A collection of premium Vue components, composables and utils built on top of Nuxt UI. <br> Focused on structure and layout, these <span class="text-default">responsive components</span> are designed to be the perfect <span class="text-default">building blocks for your next idea</span>.
</template> </template>
<template #links> <template #links>
<UButton to="/pro" size="lg"> <UButton to="/pro" size="lg">
@@ -269,8 +269,8 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
<LazyStarsBg /> <LazyStarsBg />
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<div class="relative h-[400px] border border-(--ui-border) bg-(--ui-bg-muted) overflow-hidden border-x-0 -mx-4 sm:-mx-6 lg:mx-0 lg:border-x w-screen lg:w-full"> <div class="relative h-[400px] border border-default bg-muted overflow-hidden border-x-0 -mx-4 sm:-mx-6 lg:mx-0 lg:border-x w-screen lg:w-full">
<UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -left-[100px] -top-[300px] h-[940px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }"> <UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -left-[100px] -top-[300px] h-[940px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }">
<img <img
v-for="i in 4" v-for="i in 4"
@@ -280,7 +280,7 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
height="258" height="258"
loading="lazy" loading="lazy"
:alt="`Nuxt UI Pro Screenshot ${i}`" :alt="`Nuxt UI Pro Screenshot ${i}`"
class="aspect-video border border-(--ui-border) rounded-lg bg-white" class="aspect-video border border-default rounded-lg bg-white"
> >
</UPageMarquee> </UPageMarquee>
<UPageMarquee orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -top-[400px] left-[480px] h-[1160px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }"> <UPageMarquee orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -top-[400px] left-[480px] h-[1160px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }">
@@ -292,7 +292,7 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
height="258" height="258"
loading="lazy" loading="lazy"
:alt="`Nuxt UI Pro Screenshot ${i}`" :alt="`Nuxt UI Pro Screenshot ${i}`"
class="aspect-video border border-(--ui-border) rounded-lg bg-white" class="aspect-video border border-default rounded-lg bg-white"
> >
</UPageMarquee> </UPageMarquee>
<UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: 'hidden md:flex [--duration:40s] absolute w-[460px] -top-[300px] left-[1020px] h-[1060px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }"> <UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: 'hidden md:flex [--duration:40s] absolute w-[460px] -top-[300px] left-[1020px] h-[1060px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }">
@@ -304,7 +304,7 @@ useIntersectionObserver(contributorsRef, ([entry]) => {
height="258" height="258"
loading="lazy" loading="lazy"
:alt="`Nuxt UI Pro Screenshot ${i}`" :alt="`Nuxt UI Pro Screenshot ${i}`"
class="aspect-video border border-(--ui-border) rounded-lg bg-white" class="aspect-video border border-default rounded-lg bg-white"
> >
</UPageMarquee> </UPageMarquee>
</div> </div>

View File

@@ -2,7 +2,7 @@ title: Nuxt UI Pro Pricing
description: Start for free in development mode, then upgrade to a paid plan to unlock the full features of Nuxt UI Pro when you are ready to launch. description: Start for free in development mode, then upgrade to a paid plan to unlock the full features of Nuxt UI Pro when you are ready to launch.
pricing: pricing:
headline: Pricing headline: Pricing
title: Upgrade to Nuxt UI [Pro]{class="text-(--ui-primary)"}. title: Upgrade to Nuxt UI [Pro]{class="text-primary"}.
description: On top of 40+ open source components from Nuxt UI, Pro gives you access to 50+ premium Vue components to create beautiful & responsive Nuxt applications in minutes. It includes all primitives to build landing pages, documentations, blogs, dashboards or entire SaaS products. description: On top of 40+ open source components from Nuxt UI, Pro gives you access to 50+ premium Vue components to create beautiful & responsive Nuxt applications in minutes. It includes all primitives to build landing pages, documentations, blogs, dashboards or entire SaaS products.
freePlan: freePlan:
title: Free in development title: Free in development

View File

@@ -1,8 +1,8 @@
title: Build faster with Nuxt UI Pro. title: Build faster with Nuxt UI Pro.
description: A collection of premium Vue components, composables and utils built on top of Nuxt UI, oriented on structure and layout and designed to be used as building blocks for your application. description: A collection of premium Vue components, composables and utils built on top of Nuxt UI, oriented on structure and layout and designed to be used as building blocks for your application.
hero: hero:
title: Build faster with Nuxt UI [Pro]{class="text-(--ui-primary)"}. title: Build faster with Nuxt UI [Pro]{class="text-primary"}.
description: A collection of premium Vue components, composables and utils built on top of Nuxt UI. :br Focused on structure and layout, these [responsive components]{class="text-(--ui-text)"} are designed to be the perfect [building blocks for your next idea]{class="text-(--ui-text)"}. description: A collection of premium Vue components, composables and utils built on top of Nuxt UI. :br Focused on structure and layout, these [responsive components]{class="text-default"} are designed to be the perfect [building blocks for your next idea]{class="text-default"}.
links: links:
- label: Buy a license - label: Buy a license
size: xl size: xl
@@ -62,7 +62,7 @@ testimonial:
# avatar: # avatar:
# src: https://github.com/benjamincanac.png # src: https://github.com/benjamincanac.png
mainSection: mainSection:
title: Meet the [Pro Components]{class="text-(--ui-primary)"}. title: Meet the [Pro Components]{class="text-primary"}.
description: Code with 50+ components and sections of Nuxt UI Pro to build your next application by reducing the amount of code you need to write. description: Code with 50+ components and sections of Nuxt UI Pro to build your next application by reducing the amount of code you need to write.
sections: sections:
- title: The freedom to build anything - title: The freedom to build anything

View File

@@ -1,7 +1,7 @@
title: Official Nuxt UI Pro Templates title: Official Nuxt UI Pro Templates
description: 'Ready to use templates powered by our premium Vue components and Nuxt Content. The templates are responsive, accessible and easy to customize so you can get started in no time.' description: 'Ready to use templates powered by our premium Vue components and Nuxt Content. The templates are responsive, accessible and easy to customize so you can get started in no time.'
hero: hero:
title: Ship [in minutes]{.text-(--ui-primary)} with :br Nuxt UI Pro Templates title: Ship [in minutes]{.text-primary} with :br Nuxt UI Pro Templates
description: 'Ready to use templates powered by our premium Vue components and Nuxt Content.<br class="hidden lg:block"> The templates are responsive, accessible and easy to customize so you can get started in no time.' description: 'Ready to use templates powered by our premium Vue components and Nuxt Content.<br class="hidden lg:block"> The templates are responsive, accessible and easy to customize so you can get started in no time.'
navigation: false navigation: false
links: links:

View File

@@ -73,9 +73,9 @@ onMounted(() => {
<UPageHero headline="License Activation" :title="title" :description="description" :ui="{ container: 'relative overflow-hidden', wrapper: 'lg:px-12', description: 'text-pretty' }"> <UPageHero headline="License Activation" :title="title" :description="description" :ui="{ container: 'relative overflow-hidden', wrapper: 'lg:px-12', description: 'text-pretty' }">
<LazyStarsBg /> <LazyStarsBg />
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<div class="px-4 py-10 lg:border border-(--ui-border) bg-(--ui-bg)"> <div class="px-4 py-10 lg:border border-default bg-default">
<div class="max-w-xl mx-auto"> <div class="max-w-xl mx-auto">
<UForm <UForm
:schema="schema" :schema="schema"

View File

@@ -35,9 +35,9 @@ useSeoMeta({
<LazyStarsBg /> <LazyStarsBg />
<Motion as-child :initial="{ height: 0 }" :animate="{ height: 'auto' }" :transition="{ delay: 0.2, duration: 1 }"> <Motion as-child :initial="{ height: 0 }" :animate="{ height: 'auto' }" :transition="{ delay: 0.2, duration: 1 }">
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
</Motion> </Motion>
<div class="relative h-[400px] border border-(--ui-border) bg-(--ui-bg-muted) overflow-hidden border-x-0 -mx-4 sm:-mx-6 lg:mx-0 lg:border-x w-screen lg:w-full"> <div class="relative h-[400px] border border-default bg-muted overflow-hidden border-x-0 -mx-4 sm:-mx-6 lg:mx-0 lg:border-x w-screen lg:w-full">
<UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -left-[100px] -top-[300px] h-[940px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }"> <UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -left-[100px] -top-[300px] h-[940px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }">
<img <img
v-for="i in 4" v-for="i in 4"
@@ -46,7 +46,7 @@ useSeoMeta({
width="460" width="460"
height="258" height="258"
:alt="`Nuxt UI Pro Screenshot ${i}`" :alt="`Nuxt UI Pro Screenshot ${i}`"
class="aspect-video border border-(--ui-border) rounded-lg bg-white" class="aspect-video border border-default rounded-lg bg-white"
> >
</UPageMarquee> </UPageMarquee>
<UPageMarquee orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -top-[400px] left-[480px] h-[1160px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }"> <UPageMarquee orientation="vertical" :overlay="false" :ui="{ root: '[--duration:40s] absolute w-[460px] -top-[400px] left-[480px] h-[1160px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }">
@@ -57,7 +57,7 @@ useSeoMeta({
width="460" width="460"
height="258" height="258"
:alt="`Nuxt UI Pro Screenshot ${i}`" :alt="`Nuxt UI Pro Screenshot ${i}`"
class="aspect-video border border-(--ui-border) rounded-lg bg-white" class="aspect-video border border-default rounded-lg bg-white"
> >
</UPageMarquee> </UPageMarquee>
<UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: 'hidden md:flex [--duration:40s] absolute w-[460px] -top-[300px] left-[1020px] h-[1060px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }"> <UPageMarquee reverse orientation="vertical" :overlay="false" :ui="{ root: 'hidden md:flex [--duration:40s] absolute w-[460px] -top-[300px] left-[1020px] h-[1060px] transform-3d rotate-x-55 rotate-y-0 rotate-z-30' }">
@@ -68,7 +68,7 @@ useSeoMeta({
width="460" width="460"
height="258" height="258"
:alt="`Nuxt UI Pro Screenshot ${i}`" :alt="`Nuxt UI Pro Screenshot ${i}`"
class="aspect-video border border-(--ui-border) rounded-lg bg-white" class="aspect-video border border-default rounded-lg bg-white"
> >
</UPageMarquee> </UPageMarquee>
</div> </div>
@@ -101,10 +101,10 @@ useSeoMeta({
container: 'relative', container: 'relative',
wrapper: 'sm:px-8' wrapper: 'sm:px-8'
}" }"
class="border-t border-(--ui-border)" class="border-t border-default"
> >
<Motion as-child :initial="{ height: 0 }" :while-in-view="{ height: 'auto' }" :transition="{ delay: 0.4, duration: 1 }"> <Motion as-child :initial="{ height: 0 }" :while-in-view="{ height: 'auto' }" :transition="{ delay: 0.4, duration: 1 }">
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
</Motion> </Motion>
</UPageSection> </UPageSection>
@@ -116,7 +116,7 @@ useSeoMeta({
wrapper: 'grid grid-cols-1 lg:grid-cols-2', wrapper: 'grid grid-cols-1 lg:grid-cols-2',
description: 'lg:mt-0' }" description: 'lg:mt-0' }"
orientation="horizontal" orientation="horizontal"
class="rounded-none border-t border-(--ui-border) bg-gradient-to-b from-(--ui-bg-elevated)/50 to-(--ui-bg)" class="rounded-none border-t border-default bg-gradient-to-b from-elevated/50 to-default"
> >
<template #title> <template #title>
<MDC :value="page.mainSection.title" tag="span" unwrap="p" cache-key="pro-main-section-title" /> <MDC :value="page.mainSection.title" tag="span" unwrap="p" cache-key="pro-main-section-title" />
@@ -134,7 +134,7 @@ useSeoMeta({
:reverse="section.reverse" :reverse="section.reverse"
:features="section.features" :features="section.features"
orientation="horizontal" orientation="horizontal"
:class="{ 'border-b border-(--ui-border)': index === page.sections.length - 1 }" :class="{ 'border-b border-default': index === page.sections.length - 1 }"
:ui="{ :ui="{
container: index === 0 ? 'pb-0 sm:pb-0 lg:pb-0 py-16 sm:py-16 lg:py-16' : '' container: index === 0 ? 'pb-0 sm:pb-0 lg:pb-0 py-16 sm:py-16 lg:py-16' : ''
}" }"
@@ -145,10 +145,10 @@ useSeoMeta({
<UPageSection <UPageSection
id="templates" id="templates"
v-bind="page.templates" v-bind="page.templates"
class="overflow-hidden border-x border-(--ui-border)" class="overflow-hidden border-x border-default"
:ui="{ container: 'relative' }" :ui="{ container: 'relative' }"
> >
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<UCarousel <UCarousel
v-slot="{ item }" v-slot="{ item }"
loop loop
@@ -160,7 +160,7 @@ useSeoMeta({
:ui="{ :ui="{
item: 'basis-1/2', item: 'basis-1/2',
container: 'py-2', container: 'py-2',
viewport: 'border-x border-(--ui-border)', viewport: 'border-x border-default',
arrows: 'hidden 2xl:block' arrows: 'hidden 2xl:block'
}" }"
> >
@@ -181,7 +181,7 @@ useSeoMeta({
:light="item.thumbnail.light" :light="item.thumbnail.light"
:dark="item.thumbnail.dark" :dark="item.thumbnail.dark"
:alt="item.title" :alt="item.title"
class="rounded-lg w-full border border-(--ui-border) aspect-video" class="rounded-lg w-full border border-default aspect-video"
loading="lazy" loading="lazy"
/> />
</UPageCard> </UPageCard>

View File

@@ -29,13 +29,13 @@ useSeoMeta({
<LazyStarsBg /> <LazyStarsBg />
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<div class="flex flex-col bg-(--ui-bg) gap-8 lg:gap-0"> <div class="flex flex-col bg-default gap-8 lg:gap-0">
<UPricingPlan <UPricingPlan
v-bind="page.pricing.freePlan" v-bind="page.pricing.freePlan"
variant="naked" variant="naked"
class="lg:rounded-none border-x border-(--ui-border) border-t border-b lg:border-b-0" class="lg:rounded-none border-x border-default border-t border-b lg:border-b-0"
/> />
<UPricingPlans compact> <UPricingPlans compact>
<UPricingPlan <UPricingPlan
@@ -48,7 +48,7 @@ useSeoMeta({
:billing-period="plan.billing_period" :billing-period="plan.billing_period"
:billing-cycle="plan.billing_cycle" :billing-cycle="plan.billing_cycle"
:variant="plan.highlight ? 'soft' : 'outline'" :variant="plan.highlight ? 'soft' : 'outline'"
:class="['lg:rounded-none', { 'border-2 lg:border lg:border-x-0 border-(--ui-primary) lg:border-(--ui-border)': plan.highlight }]" :class="['lg:rounded-none', { 'border-2 lg:border lg:border-x-0 border-primary lg:border-default': plan.highlight }]"
:features="plan.features" :features="plan.features"
:button="plan.button" :button="plan.button"
/> />
@@ -58,12 +58,12 @@ useSeoMeta({
variant="naked" variant="naked"
:billing-period="page.pricing.figma.billing_period" :billing-period="page.pricing.figma.billing_period"
:billing-cycle="page.pricing.figma.billing_cycle" :billing-cycle="page.pricing.figma.billing_cycle"
class="lg:rounded-none border lg:border-y-0 border-(--ui-border)" class="lg:rounded-none border lg:border-y-0 border-default"
> >
<template #features> <template #features>
<li v-for="(feature, index) in page.pricing.figma.features" :key="index" class="flex items-center gap-2 min-w-0"> <li v-for="(feature, index) in page.pricing.figma.features" :key="index" class="flex items-center gap-2 min-w-0">
<UIcon name="i-lucide-circle-check" class="size-5 text-(--ui-primary) shrink-0" /> <UIcon name="i-lucide-circle-check" class="size-5 text-primary shrink-0" />
<MDC :value="feature" unwrap="p" class="text-sm truncate text-(--ui-text-toned)" :cache-key="`pro-pricing-figma-feature-${index}`" /> <MDC :value="feature" unwrap="p" class="text-sm truncate text-toned" :cache-key="`pro-pricing-figma-feature-${index}`" />
</li> </li>
</template> </template>
</UPricingPlan> </UPricingPlan>
@@ -73,7 +73,7 @@ useSeoMeta({
<UPageSection <UPageSection
id="testimonials" id="testimonials"
v-bind="page.testimonials" v-bind="page.testimonials"
class="border-y border-(--ui-border)" class="border-y border-default"
> >
<UPageMarquee pause-on-hover :ui="{ root: '[--duration:40s]' }"> <UPageMarquee pause-on-hover :ui="{ root: '[--duration:40s]' }">
<img <img
@@ -110,7 +110,7 @@ useSeoMeta({
class="scroll-mt-(--ui-header-height)" class="scroll-mt-(--ui-header-height)"
:ui="{ container: 'relative' }" :ui="{ container: 'relative' }"
> >
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<UPageAccordion <UPageAccordion
multiple multiple
:items="(page.faq.items as any[])" :items="(page.faq.items as any[])"

View File

@@ -20,7 +20,7 @@ useSeoMeta({
<UPageHero :links="page.links" :ui="{ container: 'relative' }"> <UPageHero :links="page.links" :ui="{ container: 'relative' }">
<LazyStarsBg /> <LazyStarsBg />
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<template #title> <template #title>
<MDC :value="page.hero.title" unwrap="p" cache-key="pro-templates-hero-title" /> <MDC :value="page.hero.title" unwrap="p" cache-key="pro-templates-hero-title" />
@@ -38,10 +38,10 @@ useSeoMeta({
:links="template.links" :links="template.links"
:features="template.features" :features="template.features"
orientation="horizontal" orientation="horizontal"
class="lg:border-t border-(--ui-border)" class="lg:border-t border-default"
:ui="{ :ui="{
title: 'lg:text-4xl', title: 'lg:text-4xl',
wrapper: 'lg:py-16 lg:border-r border-(--ui-border) order-last lg:pr-16', wrapper: 'lg:py-16 lg:border-r border-default order-last lg:pr-16',
container: 'lg:py-0', container: 'lg:py-0',
links: 'gap-x-3' links: 'gap-x-3'
}" }"
@@ -50,12 +50,12 @@ useSeoMeta({
<MDC :value="template.description" unwrap="p" :cache-key="`pro-templates-${index}-description`" /> <MDC :value="template.description" unwrap="p" :cache-key="`pro-templates-${index}-description`" />
</template> </template>
<div class="lg:border-x border-(--ui-border) h-full flex items-center lg:bg-(--ui-bg-muted)/20"> <div class="lg:border-x border-default h-full flex items-center lg:bg-muted/20">
<Motion class="flex-1" :initial="{ opacity: 0, transform: 'translateY(10px)' }" :while-in-view="{ opacity: 1, transform: 'translateY(0px)' }" :in-view-options="{ once: true }" :transition="{ duration: 0.5, delay: 0.2 }"> <Motion class="flex-1" :initial="{ opacity: 0, transform: 'translateY(10px)' }" :while-in-view="{ opacity: 1, transform: 'translateY(0px)' }" :in-view-options="{ once: true }" :transition="{ duration: 0.5, delay: 0.2 }">
<UColorModeImage <UColorModeImage
v-if="template.thumbnail" v-if="template.thumbnail"
v-bind="template.thumbnail" v-bind="template.thumbnail"
class="w-full h-auto border lg:border-y lg:border-x-0 border-(--ui-border) rounded-sm lg:rounded-none" class="w-full h-auto border lg:border-y lg:border-x-0 border-default rounded-sm lg:rounded-none"
:alt="`Template ${index} thumbnail`" :alt="`Template ${index} thumbnail`"
width="656" width="656"
height="369" height="369"

View File

@@ -29,19 +29,19 @@ defineOgImageComponent('Docs', {
}" }"
> >
<template #top> <template #top>
<div class="absolute z-[-1] rounded-full bg-(--ui-primary) blur-[300px] size-60 sm:size-80 transform -translate-x-1/2 left-1/2 -translate-y-80" /> <div class="absolute z-[-1] rounded-full bg-primary blur-[300px] size-60 sm:size-80 transform -translate-x-1/2 left-1/2 -translate-y-80" />
</template> </template>
<LazyStarsBg /> <LazyStarsBg />
<div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-(--ui-border) inset-0 mx-4 sm:mx-6 lg:mx-8" /> <div aria-hidden="true" class="hidden lg:block absolute z-[-1] border-x border-default inset-0 mx-4 sm:mx-6 lg:mx-8" />
<div class="border-l border-t border-(--ui-border)"> <div class="border-l border-t border-default">
<ul class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 items-start justify-center divide-y divide-x divide-(--ui-border)"> <ul class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 items-start justify-center divide-y divide-x divide-default">
<li <li
v-for="item in page.items" v-for="item in page.items"
:key="item.name" :key="item.name"
class="group relative flex items-center justify-center flex-1 size-full p-2 last:border-r last:border-b border-(--ui-border) overflow-hidden" class="group relative flex items-center justify-center flex-1 size-full p-2 last:border-r last:border-b border-default overflow-hidden"
> >
<NuxtLink class="inset-0 absolute" :to="item.url" target="_blank"> <NuxtLink class="inset-0 absolute" :to="item.url" target="_blank">
<span class="sr-only">Go to {{ item.name }}</span> <span class="sr-only">Go to {{ item.name }}</span>

View File

@@ -41,7 +41,7 @@ const icons = {
:ui="{ title: 'text-balance', container: 'relative' }" :ui="{ title: 'text-balance', container: 'relative' }"
> >
<template #top> <template #top>
<div class="absolute z-[-1] rounded-full bg-(--ui-primary) blur-[300px] size-60 sm:size-80 transform -translate-x-1/2 left-1/2 -translate-y-80" /> <div class="absolute z-[-1] rounded-full bg-primary blur-[300px] size-60 sm:size-80 transform -translate-x-1/2 left-1/2 -translate-y-80" />
</template> </template>
<LazyStarsBg /> <LazyStarsBg />
@@ -58,7 +58,7 @@ const icons = {
container: 'gap-y-4 lg:p-8', container: 'gap-y-4 lg:p-8',
leading: 'flex justify-center', leading: 'flex justify-center',
title: 'text-center', title: 'text-center',
description: 'text-center text-(--ui-text-muted)' description: 'text-center text-muted'
}" }"
variant="subtle" variant="subtle"
> >
@@ -126,7 +126,7 @@ const icons = {
container: 'gap-y-2', container: 'gap-y-2',
leading: 'flex justify-center', leading: 'flex justify-center',
title: 'text-center', title: 'text-center',
description: 'text-center text-(--ui-text-muted)' description: 'text-center text-muted'
}" }"
> >
<template #leading> <template #leading>

View File

@@ -195,10 +195,10 @@ You can also use the new [design tokens](/getting-started/theme#neutral-palette)
```diff ```diff
<template> <template>
- <p class="text-gray-500 dark:text-gray-400" /> - <p class="text-gray-500 dark:text-gray-400" />
+ <p class="text-(--ui-text-muted)" /> + <p class="text-muted" />
- <p class="text-gray-900 dark:text-white" /> - <p class="text-gray-900 dark:text-white" />
+ <p class="text-(--ui-text-highlighted)" /> + <p class="text-highlighted" />
</template> </template>
``` ```
:: ::

View File

@@ -118,17 +118,7 @@ Learn more about automatic content detection in the detecting classes in source
## Design system ## Design system
Nuxt UI extends Tailwind CSS's theming capabilities, providing a flexible design system with pre-configured color aliases and CSS variables. This allows for easy customization and quick adaptation of the UI to your brand's aesthetic. Nuxt UI extends Tailwind CSS's theming capabilities, providing a flexible design system with pre-configured color aliases based on [Tailwind CSS colors](https://tailwindcss.com/docs/customizing-colors#color-palette-reference). This allows for easy customization and quick adaptation of the UI to your brand's aesthetic.
### Colors
::framework-only
#nuxt
Nuxt UI leverages Nuxt [App Config](https://nuxt.com/docs/guide/directory-structure/app-config#app-config-file) to provide customizable color aliases based on [Tailwind CSS colors](https://tailwindcss.com/docs/customizing-colors#color-palette-reference):
#vue
Nuxt UI leverages Vite config to provide customizable color aliases based on [Tailwind CSS colors](https://tailwindcss.com/docs/customizing-colors#color-palette-reference):
::
| Color | Default | Description | | Color | Default | Description |
| --- | --- | --- | | --- | --- | --- |
@@ -140,10 +130,27 @@ Nuxt UI leverages Vite config to provide customizable color aliases based on [Ta
| `error`{color="error"} | `red` | Used for form error validation states. | | `error`{color="error"} | `red` | Used for form error validation states. |
| `neutral` | `slate` | Neutral color for backgrounds, text, etc. | | `neutral` | `slate` | Neutral color for backgrounds, text, etc. |
These colors are used to style the components but also to generate the `color` props:
::component-code{slug="button"}
---
props:
color: primary
slots:
default: Button
---
::
::note
Try the :prose-icon{name="i-lucide-swatch-book" class="text-primary"} theme picker in the header above to change `primary` and `neutral` colors.
::
### Configuration
::framework-only ::framework-only
#nuxt #nuxt
:::div :::div
You can configure these color aliases at runtime in your `app.config.ts` file under the `ui.colors` key, allowing for dynamic theme customization without requiring an application rebuild: You can configure these color aliases at runtime in your [`app.config.ts`](https://nuxt.com/docs/guide/directory-structure/app-config#app-config-file) file under the `ui.colors` key, allowing for dynamic theme customization without requiring an application rebuild:
```ts [app.config.ts] ```ts [app.config.ts]
export default defineAppConfig({ export default defineAppConfig({
@@ -222,27 +229,14 @@ export default defineConfig({
:: ::
::note ### Extend colors
Try the :prose-icon{name="i-lucide-swatch-book" class="text-(--ui-primary)"} theme picker in the header above to change `primary` and `neutral` colors.
::
These colors are used to style the components but also to generate the `color` variants:
::component-code{slug="button"}
---
props:
color: primary
slots:
default: Button
---
::
::framework-only ::framework-only
#nuxt #nuxt
:::tip :::div
You can add you own dynamic color aliases in your `app.config.ts`, you just have to make sure to define them in the [`ui.theme.colors`](/getting-started/installation/nuxt#themecolors) option in your `nuxt.config.ts` file. You can add you own dynamic color aliases in your `app.config.ts`, you just have to make sure to define them in the [`ui.theme.colors`](/getting-started/installation/nuxt#themecolors) option in your `nuxt.config.ts` file:
```ts [app.config.ts] ```ts [app.config.ts]{4}
export default defineAppConfig({ export default defineAppConfig({
ui: { ui: {
colors: { colors: {
@@ -252,11 +246,19 @@ export default defineAppConfig({
}) })
``` ```
```ts [nuxt.config.ts] ```ts [nuxt.config.ts]{7}
export default defineNuxtConfig({ export default defineNuxtConfig({
ui: { ui: {
theme: { theme: {
colors: ['primary', 'secondary', 'tertiary', 'info', 'success', 'warning', 'error'] colors: [
'primary',
'secondary',
'tertiary',
'info',
'success',
'warning',
'error'
]
} }
} }
}) })
@@ -266,9 +268,9 @@ export default defineNuxtConfig({
#vue #vue
:::tip :::div
You can add you own dynamic color aliases in your `vite.config.ts`, you just have to make sure to also define them in the [`theme.colors`](/getting-started/installation/vue#themecolors) option of the `ui` plugin. You can add you own dynamic color aliases in your `vite.config.ts`, you just have to make sure to also define them in the [`theme.colors`](/getting-started/installation/vue#themecolors) option of the `ui` plugin:
::::module-only ::::module-only
@@ -276,7 +278,7 @@ You can add you own dynamic color aliases in your `vite.config.ts`, you just hav
:::::div :::::div
```ts [vite.config.ts] ```ts [vite.config.ts]{11,18}
import { defineConfig } from 'vite' import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue' import vue from '@vitejs/plugin-vue'
import ui from '@nuxt/ui/vite' import ui from '@nuxt/ui/vite'
@@ -291,7 +293,15 @@ export default defineConfig({
} }
}, },
theme: { theme: {
colors: ['primary', 'secondary', 'tertiary', 'info', 'success', 'warning', 'error'] colors: [
'primary',
'secondary',
'tertiary',
'info',
'success',
'warning',
'error'
]
} }
}) })
] ]
@@ -304,7 +314,7 @@ export default defineConfig({
:::::div :::::div
```ts [vite.config.ts] ```ts [vite.config.ts]{11,18}
import { defineConfig } from 'vite' import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue' import vue from '@vitejs/plugin-vue'
import uiPro from '@nuxt/ui-pro/vite' import uiPro from '@nuxt/ui-pro/vite'
@@ -316,10 +326,18 @@ export default defineConfig({
ui: { ui: {
colors: { colors: {
tertiary: 'indigo' tertiary: 'indigo'
}, }
}, },
theme: { theme: {
colors: ['primary', 'secondary', 'tertiary', 'info', 'success', 'warning', 'error'] colors: [
'primary',
'secondary',
'tertiary',
'info',
'success',
'warning',
'error'
]
} }
}) })
] ]
@@ -334,13 +352,13 @@ export default defineConfig({
:: ::
### Tokens ## CSS Variables
Nuxt UI leverages a robust system of CSS variables as design tokens to ensure consistent and flexible component styling. These tokens form the foundation of the theming system, offering smooth support for both light and dark modes. Nuxt UI leverages a robust system of CSS variables as design tokens to ensure consistent and flexible component styling. These tokens form the foundation of the theming system, offering smooth support for both light and dark modes.
#### Color Shades ### Colors
Nuxt UI automatically creates a CSS variable for each color alias you define which represent the default shade used in both light and dark modes: Nuxt UI provides a CSS variable for each color alias you define which represent the default shade used in both light and dark modes:
::code-group ::code-group
@@ -368,16 +386,125 @@ Nuxt UI automatically creates a CSS variable for each color alias you define whi
:: ::
::note These CSS variables are defined in Tailwind CSS's `@theme` so you can use them as classes:
You can use these variables in your Tailwind CSS classes in two ways:
- Using CSS variable syntax: `text-(--ui-primary)` or `bg-(--ui-primary)`
- Using color alias: `text-primary` or `bg-primary`
Both approaches will automatically adapt to the current color scheme, ensuring consistent styling across light and dark modes. ::code-preview
[Primary]{class="text-primary text-sm px-4"}
[Secondary]{class="text-secondary text-sm px-4"}
[Success]{class="text-success text-sm px-4"}
[Info]{class="text-info text-sm px-4"}
[Warning]{class="text-warning text-sm px-4"}
[Error]{class="text-error text-sm px-4"}
#code
```vue
<template>
<span class="text-primary">Primary</span>
<span class="text-secondary">Secondary</span>
<span class="text-success">Success</span>
<span class="text-info">Info</span>
<span class="text-warning">Warning</span>
<span class="text-error">Error</span>
</template>
```
:: ::
::tip ::note
You can change which shade is used for each color on light and dark mode: This is how the `@theme` is generated for each color alias:
:::code-collapse{class="[&>div]:!my-0"}
```scss
@theme default {
--color-primary: var(--ui-primary);
--color-primary-50: var(--ui-color-primary-50);
--color-primary-100: var(--ui-color-primary-100);
--color-primary-200: var(--ui-color-primary-200);
--color-primary-300: var(--ui-color-primary-300);
--color-primary-400: var(--ui-color-primary-400);
--color-primary-500: var(--ui-color-primary-500);
--color-primary-600: var(--ui-color-primary-600);
--color-primary-700: var(--ui-color-primary-700);
--color-primary-800: var(--ui-color-primary-800);
--color-primary-900: var(--ui-color-primary-900);
--color-primary-950: var(--ui-color-primary-950);
--color-secondary: var(--ui-secondary);
--color-secondary-50: var(--ui-color-secondary-50);
--color-secondary-100: var(--ui-color-secondary-100);
--color-secondary-200: var(--ui-color-secondary-200);
--color-secondary-300: var(--ui-color-secondary-300);
--color-secondary-400: var(--ui-color-secondary-400);
--color-secondary-500: var(--ui-color-secondary-500);
--color-secondary-600: var(--ui-color-secondary-600);
--color-secondary-700: var(--ui-color-secondary-700);
--color-secondary-800: var(--ui-color-secondary-800);
--color-secondary-900: var(--ui-color-secondary-900);
--color-secondary-950: var(--ui-color-secondary-950);
--color-success: var(--ui-success);
--color-success-50: var(--ui-color-success-50);
--color-success-100: var(--ui-color-success-100);
--color-success-200: var(--ui-color-success-200);
--color-success-300: var(--ui-color-success-300);
--color-success-400: var(--ui-color-success-400);
--color-success-500: var(--ui-color-success-500);
--color-success-600: var(--ui-color-success-600);
--color-success-700: var(--ui-color-success-700);
--color-success-800: var(--ui-color-success-800);
--color-success-900: var(--ui-color-success-900);
--color-success-950: var(--ui-color-success-950);
--color-info: var(--ui-info);
--color-info-50: var(--ui-color-info-50);
--color-info-100: var(--ui-color-info-100);
--color-info-200: var(--ui-color-info-200);
--color-info-300: var(--ui-color-info-300);
--color-info-400: var(--ui-color-info-400);
--color-info-500: var(--ui-color-info-500);
--color-info-600: var(--ui-color-info-600);
--color-info-700: var(--ui-color-info-700);
--color-info-800: var(--ui-color-info-800);
--color-info-900: var(--ui-color-info-900);
--color-info-950: var(--ui-color-info-950);
--color-warning: var(--ui-warning);
--color-warning-50: var(--ui-color-warning-50);
--color-warning-100: var(--ui-color-warning-100);
--color-warning-200: var(--ui-color-warning-200);
--color-warning-300: var(--ui-color-warning-300);
--color-warning-400: var(--ui-color-warning-400);
--color-warning-500: var(--ui-color-warning-500);
--color-warning-600: var(--ui-color-warning-600);
--color-warning-700: var(--ui-color-warning-700);
--color-warning-800: var(--ui-color-warning-800);
--color-warning-900: var(--ui-color-warning-900);
--color-warning-950: var(--ui-color-warning-950);
--color-error: var(--ui-error);
--color-error-50: var(--ui-color-error-50);
--color-error-100: var(--ui-color-error-100);
--color-error-200: var(--ui-color-error-200);
--color-error-300: var(--ui-color-error-300);
--color-error-400: var(--ui-color-error-400);
--color-error-500: var(--ui-color-error-500);
--color-error-600: var(--ui-color-error-600);
--color-error-700: var(--ui-color-error-700);
--color-error-800: var(--ui-color-error-800);
--color-error-900: var(--ui-color-error-900);
--color-error-950: var(--ui-color-error-950);
--color-neutral-50: var(--ui-color-neutral-50);
--color-neutral-100: var(--ui-color-neutral-100);
--color-neutral-200: var(--ui-color-neutral-200);
--color-neutral-300: var(--ui-color-neutral-300);
--color-neutral-400: var(--ui-color-neutral-400);
--color-neutral-500: var(--ui-color-neutral-500);
--color-neutral-600: var(--ui-color-neutral-600);
--color-neutral-700: var(--ui-color-neutral-700);
--color-neutral-800: var(--ui-color-neutral-800);
--color-neutral-900: var(--ui-color-neutral-900);
--color-neutral-950: var(--ui-color-neutral-950);
}
```
:::
::
You can change which shade is used for each color on light and dark mode in your `main.css` file:
::module-only ::module-only
#ui #ui
@@ -417,10 +544,6 @@ You can change which shade is used for each color on light and dark mode:
::: :::
:: ::
::
#### Black as Primary Color
::framework-only ::framework-only
#nuxt #nuxt
:::p :::p
@@ -471,7 +594,7 @@ You cannot set `primary: 'black'`{lang="ts-type"} in your [`vite.config.ts`](#co
::: :::
:: ::
#### Neutral Palette ### Neutral
Nuxt UI provides a comprehensive set of design tokens for the `neutral` color palette, ensuring consistent and accessible UI styling across both light and dark modes. These tokens offer fine-grained control over text, background, and border colors: Nuxt UI provides a comprehensive set of design tokens for the `neutral` color palette, ensuring consistent and accessible UI styling across both light and dark modes. These tokens offer fine-grained control over text, background, and border colors:
@@ -479,89 +602,153 @@ Nuxt UI provides a comprehensive set of design tokens for the `neutral` color pa
```css [Light] ```css [Light]
:root { :root {
/* Least prominent text */
--ui-text-dimmed: var(--ui-color-neutral-400); --ui-text-dimmed: var(--ui-color-neutral-400);
/* Slightly muted text */
--ui-text-muted: var(--ui-color-neutral-500); --ui-text-muted: var(--ui-color-neutral-500);
/* Moderately prominent text */
--ui-text-toned: var(--ui-color-neutral-600); --ui-text-toned: var(--ui-color-neutral-600);
/* Default text color */
--ui-text: var(--ui-color-neutral-700); --ui-text: var(--ui-color-neutral-700);
/* Most prominent text */
--ui-text-highlighted: var(--ui-color-neutral-900); --ui-text-highlighted: var(--ui-color-neutral-900);
--ui-text-inverted: var(--color-white);
/* Main background color */
--ui-bg: var(--color-white); --ui-bg: var(--color-white);
/* Subtle background */
--ui-bg-muted: var(--ui-color-neutral-50); --ui-bg-muted: var(--ui-color-neutral-50);
/* Slightly elevated background */
--ui-bg-elevated: var(--ui-color-neutral-100); --ui-bg-elevated: var(--ui-color-neutral-100);
/* More prominent background */
--ui-bg-accented: var(--ui-color-neutral-200); --ui-bg-accented: var(--ui-color-neutral-200);
/* Inverted background color */
--ui-bg-inverted: var(--ui-color-neutral-900); --ui-bg-inverted: var(--ui-color-neutral-900);
/* Default border color */
--ui-border: var(--ui-color-neutral-200); --ui-border: var(--ui-color-neutral-200);
/* Subtle border */
--ui-border-muted: var(--ui-color-neutral-200); --ui-border-muted: var(--ui-color-neutral-200);
/* More prominent border */
--ui-border-accented: var(--ui-color-neutral-300); --ui-border-accented: var(--ui-color-neutral-300);
/* Inverted border color */
--ui-border-inverted: var(--ui-color-neutral-900); --ui-border-inverted: var(--ui-color-neutral-900);
} }
``` ```
```css [Dark] ```css [Dark]
.dark { .dark {
/* Least prominent text */
--ui-text-dimmed: var(--ui-color-neutral-500); --ui-text-dimmed: var(--ui-color-neutral-500);
/* Slightly muted text */
--ui-text-muted: var(--ui-color-neutral-400); --ui-text-muted: var(--ui-color-neutral-400);
/* Moderately prominent text */
--ui-text-toned: var(--ui-color-neutral-300); --ui-text-toned: var(--ui-color-neutral-300);
/* Default text color */
--ui-text: var(--ui-color-neutral-200); --ui-text: var(--ui-color-neutral-200);
/* Most prominent text */
--ui-text-highlighted: var(--color-white); --ui-text-highlighted: var(--color-white);
--ui-text-inverted: var(--ui-color-neutral-900);
/* Main background color */
--ui-bg: var(--ui-color-neutral-900); --ui-bg: var(--ui-color-neutral-900);
/* Subtle background */
--ui-bg-muted: var(--ui-color-neutral-800); --ui-bg-muted: var(--ui-color-neutral-800);
/* Slightly elevated background */
--ui-bg-elevated: var(--ui-color-neutral-800); --ui-bg-elevated: var(--ui-color-neutral-800);
/* More prominent background */
--ui-bg-accented: var(--ui-color-neutral-700); --ui-bg-accented: var(--ui-color-neutral-700);
/* Inverted background color */
--ui-bg-inverted: var(--color-white); --ui-bg-inverted: var(--color-white);
/* Default border color */
--ui-border: var(--ui-color-neutral-800); --ui-border: var(--ui-color-neutral-800);
/* Subtle border */
--ui-border-muted: var(--ui-color-neutral-700); --ui-border-muted: var(--ui-color-neutral-700);
/* More prominent border */
--ui-border-accented: var(--ui-color-neutral-700); --ui-border-accented: var(--ui-color-neutral-700);
/* Inverted border color */
--ui-border-inverted: var(--color-white); --ui-border-inverted: var(--color-white);
} }
``` ```
:: ::
::note These CSS variables are defined in Tailwind CSS's `@theme` so you can use them as classes:
Nuxt UI automatically applies a text and background color on the `<body>` element of your app:
```css ::code-preview
body { [Dimmed]{class="text-dimmed text-sm px-4 py-1.5 inline-block rounded-md"}
@apply antialiased text-(--ui-text) bg-(--ui-bg); [Muted]{class="text-muted text-sm px-4 py-1.5 inline-block rounded-md"}
[Toned]{class="text-toned text-sm px-4 py-1.5 inline-block rounded-md"}
[Text]{class="text-default text-sm px-4 py-1.5 inline-block rounded-md"}
[Highlighted]{class="text-highlighted text-sm px-4 py-1.5 inline-block rounded-md"}
[Inverted]{class="text-inverted bg-inverted text-sm px-4 py-1.5 inline-block rounded-md"}
#code
```vue
<template>
<span class="text-dimmed">Dimmed</span>
<span class="text-muted">Muted</span>
<span class="text-toned">Toned</span>
<span class="text-default">Text</span>
<span class="text-highlighted">Highlighted</span>
<span class="text-inverted bg-inverted">Inverted</span>
</template>
```
::
::code-preview
[Default]{class="bg-default text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[Muted]{class="bg-muted text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[Elevated]{class="bg-elevated text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[Accented]{class="bg-accented text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[Inverted]{class="bg-inverted text-inverted text-sm px-4 py-1.5 inline-block rounded-md"}
#code
```vue
<template>
<div class="bg-default">Default</div>
<div class="bg-muted">Muted</div>
<div class="bg-elevated">Elevated</div>
<div class="bg-accented">Accented</div>
<div class="bg-inverted text-inverted">Inverted</div>
</template>
```
::
::code-preview
[Default]{class="border-2 border-default text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[Muted]{class="border-2 border-muted text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[Accented]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[Inverted]{class="border-2 border-inverted text-sm px-4 py-1.5 inline-block rounded-md"}
#code
```vue
<template>
<div class="border border-default">Default</div>
<div class="border border-muted">Muted</div>
<div class="border border-accented">Accented</div>
<div class="border border-inverted">Inverted</div>
</template>
```
::
::note
This is how the `@theme` is generated for each design token:
:::code-collapse{class="[&>div]:!my-0"}
```scss
@theme default {
--text-color-dimmed: var(--ui-text-dimmed);
--text-color-muted: var(--ui-text-muted);
--text-color-toned: var(--ui-text-toned);
--text-color-default: var(--ui-text);
--text-color-highlighted: var(--ui-text-highlighted);
--text-color-inverted: var(--ui-text-inverted);
--background-color-default: var(--ui-bg);
--background-color-muted: var(--ui-bg-muted);
--background-color-elevated: var(--ui-bg-elevated);
--background-color-accented: var(--ui-bg-accented);
--background-color-inverted: var(--ui-bg-inverted);
--background-color-border: var(--ui-border);
--border-color-default: var(--ui-border);
--border-color-muted: var(--ui-border-muted);
--border-color-accented: var(--ui-border-accented);
--border-color-inverted: var(--ui-border-inverted);
--ring-color-default: var(--ui-border);
--ring-color-muted: var(--ui-border-muted);
--ring-color-accented: var(--ui-border-accented);
--ring-color-inverted: var(--ui-border-inverted);
--ring-color-bg: var(--ui-bg);
--divide-color-default: var(--ui-border);
--divide-color-muted: var(--ui-border-muted);
--divide-color-accented: var(--ui-border-accented);
--divide-color-inverted: var(--ui-border-inverted);
--outline-color-default: var(--ui-border);
--outline-color-inverted: var(--ui-border-inverted);
--stroke-color-default: var(--ui-border);
--stroke-color-inverted: var(--ui-border-inverted);
--fill-color-default: var(--ui-border);
--fill-color-inverted: var(--ui-border-inverted);
} }
``` ```
:::
:: ::
::tip You can customize these CSS variables to tailor the appearance of your application in your `main.css` file:
You can customize these CSS variables to tailor the appearance of your application:
::module-only ::module-only
#ui #ui
@@ -605,13 +792,57 @@ You can customize these CSS variables to tailor the appearance of your applicati
::: :::
:: ::
:: ::note
Nuxt UI applies a text and background color on the `<body>` element of your app:
#### Border Radius
Nuxt UI provides a centralized border radius system through the `--ui-radius` CSS variable, which defaults to `0.25rem`. This system replaces Tailwind CSS's default `rounded-*` utilities while maintaining the familiar class names. The border radius system is built on a scale that multiplies the base radius value:
```css ```css
body {
@apply antialiased text-default bg-default scheme-light dark:scheme-dark;
}
```
::
### Radius
Nuxt UI provides a centralized border radius system through the `--ui-radius` CSS variable.
```css
:root {
--ui-radius: 0.25rem;
}
```
This CSS variable replaces Tailwind CSS's default `rounded-*` utilities so you can use the same class names:
::code-preview
[xs]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-xs mr-2"}
[sm]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-sm mr-2"}
[md]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-md mr-2"}
[lg]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-lg mr-2"}
[xl]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-xl mr-2"}
[2xl]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-2xl mr-2"}
[3xl]{class="border-2 border-accented text-sm px-4 py-1.5 inline-block rounded-3xl mr-2"}
#code
```vue
<template>
<div class="rounded-xs">xs</div>
<div class="rounded-sm">sm</div>
<div class="rounded-md">md</div>
<div class="rounded-lg">lg</div>
<div class="rounded-xl">xl</div>
<div class="rounded-2xl">2xl</div>
<div class="rounded-3xl">3xl</div>
</template>
```
::
::note
This is how the `@theme` is generated for each radius value:
:::code-collapse{class="[&>div]:!my-0"}
```scss
@theme default { @theme default {
--radius-xs: calc(var(--ui-radius) * 0.5); /* 0.125rem */ --radius-xs: calc(var(--ui-radius) * 0.5); /* 0.125rem */
--radius-sm: var(--ui-radius); /* 0.25rem */ --radius-sm: var(--ui-radius); /* 0.25rem */
@@ -622,6 +853,9 @@ Nuxt UI provides a centralized border radius system through the `--ui-radius` CS
--radius-3xl: calc(var(--ui-radius) * 6); /* 1.5rem */ --radius-3xl: calc(var(--ui-radius) * 6); /* 1.5rem */
} }
``` ```
:::
::
You can customize the base radius value in your `main.css` file: You can customize the base radius value in your `main.css` file:
@@ -656,12 +890,18 @@ You can customize the base radius value in your `main.css` file:
:: ::
::note ::note
Try the :prose-icon{name="i-lucide-swatch-book" class="text-(--ui-primary)"} theme picker in the header above to change the base radius value. Try the :prose-icon{name="i-lucide-swatch-book" class="text-primary"} theme picker in the header above to change the base radius value.
:: ::
#### Container ### Container
Nuxt UI provides a global `--ui-container` CSS variable that controls the maximum width of the [Container](/components/container) component, which defaults to `var(--container-7xl)`. Nuxt UI provides a `--ui-container` CSS variable that controls the maximum width of the [Container](/components/container) component.
```css
:root {
--ui-container: var(--container-7xl);
}
```
You can customize this value in your `main.css` file to adjust container widths consistently throughout your application: You can customize this value in your `main.css` file to adjust container widths consistently throughout your application:
@@ -716,7 +956,7 @@ Components in Nuxt UI can have multiple `slots`, each representing a distinct HT
```ts [src/theme/card.ts] ```ts [src/theme/card.ts]
export default { export default {
slots: { slots: {
root: 'bg-(--ui-bg) ring ring-(--ui-border) divide-y divide-(--ui-border) rounded-lg', root: 'bg-default ring ring-default divide-y divide-default rounded-lg',
header: 'p-4 sm:px-6', header: 'p-4 sm:px-6',
body: 'p-4 sm:p-6', body: 'p-4 sm:p-6',
footer: 'p-4 sm:px-6' footer: 'p-4 sm:px-6'
@@ -775,7 +1015,7 @@ Nuxt UI components use `variants` to change the `slots` styles based on props. H
```ts [src/theme/avatar.ts] ```ts [src/theme/avatar.ts]
export default { export default {
slots: { slots: {
root: 'inline-flex items-center justify-center shrink-0 select-none overflow-hidden rounded-full align-middle bg-(--ui-bg-elevated)', root: 'inline-flex items-center justify-center shrink-0 select-none overflow-hidden rounded-full align-middle bg-elevated',
image: 'h-full w-full rounded-[inherit] object-cover' image: 'h-full w-full rounded-[inherit] object-cover'
}, },
variants: { variants: {

View File

@@ -330,7 +330,7 @@ props:
color: neutral color: neutral
variant: outline variant: outline
ui: ui:
leadingIcon: 'text-(--ui-primary)' leadingIcon: 'text-primary'
slots: slots:
default: | default: |

View File

@@ -99,12 +99,12 @@ props:
slots: slots:
default: | default: |
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Right click here Right click here
</div> </div>
--- ---
:div{class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"}[Right click here] :div{class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72"}[Right click here]
:: ::
::note ::note
@@ -143,12 +143,12 @@ props:
slots: slots:
default: | default: |
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Right click here Right click here
</div> </div>
--- ---
:div{class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"}[Right click here] :div{class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72"}[Right click here]
:: ::
### Disabled ### Disabled
@@ -179,12 +179,12 @@ props:
slots: slots:
default: | default: |
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Right click here Right click here
</div> </div>
--- ---
:div{class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"}[Right click here] :div{class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72"}[Right click here]
:: ::
## Examples ## Examples

View File

@@ -248,7 +248,7 @@ Make sure to add the `data-vaul-drawer-wrapper` directive to a parent element of
```vue [app.vue] ```vue [app.vue]
<template> <template>
<UApp> <UApp>
<div class="bg-(--ui-bg)" data-vaul-drawer-wrapper> <div class="bg-default" data-vaul-drawer-wrapper>
<NuxtLayout> <NuxtLayout>
<NuxtPage /> <NuxtPage />
</NuxtLayout> </NuxtLayout>
@@ -262,7 +262,7 @@ export default defineNuxtConfig({
app: { app: {
rootAttrs: { rootAttrs: {
'data-vaul-drawer-wrapper': '', 'data-vaul-drawer-wrapper': '',
'class': 'bg-(--ui-bg)' 'class': 'bg-default'
} }
} }
}) })

View File

@@ -195,12 +195,12 @@ This will give you access to the following:
| Name | Type | | Name | Type |
| ---- | ---- | | ---- | ---- |
| `submit()`{lang="ts-type"} | `Promise<void>`{lang="ts-type"} <br> <div class="text-(--ui-text-toned) mt-1"><p>Triggers form submission.</p> | | `submit()`{lang="ts-type"} | `Promise<void>`{lang="ts-type"} <br> <div class="text-toned mt-1"><p>Triggers form submission.</p> |
| `validate(opts: { name?: keyof T \| (keyof T)[], silent?: boolean, nested?: boolean, transform?: boolean })`{lang="ts-type"} | `Promise<T>`{lang="ts-type"} <br> <div class="text-(--ui-text-toned) mt-1"><p>Triggers form validation. Will raise any errors unless `opts.silent` is set to true.</p> | | `validate(opts: { name?: keyof T \| (keyof T)[], silent?: boolean, nested?: boolean, transform?: boolean })`{lang="ts-type"} | `Promise<T>`{lang="ts-type"} <br> <div class="text-toned mt-1"><p>Triggers form validation. Will raise any errors unless `opts.silent` is set to true.</p> |
| `clear(path?: keyof T)`{lang="ts-type"} | `void` <br> <div class="text-(--ui-text-toned) mt-1"><p>Clears form errors associated with a specific path. If no path is provided, clears all form errors.</p> | | `clear(path?: keyof T)`{lang="ts-type"} | `void` <br> <div class="text-toned mt-1"><p>Clears form errors associated with a specific path. If no path is provided, clears all form errors.</p> |
| `getErrors(path?: keyof T)`{lang="ts-type"} | `FormError[]`{lang="ts-type"} <br> <div class="text-(--ui-text-toned) mt-1"><p>Retrieves form errors associated with a specific path. If no path is provided, returns all form errors.</p></div> | | `getErrors(path?: keyof T)`{lang="ts-type"} | `FormError[]`{lang="ts-type"} <br> <div class="text-toned mt-1"><p>Retrieves form errors associated with a specific path. If no path is provided, returns all form errors.</p></div> |
| `setErrors(errors: FormError[], name?: keyof T)`{lang="ts-type"} | `void` <br> <div class="text-(--ui-text-toned) mt-1"><p>Sets form errors for a given path. If no path is provided, overrides all errors.</p> | | `setErrors(errors: FormError[], name?: keyof T)`{lang="ts-type"} | `void` <br> <div class="text-toned mt-1"><p>Sets form errors for a given path. If no path is provided, overrides all errors.</p> |
| `errors`{lang="ts-type"} | `Ref<FormError[]>`{lang="ts-type"} <br> <div class="text-(--ui-text-toned) mt-1"><p>A reference to the array containing validation errors. Use this to access or manipulate the error information.</p> | | `errors`{lang="ts-type"} | `Ref<FormError[]>`{lang="ts-type"} <br> <div class="text-toned mt-1"><p>A reference to the array containing validation errors. Use this to access or manipulate the error information.</p> |
| `disabled`{lang="ts-type"} | `Ref<boolean>`{lang="ts-type"} | | `disabled`{lang="ts-type"} | `Ref<boolean>`{lang="ts-type"} |
| `dirty`{lang="ts-type"} | `Ref<boolean>`{lang="ts-type"} `true` if at least one form field has been updated by the user.| | `dirty`{lang="ts-type"} | `Ref<boolean>`{lang="ts-type"} `true` if at least one form field has been updated by the user.|
| `dirtyFields`{lang="ts-type"} | `DeepReadonly<Set<keyof T>>`{lang="ts-type"} Tracks fields that have been modified by the user. | | `dirtyFields`{lang="ts-type"} | `DeepReadonly<Set<keyof T>>`{lang="ts-type"} Tracks fields that have been modified by the user. |

View File

@@ -67,7 +67,7 @@ props:
raw: true raw: true
to: /components/link to: /components/link
activeClass: 'font-bold' activeClass: 'font-bold'
inactiveClass: 'text-(--ui-text-muted)' inactiveClass: 'text-muted'
slots: slots:
default: Link default: Link
--- ---

View File

@@ -330,7 +330,7 @@ props:
- label: Help - label: Help
icon: i-lucide-circle-help icon: i-lucide-circle-help
disabled: true disabled: true
class: 'data-[orientation=horizontal]:border-b border-(--ui-border) data-[orientation=horizontal]:w-full data-[orientation=vertical]:w-48' class: 'data-[orientation=horizontal]:border-b border-default data-[orientation=horizontal]:w-full data-[orientation=vertical]:w-48'
--- ---
:: ::

View File

@@ -75,13 +75,13 @@ props:
Use the `columns` prop as an array of [ColumnDef](https://tanstack.com/table/latest/docs/api/core/column-def) objects with properties like: Use the `columns` prop as an array of [ColumnDef](https://tanstack.com/table/latest/docs/api/core/column-def) objects with properties like:
- `accessorKey`: [The key of the row object to use when extracting the value for the column.]{class="text-(--ui-text-muted)"} - `accessorKey`: [The key of the row object to use when extracting the value for the column.]{class="text-muted"}
- `header`: [The header to display for the column. If a string is passed, it can be used as a default for the column ID. If a function is passed, it will be passed a props object for the header and should return the rendered header value (the exact type depends on the adapter being used).]{class="text-(--ui-text-muted)"} - `header`: [The header to display for the column. If a string is passed, it can be used as a default for the column ID. If a function is passed, it will be passed a props object for the header and should return the rendered header value (the exact type depends on the adapter being used).]{class="text-muted"}
- `cell`: [The cell to display each row for the column. If a function is passed, it will be passed a props object for the cell and should return the rendered cell value (the exact type depends on the adapter being used).]{class="text-(--ui-text-muted)"} - `cell`: [The cell to display each row for the column. If a function is passed, it will be passed a props object for the cell and should return the rendered cell value (the exact type depends on the adapter being used).]{class="text-muted"}
- `meta`: [Extra properties for the column.]{class="text-(--ui-text-muted)"} - `meta`: [Extra properties for the column.]{class="text-muted"}
- `class`: - `class`:
- `td`: [The classes to apply to the `td` element.]{class="text-(--ui-text-muted)"} - `td`: [The classes to apply to the `td` element.]{class="text-muted"}
- `th`: [The classes to apply to the `th` element.]{class="text-(--ui-text-muted)"} - `th`: [The classes to apply to the `th` element.]{class="text-muted"}
In order to render components or other HTML elements, you will need to use the Vue [`h` function](https://vuejs.org/api/render-function.html#h) inside the `header` and `cell` props. This is different from other components that use slots but allows for more flexibility. In order to render components or other HTML elements, you will need to use the Vue [`h` function](https://vuejs.org/api/render-function.html#h) inside the `header` and `cell` props. This is different from other components that use slots but allows for more flexibility.

View File

@@ -56,7 +56,7 @@ export default defineNuxtConfig({
}, },
rootAttrs: { rootAttrs: {
'data-vaul-drawer-wrapper': '', 'data-vaul-drawer-wrapper': '',
'class': 'bg-(--ui-bg)' 'class': 'bg-default'
} }
}, },

View File

@@ -11,7 +11,7 @@
"@nuxt/content": "https://pkg.pr.new/@nuxt/content@754e480", "@nuxt/content": "https://pkg.pr.new/@nuxt/content@754e480",
"@nuxt/image": "^1.10.0", "@nuxt/image": "^1.10.0",
"@nuxt/ui": "latest", "@nuxt/ui": "latest",
"@nuxt/ui-pro": "https://pkg.pr.new/@nuxt/ui-pro@3cc20d8", "@nuxt/ui-pro": "https://pkg.pr.new/@nuxt/ui-pro@9c280b4",
"@nuxthub/core": "^0.8.24", "@nuxthub/core": "^0.8.24",
"@nuxtjs/plausible": "^1.2.0", "@nuxtjs/plausible": "^1.2.0",
"@octokit/rest": "^21.1.1", "@octokit/rest": "^21.1.1",

View File

@@ -84,9 +84,9 @@ defineShortcuts({
<template> <template>
<UApp :toaster="(appConfig.toaster as any)"> <UApp :toaster="(appConfig.toaster as any)">
<div class="h-screen w-screen overflow-hidden flex min-h-0 bg-(--ui-bg)" data-vaul-drawer-wrapper> <div class="h-screen w-screen overflow-hidden flex min-h-0 bg-default" data-vaul-drawer-wrapper>
<UNavigationMenu :items="items" orientation="vertical" class="hidden lg:flex border-e border-(--ui-border) overflow-y-auto w-48 p-4" /> <UNavigationMenu :items="items" orientation="vertical" class="hidden lg:flex border-e border-default overflow-y-auto w-48 p-4" />
<UNavigationMenu :items="items" orientation="horizontal" class="lg:hidden border-b border-(--ui-border) [&>div]:min-w-min overflow-x-auto" /> <UNavigationMenu :items="items" orientation="horizontal" class="lg:hidden border-b border-default [&>div]:min-w-min overflow-x-auto" />
<div class="fixed top-15 lg:top-3 end-4 flex items-center gap-2"> <div class="fixed top-15 lg:top-3 end-4 flex items-center gap-2">
<UButton <UButton

View File

@@ -89,9 +89,9 @@ useHead({
<template> <template>
<template v-if="!$route.path.startsWith('/__nuxt_ui__')"> <template v-if="!$route.path.startsWith('/__nuxt_ui__')">
<UApp :toaster="appConfig.toaster"> <UApp :toaster="appConfig.toaster">
<div class="h-screen w-screen overflow-hidden flex flex-col lg:flex-row min-h-0 bg-(--ui-bg)" data-vaul-drawer-wrapper> <div class="h-screen w-screen overflow-hidden flex flex-col lg:flex-row min-h-0 bg-default" data-vaul-drawer-wrapper>
<UNavigationMenu :items="items" orientation="vertical" class="hidden lg:flex border-e border-(--ui-border) overflow-y-auto w-48 p-4" /> <UNavigationMenu :items="items" orientation="vertical" class="hidden lg:flex border-e border-default overflow-y-auto w-48 p-4" />
<UNavigationMenu :items="items" orientation="horizontal" class="lg:hidden border-b border-(--ui-border) [&>div]:min-w-min overflow-x-auto" /> <UNavigationMenu :items="items" orientation="horizontal" class="lg:hidden border-b border-default [&>div]:min-w-min overflow-x-auto" />
<div class="fixed top-15 lg:top-3 end-4 flex items-center gap-2"> <div class="fixed top-15 lg:top-3 end-4 flex items-center gap-2">
<ClientOnly v-if="!colorMode?.forced"> <ClientOnly v-if="!colorMode?.forced">

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="relative overflow-hidden rounded-sm border border-dashed border-(--ui-border-accented) opacity-75 px-4 flex items-center justify-center"> <div class="relative overflow-hidden rounded-sm border border-dashed border-accented opacity-75 px-4 flex items-center justify-center">
<svg class="absolute inset-0 h-full w-full stroke-(--ui-border-inverted)/10" fill="none"> <svg class="absolute inset-0 h-full w-full stroke-inverted/10" fill="none">
<defs> <defs>
<pattern <pattern
id="pattern-5c1e4f0e-62d5-498b-8ff0-cf77bb448c8e" id="pattern-5c1e4f0e-62d5-498b-8ff0-cf77bb448c8e"

View File

@@ -33,18 +33,18 @@ const items = [{
<UCard :ui="{ body: 'p-0 sm:p-0' }"> <UCard :ui="{ body: 'p-0 sm:p-0' }">
<UAccordion :items="items" class="w-96" :ui="{ trigger: 'px-3.5', body: 'px-3.5' }"> <UAccordion :items="items" class="w-96" :ui="{ trigger: 'px-3.5', body: 'px-3.5' }">
<template #body="{ item }"> <template #body="{ item }">
<p class="text-(--ui-text-muted)"> <p class="text-muted">
{{ item.content }} {{ item.content }}
</p> </p>
</template> </template>
<template #custom="{ item }"> <template #custom="{ item }">
<p class="text-(--ui-text-muted)"> <p class="text-muted">
Custom: {{ item.content }} Custom: {{ item.content }}
</p> </p>
</template> </template>
<template #custom-body="{ item }"> <template #custom-body="{ item }">
<p class="text-(--ui-text-muted)"> <p class="text-muted">
Custom: {{ item.content }} Custom: {{ item.content }}
</p> </p>
</template> </template>

View File

@@ -18,7 +18,7 @@ const items = [{
<template> <template>
<div class="flex flex-col gap-4"> <div class="flex flex-col gap-4">
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<UChip v-for="position in positions" :key="position" :position="position"> <UChip v-for="position in positions" :key="position" :position="position" color="neutral">
<UButton icon="i-lucide-inbox" color="neutral" variant="subtle" /> <UButton icon="i-lucide-inbox" color="neutral" variant="subtle" />
</UChip> </UChip>
</div> </div>

View File

@@ -138,7 +138,7 @@ defineShortcuts({
<UButton label="Open drawer" color="neutral" variant="outline" /> <UButton label="Open drawer" color="neutral" variant="outline" />
<template #content> <template #content>
<ReuseTemplate class="border-t border-(--ui-border) mt-4" /> <ReuseTemplate class="border-t border-default mt-4" />
</template> </template>
</UDrawer> </UDrawer>

View File

@@ -106,13 +106,13 @@ defineShortcuts(extractShortcuts(items.value))
</div> </div>
<UContextMenu :items="items" :ui="{ content: 'w-48' }" :size="size"> <UContextMenu :items="items" :ui="{ content: 'w-48' }" :size="size">
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Right click here Right click here
</div> </div>
</UContextMenu> </UContextMenu>
<UContextMenu :items="itemsWithColor" :ui="{ content: 'w-48' }" :size="size"> <UContextMenu :items="itemsWithColor" :ui="{ content: 'w-48' }" :size="size">
<div class="flex items-center justify-center rounded-md border border-dashed border-(--ui-border-accented) text-sm aspect-video w-72"> <div class="flex items-center justify-center rounded-md border border-dashed border-accented text-sm aspect-video w-72">
Color right click here Color right click here
</div> </div>
</UContextMenu> </UContextMenu>

View File

@@ -144,7 +144,7 @@ defineShortcuts(extractShortcuts(items.value))
<UButton label="Open" color="neutral" variant="outline" icon="i-lucide-menu" /> <UButton label="Open" color="neutral" variant="outline" icon="i-lucide-menu" />
<template #custom-trailing> <template #custom-trailing>
<UIcon name="i-lucide-badge-check" class="shrink-0 size-5 text-(--ui-primary)" /> <UIcon name="i-lucide-badge-check" class="shrink-0 size-5 text-primary" />
</template> </template>
</UDropdownMenu> </UDropdownMenu>

View File

@@ -54,7 +54,7 @@ const disabled = ref(false)
<FormExampleNestedList /> <FormExampleNestedList />
</div> </div>
<div class="border border-(--ui-border) rounded-lg"> <div class="border border-default rounded-lg">
<div class="py-2 px-4 flex gap-4 items-center"> <div class="py-2 px-4 flex gap-4 items-center">
<UFormField label="Validate on" class="flex items-center gap-2"> <UFormField label="Validate on" class="flex items-center gap-2">
<USelectMenu v-model="validateOn" :items="['input', 'change', 'blur']" multiple class="w-48" /> <USelectMenu v-model="validateOn" :items="['input', 'change', 'blur']" multiple class="w-48" />
@@ -62,7 +62,7 @@ const disabled = ref(false)
<UCheckbox v-model="disabled" label="Disabled" /> <UCheckbox v-model="disabled" label="Disabled" />
</div> </div>
<FormExampleElements :validate-on="validateOn" :disabled="disabled" class="border-t border-(--ui-border) p-4" /> <FormExampleElements :validate-on="validateOn" :disabled="disabled" class="border-t border-default p-4" />
</div> </div>
</div> </div>
</template> </template>

View File

@@ -8,7 +8,7 @@
<ULink active> <ULink active>
Button active Button active
</ULink> </ULink>
<ULink active class="font-medium" active-class="text-(--ui-text-highlighted)"> <ULink active class="font-medium" active-class="text-highlighted">
Button active with class Button active with class
</ULink> </ULink>
<ULink active disabled> <ULink active disabled>
@@ -18,7 +18,7 @@
<ULink> <ULink>
Button inactive Button inactive
</ULink> </ULink>
<ULink class="font-medium" inactive-class="hover:text-(--ui-primary)"> <ULink class="font-medium" inactive-class="hover:text-primary">
Button inactive with class Button inactive with class
</ULink> </ULink>
<ULink disabled> <ULink disabled>
@@ -34,7 +34,7 @@
<ULink to="/components/link"> <ULink to="/components/link">
Link active Link active
</ULink> </ULink>
<ULink to="/components/link" class="font-medium" active-class="text-(--ui-text-highlighted)"> <ULink to="/components/link" class="font-medium" active-class="text-highlighted">
Link active with class Link active with class
</ULink> </ULink>
<ULink to="/components/link" disabled> <ULink to="/components/link" disabled>
@@ -44,7 +44,7 @@
<ULink to="/components/button"> <ULink to="/components/button">
Link inactive Link inactive
</ULink> </ULink>
<ULink to="/components/button" class="font-medium" inactive-class="hover:text-(--ui-primary)"> <ULink to="/components/button" class="font-medium" inactive-class="hover:text-primary">
Link inactive with class Link inactive with class
</ULink> </ULink>
<ULink to="/components/button" disabled> <ULink to="/components/button" disabled>

View File

@@ -112,7 +112,7 @@ const items = [
:content-orientation="contentOrientation" :content-orientation="contentOrientation"
:highlight="highlight" :highlight="highlight"
:highlight-color="highlightColor" :highlight-color="highlightColor"
:class="highlight && 'data-[orientation=horizontal]:border-b border-(--ui-border)'" :class="highlight && 'data-[orientation=horizontal]:border-b border-default'"
class="data-[orientation=vertical]:data-[collapsed=false]:w-48" class="data-[orientation=vertical]:data-[collapsed=false]:w-48"
/> />
</div> </div>

View File

@@ -1,7 +1,7 @@
<template> <template>
<div class="flex flex-col gap-4"> <div class="flex flex-col gap-4">
<div> <div>
<p class="font-semibold text-(--ui-text-highlighted)"> <p class="font-semibold text-highlighted">
Nuxt UI Nuxt UI
</p> </p>
<p>An open-source UI component library.</p> <p>An open-source UI component library.</p>

View File

@@ -339,10 +339,10 @@ onMounted(() => {
getPaginationRowModel: getPaginationRowModel() getPaginationRowModel: getPaginationRowModel()
}" }"
:ui="{ :ui="{
tr: 'divide-x divide-(--ui-border)' tr: 'divide-x divide-default'
}" }"
sticky sticky
class="border border-(--ui-border-accented) rounded-sm" class="border border-accented rounded-sm"
@select="onSelect" @select="onSelect"
> >
<template #expanded="{ row }"> <template #expanded="{ row }">
@@ -351,7 +351,7 @@ onMounted(() => {
</UTable> </UTable>
<div class="flex items-center justify-between gap-3"> <div class="flex items-center justify-between gap-3">
<div class="text-sm text-(--ui-text-muted)"> <div class="text-sm text-muted">
{{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of {{ table?.tableApi?.getFilteredSelectedRowModel().rows.length || 0 }} of
{{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected. {{ table?.tableApi?.getFilteredRowModel().rows.length || 0 }} row(s) selected.
</div> </div>

View File

@@ -57,7 +57,7 @@ const items = [{
class="w-96" class="w-96"
> >
<template #custom="{ item }"> <template #custom="{ item }">
<span class="text-(--ui-text-muted)">Custom: {{ item.content }}</span> <span class="text-muted">Custom: {{ item.content }}</span>
</template> </template>
<template #list-trailing> <template #list-trailing>

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="text-center"> <div class="text-center">
<h1 class="font-semibold mb-1"> <h1 class="font-semibold text-primary mb-1">
Playground Playground
</h1> </h1>

10
pnpm-lock.yaml generated
View File

@@ -246,8 +246,8 @@ importers:
specifier: workspace:* specifier: workspace:*
version: link:.. version: link:..
'@nuxt/ui-pro': '@nuxt/ui-pro':
specifier: https://pkg.pr.new/@nuxt/ui-pro@3cc20d8 specifier: https://pkg.pr.new/@nuxt/ui-pro@9c280b4
version: https://pkg.pr.new/@nuxt/ui-pro@3cc20d8(@babel/parser@7.27.0)(joi@17.13.3)(magicast@0.3.5)(superstruct@2.0.2)(typescript@5.8.3)(valibot@1.0.0(typescript@5.8.3))(vue@3.5.13(typescript@5.8.3))(yup@1.6.1)(zod@3.24.3) version: https://pkg.pr.new/@nuxt/ui-pro@9c280b4(@babel/parser@7.27.0)(joi@17.13.3)(magicast@0.3.5)(superstruct@2.0.2)(typescript@5.8.3)(valibot@1.0.0(typescript@5.8.3))(vue@3.5.13(typescript@5.8.3))(yup@1.6.1)(zod@3.24.3)
'@nuxthub/core': '@nuxthub/core':
specifier: ^0.8.24 specifier: ^0.8.24
version: 0.8.24(db0@0.3.1(better-sqlite3@11.9.1))(ioredis@5.6.0)(magicast@0.3.5)(vite@6.3.2(@types/node@22.13.14)(jiti@2.4.2)(lightningcss@1.29.2)(terser@5.39.0)(yaml@2.7.1)) version: 0.8.24(db0@0.3.1(better-sqlite3@11.9.1))(ioredis@5.6.0)(magicast@0.3.5)(vite@6.3.2(@types/node@22.13.14)(jiti@2.4.2)(lightningcss@1.29.2)(terser@5.39.0)(yaml@2.7.1))
@@ -1348,8 +1348,8 @@ packages:
vitest: vitest:
optional: true optional: true
'@nuxt/ui-pro@https://pkg.pr.new/@nuxt/ui-pro@3cc20d8': '@nuxt/ui-pro@https://pkg.pr.new/@nuxt/ui-pro@9c280b4':
resolution: {tarball: https://pkg.pr.new/@nuxt/ui-pro@3cc20d8} resolution: {tarball: https://pkg.pr.new/@nuxt/ui-pro@9c280b4}
version: 3.0.2 version: 3.0.2
peerDependencies: peerDependencies:
joi: ^17.13.0 joi: ^17.13.0
@@ -8250,7 +8250,7 @@ snapshots:
- typescript - typescript
- yaml - yaml
'@nuxt/ui-pro@https://pkg.pr.new/@nuxt/ui-pro@3cc20d8(@babel/parser@7.27.0)(joi@17.13.3)(magicast@0.3.5)(superstruct@2.0.2)(typescript@5.8.3)(valibot@1.0.0(typescript@5.8.3))(vue@3.5.13(typescript@5.8.3))(yup@1.6.1)(zod@3.24.3)': '@nuxt/ui-pro@https://pkg.pr.new/@nuxt/ui-pro@9c280b4(@babel/parser@7.27.0)(joi@17.13.3)(magicast@0.3.5)(superstruct@2.0.2)(typescript@5.8.3)(valibot@1.0.0(typescript@5.8.3))(vue@3.5.13(typescript@5.8.3))(yup@1.6.1)(zod@3.24.3)':
dependencies: dependencies:
'@ai-sdk/vue': 1.2.8(vue@3.5.13(typescript@5.8.3))(zod@3.24.3) '@ai-sdk/vue': 1.2.8(vue@3.5.13(typescript@5.8.3))(zod@3.24.3)
'@nuxt/kit': 3.16.2(magicast@0.3.5) '@nuxt/kit': 3.16.2(magicast@0.3.5)

View File

@@ -6,7 +6,7 @@
@layer base { @layer base {
body { body {
@apply antialiased text-(--ui-text) bg-(--ui-bg) scheme-light dark:scheme-dark; @apply antialiased text-default bg-default scheme-light dark:scheme-dark;
} }
:root, .light { :root, .light {
@@ -15,6 +15,7 @@
--ui-text-toned: var(--ui-color-neutral-600); --ui-text-toned: var(--ui-color-neutral-600);
--ui-text: var(--ui-color-neutral-700); --ui-text: var(--ui-color-neutral-700);
--ui-text-highlighted: var(--ui-color-neutral-900); --ui-text-highlighted: var(--ui-color-neutral-900);
--ui-text-inverted: var(--color-white);
--ui-bg: var(--color-white); --ui-bg: var(--color-white);
--ui-bg-muted: var(--ui-color-neutral-50); --ui-bg-muted: var(--ui-color-neutral-50);
@@ -37,6 +38,7 @@
--ui-text-toned: var(--ui-color-neutral-300); --ui-text-toned: var(--ui-color-neutral-300);
--ui-text: var(--ui-color-neutral-200); --ui-text: var(--ui-color-neutral-200);
--ui-text-highlighted: var(--color-white); --ui-text-highlighted: var(--color-white);
--ui-text-inverted: var(--ui-color-neutral-900);
--ui-bg: var(--ui-color-neutral-900); --ui-bg: var(--ui-color-neutral-900);
--ui-bg-muted: var(--ui-color-neutral-800); --ui-bg-muted: var(--ui-color-neutral-800);

View File

@@ -99,6 +99,37 @@ export function getTemplates(options: ModuleOptions, uiConfig: Record<string, an
--radius-xl: calc(var(--ui-radius) * 3); --radius-xl: calc(var(--ui-radius) * 3);
--radius-2xl: calc(var(--ui-radius) * 4); --radius-2xl: calc(var(--ui-radius) * 4);
--radius-3xl: calc(var(--ui-radius) * 6); --radius-3xl: calc(var(--ui-radius) * 6);
--text-color-dimmed: var(--ui-text-dimmed);
--text-color-muted: var(--ui-text-muted);
--text-color-toned: var(--ui-text-toned);
--text-color-default: var(--ui-text);
--text-color-highlighted: var(--ui-text-highlighted);
--text-color-inverted: var(--ui-text-inverted);
--background-color-default: var(--ui-bg);
--background-color-muted: var(--ui-bg-muted);
--background-color-elevated: var(--ui-bg-elevated);
--background-color-accented: var(--ui-bg-accented);
--background-color-inverted: var(--ui-bg-inverted);
--background-color-border: var(--ui-border);
--border-color-default: var(--ui-border);
--border-color-muted: var(--ui-border-muted);
--border-color-accented: var(--ui-border-accented);
--border-color-inverted: var(--ui-border-inverted);
--ring-color-default: var(--ui-border);
--ring-color-muted: var(--ui-border-muted);
--ring-color-accented: var(--ui-border-accented);
--ring-color-inverted: var(--ui-border-inverted);
--ring-color-bg: var(--ui-bg);
--divide-color-default: var(--ui-border);
--divide-color-muted: var(--ui-border-muted);
--divide-color-accented: var(--ui-border-accented);
--divide-color-inverted: var(--ui-border-inverted);
--outline-color-default: var(--ui-border);
--outline-color-inverted: var(--ui-border-inverted);
--stroke-color-default: var(--ui-border);
--stroke-color-inverted: var(--ui-border-inverted);
--fill-color-default: var(--ui-border);
--fill-color-inverted: var(--ui-border-inverted);
} }
` `
}) })

View File

@@ -1,9 +1,9 @@
export default { export default {
slots: { slots: {
root: 'w-full', root: 'w-full',
item: 'border-b border-(--ui-border) last:border-b-0', item: 'border-b border-default last:border-b-0',
header: 'flex', header: 'flex',
trigger: 'group flex-1 flex items-center gap-1.5 font-medium text-sm py-3.5 focus-visible:outline-(--ui-primary) min-w-0', trigger: 'group flex-1 flex items-center gap-1.5 font-medium text-sm py-3.5 focus-visible:outline-primary min-w-0',
content: 'data-[state=open]:animate-[accordion-down_200ms_ease-out] data-[state=closed]:animate-[accordion-up_200ms_ease-out] overflow-hidden focus:outline-none', content: 'data-[state=open]:animate-[accordion-down_200ms_ease-out] data-[state=closed]:animate-[accordion-up_200ms_ease-out] overflow-hidden focus:outline-none',
body: 'text-sm pb-3.5', body: 'text-sm pb-3.5',
leadingIcon: 'shrink-0 size-5', leadingIcon: 'shrink-0 size-5',

View File

@@ -43,49 +43,49 @@ export default (options: Required<ModuleOptions>) => ({
color, color,
variant: 'solid', variant: 'solid',
class: { class: {
root: `bg-(--ui-${color}) text-(--ui-bg)` root: `bg-${color} text-inverted`
} }
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'outline', variant: 'outline',
class: { class: {
root: `text-(--ui-${color}) ring ring-inset ring-(--ui-${color})/25` root: `text-${color} ring ring-inset ring-${color}/25`
} }
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'soft', variant: 'soft',
class: { class: {
root: `bg-(--ui-${color})/10 text-(--ui-${color})` root: `bg-${color}/10 text-${color}`
} }
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'subtle', variant: 'subtle',
class: { class: {
root: `bg-(--ui-${color})/10 text-(--ui-${color}) ring ring-inset ring-(--ui-${color})/25` root: `bg-${color}/10 text-${color} ring ring-inset ring-${color}/25`
} }
})), { })), {
color: 'neutral', color: 'neutral',
variant: 'solid', variant: 'solid',
class: { class: {
root: 'text-(--ui-bg) bg-(--ui-bg-inverted)' root: 'text-inverted bg-inverted'
} }
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'outline', variant: 'outline',
class: { class: {
root: 'text-(--ui-text-highlighted) bg-(--ui-bg) ring ring-inset ring-(--ui-border)' root: 'text-highlighted bg-default ring ring-inset ring-default'
} }
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'soft', variant: 'soft',
class: { class: {
root: 'text-(--ui-text-highlighted) bg-(--ui-bg-elevated)/50' root: 'text-highlighted bg-elevated/50'
} }
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'subtle', variant: 'subtle',
class: { class: {
root: 'text-(--ui-text-highlighted) bg-(--ui-bg-elevated)/50 ring ring-inset ring-(--ui-border-accented)' root: 'text-highlighted bg-elevated/50 ring ring-inset ring-accented'
} }
}], }],
defaultVariants: { defaultVariants: {

View File

@@ -1,7 +1,7 @@
export default { export default {
slots: { slots: {
root: 'inline-flex flex-row-reverse justify-end', root: 'inline-flex flex-row-reverse justify-end',
base: 'relative rounded-full ring-(--ui-bg) first:me-0' base: 'relative rounded-full ring-bg first:me-0'
}, },
variants: { variants: {
size: { size: {

View File

@@ -1,9 +1,9 @@
export default { export default {
slots: { slots: {
root: 'inline-flex items-center justify-center shrink-0 select-none overflow-hidden rounded-full align-middle bg-(--ui-bg-elevated)', root: 'inline-flex items-center justify-center shrink-0 select-none overflow-hidden rounded-full align-middle bg-elevated',
image: 'h-full w-full rounded-[inherit] object-cover', image: 'h-full w-full rounded-[inherit] object-cover',
fallback: 'font-medium leading-none text-(--ui-text-muted) truncate', fallback: 'font-medium leading-none text-muted truncate',
icon: 'text-(--ui-text-muted) shrink-0' icon: 'text-muted shrink-0'
}, },
variants: { variants: {
size: { size: {

View File

@@ -58,35 +58,35 @@ export default (options: Required<ModuleOptions>) => ({
compoundVariants: [...(options.theme.colors || []).map((color: string) => ({ compoundVariants: [...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'solid', variant: 'solid',
class: `bg-(--ui-${color}) text-(--ui-bg)` class: `bg-${color} text-inverted`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'outline', variant: 'outline',
class: `text-(--ui-${color}) ring ring-inset ring-(--ui-${color})/50` class: `text-${color} ring ring-inset ring-${color}/50`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'soft', variant: 'soft',
class: `bg-(--ui-${color})/10 text-(--ui-${color})` class: `bg-${color}/10 text-${color}`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'subtle', variant: 'subtle',
class: `bg-(--ui-${color})/10 text-(--ui-${color}) ring ring-inset ring-(--ui-${color})/25` class: `bg-${color}/10 text-${color} ring ring-inset ring-${color}/25`
})), { })), {
color: 'neutral', color: 'neutral',
variant: 'solid', variant: 'solid',
class: 'text-(--ui-bg) bg-(--ui-bg-inverted)' class: 'text-inverted bg-inverted'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'outline', variant: 'outline',
class: 'ring ring-inset ring-(--ui-border-accented) text-(--ui-text) bg-(--ui-bg)' class: 'ring ring-inset ring-accented text-default bg-default'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'soft', variant: 'soft',
class: 'text-(--ui-text) bg-(--ui-bg-elevated)' class: 'text-default bg-elevated'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'subtle', variant: 'subtle',
class: 'ring ring-inset ring-(--ui-border-accented) text-(--ui-text) bg-(--ui-bg-elevated)' class: 'ring ring-inset ring-accented text-default bg-elevated'
}], }],
defaultVariants: { defaultVariants: {
color: 'primary', color: 'primary',

View File

@@ -5,21 +5,21 @@ export default (options: Required<ModuleOptions>) => ({
root: 'relative min-w-0', root: 'relative min-w-0',
list: 'flex items-center gap-1.5', list: 'flex items-center gap-1.5',
item: 'flex min-w-0', item: 'flex min-w-0',
link: 'group relative flex items-center gap-1.5 text-sm min-w-0 focus-visible:outline-(--ui-primary)', link: 'group relative flex items-center gap-1.5 text-sm min-w-0 focus-visible:outline-primary',
linkLeadingIcon: 'shrink-0 size-5', linkLeadingIcon: 'shrink-0 size-5',
linkLeadingAvatar: 'shrink-0', linkLeadingAvatar: 'shrink-0',
linkLeadingAvatarSize: '2xs', linkLeadingAvatarSize: '2xs',
linkLabel: 'truncate', linkLabel: 'truncate',
separator: 'flex', separator: 'flex',
separatorIcon: 'shrink-0 size-5 text-(--ui-text-muted)' separatorIcon: 'shrink-0 size-5 text-muted'
}, },
variants: { variants: {
active: { active: {
true: { true: {
link: 'text-(--ui-primary) font-semibold' link: 'text-primary font-semibold'
}, },
false: { false: {
link: 'text-(--ui-text-muted) font-medium' link: 'text-muted font-medium'
} }
}, },
disabled: { disabled: {
@@ -36,7 +36,7 @@ export default (options: Required<ModuleOptions>) => ({
active: false, active: false,
to: true, to: true,
class: { class: {
link: ['hover:text-(--ui-text)', options.theme.transitions && 'transition-colors'] link: ['hover:text-default', options.theme.transitions && 'transition-colors']
} }
}] }]
}) })

View File

@@ -86,51 +86,51 @@ export default (options: Required<ModuleOptions>) => ({
compoundVariants: [...(options.theme.colors || []).map((color: string) => ({ compoundVariants: [...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'solid', variant: 'solid',
class: `text-(--ui-bg) bg-(--ui-${color}) hover:bg-(--ui-${color})/75 disabled:bg-(--ui-${color}) aria-disabled:bg-(--ui-${color}) focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-(--ui-${color})` class: `text-inverted bg-${color} hover:bg-${color}/75 disabled:bg-${color} aria-disabled:bg-${color} focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-${color}`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'outline', variant: 'outline',
class: `ring ring-inset ring-(--ui-${color})/50 text-(--ui-${color}) hover:bg-(--ui-${color})/10 disabled:bg-transparent aria-disabled:bg-transparent dark:disabled:bg-transparent dark:aria-disabled:bg-transparent focus:outline-none focus-visible:ring-2 focus-visible:ring-(--ui-${color})` class: `ring ring-inset ring-${color}/50 text-${color} hover:bg-${color}/10 disabled:bg-transparent aria-disabled:bg-transparent dark:disabled:bg-transparent dark:aria-disabled:bg-transparent focus:outline-none focus-visible:ring-2 focus-visible:ring-${color}`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'soft', variant: 'soft',
class: `text-(--ui-${color}) bg-(--ui-${color})/10 hover:bg-(--ui-${color})/15 focus:outline-none focus-visible:bg-(--ui-${color})/15 disabled:bg-(--ui-${color})/10 aria-disabled:bg-(--ui-${color})/10` class: `text-${color} bg-${color}/10 hover:bg-${color}/15 focus:outline-none focus-visible:bg-${color}/15 disabled:bg-${color}/10 aria-disabled:bg-${color}/10`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'subtle', variant: 'subtle',
class: `text-(--ui-${color}) ring ring-inset ring-(--ui-${color})/25 bg-(--ui-${color})/10 hover:bg-(--ui-${color})/15 disabled:bg-(--ui-${color})/10 aria-disabled:bg-(--ui-${color})/10 focus:outline-none focus-visible:ring-2 focus-visible:ring-(--ui-${color})` class: `text-${color} ring ring-inset ring-${color}/25 bg-${color}/10 hover:bg-${color}/15 disabled:bg-${color}/10 aria-disabled:bg-${color}/10 focus:outline-none focus-visible:ring-2 focus-visible:ring-${color}`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'ghost', variant: 'ghost',
class: `text-(--ui-${color}) hover:bg-(--ui-${color})/10 focus:outline-none focus-visible:bg-(--ui-${color})/10 disabled:bg-transparent aria-disabled:bg-transparent dark:disabled:bg-transparent dark:aria-disabled:bg-transparent` class: `text-${color} hover:bg-${color}/10 focus:outline-none focus-visible:bg-${color}/10 disabled:bg-transparent aria-disabled:bg-transparent dark:disabled:bg-transparent dark:aria-disabled:bg-transparent`
})), ...(options.theme.colors || []).map((color: string) => ({ })), ...(options.theme.colors || []).map((color: string) => ({
color, color,
variant: 'link', variant: 'link',
class: `text-(--ui-${color}) hover:text-(--ui-${color})/75 disabled:text-(--ui-${color}) aria-disabled:text-(--ui-${color}) focus:outline-none focus-visible:ring-2 focus-visible:ring-inset focus-visible:ring-(--ui-${color})` class: `text-${color} hover:text-${color}/75 disabled:text-${color} aria-disabled:text-${color} focus:outline-none focus-visible:ring-2 focus-visible:ring-inset focus-visible:ring-${color}`
})), { })), {
color: 'neutral', color: 'neutral',
variant: 'solid', variant: 'solid',
class: 'text-(--ui-bg) bg-(--ui-bg-inverted) hover:bg-(--ui-bg-inverted)/90 disabled:bg-(--ui-bg-inverted) aria-disabled:bg-(--ui-bg-inverted) focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-(--ui-border-inverted)' class: 'text-inverted bg-inverted hover:bg-inverted/90 disabled:bg-inverted aria-disabled:bg-inverted focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-inverted'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'outline', variant: 'outline',
class: 'ring ring-inset ring-(--ui-border-accented) text-(--ui-text) bg-(--ui-bg) hover:bg-(--ui-bg-elevated) disabled:bg-(--ui-bg) aria-disabled:bg-(--ui-bg) focus:outline-none focus-visible:ring-2 focus-visible:ring-(--ui-border-inverted)' class: 'ring ring-inset ring-accented text-default bg-default hover:bg-elevated disabled:bg-default aria-disabled:bg-default focus:outline-none focus-visible:ring-2 focus-visible:ring-inverted'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'soft', variant: 'soft',
class: 'text-(--ui-text) bg-(--ui-bg-elevated) hover:bg-(--ui-bg-accented)/75 focus:outline-none focus-visible:bg-(--ui-bg-accented)/75 disabled:bg-(--ui-bg-elevated) aria-disabled:bg-(--ui-bg-elevated)' class: 'text-default bg-elevated hover:bg-accented/75 focus:outline-none focus-visible:bg-accented/75 disabled:bg-elevated aria-disabled:bg-elevated'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'subtle', variant: 'subtle',
class: 'ring ring-inset ring-(--ui-border-accented) text-(--ui-text) bg-(--ui-bg-elevated) hover:bg-(--ui-bg-accented)/75 disabled:bg-(--ui-bg-elevated) aria-disabled:bg-(--ui-bg-elevated) focus:outline-none focus-visible:ring-2 focus-visible:ring-(--ui-border-inverted)' class: 'ring ring-inset ring-accented text-default bg-elevated hover:bg-accented/75 disabled:bg-elevated aria-disabled:bg-elevated focus:outline-none focus-visible:ring-2 focus-visible:ring-inverted'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'ghost', variant: 'ghost',
class: 'text-(--ui-text) hover:bg-(--ui-bg-elevated) focus:outline-none focus-visible:bg-(--ui-bg-elevated) hover:disabled:bg-transparent dark:hover:disabled:bg-transparent hover:aria-disabled:bg-transparent dark:hover:aria-disabled:bg-transparent' class: 'text-default hover:bg-elevated focus:outline-none focus-visible:bg-elevated hover:disabled:bg-transparent dark:hover:disabled:bg-transparent hover:aria-disabled:bg-transparent dark:hover:aria-disabled:bg-transparent'
}, { }, {
color: 'neutral', color: 'neutral',
variant: 'link', variant: 'link',
class: 'text-(--ui-text-muted) hover:text-(--ui-text) disabled:text-(--ui-text-muted) aria-disabled:text-(--ui-text-muted) focus:outline-none focus-visible:ring-inset focus-visible:ring-2 focus-visible:ring-(--ui-border-inverted)' class: 'text-muted hover:text-default disabled:text-muted aria-disabled:text-muted focus:outline-none focus-visible:ring-inset focus-visible:ring-2 focus-visible:ring-inverted'
}, { }, {
size: 'xs', size: 'xs',
square: true, square: true,

View File

@@ -12,17 +12,17 @@ export default (options: Required<ModuleOptions>) => ({
gridBody: 'grid', gridBody: 'grid',
headCell: 'rounded-md', headCell: 'rounded-md',
cell: 'relative text-center', cell: 'relative text-center',
cellTrigger: ['m-0.5 relative flex items-center justify-center rounded-full whitespace-nowrap focus-visible:ring-2 focus:outline-none data-disabled:text-(--ui-text-muted) data-unavailable:line-through data-unavailable:text-(--ui-text-muted) data-unavailable:pointer-events-none data-[selected]:text-(--ui-bg) data-today:font-semibold data-[outside-view]:text-(--ui-text-muted)', options.theme.transitions && 'transition'] cellTrigger: ['m-0.5 relative flex items-center justify-center rounded-full whitespace-nowrap focus-visible:ring-2 focus:outline-none data-disabled:text-muted data-unavailable:line-through data-unavailable:text-muted data-unavailable:pointer-events-none data-[selected]:text-inverted data-today:font-semibold data-[outside-view]:text-muted', options.theme.transitions && 'transition']
}, },
variants: { variants: {
color: { color: {
...Object.fromEntries((options.theme.colors || []).map((color: string) => [color, { ...Object.fromEntries((options.theme.colors || []).map((color: string) => [color, {
headCell: `text-(--ui-${color})`, headCell: `text-${color}`,
cellTrigger: `focus-visible:ring-(--ui-${color}) data-[selected]:bg-(--ui-${color}) data-today:not-data-[selected]:text-(--ui-${color}) data-[highlighted]:bg-(--ui-${color})/20 hover:not-data-[selected]:bg-(--ui-${color})/20` cellTrigger: `focus-visible:ring-${color} data-[selected]:bg-${color} data-today:not-data-[selected]:text-${color} data-[highlighted]:bg-${color}/20 hover:not-data-[selected]:bg-${color}/20`
}])), }])),
neutral: { neutral: {
headCell: 'text-(--ui-bg-inverted)', headCell: 'text-highlighted',
cellTrigger: 'focus-visible:ring-(--ui-border-inverted) data-[selected]:bg-(--ui-bg-inverted) data-today:not-data-[selected]:text-(--ui-bg-inverted) data-[highlighted]:bg-(--ui-bg-inverted)/20 hover:not-data-[selected]:bg-(--ui-bg-inverted)/10' cellTrigger: 'focus-visible:ring-inverted data-[selected]:bg-inverted data-today:not-data-[selected]:text-inverted data-[highlighted]:bg-inverted/20 hover:not-data-[selected]:bg-inverted/10'
} }
}, },
size: { size: {

View File

@@ -8,16 +8,16 @@ export default {
variants: { variants: {
variant: { variant: {
solid: { solid: {
root: 'bg-(--ui-bg-inverted) text-(--ui-bg)' root: 'bg-inverted text-inverted'
}, },
outline: { outline: {
root: 'bg-(--ui-bg) ring ring-(--ui-border) divide-y divide-(--ui-border)' root: 'bg-default ring ring-default divide-y divide-default'
}, },
soft: { soft: {
root: 'bg-(--ui-bg-elevated)/50 divide-y divide-(--ui-border)' root: 'bg-elevated/50 divide-y divide-default'
}, },
subtle: { subtle: {
root: 'bg-(--ui-bg-elevated)/50 ring ring-(--ui-border) divide-y divide-(--ui-border)' root: 'bg-elevated/50 ring ring-default divide-y divide-default'
} }
} }
}, },

Some files were not shown because too many files have changed in this diff Show More