Compare commits

...

17 Commits

Author SHA1 Message Date
Ardeman
57e23adf3c feat: update UI text to Indonesian language for dashboard and form components 2025-03-25 01:25:51 +08:00
Ardeman
35c02d1643 feat: update UI text to Indonesian language across various components and pages 2025-03-25 01:20:20 +08:00
Ardeman
1f2f15d204 fix: correct spelling of 'Dashboard' to 'Dasbor' in admin menu and dashboard page 2025-03-25 01:03:52 +08:00
Ardeman
39b7720186 fix: correct spelling of 'Staff' to 'Staf' and 'Tags' to 'Tag' in various components 2025-03-25 01:00:57 +08:00
Ardeman
de1802c597 feat: implement staff management features including staff creation and dashboard layout 2025-03-25 00:56:14 +08:00
Ardeman
724a14d741 feat: update getNews API query handling and enhance social share button styles 2025-03-25 00:21:09 +08:00
Ardeman
fa5f7fbe92 feat: modify query parameter formatting in getNews API to replace spaces with plus signs 2025-03-24 19:55:47 +08:00
Ardeman
4ff1e23d25 feat: implement search functionality for news articles with query parameter 2025-03-24 19:42:27 +08:00
Ardeman
9ab67c615a feat: move views field into data response schema for news API 2025-03-24 17:22:57 +08:00
Ardeman
65f7bbe0aa feat: add views field to news response schema and update dashboard pages 2025-03-24 15:45:43 +08:00
Ardeman
1c33eba834 feat: enhance table styling in ContentsPage with additional class names for better readability 2025-03-24 13:52:31 +08:00
Ardeman
1885eab4c3 feat: add API endpoints for fetching news by ID and slug, and update routing accordingly 2025-03-24 13:42:36 +08:00
Ardeman
f2f49de86b feat: update Autoplay settings in Banner component to stop on mouse enter 2025-03-24 13:25:15 +08:00
Ardeman
e81dad4ec5 feat: update Autoplay settings in Banner component to prevent stopping on interaction 2025-03-24 13:24:24 +08:00
Ardeman
63bbf70bd6 feat: improve loading state in CarouselHero and CategorySection with ImageSkeletonIcon 2025-03-20 18:22:07 +08:00
Ardeman
33096ab7c1 feat: enhance loading state in CarouselHero with ImageSkeletonIcon 2025-03-20 18:16:15 +08:00
Ardeman
ab5b545625 feat: add ImageSkeletonIcon for loading state in carousel section 2025-03-20 17:44:04 +08:00
39 changed files with 751 additions and 162 deletions

View File

@ -0,0 +1,28 @@
import { z } from 'zod'
import { HttpServer, type THttpServer } from '~/libs/http-server'
import type { TStaffSchema } from '~/pages/form-staff'
const createStaffResponseSchema = z.object({
data: z.object({
Message: z.string(),
}),
})
type TParameter = {
payload: TStaffSchema
} & THttpServer
export const createStaffsRequest = async (parameters: TParameter) => {
const { payload, ...restParameters } = parameters
try {
const { data } = await HttpServer(restParameters).post(
'/api/staff/register',
payload,
)
return createStaffResponseSchema.parse(data)
} catch (error) {
// eslint-disable-next-line unicorn/no-useless-promise-resolve-reject
return Promise.reject(error)
}
}

View File

@ -0,0 +1,25 @@
import { z } from 'zod'
import { newsResponseSchema } from '~/apis/common/get-news'
import { HttpServer, type THttpServer } from '~/libs/http-server'
const dataResponseSchema = z.object({
data: z.object(newsResponseSchema.shape),
})
type TParameters = {
id: string
} & THttpServer
export const getNewsById = async (parameters: TParameters) => {
const { id, ...restParameters } = parameters
try {
const { data } = await HttpServer(restParameters).get(
`/api/staff/news/${encodeURIComponent(id)}`,
)
return dataResponseSchema.parse(data)
} catch (error) {
// eslint-disable-next-line unicorn/no-useless-promise-resolve-reject
return Promise.reject(error)
}
}

View File

@ -25,7 +25,11 @@ export const newsResponseSchema = z.object({
author: authorSchema, author: authorSchema,
}) })
const dataResponseSchema = z.object({ const dataResponseSchema = z.object({
data: z.array(newsResponseSchema), data: z.array(
newsResponseSchema.extend({
views: z.number(),
}),
),
}) })
export type TNewsResponse = z.infer<typeof newsResponseSchema> export type TNewsResponse = z.infer<typeof newsResponseSchema>
@ -37,10 +41,11 @@ type TParameters = {
active?: boolean active?: boolean
limit?: number limit?: number
page?: number page?: number
query?: string
} & THttpServer } & THttpServer
export const getNews = async (parameters?: TParameters) => { export const getNews = async (parameters?: TParameters) => {
const { categories, tags, active, limit, page, ...restParameters } = const { categories, tags, active, limit, page, query, ...restParameters } =
parameters || {} parameters || {}
try { try {
const { data } = await HttpServer(restParameters).get(`/api/news`, { const { data } = await HttpServer(restParameters).get(`/api/news`, {
@ -50,6 +55,7 @@ export const getNews = async (parameters?: TParameters) => {
...(active && { active }), ...(active && { active }),
...(limit && { limit }), ...(limit && { limit }),
...(page && { page }), ...(page && { page }),
...(query && { q: query }),
}, },
}) })
return dataResponseSchema.parse(data) return dataResponseSchema.parse(data)

View File

@ -117,7 +117,7 @@ export const DialogSuccess = ({ isOpen, onClose }: ModalProperties) => {
setIsSubscribeOpen(true) setIsSubscribeOpen(true)
}} }}
> >
Select Subscribe Plan Pilih Paken Berlangganan
</Button> </Button>
) : ( ) : (
<Button <Button

View File

@ -0,0 +1,19 @@
import type { JSX, SVGProps } from 'react'
export const ImageSkeletonIcon = (
properties: JSX.IntrinsicAttributes & SVGProps<SVGSVGElement>,
) => {
return (
<svg
width={40}
height={40}
aria-hidden="true"
xmlns="http://www.w3.org/2000/svg"
fill="currentColor"
viewBox="0 0 20 18"
{...properties}
>
<path d="M18 0H2a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h16a2 2 0 0 0 2-2V2a2 2 0 0 0-2-2Zm-5.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3Zm4.376 10.481A1 1 0 0 1 16 15H4a1 1 0 0 1-.895-1.447l3.5-7A1 1 0 0 1 7.468 6a.965.965 0 0 1 .9.5l2.775 4.757 1.546-1.887a1 1 0 0 1 1.618.1l2.541 4a1 1 0 0 1 .028 1.011Z" />
</svg>
)
}

View File

@ -4,6 +4,7 @@ import { Await, useRouteLoaderData } from 'react-router'
import { stripHtml } from 'string-strip-html' import { stripHtml } from 'string-strip-html'
import { ErrorAwait } from '~/components/error/await' import { ErrorAwait } from '~/components/error/await'
import { ImageSkeletonIcon } from '~/components/icons/image-skeleton'
import { CarouselButton } from '~/components/ui/button-slide' import { CarouselButton } from '~/components/ui/button-slide'
import { useNewsContext } from '~/contexts/news' import { useNewsContext } from '~/contexts/news'
import type { loader } from '~/routes/_news' import type { loader } from '~/routes/_news'
@ -74,7 +75,35 @@ export const CarouselHero = (properties: TNews) => {
ref={emblaReference} ref={emblaReference}
> >
<div className="embla__container hero flex sm:gap-x-8"> <div className="embla__container hero flex sm:gap-x-8">
<Suspense fallback={<div>Loading...</div>}> <Suspense
fallback={
<div className="embla__slide hero flex w-full min-w-0 flex-none animate-pulse justify-between gap-3 max-sm:mt-2 sm:flex">
<div className="flex aspect-[174/100] h-full items-center justify-center rounded-md bg-gray-300 dark:bg-gray-700">
<ImageSkeletonIcon />
</div>
<div className="flex w-full flex-col gap-7">
<div className="flex flex-col gap-4">
<div className="h-6 w-full rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-6 w-[20%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
</div>
<div className="flex flex-col gap-2.5">
<div className="h-5 rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 max-w-[90%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 max-w-[90%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 max-w-[90%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 max-w-[90%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 max-w-[50%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
</div>
<div className="h-[50px] w-full bg-gray-200 dark:bg-gray-700" />
</div>
<span className="sr-only">Loading...</span>
</div>
}
>
<Await <Await
resolve={items} resolve={items}
errorElement={<ErrorAwait />} errorElement={<ErrorAwait />}
@ -86,12 +115,11 @@ export const CarouselHero = (properties: TNews) => {
index, index,
) => ( ) => (
<div <div
className="embla__slide hero w-full min-w-0 flex-none" className="embla__slide hero w-full min-w-0 flex-none max-sm:mt-2 sm:flex"
key={index} key={index}
> >
<div className="max-sm:mt-2 sm:flex">
<img <img
className="col-span-2 aspect-[174/100] object-cover" className="aspect-[174/100] h-full rounded-md object-cover"
src={featured_image} src={featured_image}
alt={title} alt={title}
/> />
@ -117,7 +145,6 @@ export const CarouselHero = (properties: TNews) => {
</Button> </Button>
</div> </div>
</div> </div>
</div>
), ),
) )
} }

