feat(app): changelog version auto create
This commit is contained in:
parent
ede09bcf04
commit
1be0facd64
@ -1,17 +1,21 @@
|
|||||||
import { Link, useParams } from '@tanstack/react-router'
|
import { Link, useParams } from '@tanstack/react-router'
|
||||||
import { HandIcon, WorkflowIcon } from 'lucide-react'
|
import { HandIcon, WorkflowIcon } from 'lucide-react'
|
||||||
|
|
||||||
export const ChangelogVersionCreateStep01 = () => {
|
export const ChangelogVersionCreateStep01 = ({
|
||||||
|
nextStep,
|
||||||
|
}: { nextStep: () => void }) => {
|
||||||
const { id } = useParams({ from: '/changelog/$id' })
|
const { id } = useParams({ from: '/changelog/$id' })
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex gap-10 mt-3">
|
<div className="flex gap-10 mt-3">
|
||||||
<div className="border rounded border-muted p-5 flex items-center justify-center w-full flex-col">
|
<button
|
||||||
|
type="button"
|
||||||
|
className="flex-col hover:border-accent border rounded border-muted p-5 flex items-center justify-center w-full"
|
||||||
|
onClick={nextStep}
|
||||||
|
>
|
||||||
<WorkflowIcon />
|
<WorkflowIcon />
|
||||||
Automatic
|
Automatic
|
||||||
<small className="uppercase text-muted-foreground text-xs">
|
</button>
|
||||||
Coming soon
|
|
||||||
</small>
|
|
||||||
</div>
|
|
||||||
<Link
|
<Link
|
||||||
className="flex-col hover:border-accent border rounded border-muted p-5 flex items-center justify-center w-full"
|
className="flex-col hover:border-accent border rounded border-muted p-5 flex items-center justify-center w-full"
|
||||||
to="/changelog/$id/versionCreate"
|
to="/changelog/$id/versionCreate"
|
||||||
|
74
apps/app/src/components/Changelog/Version/Create/Step02.tsx
Normal file
74
apps/app/src/components/Changelog/Version/Create/Step02.tsx
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
import { VersionCreateAutoInput } from '@boring.tools/schema'
|
||||||
|
import {
|
||||||
|
Button,
|
||||||
|
Form,
|
||||||
|
FormControl,
|
||||||
|
FormDescription,
|
||||||
|
FormField,
|
||||||
|
FormItem,
|
||||||
|
FormLabel,
|
||||||
|
FormMessage,
|
||||||
|
Input,
|
||||||
|
} from '@boring.tools/ui'
|
||||||
|
import { zodResolver } from '@hookform/resolvers/zod'
|
||||||
|
import { useNavigate, useParams } from '@tanstack/react-router'
|
||||||
|
import { useForm } from 'react-hook-form'
|
||||||
|
import type { z } from 'zod'
|
||||||
|
import { useChangelogVersionCreateAuto } from '../../../../hooks/useChangelog'
|
||||||
|
|
||||||
|
export const ChangelogVersionCreateStep02 = () => {
|
||||||
|
const { id } = useParams({ from: '/changelog/$id' })
|
||||||
|
const navigate = useNavigate({ from: `/changelog/${id}` })
|
||||||
|
const autoVersion = useChangelogVersionCreateAuto()
|
||||||
|
|
||||||
|
const form = useForm<z.infer<typeof VersionCreateAutoInput>>({
|
||||||
|
resolver: zodResolver(VersionCreateAutoInput),
|
||||||
|
defaultValues: {
|
||||||
|
changelogId: id,
|
||||||
|
version: null,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const onSubmit = (values: z.infer<typeof VersionCreateAutoInput>) => {
|
||||||
|
autoVersion.mutate(values, {
|
||||||
|
onSuccess(data) {
|
||||||
|
navigate({
|
||||||
|
to: '/changelog/$id/version/$versionId',
|
||||||
|
params: { id, versionId: data.id },
|
||||||
|
})
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex gap-10 mt-3">
|
||||||
|
<Form {...form}>
|
||||||
|
<form
|
||||||
|
onSubmit={form.handleSubmit(onSubmit)}
|
||||||
|
className="space-y-8 w-full"
|
||||||
|
>
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="version"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel>Version</FormLabel>
|
||||||
|
<FormControl>
|
||||||
|
<Input placeholder="v1.0.1" {...field} autoFocus />
|
||||||
|
</FormControl>{' '}
|
||||||
|
<FormDescription>
|
||||||
|
Leave blank for auto generating.
|
||||||
|
</FormDescription>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div className="flex gap-5 mt-5 w-full justify-end items-end md:col-span-6">
|
||||||
|
<Button type="submit">Create</Button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</Form>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
@ -7,11 +7,23 @@ import {
|
|||||||
DialogTrigger,
|
DialogTrigger,
|
||||||
} from '@boring.tools/ui'
|
} from '@boring.tools/ui'
|
||||||
import { PlusCircleIcon } from 'lucide-react'
|
import { PlusCircleIcon } from 'lucide-react'
|
||||||
|
import { useEffect, useState } from 'react'
|
||||||
import { ChangelogVersionCreateStep01 } from './Step01'
|
import { ChangelogVersionCreateStep01 } from './Step01'
|
||||||
|
import { ChangelogVersionCreateStep02 } from './Step02'
|
||||||
|
|
||||||
export const ChangelogVersionCreate = () => {
|
export const ChangelogVersionCreate = () => {
|
||||||
|
const [isOpen, setIsOpen] = useState(false)
|
||||||
|
const [step, setStep] = useState(1)
|
||||||
|
const nextStep = () => setStep((prev) => prev + 1)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isOpen) {
|
||||||
|
setStep(1)
|
||||||
|
}
|
||||||
|
}, [isOpen])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog>
|
<Dialog open={isOpen} onOpenChange={(state) => setIsOpen(state)}>
|
||||||
<DialogTrigger>
|
<DialogTrigger>
|
||||||
<PlusCircleIcon />
|
<PlusCircleIcon />
|
||||||
</DialogTrigger>
|
</DialogTrigger>
|
||||||
@ -25,7 +37,8 @@ export const ChangelogVersionCreate = () => {
|
|||||||
</DialogDescription>
|
</DialogDescription>
|
||||||
</DialogHeader>
|
</DialogHeader>
|
||||||
|
|
||||||
<ChangelogVersionCreateStep01 />
|
{step === 1 && <ChangelogVersionCreateStep01 nextStep={nextStep} />}
|
||||||
|
{step === 2 && <ChangelogVersionCreateStep02 />}
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
)
|
)
|
||||||
|
@ -3,6 +3,7 @@ import type {
|
|||||||
ChangelogOutput,
|
ChangelogOutput,
|
||||||
ChangelogUpdateInput,
|
ChangelogUpdateInput,
|
||||||
CommitOutput,
|
CommitOutput,
|
||||||
|
VersionCreateAutoInput,
|
||||||
VersionCreateInput,
|
VersionCreateInput,
|
||||||
VersionOutput,
|
VersionOutput,
|
||||||
VersionUpdateInput,
|
VersionUpdateInput,
|
||||||
@ -18,6 +19,7 @@ type ChangelogUpdate = z.infer<typeof ChangelogUpdateInput>
|
|||||||
|
|
||||||
type Version = z.infer<typeof VersionOutput>
|
type Version = z.infer<typeof VersionOutput>
|
||||||
type VersionCreate = z.infer<typeof VersionCreateInput>
|
type VersionCreate = z.infer<typeof VersionCreateInput>
|
||||||
|
type VersionCreateAuto = z.infer<typeof VersionCreateAutoInput>
|
||||||
type VersionUpdate = z.infer<typeof VersionUpdateInput>
|
type VersionUpdate = z.infer<typeof VersionUpdateInput>
|
||||||
|
|
||||||
type Commit = z.infer<typeof CommitOutput>
|
type Commit = z.infer<typeof CommitOutput>
|
||||||
@ -155,6 +157,29 @@ export const useChangelogVersionCreate = () => {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const useChangelogVersionCreateAuto = () => {
|
||||||
|
const { getToken } = useAuth()
|
||||||
|
const queryClient = useQueryClient()
|
||||||
|
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: async (
|
||||||
|
payload: VersionCreateAuto,
|
||||||
|
): Promise<Readonly<Version>> =>
|
||||||
|
await queryFetch({
|
||||||
|
path: 'changelog/version/auto',
|
||||||
|
data: payload,
|
||||||
|
method: 'post',
|
||||||
|
token: await getToken(),
|
||||||
|
}),
|
||||||
|
onSuccess: (data) => {
|
||||||
|
queryClient.invalidateQueries({ queryKey: ['changelogList'] })
|
||||||
|
queryClient.invalidateQueries({
|
||||||
|
queryKey: ['changelogById', data.changelogId],
|
||||||
|
})
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
export const useChangelogVersionById = ({ id }: { id: string }) => {
|
export const useChangelogVersionById = ({ id }: { id: string }) => {
|
||||||
const { getToken } = useAuth()
|
const { getToken } = useAuth()
|
||||||
|
|
||||||
|
@ -43,7 +43,7 @@ import {
|
|||||||
import '@mdxeditor/editor/style.css'
|
import '@mdxeditor/editor/style.css'
|
||||||
import { format } from 'date-fns'
|
import { format } from 'date-fns'
|
||||||
import { CalendarIcon } from 'lucide-react'
|
import { CalendarIcon } from 'lucide-react'
|
||||||
import { useEffect, useRef } from 'react'
|
import { useEffect, useRef, useState } from 'react'
|
||||||
import { ChangelogVersionDelete } from '../components/Changelog/VersionDelete'
|
import { ChangelogVersionDelete } from '../components/Changelog/VersionDelete'
|
||||||
import { VersionStatus } from '../components/Changelog/VersionStatus'
|
import { VersionStatus } from '../components/Changelog/VersionStatus'
|
||||||
|
|
||||||
@ -55,6 +55,7 @@ const Component = () => {
|
|||||||
const { data, error, isPending, refetch } = useChangelogVersionById({
|
const { data, error, isPending, refetch } = useChangelogVersionById({
|
||||||
id: versionId,
|
id: versionId,
|
||||||
})
|
})
|
||||||
|
const [editorValue, setEditorValue] = useState('')
|
||||||
const form = useForm<z.infer<typeof VersionUpdateInput>>({
|
const form = useForm<z.infer<typeof VersionUpdateInput>>({
|
||||||
resolver: zodResolver(VersionUpdateInput),
|
resolver: zodResolver(VersionUpdateInput),
|
||||||
defaultValues: data,
|
defaultValues: data,
|
||||||
@ -74,9 +75,8 @@ const Component = () => {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (data) {
|
if (data) {
|
||||||
mdxEditorRef.current?.setMarkdown(data.markdown)
|
mdxEditorRef.current?.setMarkdown(data.markdown)
|
||||||
form.reset(data)
|
|
||||||
}
|
}
|
||||||
}, [data, form.reset])
|
}, [data])
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
return (
|
return (
|
||||||
|
@ -29,6 +29,7 @@ const Component = () => {
|
|||||||
title: '',
|
title: '',
|
||||||
description: '',
|
description: '',
|
||||||
isSemver: true,
|
isSemver: true,
|
||||||
|
isConventional: true,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
@ -115,6 +116,33 @@ const Component = () => {
|
|||||||
</FormItem>
|
</FormItem>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="isConventional"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem className="flex flex-row items-start space-x-3 space-y-0 rounded-md ">
|
||||||
|
<FormControl>
|
||||||
|
<Checkbox
|
||||||
|
checked={field.value}
|
||||||
|
onCheckedChange={field.onChange}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
<div className="space-y-1 leading-none">
|
||||||
|
<FormLabel>Using Conventional Commits</FormLabel>
|
||||||
|
<FormDescription>
|
||||||
|
If this changelog is using{' '}
|
||||||
|
<a
|
||||||
|
href="https://www.conventionalcommits.org/en/v1.0.0/"
|
||||||
|
className="text-emerald-700"
|
||||||
|
>
|
||||||
|
conventional commits
|
||||||
|
</a>
|
||||||
|
</FormDescription>
|
||||||
|
</div>
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
<Button type="submit">Create</Button>
|
<Button type="submit">Create</Button>
|
||||||
</form>
|
</form>
|
||||||
</Form>
|
</Form>
|
||||||
|
22
bruno/Changelog/Version/Create Auto.bru
Normal file
22
bruno/Changelog/Version/Create Auto.bru
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
meta {
|
||||||
|
name: Create Auto
|
||||||
|
type: http
|
||||||
|
seq: 1
|
||||||
|
}
|
||||||
|
|
||||||
|
post {
|
||||||
|
url: {{API_URL}}/v1/changelog/version/auto
|
||||||
|
body: json
|
||||||
|
auth: bearer
|
||||||
|
}
|
||||||
|
|
||||||
|
auth:bearer {
|
||||||
|
token: {{TOKEN}}
|
||||||
|
}
|
||||||
|
|
||||||
|
body:json {
|
||||||
|
{
|
||||||
|
"changelogId": "d83fe688-3331-4e64-9af6-318f82e511d4",
|
||||||
|
"commitIds": ["01cc79df-6d16-4496-b9ba-b238d686efc4"]
|
||||||
|
}
|
||||||
|
}
|
@ -11,5 +11,5 @@ get {
|
|||||||
}
|
}
|
||||||
|
|
||||||
auth:bearer {
|
auth:bearer {
|
||||||
token: bt_7b83481e2ae0f6ab728730511adbe491f6668eb3
|
token: {{TOKEN}}
|
||||||
}
|
}
|
||||||
|
@ -1,3 +1,6 @@
|
|||||||
vars {
|
vars {
|
||||||
API_URL: http://localhost:3000
|
API_URL: http://localhost:3000
|
||||||
}
|
}
|
||||||
|
vars:secret [
|
||||||
|
TOKEN
|
||||||
|
]
|
||||||
|
@ -7,6 +7,7 @@ export const ChangelogCreateInput = z
|
|||||||
title: z.string().min(3, 'Title must contain at least 3 charachters.'),
|
title: z.string().min(3, 'Title must contain at least 3 charachters.'),
|
||||||
description: z.string(),
|
description: z.string(),
|
||||||
isSemver: z.boolean().default(true),
|
isSemver: z.boolean().default(true),
|
||||||
|
isConventional: z.boolean().default(true),
|
||||||
})
|
})
|
||||||
.openapi({
|
.openapi({
|
||||||
required: ['title', 'userId'],
|
required: ['title', 'userId'],
|
||||||
|
@ -14,3 +14,12 @@ export const VersionCreateInput = z
|
|||||||
.openapi({
|
.openapi({
|
||||||
required: ['changelogId', 'version', 'markdown', 'releasedAt'],
|
required: ['changelogId', 'version', 'markdown', 'releasedAt'],
|
||||||
})
|
})
|
||||||
|
|
||||||
|
export const VersionCreateAutoInput = z
|
||||||
|
.object({
|
||||||
|
version: z.string().optional().nullable(),
|
||||||
|
changelogId: z.string().uuid(),
|
||||||
|
})
|
||||||
|
.openapi({
|
||||||
|
required: ['changelogId'],
|
||||||
|
})
|
||||||
|
Loading…
Reference in New Issue
Block a user