WIP - fix date picker

This commit is contained in:
Francisco Gaona
2025-12-22 01:50:53 +01:00
parent b6cb5652b7
commit b0a45d98ce
9 changed files with 74 additions and 60 deletions

View File

@@ -1,19 +1,19 @@
<script setup lang="ts"> <script setup lang="ts">
import type { PrimitiveProps } from 'reka-ui' import type { PrimitiveProps } from "reka-ui"
import type { HTMLAttributes } from 'vue' import type { HTMLAttributes } from "vue"
import type { ButtonVariants } from '.' import type { ButtonVariants } from "."
import { Primitive } from 'reka-ui' import { Primitive } from "reka-ui"
import { cn } from '@/lib/utils' import { cn } from "@/lib/utils"
import { buttonVariants } from '.' import { buttonVariants } from "."
interface Props extends PrimitiveProps { interface Props extends PrimitiveProps {
variant?: ButtonVariants['variant'] variant?: ButtonVariants["variant"]
size?: ButtonVariants['size'] size?: ButtonVariants["size"]
class?: HTMLAttributes['class'] class?: HTMLAttributes["class"]
} }
const props = withDefaults(defineProps<Props>(), { const props = withDefaults(defineProps<Props>(), {
as: 'button', as: "button",
}) })
</script> </script>

View File

@@ -1,36 +1,38 @@
import type { VariantProps } from 'class-variance-authority' import type { VariantProps } from "class-variance-authority"
import { cva } from 'class-variance-authority' import { cva } from "class-variance-authority"
export { default as Button } from './Button.vue' export { default as Button } from "./Button.vue"
export const buttonVariants = cva( export const buttonVariants = cva(
'inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0', "inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0",
{ {
variants: { variants: {
variant: { variant: {
default: 'bg-primary text-primary-foreground shadow hover:bg-primary/90', default: "bg-primary text-primary-foreground shadow hover:bg-primary/90",
destructive: 'bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90', destructive:
"bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90",
outline: outline:
'border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground', "border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground",
secondary: 'bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80', secondary:
ghost: 'hover:bg-accent hover:text-accent-foreground', "bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80",
link: 'text-primary underline-offset-4 hover:underline', ghost: "hover:bg-accent hover:text-accent-foreground",
link: "text-primary underline-offset-4 hover:underline",
}, },
size: { size: {
default: 'h-9 px-4 py-2', "default": "h-9 px-4 py-2",
xs: 'h-7 rounded px-2', "xs": "h-7 rounded px-2",
sm: 'h-8 rounded-md px-3 text-xs', "sm": "h-8 rounded-md px-3 text-xs",
lg: 'h-10 rounded-md px-8', "lg": "h-10 rounded-md px-8",
icon: 'h-9 w-9', "icon": "h-9 w-9",
'icon-sm': 'size-8', "icon-sm": "size-8",
'icon-lg': 'size-10', "icon-lg": "size-10",
}, },
}, },
defaultVariants: { defaultVariants: {
variant: 'default', variant: "default",
size: 'default', size: "default",
},
}, },
}
) )
export type ButtonVariants = VariantProps<typeof buttonVariants> export type ButtonVariants = VariantProps<typeof buttonVariants>

View File

