Initial commit
This commit is contained in:
1
src/assets/astro.svg
Normal file
1
src/assets/astro.svg
Normal file
@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" width="115" height="48"><path fill="#17191E" d="M7.77 36.35C6.4 35.11 6 32.51 6.57 30.62c.99 1.2 2.35 1.57 3.75 1.78 2.18.33 4.31.2 6.33-.78.23-.12.44-.27.7-.42.18.55.23 1.1.17 1.67a4.56 4.56 0 0 1-1.94 3.23c-.43.32-.9.61-1.34.91-1.38.94-1.76 2.03-1.24 3.62l.05.17a3.63 3.63 0 0 1-1.6-1.38 3.87 3.87 0 0 1-.63-2.1c0-.37 0-.74-.05-1.1-.13-.9-.55-1.3-1.33-1.32a1.56 1.56 0 0 0-1.63 1.26c0 .06-.03.12-.05.2Z"/><path fill="url(#a)" d="M7.77 36.35C6.4 35.11 6 32.51 6.57 30.62c.99 1.2 2.35 1.57 3.75 1.78 2.18.33 4.31.2 6.33-.78.23-.12.44-.27.7-.42.18.55.23 1.1.17 1.67a4.56 4.56 0 0 1-1.94 3.23c-.43.32-.9.61-1.34.91-1.38.94-1.76 2.03-1.24 3.62l.05.17a3.63 3.63 0 0 1-1.6-1.38 3.87 3.87 0 0 1-.63-2.1c0-.37 0-.74-.05-1.1-.13-.9-.55-1.3-1.33-1.32a1.56 1.56 0 0 0-1.63 1.26c0 .06-.03.12-.05.2Z"/><path fill="#17191E" d="M.02 30.31s4.02-1.95 8.05-1.95l3.04-9.4c.11-.45.44-.76.82-.76.37 0 .7.31.82.76l3.04 9.4c4.77 0 8.05 1.95 8.05 1.95L17 11.71c-.2-.56-.53-.91-.98-.91H7.83c-.44 0-.76.35-.97.9L.02 30.31Zm42.37-5.97c0 1.64-2.05 2.62-4.88 2.62-1.85 0-2.5-.45-2.5-1.41 0-1 .8-1.49 2.65-1.49 1.67 0 3.09.03 4.73.23v.05Zm.03-2.04a21.37 21.37 0 0 0-4.37-.36c-5.32 0-7.82 1.25-7.82 4.18 0 3.04 1.71 4.2 5.68 4.2 3.35 0 5.63-.84 6.46-2.92h.14c-.03.5-.05 1-.05 1.4 0 1.07.18 1.16 1.06 1.16h4.15a16.9 16.9 0 0 1-.36-4c0-1.67.06-2.93.06-4.62 0-3.45-2.07-5.64-8.56-5.64-2.8 0-5.9.48-8.26 1.19.22.93.54 2.83.7 4.06 2.04-.96 4.95-1.37 7.2-1.37 3.11 0 3.97.71 3.97 2.15v.57Zm11.37 3c-.56.07-1.33.07-2.12.07-.83 0-1.6-.03-2.12-.1l-.02.58c0 2.85 1.87 4.52 8.45 4.52 6.2 0 8.2-1.64 8.2-4.55 0-2.74-1.33-4.09-7.2-4.39-4.58-.2-4.99-.7-4.99-1.28 0-.66.59-1 3.65-1 3.18 0 4.03.43 4.03 1.35v.2a46.13 46.13 0 0 1 4.24.03l.02-.55c0-3.36-2.8-4.46-8.2-4.46-6.08 0-8.13 1.49-8.13 4.39 0 2.6 1.64 4.23 7.48 4.48 4.3.14 4.77.62 4.77 1.28 0 .7-.7 1.03-3.71 1.03-3.47 0-4.35-.48-4.35-1.47v-.13Zm19.82-12.05a17.5 17.5 0 0 1-6.24 3.48c.03.84.03 2.4.03 3.24l1.5.02c-.02 1.63-.04 3.6-.04 4.9 0 3.04 1.6 5.32 6.58 5.32 2.1 0 3.5-.23 5.23-.6a43.77 43.77 0 0 1-.46-4.13c-1.03.34-2.34.53-3.78.53-2 0-2.82-.55-2.82-2.13 0-1.37 0-2.65.03-3.84 2.57.02 5.13.07 6.64.11-.02-1.18.03-2.9.1-4.04-2.2.04-4.65.07-6.68.07l.07-2.93h-.16Zm13.46 6.04a767.33 767.33 0 0 1 .07-3.18H82.6c.07 1.96.07 3.98.07 6.92 0 2.95-.03 4.99-.07 6.93h5.18c-.09-1.37-.11-3.68-.11-5.65 0-3.1 1.26-4 4.12-4 1.33 0 2.28.16 3.1.46.03-1.16.26-3.43.4-4.43-.86-.25-1.81-.41-2.96-.41-2.46-.03-4.26.98-5.1 3.38l-.17-.02Zm22.55 3.65c0 2.5-1.8 3.66-4.64 3.66-2.81 0-4.61-1.1-4.61-3.66s1.82-3.52 4.61-3.52c2.82 0 4.64 1.03 4.64 3.52Zm4.71-.11c0-4.96-3.87-7.18-9.35-7.18-5.5 0-9.23 2.22-9.23 7.18 0 4.94 3.49 7.59 9.21 7.59 5.77 0 9.37-2.65 9.37-7.6Z"/><defs><linearGradient id="a" x1="6.33" x2="19.43" y1="40.8" y2="34.6" gradientUnits="userSpaceOnUse"><stop stop-color="#D83333"/><stop offset="1" stop-color="#F041FF"/></linearGradient></defs></svg>
|
After Width: | Height: | Size: 2.8 KiB |
1
src/assets/background.svg
Normal file
1
src/assets/background.svg
Normal file
@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="1440" height="1024" fill="none"><path fill="url(#a)" fill-rule="evenodd" d="M-217.58 475.75c91.82-72.02 225.52-29.38 341.2-44.74C240 415.56 372.33 315.14 466.77 384.9c102.9 76.02 44.74 246.76 90.31 366.31 29.83 78.24 90.48 136.14 129.48 210.23 57.92 109.99 169.67 208.23 155.9 331.77-13.52 121.26-103.42 264.33-224.23 281.37-141.96 20.03-232.72-220.96-374.06-196.99-151.7 25.73-172.68 330.24-325.85 315.72-128.6-12.2-110.9-230.73-128.15-358.76-12.16-90.14 65.87-176.25 44.1-264.57-26.42-107.2-167.12-163.46-176.72-273.45-10.15-116.29 33.01-248.75 124.87-320.79Z" clip-rule="evenodd" style="opacity:.154"/><path fill="url(#b)" fill-rule="evenodd" d="M1103.43 115.43c146.42-19.45 275.33-155.84 413.5-103.59 188.09 71.13 409 212.64 407.06 413.88-1.94 201.25-259.28 278.6-414.96 405.96-130 106.35-240.24 294.39-405.6 265.3-163.7-28.8-161.93-274.12-284.34-386.66-134.95-124.06-436-101.46-445.82-284.6-9.68-180.38 247.41-246.3 413.54-316.9 101.01-42.93 207.83 21.06 316.62 6.61Z" clip-rule="evenodd" style="opacity:.154"/><defs><linearGradient id="b" x1="373" x2="1995.44" y1="1100" y2="118.03" gradientUnits="userSpaceOnUse"><stop stop-color="#D83333"/><stop offset="1" stop-color="#F041FF"/></linearGradient><linearGradient id="a" x1="107.37" x2="1130.66" y1="1993.35" y2="1026.31" gradientUnits="userSpaceOnUse"><stop stop-color="#3245FF"/><stop offset="1" stop-color="#BC52EE"/></linearGradient></defs></svg>
|
After Width: | Height: | Size: 1.4 KiB |
39
src/components/TabView.tsx
Normal file
39
src/components/TabView.tsx
Normal file
@ -0,0 +1,39 @@
|
||||
import { useState } from "react";
|
||||
import InviteCodes from "./features/invites/InviteCodes";
|
||||
|
||||
const TABS = {
|
||||
invites: {
|
||||
component: InviteCodes,
|
||||
label: "Invite Codes",
|
||||
},
|
||||
} as const;
|
||||
|
||||
type TabKey = keyof typeof TABS;
|
||||
|
||||
export default function TabView() {
|
||||
const [activeTab, setActiveTab] = useState<TabKey>("invites");
|
||||
const ActiveComponent = TABS[activeTab].component;
|
||||
|
||||
return (
|
||||
<div className="container mx-auto max-w-3xl p-4">
|
||||
<div className="card bg-base-100 shadow-xl">
|
||||
<div className="card-body">
|
||||
<div className="tabs tabs-boxed">
|
||||
{Object.entries(TABS).map(([key, { label }]) => (
|
||||
<button
|
||||
key={key}
|
||||
className={`tab ${activeTab === key ? "tab-active" : ""}`}
|
||||
onClick={() => setActiveTab(key as TabKey)}
|
||||
>
|
||||
{label}
|
||||
</button>
|
||||
))}
|
||||
</div>
|
||||
<div className="mt-4">
|
||||
<ActiveComponent />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
40
src/components/auth/AuthWrapper.tsx
Normal file
40
src/components/auth/AuthWrapper.tsx
Normal file
@ -0,0 +1,40 @@
|
||||
import { useState } from "react";
|
||||
import { Settings } from "../../lib/settings";
|
||||
import NavBar from "../navigation/NavBar";
|
||||
import LoginForm from "./LoginForm";
|
||||
import { LogoutContext } from "../../lib/LogoutContext";
|
||||
import { RefreshContext } from "../../lib/RefreshContext";
|
||||
import TabView from "../TabView";
|
||||
|
||||
export default function AuthWrapper() {
|
||||
const [isConfigured, setIsConfigured] = useState(
|
||||
Boolean(Settings.getServiceUrl() && Settings.getAdminPassword()),
|
||||
);
|
||||
const [lastUpdate, setLastUpdate] = useState(Date.now());
|
||||
|
||||
const handleLogout = () => {
|
||||
Settings.clearSettings();
|
||||
setIsConfigured(false);
|
||||
};
|
||||
|
||||
const refresh = () => {
|
||||
setLastUpdate(Date.now());
|
||||
};
|
||||
|
||||
if (!isConfigured) {
|
||||
return <LoginForm onLogin={() => setIsConfigured(true)} />;
|
||||
}
|
||||
|
||||
return (
|
||||
<LogoutContext.Provider value={handleLogout}>
|
||||
<RefreshContext.Provider value={{ refresh, lastUpdate }}>
|
||||
<div className="min-h-screen flex flex-col">
|
||||
<NavBar />
|
||||
<div className="flex-1">
|
||||
<TabView />
|
||||
</div>
|
||||
</div>
|
||||
</RefreshContext.Provider>
|
||||
</LogoutContext.Provider>
|
||||
);
|
||||
}
|
63
src/components/auth/LoginForm.tsx
Normal file
63
src/components/auth/LoginForm.tsx
Normal file
@ -0,0 +1,63 @@
|
||||
import { useState } from "react";
|
||||
import { Settings } from "../../lib/settings";
|
||||
|
||||
interface LoginFormProps {
|
||||
onLogin: () => void;
|
||||
}
|
||||
|
||||
export default function LoginForm({ onLogin }: LoginFormProps) {
|
||||
const [serviceUrl, setServiceUrl] = useState("");
|
||||
const [adminPassword, setAdminPassword] = useState("");
|
||||
|
||||
const handleSubmit = (e: React.FormEvent) => {
|
||||
e.preventDefault();
|
||||
Settings.saveSettings(serviceUrl, adminPassword);
|
||||
onLogin();
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="min-h-screen bg-base-200 flex items-center justify-center">
|
||||
<div className="card w-full max-w-md bg-base-100 shadow-xl">
|
||||
<div className="card-body">
|
||||
<h1 className="card-title text-2xl font-bold text-center">
|
||||
BlueSky PDS Manager
|
||||
</h1>
|
||||
<p className="text-center text-base-content/70 mb-4">
|
||||
Enter your PDS credentials to get started
|
||||
</p>
|
||||
<form onSubmit={handleSubmit} className="space-y-4">
|
||||
<div className="form-control w-full">
|
||||
<label className="label">
|
||||
<span className="label-text">PDS URL</span>
|
||||
</label>
|
||||
<input
|
||||
type="url"
|
||||
value={serviceUrl}
|
||||
onChange={(e) => setServiceUrl(e.target.value)}
|
||||
placeholder="https://bsky.web.site"
|
||||
required
|
||||
className="input input-bordered w-full"
|
||||
/>
|
||||
</div>
|
||||
<div className="form-control w-full">
|
||||
<label className="label">
|
||||
<span className="label-text">Admin Password</span>
|
||||
</label>
|
||||
<input
|
||||
type="password"
|
||||
value={adminPassword}
|
||||
onChange={(e) => setAdminPassword(e.target.value)}
|
||||
required
|
||||
className="input input-bordered w-full"
|
||||
/>
|
||||
</div>
|
||||
<div className="divider"></div>
|
||||
<button type="submit" className="btn btn-primary w-full">
|
||||
Connect to PDS
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
288
src/components/features/invites/InviteCodes.tsx
Normal file
288
src/components/features/invites/InviteCodes.tsx
Normal file
@ -0,0 +1,288 @@
|
||||
import { useState, useEffect } from "react";
|
||||
import { Settings } from "../../../lib/settings";
|
||||
import { useRefresh } from "../../../lib/RefreshContext";
|
||||
|
||||
interface InviteCodeUse {
|
||||
usedBy: string;
|
||||
usedAt: string;
|
||||
}
|
||||
|
||||
interface InviteCode {
|
||||
code: string;
|
||||
available: number;
|
||||
disabled: boolean;
|
||||
forAccount: string;
|
||||
createdBy: string;
|
||||
createdAt: string;
|
||||
uses: InviteCodeUse[];
|
||||
}
|
||||
|
||||
function getAuthHeader(adminPassword: string) {
|
||||
const authString = `admin:${adminPassword}`;
|
||||
const base64Auth = btoa(authString);
|
||||
return `Basic ${base64Auth}`;
|
||||
}
|
||||
|
||||
export default function InviteCodes() {
|
||||
const [codes, setCodes] = useState<InviteCode[]>([]);
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const { lastUpdate } = useRefresh();
|
||||
|
||||
const resetState = () => {
|
||||
setCodes([]);
|
||||
setError(null);
|
||||
setLoading(false);
|
||||
};
|
||||
|
||||
const sortCodes = (codes: InviteCode[]) => {
|
||||
return [...codes].sort((a, b) => {
|
||||
if (a.disabled && !b.disabled) return 1;
|
||||
if (!a.disabled && b.disabled) return -1;
|
||||
return new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime();
|
||||
});
|
||||
};
|
||||
|
||||
const fetchCodes = async () => {
|
||||
try {
|
||||
setLoading(true);
|
||||
setError(null);
|
||||
|
||||
const url = `${Settings.getServiceUrl()}/xrpc/com.atproto.admin.getInviteCodes`;
|
||||
const headers = {
|
||||
Authorization: getAuthHeader(Settings.getAdminPassword()),
|
||||
"Content-Type": "application/json",
|
||||
};
|
||||
|
||||
const params = new URLSearchParams({
|
||||
sort: "recent",
|
||||
limit: "100",
|
||||
});
|
||||
|
||||
const response = await fetch(`${url}?${params}`, { headers });
|
||||
|
||||
if (!response.ok) {
|
||||
const errorText = await response.text();
|
||||
console.error("Error response:", errorText);
|
||||
resetState();
|
||||
throw new Error(
|
||||
`Failed to fetch codes: ${response.status} ${response.statusText}`,
|
||||
);
|
||||
}
|
||||
|
||||
const data = await response.json();
|
||||
setCodes(data.codes || []);
|
||||
} catch (err) {
|
||||
console.error("Fetch error:", err);
|
||||
resetState();
|
||||
setError(err instanceof Error ? err.message : "Failed to fetch codes");
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const createCode = async () => {
|
||||
try {
|
||||
setLoading(true);
|
||||
setError(null);
|
||||
|
||||
const url = `${Settings.getServiceUrl()}/xrpc/com.atproto.server.createInviteCode`;
|
||||
const headers = {
|
||||
Authorization: getAuthHeader(Settings.getAdminPassword()),
|
||||
"Content-Type": "application/json",
|
||||
};
|
||||
|
||||
const response = await fetch(url, {
|
||||
method: "POST",
|
||||
headers,
|
||||
body: JSON.stringify({
|
||||
useCount: 1,
|
||||
}),
|
||||
});
|
||||
|
||||
if (!response.ok) {
|
||||
const errorText = await response.text();
|
||||
console.error("Error response:", errorText);
|
||||
throw new Error(
|
||||
`Failed to create code: ${response.status} ${response.statusText}`,
|
||||
);
|
||||
}
|
||||
|
||||
await fetchCodes();
|
||||
} catch (err) {
|
||||
console.error("Create error:", err);
|
||||
setError(err instanceof Error ? err.message : "Failed to create code");
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const disableCode = async (code: string) => {
|
||||
try {
|
||||
setLoading(true);
|
||||
setError(null);
|
||||
|
||||
const url = `${Settings.getServiceUrl()}/xrpc/com.atproto.admin.disableInviteCodes`;
|
||||
const headers = {
|
||||
Authorization: getAuthHeader(Settings.getAdminPassword()),
|
||||
"Content-Type": "application/json",
|
||||
};
|
||||
|
||||
const response = await fetch(url, {
|
||||
method: "POST",
|
||||
headers,
|
||||
body: JSON.stringify({
|
||||
codes: [code],
|
||||
}),
|
||||
});
|
||||
|
||||
if (!response.ok) {
|
||||
const errorText = await response.text();
|
||||
console.error("Error response:", errorText);
|
||||
throw new Error(
|
||||
`Failed to disable code: ${response.status} ${response.statusText}`,
|
||||
);
|
||||
}
|
||||
|
||||
await fetchCodes();
|
||||
} catch (err) {
|
||||
console.error("Disable error:", err);
|
||||
setError(err instanceof Error ? err.message : "Failed to disable code");
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
return () => {
|
||||
resetState();
|
||||
};
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
if (typeof window !== "undefined") {
|
||||
fetchCodes();
|
||||
}
|
||||
}, [lastUpdate]);
|
||||
|
||||
if (typeof window === "undefined") {
|
||||
return <div></div>;
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="card bg-base-100 shadow-xl">
|
||||
<div className="card-body">
|
||||
<div className="flex justify-between items-center mb-4">
|
||||
<h1 className="card-title text-2xl">Invite Codes</h1>
|
||||
<button
|
||||
onClick={createCode}
|
||||
disabled={loading}
|
||||
className="btn btn-primary"
|
||||
>
|
||||
Create New Code
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{error && (
|
||||
<div className="alert alert-error">
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
className="stroke-current shrink-0 h-6 w-6"
|
||||
fill="none"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
strokeLinecap="round"
|
||||
strokeLinejoin="round"
|
||||
strokeWidth="2"
|
||||
d="M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z"
|
||||
/>
|
||||
</svg>
|
||||
<div>
|
||||
<h3 className="font-bold">Error!</h3>
|
||||
<div className="text-xs">{error}</div>
|
||||
</div>
|
||||
<button onClick={fetchCodes} className="btn btn-sm">
|
||||
Retry
|
||||
</button>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{loading ? (
|
||||
<div className="flex justify-center p-8">
|
||||
<span className="loading loading-spinner loading-lg"></span>
|
||||
</div>
|
||||
) : (
|
||||
<div className="space-y-4">
|
||||
{sortCodes(codes).map((code, index) => (
|
||||
<div
|
||||
key={index}
|
||||
className={`card ${
|
||||
code.disabled ? "bg-base-300 opacity-75" : "bg-base-200"
|
||||
} hover:shadow-md transition-shadow`}
|
||||
>
|
||||
<div className="card-body">
|
||||
<div className="flex justify-between items-start">
|
||||
<div
|
||||
className={`font-mono text-lg bg-base-100 p-3 rounded-box select-all flex-1 mr-4 ${
|
||||
code.disabled ? "opacity-50" : ""
|
||||
}`}
|
||||
>
|
||||
{code.code}
|
||||
</div>
|
||||
<div className="flex flex-col items-end gap-2">
|
||||
{code.disabled ? (
|
||||
<span className="badge badge-error">Disabled</span>
|
||||
) : (
|
||||
<button
|
||||
onClick={() => disableCode(code.code)}
|
||||
className="btn btn-sm btn-warning"
|
||||
title="Disable code"
|
||||
>
|
||||
Disable
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex flex-wrap gap-4 mt-2 text-sm">
|
||||
{!code.disabled && (
|
||||
<div className="badge badge-neutral">
|
||||
Uses: {code.uses.length} / {code.available}
|
||||
</div>
|
||||
)}
|
||||
<div className="badge badge-ghost">
|
||||
Created: {new Date(code.createdAt).toLocaleString()}
|
||||
</div>
|
||||
{code.uses.length > 0 && (
|
||||
<div className="badge badge-info">
|
||||
{code.uses.length}{" "}
|
||||
{code.uses.length === 1 ? "use" : "uses"}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
{code.uses.length > 0 && (
|
||||
<div
|
||||
className={`text-sm mt-2 ${
|
||||
code.disabled ? "opacity-75" : "opacity-70"
|
||||
}`}
|
||||
>
|
||||
<span className="font-semibold">Used by:</span>{" "}
|
||||
{code.uses.map((use) => use.usedBy).join(", ")}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
{!loading && codes.length === 0 && (
|
||||
<div className="text-center p-8 bg-base-200 rounded-box">
|
||||
<div className="text-base-content/70">
|
||||
No invite codes found
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
119
src/components/navigation/NavBar.tsx
Normal file
119
src/components/navigation/NavBar.tsx
Normal file
@ -0,0 +1,119 @@
|
||||
import { useLogout } from "../../lib/LogoutContext";
|
||||
import { useRefresh } from "../../lib/RefreshContext";
|
||||
import { useState } from "react";
|
||||
import { Settings } from "../../lib/settings";
|
||||
|
||||
export default function NavBar() {
|
||||
const logout = useLogout();
|
||||
const { refresh } = useRefresh();
|
||||
const [showSettings, setShowSettings] = useState(false);
|
||||
const [newUrl, setNewUrl] = useState(Settings.getServiceUrl());
|
||||
const [newPassword, setNewPassword] = useState(Settings.getAdminPassword());
|
||||
|
||||
const saveSettings = () => {
|
||||
Settings.saveSettings(newUrl, newPassword);
|
||||
setShowSettings(false);
|
||||
refresh();
|
||||
};
|
||||
|
||||
return (
|
||||
<>
|
||||
<div className="navbar bg-base-100 shadow-md">
|
||||
<div className="container mx-auto max-w-3xl px-4">
|
||||
<div className="flex-1">
|
||||
<span className="text-xl font-bold">BlueSky PDS Manager</span>
|
||||
</div>
|
||||
<div className="flex-none">
|
||||
{/* Regular buttons for larger screens */}
|
||||
<div className="hidden sm:flex items-center gap-2">
|
||||
<button
|
||||
onClick={() => setShowSettings(true)}
|
||||
className="btn btn-ghost"
|
||||
>
|
||||
⚙️ Settings
|
||||
</button>
|
||||
<button onClick={logout} className="btn btn-ghost text-error">
|
||||
🚪 Logout
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{/* Dropdown for mobile */}
|
||||
<div className="dropdown dropdown-end sm:hidden">
|
||||
<label tabIndex={0} className="btn btn-ghost btn-circle">
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
fill="none"
|
||||
viewBox="0 0 24 24"
|
||||
className="inline-block w-5 h-5 stroke-current"
|
||||
>
|
||||
<path
|
||||
strokeLinecap="round"
|
||||
strokeLinejoin="round"
|
||||
strokeWidth="2"
|
||||
d="M5 12h.01M12 12h.01M19 12h.01M6 12a1 1 0 11-2 0 1 1 0 012 0zm7 0a1 1 0 11-2 0 1 1 0 012 0zm7 0a1 1 0 11-2 0 1 1 0 012 0z"
|
||||
/>
|
||||
</svg>
|
||||
</label>
|
||||
<ul
|
||||
tabIndex={0}
|
||||
className="dropdown-content z-[1] menu p-2 shadow bg-base-100 rounded-box w-52"
|
||||
>
|
||||
<li>
|
||||
<button onClick={() => setShowSettings(true)}>
|
||||
⚙️ Settings
|
||||
</button>
|
||||
</li>
|
||||
<li>
|
||||
<button onClick={logout} className="text-error">
|
||||
🚪 Logout
|
||||
</button>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Settings Modal */}
|
||||
<dialog className={`modal ${showSettings ? "modal-open" : ""}`}>
|
||||
<div className="modal-box">
|
||||
<h3 className="font-bold text-lg mb-4">PDS Settings</h3>
|
||||
<div className="form-control w-full">
|
||||
<label className="label">
|
||||
<span className="label-text">PDS URL</span>
|
||||
</label>
|
||||
<input
|
||||
type="url"
|
||||
value={newUrl}
|
||||
onChange={(e) => setNewUrl(e.target.value)}
|
||||
className="input input-bordered w-full"
|
||||
placeholder="https://bsky.atri.dad"
|
||||
/>
|
||||
</div>
|
||||
<div className="form-control w-full mt-4">
|
||||
<label className="label">
|
||||
<span className="label-text">Admin Password</span>
|
||||
</label>
|
||||
<input
|
||||
type="password"
|
||||
value={newPassword}
|
||||
onChange={(e) => setNewPassword(e.target.value)}
|
||||
className="input input-bordered w-full"
|
||||
/>
|
||||
</div>
|
||||
<div className="modal-action">
|
||||
<button onClick={() => setShowSettings(false)} className="btn">
|
||||
Cancel
|
||||
</button>
|
||||
<button onClick={saveSettings} className="btn btn-primary">
|
||||
Save
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<form method="dialog" className="modal-backdrop">
|
||||
<button onClick={() => setShowSettings(false)}>close</button>
|
||||
</form>
|
||||
</dialog>
|
||||
</>
|
||||
);
|
||||
}
|
16
src/layouts/Layout.astro
Normal file
16
src/layouts/Layout.astro
Normal file
@ -0,0 +1,16 @@
|
||||
<!doctype html>
|
||||
<html data-theme="dark" class="min-h-screen">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width" />
|
||||
<link
|
||||
rel="icon"
|
||||
href="data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 100 100'><text y='.9em' font-size='90'>🦋</text></svg>"
|
||||
/>
|
||||
<meta name="generator" content={Astro.generator} />
|
||||
<title>PDS Manager</title>
|
||||
</head>
|
||||
<body class="min-h-screen bg-base-200">
|
||||
<slot />
|
||||
</body>
|
||||
</html>
|
13
src/lib/LogoutContext.tsx
Normal file
13
src/lib/LogoutContext.tsx
Normal file
@ -0,0 +1,13 @@
|
||||
import * as React from "react";
|
||||
|
||||
export const LogoutContext = React.createContext<(() => void) | undefined>(
|
||||
undefined,
|
||||
);
|
||||
|
||||
export function useLogout() {
|
||||
const context = React.useContext(LogoutContext);
|
||||
if (!context) {
|
||||
throw new Error("useLogout must be used within a LogoutProvider");
|
||||
}
|
||||
return context;
|
||||
}
|
15
src/lib/RefreshContext.tsx
Normal file
15
src/lib/RefreshContext.tsx
Normal file
@ -0,0 +1,15 @@
|
||||
import * as React from "react";
|
||||
|
||||
interface RefreshContextType {
|
||||
refresh: () => void;
|
||||
lastUpdate: number;
|
||||
}
|
||||
|
||||
export const RefreshContext = React.createContext<RefreshContextType>({
|
||||
refresh: () => {},
|
||||
lastUpdate: Date.now(),
|
||||
});
|
||||
|
||||
export function useRefresh() {
|
||||
return React.useContext(RefreshContext);
|
||||
}
|
23
src/lib/settings.ts
Normal file
23
src/lib/settings.ts
Normal file
@ -0,0 +1,23 @@
|
||||
export class Settings {
|
||||
static getServiceUrl(): string {
|
||||
if (typeof window === "undefined") return "";
|
||||
return window.localStorage.getItem("pds_url") || "";
|
||||
}
|
||||
|
||||
static getAdminPassword(): string {
|
||||
if (typeof window === "undefined") return "";
|
||||
return window.localStorage.getItem("pds_admin_password") || "";
|
||||
}
|
||||
|
||||
static saveSettings(url: string, password: string) {
|
||||
if (typeof window === "undefined") return;
|
||||
window.localStorage.setItem("pds_url", url);
|
||||
window.localStorage.setItem("pds_admin_password", password);
|
||||
}
|
||||
|
||||
static clearSettings() {
|
||||
if (typeof window === "undefined") return;
|
||||
window.localStorage.removeItem("pds_url");
|
||||
window.localStorage.removeItem("pds_admin_password");
|
||||
}
|
||||
}
|
10
src/pages/index.astro
Normal file
10
src/pages/index.astro
Normal file
@ -0,0 +1,10 @@
|
||||
---
|
||||
import Layout from "../layouts/Layout.astro";
|
||||
import AuthWrapper from "../components/auth/AuthWrapper";
|
||||
---
|
||||
|
||||
<Layout>
|
||||
<main class="min-h-screen">
|
||||
<AuthWrapper client:only="react" />
|
||||
</main>
|
||||
</Layout>
|
0
src/types/index.ts
Normal file
0
src/types/index.ts
Normal file
Reference in New Issue
Block a user