feat: un/assigned commits on version update
This commit is contained in:
parent
d6cb69ec3b
commit
b7f0713d6f
@ -1,7 +1,12 @@
|
|||||||
import { changelog, changelog_version, db } from '@boring.tools/database'
|
import {
|
||||||
|
changelog,
|
||||||
|
changelog_commit,
|
||||||
|
changelog_version,
|
||||||
|
db,
|
||||||
|
} from '@boring.tools/database'
|
||||||
import { VersionByIdParams, VersionOutput } from '@boring.tools/schema'
|
import { VersionByIdParams, VersionOutput } from '@boring.tools/schema'
|
||||||
import { createRoute } from '@hono/zod-openapi'
|
import { createRoute } from '@hono/zod-openapi'
|
||||||
import { and, eq } from 'drizzle-orm'
|
import { and, desc, eq } from 'drizzle-orm'
|
||||||
|
|
||||||
import { HTTPException } from 'hono/http-exception'
|
import { HTTPException } from 'hono/http-exception'
|
||||||
import type changelogVersionApi from '.'
|
import type changelogVersionApi from '.'
|
||||||
@ -37,7 +42,9 @@ export const registerVersionById = (api: typeof changelogVersionApi) => {
|
|||||||
const versionResult = await db.query.changelog_version.findFirst({
|
const versionResult = await db.query.changelog_version.findFirst({
|
||||||
where: eq(changelog_version.id, id),
|
where: eq(changelog_version.id, id),
|
||||||
with: {
|
with: {
|
||||||
commits: true,
|
commits: {
|
||||||
|
orderBy: () => desc(changelog_commit.createdAt),
|
||||||
|
},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -1,7 +1,12 @@
|
|||||||
import { changelog, changelog_version, db } from '@boring.tools/database'
|
import {
|
||||||
|
changelog,
|
||||||
|
changelog_commit,
|
||||||
|
changelog_version,
|
||||||
|
db,
|
||||||
|
} from '@boring.tools/database'
|
||||||
import { VersionUpdateInput, VersionUpdateOutput } from '@boring.tools/schema'
|
import { VersionUpdateInput, VersionUpdateOutput } from '@boring.tools/schema'
|
||||||
import { createRoute, type z } from '@hono/zod-openapi'
|
import { createRoute, type z } from '@hono/zod-openapi'
|
||||||
import { and, eq } from 'drizzle-orm'
|
import { and, eq, inArray, notInArray } from 'drizzle-orm'
|
||||||
import { HTTPException } from 'hono/http-exception'
|
import { HTTPException } from 'hono/http-exception'
|
||||||
|
|
||||||
import type changelogVersionApi from '.'
|
import type changelogVersionApi from '.'
|
||||||
@ -74,6 +79,18 @@ export const registerVersionUpdate = (api: typeof changelogVersionApi) => {
|
|||||||
.where(and(eq(changelog_version.id, id)))
|
.where(and(eq(changelog_version.id, id)))
|
||||||
.returning()
|
.returning()
|
||||||
|
|
||||||
|
if (payload.commitIds) {
|
||||||
|
await db
|
||||||
|
.update(changelog_commit)
|
||||||
|
.set({ versionId: null })
|
||||||
|
.where(notInArray(changelog_commit.id, payload.commitIds))
|
||||||
|
|
||||||
|
await db
|
||||||
|
.update(changelog_commit)
|
||||||
|
.set({ versionId: versionUpdateResult.id })
|
||||||
|
.where(inArray(changelog_commit.id, payload.commitIds))
|
||||||
|
}
|
||||||
|
|
||||||
if (findChangelog.pageId) {
|
if (findChangelog.pageId) {
|
||||||
redis.del(findChangelog.pageId)
|
redis.del(findChangelog.pageId)
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,11 @@ import { VersionUpdateInput } from '@boring.tools/schema'
|
|||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
Calendar,
|
Calendar,
|
||||||
|
Card,
|
||||||
|
CardContent,
|
||||||
|
CardHeader,
|
||||||
|
CardTitle,
|
||||||
|
Checkbox,
|
||||||
Form,
|
Form,
|
||||||
FormControl,
|
FormControl,
|
||||||
FormField,
|
FormField,
|
||||||
@ -12,12 +17,17 @@ import {
|
|||||||
Popover,
|
Popover,
|
||||||
PopoverContent,
|
PopoverContent,
|
||||||
PopoverTrigger,
|
PopoverTrigger,
|
||||||
|
ScrollArea,
|
||||||
Select,
|
Select,
|
||||||
SelectContent,
|
SelectContent,
|
||||||
SelectItem,
|
SelectItem,
|
||||||
SelectTrigger,
|
SelectTrigger,
|
||||||
SelectValue,
|
SelectValue,
|
||||||
Separator,
|
Separator,
|
||||||
|
Tabs,
|
||||||
|
TabsContent,
|
||||||
|
TabsList,
|
||||||
|
TabsTrigger,
|
||||||
cn,
|
cn,
|
||||||
} from '@boring.tools/ui'
|
} from '@boring.tools/ui'
|
||||||
import { zodResolver } from '@hookform/resolvers/zod'
|
import { zodResolver } from '@hookform/resolvers/zod'
|
||||||
@ -38,6 +48,7 @@ import { createFileRoute, useNavigate } from '@tanstack/react-router'
|
|||||||
import { useForm } from 'react-hook-form'
|
import { useForm } from 'react-hook-form'
|
||||||
import type { z } from 'zod'
|
import type { z } from 'zod'
|
||||||
import {
|
import {
|
||||||
|
useChangelogCommitList,
|
||||||
useChangelogVersionById,
|
useChangelogVersionById,
|
||||||
useChangelogVersionUpdate,
|
useChangelogVersionUpdate,
|
||||||
} from '../hooks/useChangelog'
|
} from '../hooks/useChangelog'
|
||||||
@ -56,6 +67,7 @@ const Component = () => {
|
|||||||
const { data, error, isPending, refetch } = useChangelogVersionById({
|
const { data, error, isPending, refetch } = useChangelogVersionById({
|
||||||
id: versionId,
|
id: versionId,
|
||||||
})
|
})
|
||||||
|
const commitResult = useChangelogCommitList({ id })
|
||||||
|
|
||||||
const form = useForm<z.infer<typeof VersionUpdateInput>>({
|
const form = useForm<z.infer<typeof VersionUpdateInput>>({
|
||||||
resolver: zodResolver(VersionUpdateInput),
|
resolver: zodResolver(VersionUpdateInput),
|
||||||
@ -76,7 +88,10 @@ const Component = () => {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (data) {
|
if (data) {
|
||||||
mdxEditorRef.current?.setMarkdown(data.markdown)
|
mdxEditorRef.current?.setMarkdown(data.markdown)
|
||||||
form.reset(data)
|
form.reset({
|
||||||
|
...data,
|
||||||
|
commitIds: data.commits?.map((commit) => commit.id),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}, [data, form])
|
}, [data, form])
|
||||||
|
|
||||||
@ -95,102 +110,27 @@ const Component = () => {
|
|||||||
<div className="flex flex-col gap-5">
|
<div className="flex flex-col gap-5">
|
||||||
<Separator />
|
<Separator />
|
||||||
{!isPending && data && (
|
{!isPending && data && (
|
||||||
<div>
|
<Form {...form}>
|
||||||
<Form {...form}>
|
<form
|
||||||
<form
|
onSubmit={form.handleSubmit(onSubmit)}
|
||||||
onSubmit={form.handleSubmit(onSubmit)}
|
className="flex gap-4 w-full"
|
||||||
className="space-y-8 max-w-screen-md"
|
>
|
||||||
>
|
<Card className="w-full">
|
||||||
<FormField
|
<CardHeader>
|
||||||
control={form.control}
|
<div className="flex items-center justify-between">
|
||||||
name="version"
|
<CardTitle>Details</CardTitle>
|
||||||
render={({ field }) => (
|
</div>
|
||||||
<FormItem>
|
</CardHeader>
|
||||||
<FormLabel>Version</FormLabel>
|
<CardContent>
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="v1.0.1" {...field} />
|
|
||||||
</FormControl>{' '}
|
|
||||||
<FormMessage />
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="markdown"
|
|
||||||
render={({ field }) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Notes</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<MDXEditor
|
|
||||||
className="dark-theme"
|
|
||||||
contentEditableClassName="prose dark:prose-invert max-w-none"
|
|
||||||
markdown={''}
|
|
||||||
ref={mdxEditorRef}
|
|
||||||
onChange={field.onChange}
|
|
||||||
onBlur={field.onBlur}
|
|
||||||
plugins={[
|
|
||||||
headingsPlugin(),
|
|
||||||
listsPlugin(),
|
|
||||||
thematicBreakPlugin(),
|
|
||||||
quotePlugin(),
|
|
||||||
|
|
||||||
toolbarPlugin({
|
|
||||||
toolbarContents: () => (
|
|
||||||
<>
|
|
||||||
<BlockTypeSelect />
|
|
||||||
<BoldItalicUnderlineToggles />
|
|
||||||
<ListsToggle />
|
|
||||||
<UndoRedo />
|
|
||||||
</>
|
|
||||||
),
|
|
||||||
}),
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
</FormControl>{' '}
|
|
||||||
<FormMessage />
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<div className="flex gap-5 items-center">
|
|
||||||
<FormField
|
<FormField
|
||||||
control={form.control}
|
control={form.control}
|
||||||
name="status"
|
name="version"
|
||||||
render={({ field }) => (
|
render={({ field }) => (
|
||||||
<FormItem>
|
<FormItem>
|
||||||
<FormLabel>Status</FormLabel>
|
<FormLabel>Version</FormLabel>
|
||||||
<Select
|
<FormControl>
|
||||||
onValueChange={field.onChange}
|
<Input placeholder="v1.0.1" {...field} />
|
||||||
defaultValue={field.value}
|
</FormControl>{' '}
|
||||||
value={field.value}
|
|
||||||
>
|
|
||||||
<FormControl>
|
|
||||||
<SelectTrigger>
|
|
||||||
<SelectValue placeholder="Select your version status" />
|
|
||||||
</SelectTrigger>
|
|
||||||
</FormControl>
|
|
||||||
<SelectContent>
|
|
||||||
<SelectItem value="draft">
|
|
||||||
<div className="flex gap-2 items-center">
|
|
||||||
<VersionStatus status={'draft'} />
|
|
||||||
<span>Draft</span>
|
|
||||||
</div>
|
|
||||||
</SelectItem>
|
|
||||||
<SelectItem value="review">
|
|
||||||
<div className="flex gap-2 items-center">
|
|
||||||
<VersionStatus status={'review'} />
|
|
||||||
<span>Review</span>
|
|
||||||
</div>
|
|
||||||
</SelectItem>
|
|
||||||
<SelectItem value="published">
|
|
||||||
<div className="flex gap-2 items-center">
|
|
||||||
<VersionStatus status={'published'} />
|
|
||||||
<span>Published</span>
|
|
||||||
</div>
|
|
||||||
</SelectItem>
|
|
||||||
</SelectContent>
|
|
||||||
</Select>
|
|
||||||
<FormMessage />
|
<FormMessage />
|
||||||
</FormItem>
|
</FormItem>
|
||||||
)}
|
)}
|
||||||
@ -198,62 +138,286 @@ const Component = () => {
|
|||||||
|
|
||||||
<FormField
|
<FormField
|
||||||
control={form.control}
|
control={form.control}
|
||||||
name="releasedAt"
|
name="markdown"
|
||||||
render={({ field }) => (
|
render={({ field }) => (
|
||||||
<FormItem className="flex flex-col">
|
<FormItem>
|
||||||
<FormLabel className="mb-2">Released at</FormLabel>
|
<FormLabel>Notes</FormLabel>
|
||||||
<Popover>
|
<FormControl>
|
||||||
<PopoverTrigger asChild>
|
<MDXEditor
|
||||||
<FormControl>
|
className="dark-theme"
|
||||||
<Button
|
contentEditableClassName="prose dark:prose-invert max-w-none"
|
||||||
variant={'outline'}
|
markdown={''}
|
||||||
size={'lg'}
|
ref={mdxEditorRef}
|
||||||
className={cn(
|
onChange={field.onChange}
|
||||||
'w-[240px] pl-3 text-left font-normal',
|
onBlur={field.onBlur}
|
||||||
!field.value && 'text-muted-foreground',
|
plugins={[
|
||||||
)}
|
headingsPlugin(),
|
||||||
>
|
listsPlugin(),
|
||||||
{field.value ? (
|
thematicBreakPlugin(),
|
||||||
format(field.value, 'PPP')
|
quotePlugin(),
|
||||||
) : (
|
|
||||||
<span>Pick a date</span>
|
toolbarPlugin({
|
||||||
)}
|
toolbarContents: () => (
|
||||||
<CalendarIcon className="ml-auto h-4 w-4 opacity-50" />
|
<>
|
||||||
</Button>
|
<BlockTypeSelect />
|
||||||
</FormControl>
|
<BoldItalicUnderlineToggles />
|
||||||
</PopoverTrigger>
|
<ListsToggle />
|
||||||
<PopoverContent className="w-auto p-0" align="start">
|
<UndoRedo />
|
||||||
<Calendar
|
</>
|
||||||
mode="single"
|
),
|
||||||
selected={field.value as Date}
|
}),
|
||||||
onSelect={(date) => field.onChange(date)}
|
]}
|
||||||
weekStartsOn={1}
|
/>
|
||||||
/>
|
</FormControl>{' '}
|
||||||
</PopoverContent>
|
|
||||||
</Popover>
|
|
||||||
<FormMessage />
|
<FormMessage />
|
||||||
</FormItem>
|
</FormItem>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="flex gap-5">
|
<div className="flex gap-5 items-center">
|
||||||
<Button
|
<FormField
|
||||||
type="button"
|
control={form.control}
|
||||||
variant={'ghost'}
|
name="status"
|
||||||
onClick={() =>
|
render={({ field }) => (
|
||||||
navigate({ to: '/changelog/$id', params: { id } })
|
<FormItem>
|
||||||
}
|
<FormLabel>Status</FormLabel>
|
||||||
>
|
<Select
|
||||||
Cancel
|
onValueChange={field.onChange}
|
||||||
</Button>
|
defaultValue={field.value}
|
||||||
<Button type="submit">Update</Button>
|
value={field.value}
|
||||||
</div>
|
>
|
||||||
</form>
|
<FormControl>
|
||||||
</Form>
|
<SelectTrigger>
|
||||||
|
<SelectValue placeholder="Select your version status" />
|
||||||
|
</SelectTrigger>
|
||||||
|
</FormControl>
|
||||||
|
<SelectContent>
|
||||||
|
<SelectItem value="draft">
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<VersionStatus status={'draft'} />
|
||||||
|
<span>Draft</span>
|
||||||
|
</div>
|
||||||
|
</SelectItem>
|
||||||
|
<SelectItem value="review">
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<VersionStatus status={'review'} />
|
||||||
|
<span>Review</span>
|
||||||
|
</div>
|
||||||
|
</SelectItem>
|
||||||
|
<SelectItem value="published">
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<VersionStatus status={'published'} />
|
||||||
|
<span>Published</span>
|
||||||
|
</div>
|
||||||
|
</SelectItem>
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
|
||||||
<ChangelogVersionDelete id={id} versionId={versionId} />
|
<FormField
|
||||||
</div>
|
control={form.control}
|
||||||
|
name="releasedAt"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem className="flex flex-col">
|
||||||
|
<FormLabel className="mb-2">Released at</FormLabel>
|
||||||
|
<Popover>
|
||||||
|
<PopoverTrigger asChild>
|
||||||
|
<FormControl>
|
||||||
|
<Button
|
||||||
|
variant={'outline'}
|
||||||
|
size={'lg'}
|
||||||
|
className={cn(
|
||||||
|
'w-[240px] pl-3 text-left font-normal',
|
||||||
|
!field.value && 'text-muted-foreground',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{field.value ? (
|
||||||
|
format(field.value, 'PPP')
|
||||||
|
) : (
|
||||||
|
<span>Pick a date</span>
|
||||||
|
)}
|
||||||
|
<CalendarIcon className="ml-auto h-4 w-4 opacity-50" />
|
||||||
|
</Button>
|
||||||
|
</FormControl>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent className="w-auto p-0" align="start">
|
||||||
|
<Calendar
|
||||||
|
mode="single"
|
||||||
|
selected={field.value as Date}
|
||||||
|
onSelect={(date) => field.onChange(date)}
|
||||||
|
weekStartsOn={1}
|
||||||
|
/>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex gap-5">
|
||||||
|
<Button
|
||||||
|
type="button"
|
||||||
|
variant={'ghost'}
|
||||||
|
onClick={() =>
|
||||||
|
navigate({ to: '/changelog/$id', params: { id } })
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button type="submit">Update</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ChangelogVersionDelete id={id} versionId={versionId} />
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
<div className="w-full">
|
||||||
|
<Card className="w-full max-w-screen-sm">
|
||||||
|
<CardHeader>
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<CardTitle>Commits ({data.commits?.length})</CardTitle>
|
||||||
|
</div>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<Tabs defaultValue="assigned" className="w-full">
|
||||||
|
<TabsList>
|
||||||
|
<TabsTrigger value="assigned">Assigend</TabsTrigger>
|
||||||
|
<TabsTrigger value="unassigned">Unassigned</TabsTrigger>
|
||||||
|
</TabsList>
|
||||||
|
<TabsContent value="assigned">
|
||||||
|
<ScrollArea className="w-full h-[350px]">
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
{data?.commits?.map((commit) => {
|
||||||
|
return (
|
||||||
|
<FormField
|
||||||
|
key={commit.id}
|
||||||
|
control={form.control}
|
||||||
|
name={'commitIds'}
|
||||||
|
render={({ field }) => {
|
||||||
|
return (
|
||||||
|
<FormItem className="flex flex-row items-start space-x-3 space-y-0 rounded-md ">
|
||||||
|
<FormControl>
|
||||||
|
<Checkbox
|
||||||
|
value={commit.id}
|
||||||
|
checked={field.value?.includes(
|
||||||
|
commit.id,
|
||||||
|
)}
|
||||||
|
onCheckedChange={() => {
|
||||||
|
const exist = field.value?.includes(
|
||||||
|
commit.id,
|
||||||
|
)
|
||||||
|
if (exist) {
|
||||||
|
return field.onChange(
|
||||||
|
field.value?.filter(
|
||||||
|
(value) =>
|
||||||
|
value !== commit.id,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return field.onChange([
|
||||||
|
...(field.value as string[]),
|
||||||
|
commit.id,
|
||||||
|
])
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
<div className="space-y-1 leading-none w-full">
|
||||||
|
<FormLabel className="flex gap-2 w-full">
|
||||||
|
<span className="text-muted-foreground font-mono">
|
||||||
|
{commit.commit}{' '}
|
||||||
|
</span>
|
||||||
|
<span className="w-full">
|
||||||
|
{commit.subject}
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
{format(
|
||||||
|
new Date(commit.commiter.date),
|
||||||
|
'dd.MM.',
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
</FormLabel>
|
||||||
|
</div>
|
||||||
|
</FormItem>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</ScrollArea>
|
||||||
|
</TabsContent>
|
||||||
|
<TabsContent value="unassigned">
|
||||||
|
<ScrollArea className="w-full h-[350px]">
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
{commitResult.data?.map((commit) => {
|
||||||
|
return (
|
||||||
|
<FormField
|
||||||
|
key={commit.id}
|
||||||
|
control={form.control}
|
||||||
|
name={'commitIds'}
|
||||||
|
render={({ field }) => {
|
||||||
|
return (
|
||||||
|
<FormItem className="flex flex-row items-start space-x-3 space-y-0 rounded-md ">
|
||||||
|
<FormControl>
|
||||||
|
<Checkbox
|
||||||
|
value={commit.id}
|
||||||
|
checked={field.value?.includes(
|
||||||
|
commit.id,
|
||||||
|
)}
|
||||||
|
onCheckedChange={() => {
|
||||||
|
const exist = field.value?.includes(
|
||||||
|
commit.id,
|
||||||
|
)
|
||||||
|
if (exist) {
|
||||||
|
return field.onChange(
|
||||||
|
field.value?.filter(
|
||||||
|
(value) =>
|
||||||
|
value !== commit.id,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return field.onChange([
|
||||||
|
...(field.value as string[]),
|
||||||
|
commit.id,
|
||||||
|
])
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
<div className="space-y-1 leading-none w-full">
|
||||||
|
<FormLabel className="flex gap-2 w-full">
|
||||||
|
<span className="text-muted-foreground font-mono">
|
||||||
|
{commit.commit}
|
||||||
|
</span>
|
||||||
|
<span className="w-full">
|
||||||
|
{commit.subject}
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
{format(
|
||||||
|
new Date(commit.commiter.date),
|
||||||
|
'dd.MM.',
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
</FormLabel>
|
||||||
|
</div>
|
||||||
|
</FormItem>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</ScrollArea>
|
||||||
|
</TabsContent>
|
||||||
|
</Tabs>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</Form>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import { z } from '@hono/zod-openapi'
|
import { z } from '@hono/zod-openapi'
|
||||||
|
import { CommitOutput } from '../commit'
|
||||||
|
|
||||||
export const VersionOutput = z
|
export const VersionOutput = z
|
||||||
.object({
|
.object({
|
||||||
@ -20,5 +21,6 @@ export const VersionOutput = z
|
|||||||
status: z.enum(['draft', 'review', 'published']).default('draft').openapi({
|
status: z.enum(['draft', 'review', 'published']).default('draft').openapi({
|
||||||
example: 'draft',
|
example: 'draft',
|
||||||
}),
|
}),
|
||||||
|
commits: z.array(CommitOutput).optional(),
|
||||||
})
|
})
|
||||||
.openapi('Version')
|
.openapi('Version')
|
||||||
|
@ -11,6 +11,7 @@ export const VersionUpdateInput = z
|
|||||||
.default('draft')
|
.default('draft')
|
||||||
.optional(),
|
.optional(),
|
||||||
releasedAt: z.date().or(z.string()).optional().nullable(),
|
releasedAt: z.date().or(z.string()).optional().nullable(),
|
||||||
|
commitIds: z.array(z.string()).optional(),
|
||||||
})
|
})
|
||||||
.openapi({})
|
.openapi({})
|
||||||
export const VersionUpdateParams = z
|
export const VersionUpdateParams = z
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
"@radix-ui/react-select": "^2.1.2",
|
"@radix-ui/react-select": "^2.1.2",
|
||||||
"@radix-ui/react-separator": "^1.1.0",
|
"@radix-ui/react-separator": "^1.1.0",
|
||||||
"@radix-ui/react-slot": "^1.1.0",
|
"@radix-ui/react-slot": "^1.1.0",
|
||||||
|
"@radix-ui/react-tabs": "^1.1.1",
|
||||||
"@radix-ui/react-tooltip": "^1.1.3",
|
"@radix-ui/react-tooltip": "^1.1.3",
|
||||||
"class-variance-authority": "^0.7.0",
|
"class-variance-authority": "^0.7.0",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
|
@ -30,3 +30,4 @@ export * from './command'
|
|||||||
export * from './dialog'
|
export * from './dialog'
|
||||||
export * from './scroll-area'
|
export * from './scroll-area'
|
||||||
export * from './table'
|
export * from './table'
|
||||||
|
export * from './tabs'
|
||||||
|
53
packages/ui/src/tabs.tsx
Normal file
53
packages/ui/src/tabs.tsx
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import * as TabsPrimitive from '@radix-ui/react-tabs'
|
||||||
|
import * as React from 'react'
|
||||||
|
|
||||||
|
import { cn } from './lib/cn'
|
||||||
|
|
||||||
|
const Tabs = TabsPrimitive.Root
|
||||||
|
|
||||||
|
const TabsList = React.forwardRef<
|
||||||
|
React.ElementRef<typeof TabsPrimitive.List>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof TabsPrimitive.List>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<TabsPrimitive.List
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
'inline-flex h-10 items-center justify-center rounded-md bg-muted p-1 text-muted-foreground',
|
||||||
|
className,
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
TabsList.displayName = TabsPrimitive.List.displayName
|
||||||
|
|
||||||
|
const TabsTrigger = React.forwardRef<
|
||||||
|
React.ElementRef<typeof TabsPrimitive.Trigger>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof TabsPrimitive.Trigger>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<TabsPrimitive.Trigger
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
'inline-flex items-center justify-center whitespace-nowrap rounded-sm px-3 py-1.5 text-sm font-medium ring-offset-background transition-all focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:shadow-sm',
|
||||||
|
className,
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
TabsTrigger.displayName = TabsPrimitive.Trigger.displayName
|
||||||
|
|
||||||
|
const TabsContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof TabsPrimitive.Content>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof TabsPrimitive.Content>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<TabsPrimitive.Content
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
'mt-2 ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2',
|
||||||
|
className,
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
TabsContent.displayName = TabsPrimitive.Content.displayName
|
||||||
|
|
||||||
|
export { Tabs, TabsList, TabsTrigger, TabsContent }
|
Loading…
Reference in New Issue
Block a user