@@ -5,6 +5,7 @@ import { Button } from '@/components/ui/button'
import { Popover, PopoverContent, PopoverTrigger } from '@/components/ui/popover' import { Popover, PopoverContent, PopoverTrigger } from '@/components/ui/popover'
import { CalendarIcon } from 'lucide-vue-next' import { CalendarIcon } from 'lucide-vue-next'
import { cn } from '@/lib/utils' import { cn } from '@/lib/utils'
import { CalendarDate, type DateValue } from '@internationalized/date'
interface Props { interface Props {
modelValue?: Date | string | null modelValue?: Date | string | null
@@ -22,18 +23,27 @@ const emit = defineEmits<{
'update:modelValue': [value: Date | null] 'update:modelValue': [value: Date | null]
}>() }>()
const value = computed({ const placeholder = ref<DateValue>(new CalendarDate(new Date().getFullYear(), new Date().getMonth() + 1, new Date().getDate()))
const value = computed<DateValue | undefined>({
get: () => { get: () => {
if (!props.modelValue) return undefined if (!props.modelValue) return undefined
return props.modelValue instanceof Date ? props.modelValue : new Date(props.modelValue) const date = props.modelValue instanceof Date ? props.modelValue : new Date(props.modelValue)
return new CalendarDate(date.getFullYear(), date.getMonth() + 1, date.getDate())
}, },
set: (date) => { set: (dateValue) => {
emit('update:modelValue', date || null) if (!dateValue) {
emit('update:modelValue', null)
return
}
const jsDate = new Date(dateValue.year, dateValue.month - 1, dateValue.day)
emit('update:modelValue', jsDate)
}, },
}) })
const formatDate = (date: Date | undefined) => { const formatDate = (dateValue: DateValue | undefined) => {
if (!date) return props.placeholder if (!dateValue) return props.placeholder
const date = new Date(dateValue.year, dateValue.month - 1, dateValue.day)
return date.toLocaleDateString('en-US', { return date.toLocaleDateString('en-US', {
year: 'numeric', year: 'numeric',
month: 'long', month: 'long',
@@ -58,7 +68,7 @@ const formatDate = (date: Date | undefined) => {
</Button> </Button>
</PopoverTrigger> </PopoverTrigger>
<PopoverContent class="w-auto p-0"> <PopoverContent class="w-auto p-0">
<Calendar v-model="value" initial-focus /> <Calendar v-model="value" :placeholder="placeholder" />
</PopoverContent> </PopoverContent>
</Popover> </Popover>
</template> </template>

View File

@@ -1,6 +1,6 @@
<script setup lang="ts"> <script setup lang="ts">
import type { PopoverRootEmits, PopoverRootProps } from "radix-vue" import type { PopoverRootEmits, PopoverRootProps } from "reka-ui"
import { PopoverRoot, useForwardPropsEmits } from "radix-vue" import { PopoverRoot, useForwardPropsEmits } from "reka-ui"
const props = defineProps<PopoverRootProps>() const props = defineProps<PopoverRootProps>()
const emits = defineEmits<PopoverRootEmits>() const emits = defineEmits<PopoverRootEmits>()

View File

@@ -1,12 +1,11 @@
<script setup lang="ts"> <script setup lang="ts">
import type { PopoverContentEmits, PopoverContentProps } from "radix-vue" import type { PopoverContentEmits, PopoverContentProps } from "reka-ui"
import type { HTMLAttributes } from "vue" import type { HTMLAttributes } from "vue"
import { reactiveOmit } from "@vueuse/core" import { reactiveOmit } from "@vueuse/core"
import { import {
PopoverContent, PopoverContent,
PopoverPortal,
useForwardPropsEmits, useForwardPropsEmits,
} from "radix-vue" } from "reka-ui"
import { cn } from "@/lib/utils" import { cn } from "@/lib/utils"
defineOptions({ defineOptions({
@@ -28,7 +27,6 @@ const forwarded = useForwardPropsEmits(delegatedProps, emits)
</script> </script>
<template> <template>
<PopoverPortal to="body">
<PopoverContent <PopoverContent
v-bind="{ ...forwarded, ...$attrs }" v-bind="{ ...forwarded, ...$attrs }"
:class=" :class="
@@ -40,5 +38,4 @@ const forwarded = useForwardPropsEmits(delegatedProps, emits)
> >
<slot /> <slot />
</PopoverContent> </PopoverContent>
</PopoverPortal>
</template> </template>

View File

@@ -1,6 +1,6 @@
<script setup lang="ts"> <script setup lang="ts">
import type { PopoverTriggerProps } from "radix-vue" import type { PopoverTriggerProps } from "reka-ui"
import { PopoverTrigger } from "radix-vue" import { PopoverTrigger } from "reka-ui"
const props = defineProps<PopoverTriggerProps>() const props = defineProps<PopoverTriggerProps>()
</script> </script>

View File

@@ -53,6 +53,9 @@ export default defineNuxtConfig({
}, },
vite: { vite: {
optimizeDeps: {
include: ['@internationalized/date'],
},
server: { server: {
hmr: { hmr: {
clientPort: 3001, clientPort: 3001,

View File

@@ -9,6 +9,7 @@
"version": "0.0.1", "version": "0.0.1",
"hasInstallScript": true, "hasInstallScript": true,
"dependencies": { "dependencies": {
"@internationalized/date": "^3.10.1",
"@nuxtjs/tailwindcss": "^6.11.4", "@nuxtjs/tailwindcss": "^6.11.4",
"@vueuse/core": "^10.11.1", "@vueuse/core": "^10.11.1",
"class-variance-authority": "^0.7.0", "class-variance-authority": "^0.7.0",
@@ -1258,9 +1259,9 @@
"peer": true "peer": true
}, },
"node_modules/@internationalized/date": { "node_modules/@internationalized/date": {
"version": "3.10.0", "version": "3.10.1",
"resolved": "https://registry.npmjs.org/@internationalized/date/-/date-3.10.0.tgz", "resolved": "https://registry.npmjs.org/@internationalized/date/-/date-3.10.1.tgz",
"integrity": "sha512-oxDR/NTEJ1k+UFVQElaNIk65E/Z83HK1z1WI3lQyhTtnNg4R5oVXaPzK3jcpKG8UHKDVuDQHzn+wsxSz8RP3aw==", "integrity": "sha512-oJrXtQiAXLvT9clCf1K4kxp3eKsQhIaZqxEyowkBcsvZDdZkbWrVmnGknxs5flTD0VGsxrxKgBCZty1EzoiMzA==",
"license": "Apache-2.0", "license": "Apache-2.0",
"dependencies": { "dependencies": {
"@swc/helpers": "^0.5.0" "@swc/helpers": "^0.5.0"

View File

@@ -15,6 +15,7 @@
"format": "prettier --write \"**/*.{js,ts,vue,json,css,scss,md}\"" "format": "prettier --write \"**/*.{js,ts,vue,json,css,scss,md}\""
}, },
"dependencies": { "dependencies": {
"@internationalized/date": "^3.10.1",
"@nuxtjs/tailwindcss": "^6.11.4", "@nuxtjs/tailwindcss": "^6.11.4",
"@vueuse/core": "^10.11.1", "@vueuse/core": "^10.11.1",
"class-variance-authority": "^0.7.0", "class-variance-authority": "^0.7.0",