View File

@ -4,6 +4,7 @@ import { Await, useRouteLoaderData } from 'react-router'
import { stripHtml } from 'string-strip-html' import { stripHtml } from 'string-strip-html'
import { ErrorAwait } from '~/components/error/await' import { ErrorAwait } from '~/components/error/await'
import { ImageSkeletonIcon } from '~/components/icons/image-skeleton'
import { CarouselButton } from '~/components/ui/button-slide' import { CarouselButton } from '~/components/ui/button-slide'
import { useNewsContext } from '~/contexts/news' import { useNewsContext } from '~/contexts/news'
import type { loader } from '~/routes/_news' import type { loader } from '~/routes/_news'
@ -80,7 +81,31 @@ export const CarouselSection = (properties: TNews) => {
ref={emblaReference} ref={emblaReference}
> >
<div className="embla__container col-span-3 flex max-h-[586px] sm:gap-x-8"> <div className="embla__container col-span-3 flex max-h-[586px] sm:gap-x-8">
<Suspense fallback={<div>Loading...</div>}> <Suspense
fallback={Array.from({ length: 3 }).map((_, index) => (
<div
key={index}
className="embla__slide flex w-full min-w-0 flex-none animate-pulse flex-col justify-between gap-3 sm:w-1/3"
>
<div className="flex h-[280px] w-full items-center justify-center rounded-md bg-gray-300 dark:bg-gray-700">
<ImageSkeletonIcon />
</div>
<div className="flex w-full flex-col gap-4">
<div className="flex flex-col gap-4">
<div className="h-6 w-full rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-6 w-[20%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
</div>
<div className="flex flex-col gap-2.5">
<div className="h-5 max-w-[80%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 max-w-[50%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
</div>
</div>
<div className="h-[50px] w-full bg-gray-200 dark:bg-gray-700" />
<span className="sr-only">Loading...</span>
</div>
))}
>
<Await <Await
resolve={items} resolve={items}
errorElement={<ErrorAwait />} errorElement={<ErrorAwait />}
@ -92,10 +117,9 @@ export const CarouselSection = (properties: TNews) => {
index, index,
) => ( ) => (
<div <div
className="embla__slide w-full min-w-0 flex-none sm:w-1/3" className="embla__slide flex w-full min-w-0 flex-none flex-col justify-between gap-3 sm:w-1/3"
key={index} key={index}
> >
<div className="flex flex-col justify-between gap-3">
<img <img
className="aspect-[174/100] max-h-[280px] w-full rounded-md object-cover sm:aspect-[5/4]" className="aspect-[174/100] max-h-[280px] w-full rounded-md object-cover sm:aspect-[5/4]"
src={featured_image} src={featured_image}
@ -111,7 +135,7 @@ export const CarouselSection = (properties: TNews) => {
{title} {title}
</h3> </h3>
</div> </div>
<p className="text-md line-clamp-3 text-[#777777] sm:text-xl"> <p className="line-clamp-3 text-base text-[#777777] sm:text-xl">
{stripHtml(content).result} {stripHtml(content).result}
</p> </p>
<Button <Button
@ -127,7 +151,6 @@ export const CarouselSection = (properties: TNews) => {
</Button> </Button>
</div> </div>
</div> </div>
</div>
), ),
) )
} }

View File

