This repository has been archived on 2024-04-04. You can view files and clone it, but cannot push or open issues or pull requests.
tailwindui/vue/components/application-ui/data-display/calendars/month_view.vue
2024-01-24 19:02:44 +08:00

226 lines
14 KiB
Vue

<template>
<div class="lg:flex lg:h-full lg:flex-col">
<header class="flex items-center justify-between border-b border-gray-200 px-6 py-4 lg:flex-none">
<h1 class="text-base font-semibold leading-6 text-gray-900">
<time datetime="2022-01">January 2022</time>
</h1>
<div class="flex items-center">
<div class="relative flex items-center rounded-md bg-white shadow-sm md:items-stretch">
<div class="pointer-events-none absolute inset-0 rounded-md ring-1 ring-inset ring-gray-300" aria-hidden="true" />
<button type="button" class="flex items-center justify-center rounded-l-md py-2 pl-3 pr-4 text-gray-400 hover:text-gray-500 focus:relative md:w-9 md:px-2 md:hover:bg-gray-50">
<span class="sr-only">Previous month</span>
<ChevronLeftIcon class="h-5 w-5" aria-hidden="true" />
</button>
<button type="button" class="hidden px-3.5 text-sm font-semibold text-gray-900 hover:bg-gray-50 focus:relative md:block">Today</button>
<span class="relative -mx-px h-5 w-px bg-gray-300 md:hidden" />
<button type="button" class="flex items-center justify-center rounded-r-md py-2 pl-4 pr-3 text-gray-400 hover:text-gray-500 focus:relative md:w-9 md:px-2 md:hover:bg-gray-50">
<span class="sr-only">Next month</span>
<ChevronRightIcon class="h-5 w-5" aria-hidden="true" />
</button>
</div>
<div class="hidden md:ml-4 md:flex md:items-center">
<Menu as="div" class="relative">
<MenuButton type="button" class="flex items-center gap-x-1.5 rounded-md bg-white px-3 py-2 text-sm font-semibold text-gray-900 shadow-sm ring-1 ring-inset ring-gray-300 hover:bg-gray-50">
Month view
<ChevronDownIcon class="-mr-1 h-5 w-5 text-gray-400" aria-hidden="true" />
</MenuButton>
<transition enter-active-class="transition ease-out duration-100" enter-from-class="transform opacity-0 scale-95" enter-to-class="transform opacity-100 scale-100" leave-active-class="transition ease-in duration-75" leave-from-class="transform opacity-100 scale-100" leave-to-class="transform opacity-0 scale-95">
<MenuItems class="absolute right-0 z-10 mt-3 w-36 origin-top-right overflow-hidden rounded-md bg-white shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none">
<div class="py-1">
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Day view</a>
</MenuItem>
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Week view</a>
</MenuItem>
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Month view</a>
</MenuItem>
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Year view</a>
</MenuItem>
</div>
</MenuItems>
</transition>
</Menu>
<div class="ml-6 h-6 w-px bg-gray-300" />
<button type="button" class="ml-6 rounded-md bg-indigo-600 px-3 py-2 text-sm font-semibold text-white shadow-sm hover:bg-indigo-500 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-indigo-500">Add event</button>
</div>
<Menu as="div" class="relative ml-6 md:hidden">
<MenuButton class="-mx-2 flex items-center rounded-full border border-transparent p-2 text-gray-400 hover:text-gray-500">
<span class="sr-only">Open menu</span>
<EllipsisHorizontalIcon class="h-5 w-5" aria-hidden="true" />
</MenuButton>
<transition enter-active-class="transition ease-out duration-100" enter-from-class="transform opacity-0 scale-95" enter-to-class="transform opacity-100 scale-100" leave-active-class="transition ease-in duration-75" leave-from-class="transform opacity-100 scale-100" leave-to-class="transform opacity-0 scale-95">
<MenuItems class="absolute right-0 z-10 mt-3 w-36 origin-top-right divide-y divide-gray-100 overflow-hidden rounded-md bg-white shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none">
<div class="py-1">
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Create event</a>
</MenuItem>
</div>
<div class="py-1">
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Go to today</a>
</MenuItem>
</div>
<div class="py-1">
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Day view</a>
</MenuItem>
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Week view</a>
</MenuItem>
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Month view</a>
</MenuItem>
<MenuItem v-slot="{ active }">
<a href="#" :class="[active ? 'bg-gray-100 text-gray-900' : 'text-gray-700', 'block px-4 py-2 text-sm']">Year view</a>
</MenuItem>
</div>
</MenuItems>
</transition>
</Menu>
</div>
</header>
<div class="shadow ring-1 ring-black ring-opacity-5 lg:flex lg:flex-auto lg:flex-col">
<div class="grid grid-cols-7 gap-px border-b border-gray-300 bg-gray-200 text-center text-xs font-semibold leading-6 text-gray-700 lg:flex-none">
<div class="bg-white py-2">M<span class="sr-only sm:not-sr-only">on</span></div>
<div class="bg-white py-2">T<span class="sr-only sm:not-sr-only">ue</span></div>
<div class="bg-white py-2">W<span class="sr-only sm:not-sr-only">ed</span></div>
<div class="bg-white py-2">T<span class="sr-only sm:not-sr-only">hu</span></div>
<div class="bg-white py-2">F<span class="sr-only sm:not-sr-only">ri</span></div>
<div class="bg-white py-2">S<span class="sr-only sm:not-sr-only">at</span></div>
<div class="bg-white py-2">S<span class="sr-only sm:not-sr-only">un</span></div>
</div>
<div class="flex bg-gray-200 text-xs leading-6 text-gray-700 lg:flex-auto">
<div class="hidden w-full lg:grid lg:grid-cols-7 lg:grid-rows-6 lg:gap-px">
<div v-for="day in days" :key="day.date" :class="[day.isCurrentMonth ? 'bg-white' : 'bg-gray-50 text-gray-500', 'relative px-3 py-2']">
<time :datetime="day.date" :class="day.isToday ? 'flex h-6 w-6 items-center justify-center rounded-full bg-indigo-600 font-semibold text-white' : undefined">{{ day.date.split('-').pop().replace(/^0/, '') }}</time>
<ol v-if="day.events.length > 0" class="mt-2">
<li v-for="event in day.events.slice(0, 2)" :key="event.id">
<a :href="event.href" class="group flex">
<p class="flex-auto truncate font-medium text-gray-900 group-hover:text-indigo-600">
{{ event.name }}
</p>
<time :datetime="event.datetime" class="ml-3 hidden flex-none text-gray-500 group-hover:text-indigo-600 xl:block">{{ event.time }}</time>
</a>
</li>
<li v-if="day.events.length > 2" class="text-gray-500">+ {{ day.events.length - 2 }} more</li>
</ol>
</div>
</div>
<div class="isolate grid w-full grid-cols-7 grid-rows-6 gap-px lg:hidden">
<button v-for="day in days" :key="day.date" type="button" :class="[day.isCurrentMonth ? 'bg-white' : 'bg-gray-50', (day.isSelected || day.isToday) && 'font-semibold', day.isSelected && 'text-white', !day.isSelected && day.isToday && 'text-indigo-600', !day.isSelected && day.isCurrentMonth && !day.isToday && 'text-gray-900', !day.isSelected && !day.isCurrentMonth && !day.isToday && 'text-gray-500', 'flex h-14 flex-col px-3 py-2 hover:bg-gray-100 focus:z-10']">
<time :datetime="day.date" :class="[day.isSelected && 'flex h-6 w-6 items-center justify-center rounded-full', day.isSelected && day.isToday && 'bg-indigo-600', day.isSelected && !day.isToday && 'bg-gray-900', 'ml-auto']">{{ day.date.split('-').pop().replace(/^0/, '') }}</time>
<span class="sr-only">{{ day.events.length }} events</span>
<span v-if="day.events.length > 0" class="-mx-0.5 mt-auto flex flex-wrap-reverse">
<span v-for="event in day.events" :key="event.id" class="mx-0.5 mb-1 h-1.5 w-1.5 rounded-full bg-gray-400" />
</span>
</button>
</div>
</div>
</div>
<div v-if="selectedDay?.events.length > 0" class="px-4 py-10 sm:px-6 lg:hidden">
<ol class="divide-y divide-gray-100 overflow-hidden rounded-lg bg-white text-sm shadow ring-1 ring-black ring-opacity-5">
<li v-for="event in selectedDay.events" :key="event.id" class="group flex p-4 pr-6 focus-within:bg-gray-50 hover:bg-gray-50">
<div class="flex-auto">
<p class="font-semibold text-gray-900">{{ event.name }}</p>
<time :datetime="event.datetime" class="mt-2 flex items-center text-gray-700">
<ClockIcon class="mr-2 h-5 w-5 text-gray-400" aria-hidden="true" />
{{ event.time }}
</time>
</div>
<a :href="event.href" class="ml-6 flex-none self-center rounded-md bg-white px-3 py-2 font-semibold text-gray-900 opacity-0 shadow-sm ring-1 ring-inset ring-gray-300 hover:ring-gray-400 focus:opacity-100 group-hover:opacity-100"
>Edit<span class="sr-only">, {{ event.name }}</span></a
>
</li>
</ol>
</div>
</div>
</template>
<script setup>
import {
ChevronDownIcon,
ChevronLeftIcon,
ChevronRightIcon,
ClockIcon,
EllipsisHorizontalIcon,
} from '@heroicons/vue/20/solid'
import { Menu, MenuButton, MenuItem, MenuItems } from '@headlessui/vue'
const days = [
{ date: '2021-12-27', events: [] },
{ date: '2021-12-28', events: [] },
{ date: '2021-12-29', events: [] },
{ date: '2021-12-30', events: [] },
{ date: '2021-12-31', events: [] },
{ date: '2022-01-01', isCurrentMonth: true, events: [] },
{ date: '2022-01-02', isCurrentMonth: true, events: [] },
{
date: '2022-01-03',
isCurrentMonth: true,
events: [
{ id: 1, name: 'Design review', time: '10AM', datetime: '2022-01-03T10:00', href: '#' },
{ id: 2, name: 'Sales meeting', time: '2PM', datetime: '2022-01-03T14:00', href: '#' },
],
},
{ date: '2022-01-04', isCurrentMonth: true, events: [] },
{ date: '2022-01-05', isCurrentMonth: true, events: [] },
{ date: '2022-01-06', isCurrentMonth: true, events: [] },
{
date: '2022-01-07',
isCurrentMonth: true,
events: [{ id: 3, name: 'Date night', time: '6PM', datetime: '2022-01-08T18:00', href: '#' }],
},
{ date: '2022-01-08', isCurrentMonth: true, events: [] },
{ date: '2022-01-09', isCurrentMonth: true, events: [] },
{ date: '2022-01-10', isCurrentMonth: true, events: [] },
{ date: '2022-01-11', isCurrentMonth: true, events: [] },
{
date: '2022-01-12',
isCurrentMonth: true,
isToday: true,
events: [{ id: 6, name: "Sam's birthday party", time: '2PM', datetime: '2022-01-25T14:00', href: '#' }],
},
{ date: '2022-01-13', isCurrentMonth: true, events: [] },
{ date: '2022-01-14', isCurrentMonth: true, events: [] },
{ date: '2022-01-15', isCurrentMonth: true, events: [] },
{ date: '2022-01-16', isCurrentMonth: true, events: [] },
{ date: '2022-01-17', isCurrentMonth: true, events: [] },
{ date: '2022-01-18', isCurrentMonth: true, events: [] },
{ date: '2022-01-19', isCurrentMonth: true, events: [] },
{ date: '2022-01-20', isCurrentMonth: true, events: [] },
{ date: '2022-01-21', isCurrentMonth: true, events: [] },
{
date: '2022-01-22',
isCurrentMonth: true,
isSelected: true,
events: [
{ id: 4, name: 'Maple syrup museum', time: '3PM', datetime: '2022-01-22T15:00', href: '#' },
{ id: 5, name: 'Hockey game', time: '7PM', datetime: '2022-01-22T19:00', href: '#' },
],
},
{ date: '2022-01-23', isCurrentMonth: true, events: [] },
{ date: '2022-01-24', isCurrentMonth: true, events: [] },
{ date: '2022-01-25', isCurrentMonth: true, events: [] },
{ date: '2022-01-26', isCurrentMonth: true, events: [] },
{ date: '2022-01-27', isCurrentMonth: true, events: [] },
{ date: '2022-01-28', isCurrentMonth: true, events: [] },
{ date: '2022-01-29', isCurrentMonth: true, events: [] },
{ date: '2022-01-30', isCurrentMonth: true, events: [] },
{ date: '2022-01-31', isCurrentMonth: true, events: [] },
{ date: '2022-02-01', events: [] },
{ date: '2022-02-02', events: [] },
{ date: '2022-02-03', events: [] },
{
date: '2022-02-04',
events: [{ id: 7, name: 'Cinema with friends', time: '9PM', datetime: '2022-02-04T21:00', href: '#' }],
},
{ date: '2022-02-05', events: [] },
{ date: '2022-02-06', events: [] },
]
const selectedDay = days.find((day) => day.isSelected)
</script>