toast added / table in logs added / favicon added
This commit is contained in:
parent
5cafff6f30
commit
ffe93a3257
7 changed files with 151 additions and 9 deletions
|
@ -1,6 +1,6 @@
|
||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { useForm, SubmitHandler, useFieldArray } from "react-hook-form";
|
import { useForm, SubmitHandler, useFieldArray } from "react-hook-form";
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
|
@ -12,13 +12,35 @@ import {
|
||||||
FormLabel,
|
FormLabel,
|
||||||
FormMessage,
|
FormMessage,
|
||||||
} from "@/components/ui/form";
|
} from "@/components/ui/form";
|
||||||
|
import {
|
||||||
|
Table,
|
||||||
|
TableBody,
|
||||||
|
TableCaption,
|
||||||
|
TableCell,
|
||||||
|
TableHead,
|
||||||
|
TableHeader,
|
||||||
|
TableRow,
|
||||||
|
} from "@/components/ui/table";
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
import { logsSchema } from "@/lib/validations/validation";
|
import { logsSchema } from "@/lib/validations/validation";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
import { useToast } from "@/components/ui/use-toast";
|
||||||
|
|
||||||
|
interface LogEntry {
|
||||||
|
_id: string;
|
||||||
|
version: string;
|
||||||
|
date: string;
|
||||||
|
bullets: { point: string }[];
|
||||||
|
}
|
||||||
|
|
||||||
type LogsFormValues = z.infer<typeof logsSchema>;
|
type LogsFormValues = z.infer<typeof logsSchema>;
|
||||||
|
|
||||||
const AdminLogPage = () => {
|
const AdminLogPage = () => {
|
||||||
|
const [logs, setLogs] = useState<LogEntry[]>([]);
|
||||||
|
const [error, setError] = useState("");
|
||||||
|
const { toast } = useToast();
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
const form = useForm<LogsFormValues>({
|
const form = useForm<LogsFormValues>({
|
||||||
resolver: zodResolver(logsSchema),
|
resolver: zodResolver(logsSchema),
|
||||||
defaultValues: {
|
defaultValues: {
|
||||||
|
@ -33,7 +55,33 @@ const AdminLogPage = () => {
|
||||||
name: "bullets",
|
name: "bullets",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const fetchLogs = async () => {
|
||||||
|
try {
|
||||||
|
const response = await fetch("/api/logs", {
|
||||||
|
method: "GET",
|
||||||
|
});
|
||||||
|
if (response.ok) {
|
||||||
|
const data: LogEntry[] = await response.json();
|
||||||
|
setLogs(data);
|
||||||
|
} else {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`);
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
setError(error.message || "Failed to fetch logs");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
fetchLogs();
|
||||||
|
const interval = setInterval(() => {
|
||||||
|
fetchLogs();
|
||||||
|
}, 10000);
|
||||||
|
|
||||||
|
return () => clearInterval(interval);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const onSubmit: SubmitHandler<LogsFormValues> = async (data) => {
|
const onSubmit: SubmitHandler<LogsFormValues> = async (data) => {
|
||||||
|
setLoading(true);
|
||||||
const response = await fetch("/api/uploadlogs", {
|
const response = await fetch("/api/uploadlogs", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
|
@ -44,11 +92,30 @@ const AdminLogPage = () => {
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
form.reset();
|
form.reset();
|
||||||
|
fetchLogs();
|
||||||
|
setLoading(false);
|
||||||
|
toast({ description: "Logs successfully added" });
|
||||||
console.log("Upload successful");
|
console.log("Upload successful");
|
||||||
alert("Uploaded");
|
|
||||||
} else {
|
} else {
|
||||||
console.error("Upload failed");
|
console.error("Upload failed");
|
||||||
alert("Upload Failed");
|
setLoading(false);
|
||||||
|
toast({ description: "Upload Failed", variant: "destructive" });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const deleteLog = async (id: string) => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/delete/${id}`, {
|
||||||
|
method: "DELETE",
|
||||||
|
});
|
||||||
|
if (response.ok) {
|
||||||
|
fetchLogs();
|
||||||
|
console.log("Delete successful");
|
||||||
|
} else {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`);
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
setError(error.message || "Failed to delete log");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -83,6 +150,7 @@ const AdminLogPage = () => {
|
||||||
</FormItem>
|
</FormItem>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{fields.map((field, index) => (
|
{fields.map((field, index) => (
|
||||||
<FormField
|
<FormField
|
||||||
key={field.id}
|
key={field.id}
|
||||||
|
@ -98,7 +166,7 @@ const AdminLogPage = () => {
|
||||||
<Button
|
<Button
|
||||||
type="button"
|
type="button"
|
||||||
className="mt-2"
|
className="mt-2"
|
||||||
variant={"destructive"}
|
variant={"secondary"}
|
||||||
onClick={() => remove(index)}
|
onClick={() => remove(index)}
|
||||||
>
|
>
|
||||||
Remove
|
Remove
|
||||||
|
@ -119,12 +187,46 @@ const AdminLogPage = () => {
|
||||||
<Button
|
<Button
|
||||||
type="submit"
|
type="submit"
|
||||||
className="w-full text-lg rounded-full"
|
className="w-full text-lg rounded-full"
|
||||||
|
disabled={loading}
|
||||||
size={"lg"}
|
size={"lg"}
|
||||||
>
|
>
|
||||||
Submit
|
Submit
|
||||||
</Button>
|
</Button>
|
||||||
</form>
|
</form>
|
||||||
</Form>
|
</Form>
|
||||||
|
|
||||||
|
{/* Section to list and delete logs */}
|
||||||
|
<section id="logs-list" className="py-16 md:py-24">
|
||||||
|
<h2 className="text-3xl md:text-4xl font-bold">Existing Logs</h2>
|
||||||
|
{error && <p className="text-red-500">{error}</p>}
|
||||||
|
<Table className="w-full mt-4 border-muted">
|
||||||
|
<TableHeader>
|
||||||
|
<TableRow>
|
||||||
|
<TableHead className="border-b px-4 py-2">Version</TableHead>
|
||||||
|
<TableHead className="border-b px-4 py-2">Date</TableHead>
|
||||||
|
<TableHead className="border-b px-4 py-2">Actions</TableHead>
|
||||||
|
</TableRow>
|
||||||
|
</TableHeader>
|
||||||
|
<TableBody>
|
||||||
|
{logs.map((log) => (
|
||||||
|
<TableRow key={log._id}>
|
||||||
|
<TableCell className="border-b px-4 py-2">
|
||||||
|
{log.version}
|
||||||
|
</TableCell>
|
||||||
|
<TableCell className="border-b px-4 py-2">{log.date}</TableCell>
|
||||||
|
<TableCell className="border-b px-4 py-2">
|
||||||
|
<Button
|
||||||
|
variant={"destructive"}
|
||||||
|
onClick={() => deleteLog(log._id)}
|
||||||
|
>
|
||||||
|
Delete
|
||||||
|
</Button>
|
||||||
|
</TableCell>
|
||||||
|
</TableRow>
|
||||||
|
))}
|
||||||
|
</TableBody>
|
||||||
|
</Table>
|
||||||
|
</section>
|
||||||
</section>
|
</section>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -16,8 +16,11 @@ import {
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
import { UploadButton, UploadDropzone } from "@/lib/uploadthing";
|
import { UploadButton, UploadDropzone } from "@/lib/uploadthing";
|
||||||
import { downloadSchema } from "@/lib/validations/validation";
|
import { downloadSchema } from "@/lib/validations/validation";
|
||||||
|
import { useToast } from "@/components/ui/use-toast";
|
||||||
|
|
||||||
const DownloadsPage = () => {
|
const DownloadsPage = () => {
|
||||||
|
const { toast } = useToast();
|
||||||
|
|
||||||
const form = useForm<z.infer<typeof downloadSchema>>({
|
const form = useForm<z.infer<typeof downloadSchema>>({
|
||||||
resolver: zodResolver(downloadSchema),
|
resolver: zodResolver(downloadSchema),
|
||||||
defaultValues: {
|
defaultValues: {
|
||||||
|
@ -41,11 +44,11 @@ const DownloadsPage = () => {
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
form.reset();
|
form.reset();
|
||||||
|
toast({ description: "Download Sucessfully Updated" });
|
||||||
console.log("Upload successful");
|
console.log("Upload successful");
|
||||||
alert("Uploaded");
|
|
||||||
} else {
|
} else {
|
||||||
console.error("Upload failed");
|
console.error("Upload failed");
|
||||||
alert("Upload Failed");
|
toast({ description: "Uploading Failed", variant: "destructive" });
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -16,8 +16,10 @@ import {
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
import { UploadButton, UploadDropzone } from "@/lib/uploadthing";
|
import { UploadButton, UploadDropzone } from "@/lib/uploadthing";
|
||||||
import { modsSchema } from "@/lib/validations/validation";
|
import { modsSchema } from "@/lib/validations/validation";
|
||||||
|
import { useToast } from "@/components/ui/use-toast";
|
||||||
|
|
||||||
const SvrjsModsAdminPage = () => {
|
const SvrjsModsAdminPage = () => {
|
||||||
|
const { toast } = useToast();
|
||||||
const form = useForm<z.infer<typeof modsSchema>>({
|
const form = useForm<z.infer<typeof modsSchema>>({
|
||||||
resolver: zodResolver(modsSchema),
|
resolver: zodResolver(modsSchema),
|
||||||
defaultValues: {
|
defaultValues: {
|
||||||
|
@ -39,11 +41,16 @@ const SvrjsModsAdminPage = () => {
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
form.reset();
|
form.reset();
|
||||||
|
toast({
|
||||||
|
description: "Successfully Uploaded Mods",
|
||||||
|
});
|
||||||
console.log("Upload successful");
|
console.log("Upload successful");
|
||||||
alert("Uploaded");
|
|
||||||
} else {
|
} else {
|
||||||
console.error("Upload failed");
|
console.error("Upload failed");
|
||||||
alert("Upload Failed");
|
toast({
|
||||||
|
description: "Upload failed",
|
||||||
|
variant: "destructive",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
30
app/api/delete/[id]/route.ts
Normal file
30
app/api/delete/[id]/route.ts
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
// app/api/delete/[id]/route.ts
|
||||||
|
import clientPromise from "@/lib/db";
|
||||||
|
import { ObjectId } from "mongodb";
|
||||||
|
import { NextResponse } from "next/server";
|
||||||
|
|
||||||
|
export async function DELETE(
|
||||||
|
request: Request,
|
||||||
|
{ params }: { params: { id: string } }
|
||||||
|
) {
|
||||||
|
const { id } = params;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const client = await clientPromise;
|
||||||
|
const db = client.db("downloadsDatabase");
|
||||||
|
const collection = db.collection("logs");
|
||||||
|
|
||||||
|
const result = await collection.deleteOne({ _id: new ObjectId(id) });
|
||||||
|
|
||||||
|
if (result.deletedCount === 1) {
|
||||||
|
return NextResponse.json({ message: "Log deleted successfully" });
|
||||||
|
} else {
|
||||||
|
return NextResponse.json({ message: "Log not found" }, { status: 404 });
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ message: "Failed to delete log", error: error },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
BIN
app/favicon.ico
BIN
app/favicon.ico
Binary file not shown.
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB |
|
@ -30,7 +30,7 @@ export default function RootLayout({
|
||||||
disableTransitionOnChange
|
disableTransitionOnChange
|
||||||
>
|
>
|
||||||
<AuthProvider>
|
<AuthProvider>
|
||||||
{children}
|
<main>{children}</main>
|
||||||
<Toaster />
|
<Toaster />
|
||||||
</AuthProvider>
|
</AuthProvider>
|
||||||
</ThemeProvider>
|
</ThemeProvider>
|
||||||
|
|
BIN
bun.lockb
BIN
bun.lockb
Binary file not shown.
Loading…
Reference in a new issue