Skip to content

Commit

Permalink
first commit
Browse files Browse the repository at this point in the history
  • Loading branch information
hamster1963 committed Jul 26, 2024
0 parents commit 3085a41
Show file tree
Hide file tree
Showing 66 changed files with 2,172 additions and 0 deletions.
2 changes: 2 additions & 0 deletions .env.example
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
NezhaBaseUrl=http://0.0.0.0:8008
NezhaAuth=5hAY3QX6Nl9B3UOQgB26KdsdS1dsdUdM
Binary file added .github/shotOne.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added .github/shotTwo.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
48 changes: 48 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.

# dependencies
/node_modules
/.pnp
.pnp.js
.yarn/install-state.gz

# testing
/coverage

# next.js
/.next/
/out/

# production
/build

# misc
.DS_Store
*.pem

# debug
npm-debug.log*
yarn-debug.log*
yarn-error.log*

# local env files
.env*.local

# vercel
.vercel

# typescript
*.tsbuildinfo
next-env.d.ts

# pwa
/public/sw.js
/public/sw.js.map
/public/swe-worker-development.js
/public/workbox*.js
/public/workbox*.js.map

/.idea/

.env

10 changes: 10 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
<h1 align="center">HomeDash</h1>
<div align="center">
<strong>HomeDash 是一个基于 Next.js 和 Shadcn 的仪表盘</strong>
<br>
<strong>Demo地址: https://dash.buycoffee.top</strong>

![screen-shot-one](/.github/shotOne.png)
![screen-shot-two](/.github/shotTwo.png)

</div>
43 changes: 43 additions & 0 deletions app/(main)/ClientComponents/LiveTag.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
"use client";
import { useEffect, useState } from "react";
import { toast } from "sonner";

import { Badge } from "@/components/ui/badge";
import { verifySSEConnection } from "@/lib/sseFetch";

export default function LiveTag() {
const [connected, setConnected] = useState(false);

useEffect(() => {
// Store the promise in a variable
const ssePromise = verifySSEConnection(
"https://home.buycoffee.tech/v2/VerifySSEConnect",
);
setTimeout(() => {
toast.promise(ssePromise, {
loading: "Connecting to SSE...",
success: "HomeDash SSE Connected",
error: "Error connecting to SSE",
});
});
// Handle promise resolution separately
ssePromise
.then(() => {
setConnected(true);
})
.catch(() => {
setConnected(false);
});
}, []);

return connected ? (
<Badge className={"flex items-center justify-center gap-1 px-2"}>
Synced
<span className="h-2 w-2 rounded-full bg-green-500"></span>
</Badge>
) : (
<Badge className={"flex items-center justify-center gap-1 px-2"}>
Static<span className="h-2 w-2 rounded-full bg-red-500"></span>
</Badge>
);
}
34 changes: 34 additions & 0 deletions app/(main)/ClientComponents/ServerListClient.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
"use client";

import ServerCard from "@/components/ServerCard";
import { nezhaFetcher } from "@/lib/utils";
import useSWR from "swr";

export default function ServerListClient() {
const { data } = useSWR('/api/server', nezhaFetcher, {
refreshInterval: 2000,
});
if (!data) return null;
const sortedResult = data.result.sort((a: any, b: any) => a.id - b.id);

return (
<section className={"grid grid-cols-1 gap-2 md:grid-cols-2"}>
{sortedResult.map(
(server: any) => (
<ServerCard
key={server.id}
id={server.id}
cpu={server.status.CPU}
name={server.name}
up={server.status.NetOutSpeed / 1024 / 1024}
down={server.status.NetInSpeed / 1024 / 1024}
status={server.status.Uptime !== 0 ? "online" : "offline"}
uptime={server.status.Uptime / 86400}
mem={(server.status.MemUsed / server.host.MemTotal) * 100}
stg={server.status.DiskUsed / server.host.DiskTotal}
/>
),
)}
</section>
);
}
75 changes: 75 additions & 0 deletions app/(main)/ClientComponents/ServerOverviewClient.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,75 @@
"use client";

import { Card, CardContent } from "@/components/ui/card";
import blogMan from "@/public/blog-man.webp";
import Image from "next/image";
import useSWR from "swr";
import { formatBytes, nezhaFetcher } from "@/lib/utils";
import { Loader } from "@/components/loading/Loader";

