Files
k-tv/k-tv-frontend/app/(main)/dashboard/components/create-channel-dialog.tsx
Gabriel Kaszewski 8d8d320a02 feat: implement authentication context and hooks for user management
- Add AuthContext to manage user authentication state and token storage.
- Create hooks for login, registration, and logout functionalities.
- Implement dashboard layout with authentication check and loading state.
- Enhance dashboard page with channel management features including create, edit, and delete channels.
- Integrate API calls for channel operations and current broadcast retrieval.
- Add stream URL resolution via server-side API route to handle redirects.
- Update TV page to utilize new hooks for channel and broadcast management.
- Refactor components for better organization and user experience.
- Update application metadata for improved branding.
2026-03-11 19:32:49 +01:00

122 lines
3.7 KiB
TypeScript

"use client";
import { useState } from "react";
import {
Dialog,
DialogContent,
DialogHeader,
DialogTitle,
DialogFooter,
} from "@/components/ui/dialog";
import { Button } from "@/components/ui/button";
interface CreateChannelDialogProps {
open: boolean;
onOpenChange: (open: boolean) => void;
onSubmit: (data: {
name: string;
timezone: string;
description: string;
}) => void;
isPending: boolean;
error?: string | null;
}
export function CreateChannelDialog({
open,
onOpenChange,
onSubmit,
isPending,
error,
}: CreateChannelDialogProps) {
const [name, setName] = useState("");
const [timezone, setTimezone] = useState("UTC");
const [description, setDescription] = useState("");
const handleSubmit = (e: React.FormEvent) => {
e.preventDefault();
onSubmit({ name, timezone, description });
};
const handleOpenChange = (next: boolean) => {
if (!isPending) {
onOpenChange(next);
if (!next) {
setName("");
setTimezone("UTC");
setDescription("");
}
}
};
return (
<Dialog open={open} onOpenChange={handleOpenChange}>
<DialogContent className="bg-zinc-900 border-zinc-800 text-zinc-100 sm:max-w-md">
<DialogHeader>
<DialogTitle>New channel</DialogTitle>
</DialogHeader>
<form onSubmit={handleSubmit} className="space-y-4 py-2">
<div className="space-y-1.5">
<label className="block text-xs font-medium text-zinc-400">
Name <span className="text-red-400">*</span>
</label>
<input
required
value={name}
onChange={(e) => setName(e.target.value)}
placeholder="90s Sitcom Network"
className="w-full rounded-md border border-zinc-700 bg-zinc-800 px-3 py-2 text-sm text-zinc-100 placeholder:text-zinc-600 focus:border-zinc-500 focus:outline-none"
/>
</div>
<div className="space-y-1.5">
<label className="block text-xs font-medium text-zinc-400">
Timezone <span className="text-red-400">*</span>
</label>
<input
required
value={timezone}
onChange={(e) => setTimezone(e.target.value)}
placeholder="America/New_York"
className="w-full rounded-md border border-zinc-700 bg-zinc-800 px-3 py-2 text-sm text-zinc-100 placeholder:text-zinc-600 focus:border-zinc-500 focus:outline-none"
/>
<p className="text-[11px] text-zinc-600">
IANA timezone, e.g. America/New_York, Europe/London, UTC
</p>
</div>
<div className="space-y-1.5">
<label className="block text-xs font-medium text-zinc-400">
Description
</label>
<textarea
value={description}
onChange={(e) => setDescription(e.target.value)}
placeholder="Nothing but classic sitcoms, all day"
rows={2}
className="w-full resize-none rounded-md border border-zinc-700 bg-zinc-800 px-3 py-2 text-sm text-zinc-100 placeholder:text-zinc-600 focus:border-zinc-500 focus:outline-none"
/>
</div>
{error && <p className="text-xs text-red-400">{error}</p>}
<DialogFooter>
<Button
type="button"
variant="ghost"
onClick={() => handleOpenChange(false)}
disabled={isPending}
>
Cancel
</Button>
<Button type="submit" disabled={isPending}>
{isPending ? "Creating…" : "Create channel"}
</Button>
</DialogFooter>
</form>
</DialogContent>
</Dialog>
);
}