@ -6,6 +6,7 @@ import { twMerge } from 'tailwind-merge'
import { ErrorAwait } from '~/components/error/await' import { ErrorAwait } from '~/components/error/await'
import { CarouselNextIcon } from '~/components/icons/carousel-next' import { CarouselNextIcon } from '~/components/icons/carousel-next'
import { CarouselPreviousIcon } from '~/components/icons/carousel-previous' import { CarouselPreviousIcon } from '~/components/icons/carousel-previous'
import { ImageSkeletonIcon } from '~/components/icons/image-skeleton'
import { Button } from '~/components/ui/button' import { Button } from '~/components/ui/button'
import { useNewsContext } from '~/contexts/news' import { useNewsContext } from '~/contexts/news'
import type { loader } from '~/routes/_news' import type { loader } from '~/routes/_news'
@ -43,7 +44,31 @@ export const CategorySection = (properties: TNews) => {
</div> </div>
<div className="grid sm:grid-cols-3 sm:gap-x-8"> <div className="grid sm:grid-cols-3 sm:gap-x-8">
<Suspense fallback={<div>Loading...</div>}> <Suspense
fallback={Array.from({ length: 3 }).map((_, index) => (
<div
key={index}
className="grid gap-3 sm:gap-x-8"
>
<div className="flex h-[280px] w-full items-center justify-center rounded-md bg-gray-300 dark:bg-gray-700">
<ImageSkeletonIcon />
</div>
<div className="flex w-full flex-col gap-4">
<div className="flex flex-col gap-4">
<div className="h-6 w-full rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-6 w-[20%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
</div>
<div className="flex flex-col gap-2.5">
<div className="h-5 max-w-[80%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 rounded-full bg-gray-200 dark:bg-gray-700"></div>
<div className="h-5 max-w-[50%] rounded-full bg-gray-200 dark:bg-gray-700"></div>
</div>
</div>
<div className="h-[50px] w-full bg-gray-200 dark:bg-gray-700" />
<span className="sr-only">Loading...</span>
</div>
))}
>
<Await <Await
resolve={items} resolve={items}
errorElement={<ErrorAwait />} errorElement={<ErrorAwait />}
@ -56,18 +81,14 @@ export const CategorySection = (properties: TNews) => {
) => ( ) => (
<div <div
key={index} key={index}
className={twMerge('grid gap-3 sm:gap-x-8')} className="grid gap-3 sm:gap-x-8"
> >
<img <img
className={twMerge( className="aspect-[174/100] w-full rounded-md object-cover sm:aspect-[5/4]"
'aspect-[174/100] w-full rounded-md object-cover sm:aspect-[5/4]',
)}
src={featured_image} src={featured_image}
alt={title} alt={title}
/> />
<div <div className="flex flex-col justify-between gap-4">
className={twMerge('flex flex-col justify-between gap-4')}
>
<Tags <Tags
tags={tags} tags={tags}
is_premium={is_premium} is_premium={is_premium}

View File

@ -39,7 +39,7 @@ export const SocialShareButtons = ({
onClick={handleCopyLink} onClick={handleCopyLink}
className="relative cursor-pointer" className="relative cursor-pointer"
> >
<LinkIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 sm:h-10 sm:w-10" /> <LinkIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 transition hover:bg-gray-200 hover:shadow active:bg-gray-300 sm:h-10 sm:w-10" />
{showPopup && ( {showPopup && (
<div className="absolute top-12 w-48 rounded-lg border-2 border-gray-400 bg-white p-2 shadow-lg"> <div className="absolute top-12 w-48 rounded-lg border-2 border-gray-400 bg-white p-2 shadow-lg">
Link berhasil disalin! Link berhasil disalin!
@ -51,28 +51,28 @@ export const SocialShareButtons = ({
url={url} url={url}
title={title} title={title}
> >
<FacebookIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 sm:h-10 sm:w-10" /> <FacebookIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 transition hover:bg-gray-200 hover:shadow active:bg-gray-300 sm:h-10 sm:w-10" />
</FacebookShareButton> </FacebookShareButton>
<LinkedinShareButton <LinkedinShareButton
url={url} url={url}
title={title} title={title}
> >
<LinkedinIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 sm:h-10 sm:w-10" /> <LinkedinIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 transition hover:bg-gray-200 hover:shadow active:bg-gray-300 sm:h-10 sm:w-10" />
</LinkedinShareButton> </LinkedinShareButton>
<TwitterShareButton <TwitterShareButton
url={url} url={url}
title={title} title={title}
> >
<XIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 sm:h-10 sm:w-10" /> <XIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 transition hover:bg-gray-200 hover:shadow active:bg-gray-300 sm:h-10 sm:w-10" />
</TwitterShareButton> </TwitterShareButton>
<button <button
onClick={handleInstagramShare} onClick={handleInstagramShare}
className="cursor-pointer" className="cursor-pointer"
> >
<InstagramIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 sm:h-10 sm:w-10" /> <InstagramIcon className="size-8 rounded-full bg-[#F4F4F4] p-2 transition hover:bg-gray-200 hover:shadow active:bg-gray-300 sm:h-10 sm:w-10" />
</button> </button>
</div> </div>
) )

View File

@ -21,7 +21,7 @@ export const profileSchema = z.object({
name: z.string().min(1, 'Name is required'), name: z.string().min(1, 'Name is required'),
email: z.string().email('Email is invalid'), email: z.string().email('Email is invalid'),
profile_picture: z.string().url({ profile_picture: z.string().url({
message: 'URL must be valid', message: 'Gambar profil must be valid',
}), }),
}) })
@ -93,7 +93,7 @@ export const DialogProfile = () => {
<Input <Input
name="name" name="name"
id="name" id="name"
label="Name" label="Nama"
placeholder="Enter your name" placeholder="Enter your name"
/> />
<Input <Input
@ -105,8 +105,8 @@ export const DialogProfile = () => {
<InputFile <InputFile
name="profile_picture" name="profile_picture"
id="profile_picture" id="profile_picture"
label="Profile Picture" label="Gambar Profil"
placeholder="Upload your profile picture" placeholder="Unggah gambar profil Anda"
category="profile_picture" category="profile_picture"
/> />
<Button <Button

View File

@ -1,4 +1,5 @@
import { import {
BriefcaseIcon,
ChartBarSquareIcon, ChartBarSquareIcon,
ClipboardDocumentCheckIcon, ClipboardDocumentCheckIcon,
DocumentCurrencyDollarIcon, DocumentCurrencyDollarIcon,
@ -24,12 +25,12 @@ export const MENU: TMenu[] = [
group: 'Menu', group: 'Menu',
items: [ items: [
{ {
title: 'Dashboard', title: 'Dasbor',
url: '/lg-admin', url: '/lg-admin',
icon: ChartBarSquareIcon, icon: ChartBarSquareIcon,
}, },
{ {
title: 'User', title: 'Pengguna',
url: '/lg-admin/users', url: '/lg-admin/users',
icon: UsersIcon, icon: UsersIcon,
}, },
@ -39,12 +40,12 @@ export const MENU: TMenu[] = [
icon: NewspaperIcon, icon: NewspaperIcon,
}, },
{ {
title: 'Banner Iklan', title: 'Spanduk Iklan',
url: '/lg-admin/advertisements', url: '/lg-admin/advertisements',
icon: MegaphoneIcon, icon: MegaphoneIcon,
}, },
{ {
title: 'Subscription', title: 'Pelanggan',
url: '/lg-admin/subscriptions', url: '/lg-admin/subscriptions',
icon: PresentationChartLineIcon, icon: PresentationChartLineIcon,
}, },
@ -64,10 +65,15 @@ export const MENU: TMenu[] = [
icon: TagIcon, icon: TagIcon,
}, },
{ {
title: 'Subscribe Plan', title: 'Paket Berlangganan',
url: '/lg-admin/subscribe-plan', url: '/lg-admin/subscribe-plan',
icon: DocumentCurrencyDollarIcon, icon: DocumentCurrencyDollarIcon,
}, },
{
title: 'Staf',
url: '/lg-admin/staffs',
icon: BriefcaseIcon,
},
], ],
}, },
] ]

View File

@ -22,7 +22,7 @@ export const Sidebar = () => {
key={`${group}-${title}`} key={`${group}-${title}`}
className={twMerge( className={twMerge(
path === url ? 'bg-[#707FDD]/10 font-bold' : '', path === url ? 'bg-[#707FDD]/10 font-bold' : '',
'group/menu flex h-[42px] w-[200px] items-center gap-x-3 rounded-md px-5 transition hover:bg-[#707FDD]/10 active:bg-[#707FDD]/20', 'group/menu flex h-[42px] w-[240px] items-center gap-x-3 rounded-md px-5 transition hover:bg-[#707FDD]/10 active:bg-[#707FDD]/20',
)} )}
> >
<Icon <Icon

View File

@ -8,7 +8,12 @@ import type { loader } from '~/routes/_news'
export const Banner = () => { export const Banner = () => {
const loaderData = useRouteLoaderData<typeof loader>('routes/_news') const loaderData = useRouteLoaderData<typeof loader>('routes/_news')
const { adsData } = loaderData || {} const { adsData } = loaderData || {}
const [emblaReference] = useEmblaCarousel({ loop: true }, [Autoplay()]) const [emblaReference] = useEmblaCarousel({ loop: true }, [
Autoplay({
stopOnInteraction: false,
stopOnMouseEnter: true,
}),
])
const fetcher = useFetcher() const fetcher = useFetcher()
return ( return (

View File

@ -28,7 +28,7 @@ export const registerSchema = z
.optional() .optional()
.nullable() .nullable()
.refine((data) => !!data, { .refine((data) => !!data, {
message: 'Please select a Subscribe Plan', message: 'Silakan pilih paket berlangganan',
}), }),
}) })
.refine((field) => field.password === field.rePassword, { .refine((field) => field.password === field.rePassword, {
@ -121,8 +121,8 @@ export const DialogRegister = () => {
<Combobox <Combobox
id="subscribe_plan" id="subscribe_plan"
name="subscribe_plan" name="subscribe_plan"
label="Subscribe Plan" label="Paket Berlangganan"
placeholder="Pilih Subscribe Plan" placeholder="Pilih Paket Berlangganan"
options={subscribePlan} options={subscribePlan}
/> />

View File

@ -21,7 +21,7 @@ export const subscribeSchema = z.object({
.optional() .optional()
.nullable() .nullable()
.refine((data) => !!data, { .refine((data) => !!data, {
message: 'Please select a subscription', message: 'Silakan pilih paket berlangganan',
}), }),
}) })
@ -62,7 +62,7 @@ export const DialogSubscribePlan = () => {
setIsSubscribeOpen(false) setIsSubscribeOpen(false)
} }
}} }}
description="Selamat Datang, silakan Pilih Subscribe Plan Anda untuk melanjutkan!" description="Selamat Datang, silakan Pilih Paket Berlangganan Anda untuk melanjutkan!"
> >
<div className="flex flex-col items-center justify-center"> <div className="flex flex-col items-center justify-center">
<RemixFormProvider {...formMethods}> <RemixFormProvider {...formMethods}>
@ -75,8 +75,8 @@ export const DialogSubscribePlan = () => {
<Combobox <Combobox
id="subscribe_plan" id="subscribe_plan"
name="subscribe_plan" name="subscribe_plan"
label="Subscribe Plan" label="Paket Berlangganan"
placeholder="Pilih Subscribe Plan" placeholder="Pilih Paket Berlangganan"
options={subscribePlan} options={subscribePlan}
/> />

View File

@ -3,11 +3,16 @@ import { Button } from '~/components/ui/button'
export const HeaderSearch = () => { export const HeaderSearch = () => {
return ( return (
<form className="flex flex-1 justify-between gap-[15px] px-[35px]"> <form
className="flex flex-1 justify-between gap-[15px] px-[35px]"
method="get"
action="/search"
>
<input <input
placeholder="Cari..." placeholder="Cari..."
className="flex-1 text-xl placeholder:text-white focus:ring-0 focus:outline-none" className="flex-1 text-xl placeholder:text-white focus:ring-0 focus:outline-none"
size={1} size={1}
name="q"
/> />
<Button <Button
type="submit" type="submit"

View File

@ -14,7 +14,7 @@ import { Button } from '~/components/ui/button'
import { UiTable } from '~/components/ui/table' import { UiTable } from '~/components/ui/table'
import { TitleDashboard } from '~/components/ui/title-dashboard' import { TitleDashboard } from '~/components/ui/title-dashboard'
import type { loader } from '~/routes/_admin.lg-admin._dashboard.advertisements._index' import type { loader } from '~/routes/_admin.lg-admin._dashboard.advertisements._index'
import { formatDate } from '~/utils/formatter' import { formatDate, formatNumberWithPeriods } from '~/utils/formatter'
export const AdvertisementsPage = () => { export const AdvertisementsPage = () => {
const loaderData = useRouteLoaderData<typeof loader>( const loaderData = useRouteLoaderData<typeof loader>(
@ -70,13 +70,14 @@ export const AdvertisementsPage = () => {
/> />
) )
}, },
5: (value: number) => formatNumberWithPeriods(value),
6: (value: string, _type: unknown, data: TAdResponse) => ( 6: (value: string, _type: unknown, data: TAdResponse) => (
<div className="flex space-x-2"> <div className="flex space-x-2">
<Button <Button
as="a" as="a"
href={`/lg-admin/advertisements/update/${value}`} href={`/lg-admin/advertisements/update/${value}`}
size="icon" size="icon"
title="Update Banner Iklan" title="Update Spanduk Iklan"
> >
<PencilSquareIcon className="size-4" /> <PencilSquareIcon className="size-4" />
</Button> </Button>
@ -85,7 +86,7 @@ export const AdvertisementsPage = () => {
size="icon" size="icon"
variant="danger" variant="danger"
onClick={() => setSelectedAds(data)} onClick={() => setSelectedAds(data)}
title="Hapus Banner Iklan" title="Hapus Spanduk Iklan"
> >
<TrashIcon className="size-4" /> <TrashIcon className="size-4" />
</Button> </Button>
@ -95,7 +96,7 @@ export const AdvertisementsPage = () => {
return ( return (
<div className="relative"> <div className="relative">
<TitleDashboard title="Banner Iklan" /> <TitleDashboard title="Spanduk Iklan" />
<div className="mb-8 flex items-end justify-between gap-5"> <div className="mb-8 flex items-end justify-between gap-5">
<div className="flex-1">{/* TODO: Filter */}</div> <div className="flex-1">{/* TODO: Filter */}</div>
@ -105,7 +106,7 @@ export const AdvertisementsPage = () => {
size="lg" size="lg"
className="text-md h-[42px] px-4" className="text-md h-[42px] px-4"
> >
<PlusIcon className="size-8" /> Buat Banner Iklan <PlusIcon className="size-8" /> Buat Spanduk Iklan
</Button> </Button>
</div> </div>
@ -113,13 +114,13 @@ export const AdvertisementsPage = () => {
data={dataTable} data={dataTable}
columns={dataColumns} columns={dataColumns}
slots={dataSlot} slots={dataSlot}
title="Daftar Banner Iklan" title="Daftar Spanduk Iklan"
/> />
<DialogDelete <DialogDelete
selectedId={selectedAds?.id} selectedId={selectedAds?.id}
close={() => setSelectedAds(undefined)} close={() => setSelectedAds(undefined)}
title="Banner iklan" title="Spanduk iklan"
fetcherAction={`/actions/admin/advertisements/delete/${selectedAds?.id}`} fetcherAction={`/actions/admin/advertisements/delete/${selectedAds?.id}`}
> >
<img <img

View File

@ -16,7 +16,7 @@ import { Button } from '~/components/ui/button'
import { UiTable } from '~/components/ui/table' import { UiTable } from '~/components/ui/table'
import { TitleDashboard } from '~/components/ui/title-dashboard' import { TitleDashboard } from '~/components/ui/title-dashboard'
import type { loader } from '~/routes/_admin.lg-admin._dashboard.contents._index' import type { loader } from '~/routes/_admin.lg-admin._dashboard.contents._index'
import { formatDate } from '~/utils/formatter' import { formatDate, formatNumberWithPeriods } from '~/utils/formatter'
export const ContentsPage = () => { export const ContentsPage = () => {
const loaderData = useRouteLoaderData<typeof loader>( const loaderData = useRouteLoaderData<typeof loader>(
@ -41,7 +41,7 @@ export const ContentsPage = () => {
}, },
}, },
{ {
title: 'Tanggal Live', title: 'Mulai Tayang',
data: 'live_at', data: 'live_at',
}, },
{ {
@ -58,25 +58,33 @@ export const ContentsPage = () => {
title: 'Subscription', title: 'Subscription',
data: 'is_premium', data: 'is_premium',
}, },
{
title: 'Jumlah Pembaca',
data: 'views',
},
{ {
title: 'Action', title: 'Action',
data: 'slug', data: 'id',
}, },
] ]
const dataSlot: DataTableSlots = { const dataSlot: DataTableSlots = {
1: (value: string) => formatDate(value), 1: (value: string) => formatDate(value),
2: (value: TAuthorResponse) => ( 2: (value: TAuthorResponse) => (
<div> <>
<div>{value.name}</div> <div>{value.name}</div>
<div className="text-sm text-[#7C7C7C]">ID: {value.id.slice(0, 8)}</div> <div className="text-sm text-[#7C7C7C]">ID: {value.id.slice(0, 8)}</div>
</div> </>
), ),
3: (value: string) => <span className="text-sm">{value}</span>, 3: (value: string) => <span className="text-sm">{value}</span>,
4: (value: TCategoryResponse[]) => ( 4: (value: TCategoryResponse[]) => (
<div className="text-xs">{value.map((item) => item.name).join(', ')}</div> <span className="text-xs">
{value.map((item) => item.name).join(', ')}
</span>
), ),
5: (value: TTagResponse[]) => ( 5: (value: TTagResponse[]) => (
<div className="text-xs">{value.map((item) => item.name).join(', ')}</div> <span className="text-xs">
{value.map((item) => item.name).join(', ')}
</span>
), ),
6: (value: string) => 6: (value: string) =>
value ? ( value ? (
@ -85,10 +93,11 @@ export const ContentsPage = () => {
</div> </div>
) : ( ) : (
<div className="rounded-full bg-[#F5F5F5] px-2 text-center text-[#4C5CA0]"> <div className="rounded-full bg-[#F5F5F5] px-2 text-center text-[#4C5CA0]">
Normal Biasa
</div> </div>
), ),
7: (value: string, _type: unknown, data: TNewsResponse) => ( 7: (value: number) => formatNumberWithPeriods(value),
8: (value: string, _type: unknown, data: TNewsResponse) => (
<div className="flex space-x-2"> <div className="flex space-x-2">
<Button <Button
as="a" as="a"

View File

@ -0,0 +1,89 @@
import { PlusIcon } from '@heroicons/react/24/solid'
import { Link } from 'react-router'
import { Button } from '~/components/ui/button'
import { TitleDashboard } from '~/components/ui/title-dashboard'
export const StaffsPage = () => {
// const loaderData = useRouteLoaderData<typeof loader>(
// 'routes/_admin.lg-admin._dashboard.staffs._index',
// )
// DataTable.use(DT)
// const dataTable =
// loaderData?.usersData?.sort(
// (a, b) =>
// new Date(b.subscribe.start_date).getTime() -
// new Date(a.subscribe.start_date).getTime(),
// ) || []
// const dataColumns: ConfigColumns[] = [
// {
// title: 'No',
// render: (
// _data: unknown,
// _type: unknown,
// _row: unknown,
// meta: { row: number },
// ) => {
// return meta.row + 1
// },
// },
// {
// title: 'Tanggal Daftar',
// data: 'created_at',
// },
// {
// title: 'User',
// },
// {
// title: 'Phone',
// data: 'phone',
// },
// {
// title: 'Status',
// data: 'subscribe.subscribe_plan.name',
// },
// ]
// const dataSlot: DataTableSlots = {
// 1: (value: string) => formatDate(value),
// 2: (_value: unknown, _type: unknown, data: TUserResponse) => (
// <div>
// <div>{data.email}</div>
// <div className="text-sm text-[#7C7C7C]">ID: {data.id.slice(0, 8)}</div>
// </div>
// ),
// 3: (value: string) => <span>{value}</span>,
// 4: (value: TColorBadge, _type: unknown, data: TUserResponse) => (
// <span
// className={`rounded-lg px-2 text-sm ${getStatusBadge(data.subscribe.status as TColorBadge)}`}
// >
// {value}
// </span>
// ),
// }
return (
<div className="relative">
<TitleDashboard title="Staf" />
<div className="mb-8 flex items-end justify-between gap-5">
<div className="flex-1">{/* TODO: Filter */}</div>
<Button
as={Link}
to="/lg-admin/staffs/create"
size="lg"
className="text-md h-[42px] px-4"
>
<PlusIcon className="size-8" /> Buat Staf
</Button>
</div>
{/* <UiTable
data={dataTable || []}
columns={dataColumns}
slots={dataSlot}
title="Daftar Users"
/> */}
</div>
)
}

View File

@ -3,8 +3,8 @@ import {
PlusIcon, PlusIcon,
TrashIcon, TrashIcon,
} from '@heroicons/react/24/solid' } from '@heroicons/react/24/solid'
import DT from 'datatables.net-dt' import DT, { type ConfigColumns } from 'datatables.net-dt'
import DataTable from 'datatables.net-react' import DataTable, { type DataTableSlots } from 'datatables.net-react'
import { useState } from 'react' import { useState } from 'react'
import { Link, useRouteLoaderData } from 'react-router' import { Link, useRouteLoaderData } from 'react-router'
@ -27,7 +27,7 @@ export const SubscribePlanPage = () => {
DataTable.use(DT) DataTable.use(DT)
const { subscribePlanData: dataTable } = loaderData || {} const { subscribePlanData: dataTable } = loaderData || {}
const dataColumns = [ const dataColumns: ConfigColumns[] = [
{ {
title: 'No', title: 'No',
render: ( render: (
@ -48,12 +48,13 @@ export const SubscribePlanPage = () => {
data: 'code', data: 'code',
}, },
{ {
title: 'Length', title: 'Durasi',
data: 'length', data: 'length',
}, },
{ {
title: 'Harga', title: 'Harga',
data: 'price', data: 'price',
className: 'dt-type-numeric',
}, },
{ {
title: 'Status', title: 'Status',
@ -64,10 +65,8 @@ export const SubscribePlanPage = () => {
data: 'id', data: 'id',
}, },
] ]
const dataSlot = { const dataSlot: DataTableSlots = {
4: (value: number) => ( 4: (value: number) => `Rp. ${formatNumberWithPeriods(value)}`,
<div className="text-right">Rp. {formatNumberWithPeriods(value)}</div>
),
5: (value: number) => ( 5: (value: number) => (
<span <span
className={`rounded-lg px-2 text-sm ${getStatusBadge(value as TColorBadge)}`} className={`rounded-lg px-2 text-sm ${getStatusBadge(value as TColorBadge)}`}
@ -84,7 +83,7 @@ export const SubscribePlanPage = () => {
as="a" as="a"
href={`/lg-admin/subscribe-plan/update/${value}`} href={`/lg-admin/subscribe-plan/update/${value}`}
size="icon" size="icon"
title="Update Subscribe Plan" title="Update Paket Berlangganan"
> >
<PencilSquareIcon className="size-4" /> <PencilSquareIcon className="size-4" />
</Button> </Button>
@ -93,7 +92,7 @@ export const SubscribePlanPage = () => {
size="icon" size="icon"
variant="danger" variant="danger"
onClick={() => setSelectedSubscribePlan(data)} onClick={() => setSelectedSubscribePlan(data)}
title="Hapus Subscribe Plan" title="Hapus Paket Berlangganan"
> >
<TrashIcon className="size-4" /> <TrashIcon className="size-4" />
</Button> </Button>
@ -102,7 +101,7 @@ export const SubscribePlanPage = () => {
} }
return ( return (
<div className="relative"> <div className="relative">
<TitleDashboard title="Subscribe Plan" /> <TitleDashboard title="Paket Berlangganan" />
<div className="mb-8 flex items-end justify-between"> <div className="mb-8 flex items-end justify-between">
<div className="flex-1">{/* TODO: Filter */}</div> <div className="flex-1">{/* TODO: Filter */}</div>
<Button <Button
@ -111,7 +110,7 @@ export const SubscribePlanPage = () => {
size="lg" size="lg"
className="text-md h-[42px] px-4" className="text-md h-[42px] px-4"
> >
<PlusIcon className="size-8" /> Buat Subscribe Plan <PlusIcon className="size-8" /> Buat Paket Berlangganan
</Button> </Button>
</div> </div>
@ -125,13 +124,13 @@ export const SubscribePlanPage = () => {
ordering: true, ordering: true,
info: true, info: true,
}} }}
title=" Daftar Subscribe Plan" title=" Daftar Paket Berlangganan"
/> />
<DialogDelete <DialogDelete
selectedId={selectedSubscribePlan?.id} selectedId={selectedSubscribePlan?.id}
close={() => setSelectedSubscribePlan(undefined)} close={() => setSelectedSubscribePlan(undefined)}
title="Subscribe plan" title="Paket Berlangganan"
fetcherAction={`/actions/admin/subscribe-plan/delete/${selectedSubscribePlan?.id}`} fetcherAction={`/actions/admin/subscribe-plan/delete/${selectedSubscribePlan?.id}`}
> >
<p>{selectedSubscribePlan?.name}</p> <p>{selectedSubscribePlan?.name}</p>

View File

@ -27,7 +27,7 @@ export const SubscriptionsPage = () => {
return ( return (
<div className="relative"> <div className="relative">
<TitleDashboard title="Subscription" /> <TitleDashboard title="Pelanggan" />
<div className="mb-8 flex items-end justify-between"> <div className="mb-8 flex items-end justify-between">
<div className="flex items-center gap-5 rounded-lg bg-gray-50 text-[#363636]"> <div className="flex items-center gap-5 rounded-lg bg-gray-50 text-[#363636]">
@ -71,7 +71,7 @@ export const SubscriptionsPage = () => {
ordering: true, ordering: true,
info: true, info: true,
}} }}
title="Daftar Subscription" title="Daftar Pelanggan"
/> />
</div> </div>
) )

View File

@ -88,7 +88,7 @@ export const TagsPage = () => {
}) })
return ( return (
<div className="relative"> <div className="relative">
<TitleDashboard title="Tags" /> <TitleDashboard title="Tag" />
<div className="mb-8 flex items-end justify-between gap-5"> <div className="mb-8 flex items-end justify-between gap-5">
<div className="flex-1"> <div className="flex-1">
<TableSearchFilter <TableSearchFilter
@ -111,7 +111,7 @@ export const TagsPage = () => {
columns={dataColumns} columns={dataColumns}
options={dataOptions} options={dataOptions}
slots={dataSlot} slots={dataSlot}
title="Daftar Tags" title="Daftar Tag"
/> />
<DialogDelete <DialogDelete

View File

@ -39,10 +39,10 @@ export const UsersPage = () => {
data: 'created_at', data: 'created_at',
}, },
{ {
title: 'User', title: 'Pengguna',
}, },
{ {
title: 'Phone', title: 'No. Telepon',
data: 'phone', data: 'phone',
}, },
{ {
@ -70,7 +70,7 @@ export const UsersPage = () => {
return ( return (
<div className="relative"> <div className="relative">
<TitleDashboard title="Users" /> <TitleDashboard title="Pengguna" />
<div className="mb-8 flex items-end justify-between gap-5"> <div className="mb-8 flex items-end justify-between gap-5">
<div className="flex-1">{/* TODO: Filter */}</div> <div className="flex-1">{/* TODO: Filter */}</div>
@ -80,7 +80,7 @@ export const UsersPage = () => {
data={dataTable || []} data={dataTable || []}
columns={dataColumns} columns={dataColumns}
slots={dataSlot} slots={dataSlot}
title="Daftar Users" title="Daftar Pengguna"
/> />
</div> </div>
) )

View File

@ -55,7 +55,7 @@ export const ChartDonut = () => {
return ( return (
<div className="rounded-xl bg-white p-6 shadow-sm"> <div className="rounded-xl bg-white p-6 shadow-sm">
<h2 className="mb-4 text-[20px]">Subscription Selesai</h2> <h2 className="mb-4 text-[20px]">Langganan Selesai</h2>
<div className="flex items-center justify-between"> <div className="flex items-center justify-between">
<div style={{ height: 'auto', width: '100%' }}> <div style={{ height: 'auto', width: '100%' }}>
<Doughnut <Doughnut

View File

@ -66,7 +66,7 @@ export const ChartPie = () => {
return ( return (
<div className="h-[300px] w-full items-center justify-center rounded-xl bg-white p-5 text-center shadow-sm"> <div className="h-[300px] w-full items-center justify-center rounded-xl bg-white p-5 text-center shadow-sm">
<h2 className="text-xl font-bold">Top 5 Artikel</h2> <h2 className="text-xl font-bold">5 Artikel Teratas</h2>
<Pie <Pie
height={225} height={225}
width={450} width={450}

View File

@ -1,11 +1,11 @@
import { ChartBarIcon, ChartPieIcon } from '@heroicons/react/24/solid' import { ChartBarIcon, ChartPieIcon } from '@heroicons/react/24/solid'
export const REPORT = [ export const REPORT = [
{ title: 'Total User', amount: 10_800, icon: ChartBarIcon }, { title: 'Total Pengguna', amount: 8, icon: ChartBarIcon },
{ title: 'Total User Subscribe', amount: 5000, icon: ChartBarIcon }, { title: 'Total Pelanggan', amount: 0, icon: ChartBarIcon },
{ {
title: 'Total Nilai Subscribe', title: 'Total Nilai Berlangganan',
amount: 250_000_000, amount: 0,
icon: ChartBarIcon, icon: ChartBarIcon,
currency: 'Rp. ', currency: 'Rp. ',
}, },
@ -13,13 +13,13 @@ export const REPORT = [
export const HISTORY = [ export const HISTORY = [
{ {
title: 'Total Content Biasa', title: 'Total Artikel Biasa',
amount: 2890, amount: 7,
icon: ChartPieIcon, icon: ChartPieIcon,
}, },
{ {
title: 'Total Content Premium', title: 'Total Artikel Premium',
amount: 274, amount: 3,
icon: ChartPieIcon, icon: ChartPieIcon,
}, },
] ]

View File

@ -8,7 +8,7 @@ export const DashboardPage = () => {
return ( return (
<div className="relative"> <div className="relative">
<section className="mb-5 flex items-center justify-between"> <section className="mb-5 flex items-center justify-between">
<h1 className="text-xl font-bold">Dashboard</h1> <h1 className="text-xl font-bold">Dasbor</h1>
<div className="flex items-center gap-2"> <div className="flex items-center gap-2">
<span>Tanggal:</span> <span>Tanggal:</span>
<input <input

View File

@ -57,7 +57,7 @@ export const FormAdvertisementsPage = (properties: TProperties) => {
} }
if (fetcher.data?.success) { if (fetcher.data?.success) {
toast.success(`Banner iklan berhasil ${adData ? 'diupdate' : 'dibuat'}!`) toast.success(`Spanduk iklan berhasil ${adData ? 'diupdate' : 'dibuat'}!`)
navigate('/lg-admin/advertisements') navigate('/lg-admin/advertisements')
} }
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
@ -65,7 +65,7 @@ export const FormAdvertisementsPage = (properties: TProperties) => {
return ( return (
<div className="relative"> <div className="relative">
<TitleDashboard title={`${adData ? 'Update' : 'Buat'} Banner Iklan`} /> <TitleDashboard title={`${adData ? 'Update' : 'Buat'} Spanduk Iklan`} />
<div> <div>
<RemixFormProvider {...formMethods}> <RemixFormProvider {...formMethods}>
<fetcher.Form <fetcher.Form

View File

@ -56,7 +56,7 @@ export const contentSchema = z.object({
}), }),
is_premium: z.boolean().optional(), is_premium: z.boolean().optional(),
live_at: z.string().min(1, { live_at: z.string().min(1, {
message: 'Tanggal live is required', message: 'Tanggal mulai tayang is required',
}), }),
}) })
@ -126,7 +126,6 @@ export const FormContentsPage = (properties: TProperties) => {
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100" className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
labelClassName="text-sm font-medium text-[#363636]" labelClassName="text-sm font-medium text-[#363636]"
containerClassName="flex-1" containerClassName="flex-1"
disabled={!!newsData}
/> />
<InputFile <InputFile
id="featured_image" id="featured_image"
@ -181,7 +180,7 @@ export const FormContentsPage = (properties: TProperties) => {
/> />
<Input <Input
id="live_at" id="live_at"
label="Tanggal Live" label="Tanggal Mulai Tayang"
placeholder="Pilih Tanggal" placeholder="Pilih Tanggal"
name="live_at" name="live_at"
type="date" type="date"
@ -194,7 +193,7 @@ export const FormContentsPage = (properties: TProperties) => {
label="Subscription" label="Subscription"
labelClassName="text-sm font-medium text-[#363636]" labelClassName="text-sm font-medium text-[#363636]"
className="h-[42px]" className="h-[42px]"
options={{ true: 'Premium', false: 'Normal' }} options={{ true: 'Premium', false: 'Biasa' }}
/> />
</div> </div>

View File

@ -0,0 +1,134 @@
import { zodResolver } from '@hookform/resolvers/zod'
import { useEffect } from 'react'
import toast from 'react-hot-toast'
import { useFetcher, useNavigate } from 'react-router'
import { RemixFormProvider, useRemixForm } from 'remix-hook-form'
import { z } from 'zod'
import { Button } from '~/components/ui/button'
import { Input } from '~/components/ui/input'
import { InputFile } from '~/components/ui/input-file'
import { TitleDashboard } from '~/components/ui/title-dashboard'
export const staffSchema = z
.object({
profile_picture: z
.string()
.url({
message: 'Gambar profil must be a valid URL',
})
.or(z.literal('')),
name: z.string().min(1, {
message: 'Nama staf is required',
}),
password: z.string().min(6, 'Kata sandi minimal 6 karakter'),
rePassword: z.string().min(6, 'Kata sandi minimal 6 karakter'),
email: z.string().email('Email tidak valid'),
})
.refine((field) => field.password === field.rePassword, {
message: 'Kata sandi tidak sama',
path: ['rePassword'],
})
export type TStaffSchema = z.infer<typeof staffSchema>
export const FormStaffPage = () => {
const fetcher = useFetcher()
const navigate = useNavigate()
const formMethods = useRemixForm<TStaffSchema>({
mode: 'onSubmit',
fetcher,
resolver: zodResolver(staffSchema),
})
const { handleSubmit } = formMethods
useEffect(() => {
if (!fetcher.data?.success && fetcher.data?.message) {
toast.error(fetcher.data.message)
}
if (fetcher.data?.success) {
toast.success(`Staff berhasil dibuat!`)
navigate('/lg-admin/staffs')
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [fetcher.data])
return (
<div className="relative">
<TitleDashboard title={`Buat Staf`} />
<div>
<RemixFormProvider {...formMethods}>
<fetcher.Form
method="post"
onSubmit={handleSubmit}
action={`/actions/admin/staffs/create`}
className="space-y-4"
>
<div className="flex items-end justify-between gap-4">
<Input
id="name"
label="Nama Staf"
placeholder="Masukkan nama staf"
name="name"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
labelClassName="text-sm font-medium text-[#363636]"
containerClassName="flex-1"
/>
<Input
id="email"
label="Email"
placeholder="Contoh: legal@legalgo.id"
name="email"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
labelClassName="text-sm font-medium text-[#363636]"
containerClassName="flex-1"
/>
<Button
isLoading={fetcher.state !== 'idle'}
disabled={fetcher.state !== 'idle'}
type="submit"
size="lg"
className="text-md h-[42px] rounded-md"
>
Save
</Button>
</div>
<div className="flex items-end justify-between gap-4">
<Input
id="password"
label="Kata Sandi"
placeholder="Masukkan Kata Sandi"
name="password"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
labelClassName="text-sm font-medium text-[#363636]"
containerClassName="flex-1"
type="password"
/>
<Input
id="re-password"
label="Ulangi Kata Sandi"
placeholder="Masukkan Kata Sandi"
name="rePassword"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
labelClassName="text-sm font-medium text-[#363636]"
containerClassName="flex-1"
type="password"
/>
<InputFile
id="profile_picture"
label="Gambar Profil"
placeholder="Unggah gambar profil Anda"
name="profile_picture"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
labelClassName="text-sm font-medium text-[#363636]"
containerClassName="flex-1"
category="profile_picture"
/>
</div>
</fetcher.Form>
</RemixFormProvider>
</div>
</div>
)
}

View File

@ -54,7 +54,7 @@ export const FormSubscribePlanPage = (properties: TProperties) => {
if (fetcher.data?.success) { if (fetcher.data?.success) {
toast.success( toast.success(
`Subscribe Plan berhasil ${subscribePlanData ? 'diupdate' : 'dibuat'}!`, `Paket Berlangganan berhasil ${subscribePlanData ? 'diupdate' : 'dibuat'}!`,
) )
navigate('/lg-admin/subscribe-plan') navigate('/lg-admin/subscribe-plan')
} }
@ -69,7 +69,7 @@ export const FormSubscribePlanPage = (properties: TProperties) => {
return ( return (
<div className="relative"> <div className="relative">
<TitleDashboard <TitleDashboard
title={`${subscribePlanData ? 'Update' : 'Buat'} Subscribe Plan`} title={`${subscribePlanData ? 'Update' : 'Buat'} Paket Berlangganan`}
/> />
<div> <div>
<RemixFormProvider {...formMethods}> <RemixFormProvider {...formMethods}>
@ -82,8 +82,8 @@ export const FormSubscribePlanPage = (properties: TProperties) => {
<div className="flex items-end justify-between gap-4"> <div className="flex items-end justify-between gap-4">
<Input <Input
id="name" id="name"
label="Subscribe Plan" label="Paket Berlangganan"
placeholder="Masukkan Nama Subscribe Plan" placeholder="Masukkan Nama Paket Berlangganan"
name="name" name="name"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100" className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
labelClassName="text-sm font-medium text-[#363636]" labelClassName="text-sm font-medium text-[#363636]"
@ -92,7 +92,7 @@ export const FormSubscribePlanPage = (properties: TProperties) => {
<Input <Input
id="code" id="code"
label="Kode" label="Kode"
placeholder="Masukkan Kode Subscribe Plan" placeholder="Masukkan Kode Paket Berlangganan"
readOnly readOnly
name="code" name="code"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100" className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none disabled:bg-gray-100"
@ -112,9 +112,9 @@ export const FormSubscribePlanPage = (properties: TProperties) => {
<div className="flex items-end justify-between gap-4"> <div className="flex items-end justify-between gap-4">
<Input <Input
id="length" id="length"
label="Length" label="Durasi"
type="number" type="number"
placeholder="Masukkan Subscribe Plan Length (days)" placeholder="Masukkan Durasi Paket Berlangganan (hari)"
name="length" name="length"
className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none" className="border-0 bg-white shadow read-only:bg-gray-100 focus:ring-1 focus:ring-[#2E2F7C] focus:outline-none"
labelClassName="text-sm font-medium text-[#363636]" labelClassName="text-sm font-medium text-[#363636]"

View File

@ -0,0 +1,22 @@
import { useRouteLoaderData } from 'react-router'
import { Card } from '~/components/ui/card'
import { CategorySection } from '~/components/ui/category-section'
import type { loader } from '~/routes/_news.search'
export const NewsSearchPage = () => {
const loaderData = useRouteLoaderData<typeof loader>('routes/_news.search')
const { newsData, query } = loaderData || {}
return (
<div className="relative">
<Card>
<CategorySection
title="Hasil pencarian:"
description={query || ''}
items={newsData || Promise.resolve({ data: [] })}
/>
</Card>
</div>
)
}

View File

@ -1,15 +1,15 @@
import { isRouteErrorResponse } from 'react-router' import { isRouteErrorResponse } from 'react-router'
import { getNewsBySlug } from '~/apis/common/get-news-by-slug' import { getNewsById } from '~/apis/admin/get-news-by-id'
import { handleCookie } from '~/libs/cookies' import { handleCookie } from '~/libs/cookies'
import { FormContentsPage } from '~/pages/form-contents' import { FormContentsPage } from '~/pages/form-contents'
import type { Route } from './+types/_admin.lg-admin._dashboard.contents.update.$slug' import type { Route } from './+types/_admin.lg-admin._dashboard.contents.update.$id'
export const loader = async ({ request, params }: Route.LoaderArgs) => { export const loader = async ({ request, params }: Route.LoaderArgs) => {
const { staffToken: accessToken } = await handleCookie(request) const { staffToken: accessToken } = await handleCookie(request)
const { slug } = params const { id } = params
const { data: newsData } = await getNewsBySlug({ accessToken, slug }) const { data: newsData } = await getNewsById({ accessToken, id })
return { newsData } return { newsData }
} }

View File

@ -0,0 +1,44 @@
import { isRouteErrorResponse } from 'react-router'
import { StaffsPage } from '~/pages/dashboard-staffs'
import type { Route } from './+types/_admin.lg-admin._dashboard.staffs._index'
// export const loader = async ({ request }: Route.LoaderArgs) => {
// const { staffToken: accessToken } = await handleCookie(request)
// const { data: usersData } = await getUsers({ accessToken })
// return { usersData }
// }
export const ErrorBoundary = ({ error }: Route.ErrorBoundaryProps) => {
let message = 'Oops!'
let details = 'An unexpected error occurred.'
let stack: string | undefined
if (isRouteErrorResponse(error)) {
message = error.status === 404 ? '404' : 'Error'
details =
error.status === 404
? 'The requested page could not be found.'
: error.statusText || details
} else if (import.meta.env.DEV && error && error instanceof Error) {
details = error.message
stack = error.stack
}
return (
<div className="container mx-auto p-4">
<h1>{message}</h1>
<p>{details}</p>
{stack && (
<pre className="w-full p-4 whitespace-pre-wrap">
<code>{stack}</code>
</pre>
)}
</div>
)
}
const DashboardStaffsLayout = () => <StaffsPage />
export default DashboardStaffsLayout

View File

@ -0,0 +1,4 @@
import { FormStaffPage } from '~/pages/form-staff'
const DashboardStaffsCreateLayout = () => <FormStaffPage />
export default DashboardStaffsCreateLayout

View File

@ -4,7 +4,7 @@ import { stripHtml } from 'string-strip-html'
import { getCategories } from '~/apis/common/get-categories' import { getCategories } from '~/apis/common/get-categories'
import { getNews } from '~/apis/common/get-news' import { getNews } from '~/apis/common/get-news'
import { getNewsBySlug } from '~/apis/common/get-news-by-slug' import { getNewsBySlug } from '~/apis/news/get-news-by-slug'
import { getUser } from '~/apis/news/get-user' import { getUser } from '~/apis/news/get-user'
import { APP } from '~/configs/meta' import { APP } from '~/configs/meta'
import { handleCookie } from '~/libs/cookies' import { handleCookie } from '~/libs/cookies'

View File

@ -0,0 +1,59 @@
import { isRouteErrorResponse } from 'react-router'
import { getNews } from '~/apis/common/get-news'
import { APP } from '~/configs/meta'
import { NewsSearchPage } from '~/pages/news-search'
import type { Route } from './+types/_news.search'
export const loader = async ({ request }: Route.LoaderArgs) => {
const url = new URL(request.url)
const query = url.searchParams.get('q') || ''
const newsData = getNews({ query, active: true })
return { query, newsData }
}
export const meta = ({ data }: Route.MetaArgs) => {
const { query } = data
const metaTitle = APP.title
const title = `Pencarian: ${query} - ${metaTitle}`
return [
{
title,
},
]
}
export const ErrorBoundary = ({ error }: Route.ErrorBoundaryProps) => {
let message = 'Oops!'
let details = 'An unexpected error occurred.'
let stack: string | undefined
if (isRouteErrorResponse(error)) {
message = error.status === 404 ? '404' : 'Error'
details =
error.status === 404
? 'The requested page could not be found.'
: error.statusText || details
} else if (import.meta.env.DEV && error && error instanceof Error) {
details = error.message
stack = error.stack
}
return (
<div className="container mx-auto p-4">
<h1>{message}</h1>
<p>{details}</p>
{stack && (
<pre className="w-full p-4 whitespace-pre-wrap">
<code>{stack}</code>
</pre>
)}
</div>
)
}
const NewsSearchLayout = () => <NewsSearchPage />
export default NewsSearchLayout

View File

@ -0,0 +1,64 @@
import { zodResolver } from '@hookform/resolvers/zod'
import { data } from 'react-router'
import { getValidatedFormData } from 'remix-hook-form'
import { XiorError } from 'xior'
import { createStaffsRequest } from '~/apis/admin/create-staffs'
import { handleCookie } from '~/libs/cookies'
import { staffSchema, type TStaffSchema } from '~/pages/form-staff'
import type { Route } from './+types/actions.admin.staffs.create'
export const action = async ({ request }: Route.ActionArgs) => {
const { staffToken: accessToken } = await handleCookie(request)
try {
const {
errors,
data: payload,
receivedValues: defaultValues,
} = await getValidatedFormData<TStaffSchema>(
request,
zodResolver(staffSchema),
false,
)
if (errors) {
return data({ success: false, errors, defaultValues }, { status: 400 })
}
const { data: staffData } = await createStaffsRequest({
accessToken,
payload,
})
return data(
{
success: true,
staffData,
},
{
status: 200,
statusText: 'OK',
},
)
} catch (error) {
if (error instanceof XiorError) {
return data(
{
success: false,
message: error?.response?.data?.error?.message || error.message,
},
{
status: error?.response?.status || 500,
},
)
}
return data(
{
success: false,
message: 'Internal server error',
},
{ status: 500 },
)
}
}