export default function ServerOverviewClient() {
const { data } = useSWR('/api/server', nezhaFetcher);

return (
<section className="grid md:grid-cols-4 gap-4 grid-cols-2">
<Card>
<CardContent className="px-6 py-3" >
<section className="flex gap-1 flex-col">
<p className="text-md font-medium">Total servers</p>
<div className="flex items-center gap-2">
<span className="relative flex h-2 w-2">
<span className="relative inline-flex h-2 w-2 rounded-full bg-blue-500"></span>
</span>
{data ? <p className="text-lg font-semibold">{data?.result.length}</p> : <div className="h-7 flex items-center"><Loader visible={true} /></div>}
</div>
</section>
</CardContent>
</Card>
<Card>
<CardContent className="px-6 py-3" >
<section className="flex gap-1 flex-col">
<p className="text-md font-medium">Online servers</p>
<div className="flex items-center gap-2">
<span className="relative flex h-2 w-2">
<span className="absolute inline-flex h-full w-full animate-ping rounded-full bg-green-500 opacity-75"></span>
<span className="relative inline-flex h-2 w-2 rounded-full bg-green-500"></span>
</span>
{data ? <p className="text-lg font-semibold">{data?.live_servers}</p> : <div className="h-7 flex items-center"><Loader visible={true} /></div>}
</div>
</section>
</CardContent>
</Card>
<Card>
<CardContent className="px-6 py-3" >
<section className="flex gap-1 flex-col">
<p className="text-md font-medium">Offline servers</p>
<div className="flex items-center gap-2">
<span className="relative flex h-2 w-2">
<span className="absolute inline-flex h-full w-full animate-ping rounded-full bg-red-500 opacity-75"></span>
<span className="relative inline-flex h-2 w-2 rounded-full bg-red-500"></span>
</span>
{data ? <p className="text-lg font-semibold">{data?.offline_servers}</p> : <div className="h-7 flex items-center"><Loader visible={true} /></div>}
</div>


</section>
</CardContent>
</Card>
<Card>
<CardContent className="px-6 py-3 relative" >
<section className="flex gap-1 flex-col">
<p className="text-md font-medium">Total bandwidth</p>
{data ? <p className="text-lg font-semibold">{formatBytes(data?.total_bandwidth)}</p> : <div className="h-7 flex items-center"><Loader visible={true} /></div>}
</section>
<Image
className="pointer-events-none absolute right-3 top-[-85px] z-10 w-20"
alt={'Hamster1963'}
src={blogMan}
priority
placeholder=""
/>
</CardContent>
</Card>
</section>
)
}
62 changes: 62 additions & 0 deletions app/(main)/header.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
"use client";

import React, { useEffect, useState } from "react";
import Image from "next/image";
import { Separator } from "@/components/ui/separator";
import { DateTime } from "luxon";

function Header() {
return (
<div className="mx-auto w-full max-w-5xl">
<section className="flex items-center justify-between">
<section className="text-md flex items-center font-medium">
<div className="mr-1 flex flex-row items-center justify-start">
<Image
width={40}
height={40}
unoptimized
alt="apple-touch-icon"
src={"/apple-touch-icon.png"}
className="relative !m-0 h-6 w-6 border-2 border-white object-cover object-top !p-0 transition duration-500 group-hover:z-30 group-hover:scale-105"
/>
</div>
HomeDash
<Separator
orientation="vertical"
className="mx-2 hidden h-4 w-[1px] md:block"
/>
<p className="hidden text-sm font-medium opacity-40 md:block">
Simple and beautiful dashboard
</p>
</section>
{/* <LiveTag /> */}
</section>
<Overview />
</div>
);
}

function Overview() {
const [mouted, setMounted] = useState(false);
useEffect(() => {
setMounted(true);
}, []);
const time = DateTime.TIME_SIMPLE;
time.hour12 = true;

return (
<section className={"md:mt-16 mt-10 flex flex-col"}>
<p className="text-md font-semibold">👋 Overview</p>
<div className="flex items-center gap-1.5">
<p className="text-sm font-medium opacity-50">where the time is</p>
{mouted && (
<p className="opacity-1 text-sm font-medium">
{DateTime.now().setLocale("en-US").toLocaleString(time)}
</p>
)}
</div>
</section>
);
}

export default Header;
21 changes: 21 additions & 0 deletions app/(main)/layout.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
import React from "react";

import Header from "@/app/(main)/header";
import BlurLayers from "@/components/BlurLayer";

type DashboardProps = {
children: React.ReactNode;
};

export default function MainLayout({ children }: DashboardProps) {
return (
<div className="flex min-h-screen w-full flex-col">
<main className="flex min-h-[calc(100vh_-_theme(spacing.16))] flex-1 flex-col gap-4 bg-muted/40 p-4 md:p-10 md:pt-8">
<Header />
<BlurLayers />
{/* <Nav /> */}
{children}
</main>
</div>
);
}
57 changes: 57 additions & 0 deletions app/(main)/nav.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
"use client";

import { clsx } from "clsx";
import Link from "next/link";
import { usePathname } from "next/navigation";
import React from "react";

import { cn } from "@/lib/utils";

export const siteUrlList = [
{
name: "Home",
header: "👋 Overview",
url: "/",
},
{
name: "Service",
header: "🎛️ Service",
url: "/service",
},
];
export default function Nav() {
const nowPath = usePathname();
return (
<div className={"flex flex-col items-center justify-center"}>
<div
className={
"fixed bottom-6 z-50 flex items-center gap-1 rounded-[50px] bg-stone-700 bg-opacity-80 px-2 py-1.5 backdrop-blur-lg"
}
>
{siteUrlList.map((site, index) => (
<div key={site.name} className={"flex items-center gap-1"}>
{index !== 0 && (
<p key={index} className={"pointer-events-none text-stone-500"}>
/
</p>
)}
<Link
key={site.name}
href={site.url}
scroll={true}
className={cn(
"rounded-[50px] px-2.5 py-1.5 text-[16px] font-[500] text-stone-400 transition-colors sm:hover:text-white",
clsx(
nowPath === site.url &&
"bg-stone-500 text-white dark:bg-stone-600",
),
)}
>
{site.name}
</Link>
</div>
))}
</div>
</div>
);
}
Loading

0 comments on commit 3085a41

Please sign in to comment.