This commit is contained in:
parent
27f2ece380
commit
d083f5e1f6
14 changed files with 590 additions and 702 deletions
|
@ -1,7 +1,7 @@
|
||||||
{
|
{
|
||||||
"name": "",
|
"name": "pdsmanager",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"version": "0.0.1",
|
"version": "0.1.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "astro dev",
|
"dev": "astro dev",
|
||||||
"build": "astro build",
|
"build": "astro build",
|
||||||
|
@ -27,4 +27,4 @@
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"daisyui": "^4.12.23"
|
"daisyui": "^4.12.23"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,120 +0,0 @@
|
||||||
import { useState } from "react";
|
|
||||||
import { Settings } from "../lib/settings";
|
|
||||||
import InviteCodes from "./InviteCodes";
|
|
||||||
|
|
||||||
export default function App() {
|
|
||||||
const [serviceUrl, setServiceUrl] = useState(Settings.getServiceUrl());
|
|
||||||
const [adminPassword, setAdminPassword] = useState(
|
|
||||||
Settings.getAdminPassword(),
|
|
||||||
);
|
|
||||||
const [isConfigured, setIsConfigured] = useState(
|
|
||||||
Boolean(Settings.getServiceUrl() && Settings.getAdminPassword()),
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleSubmit = (e: React.FormEvent) => {
|
|
||||||
e.preventDefault();
|
|
||||||
Settings.saveSettings(serviceUrl, adminPassword);
|
|
||||||
setIsConfigured(true);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClearSettings = () => {
|
|
||||||
Settings.clearSettings();
|
|
||||||
setServiceUrl("");
|
|
||||||
setAdminPassword("");
|
|
||||||
setIsConfigured(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!isConfigured) {
|
|
||||||
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"
|
|
||||||
id="serviceUrl"
|
|
||||||
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"
|
|
||||||
id="adminPassword"
|
|
||||||
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>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="min-h-screen bg-base-200 flex flex-col">
|
|
||||||
<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">
|
|
||||||
<div className="dropdown dropdown-end">
|
|
||||||
<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={handleClearSettings} className="text-error">
|
|
||||||
Logout
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className="flex-1">
|
|
||||||
<InviteCodes onLogout={handleClearSettings} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,356 +0,0 @@
|
||||||
import { useState, useEffect } from "react";
|
|
||||||
import { Settings } from "../lib/settings";
|
|
||||||
|
|
||||||
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({ onLogout }: { onLogout: () => void }) {
|
|
||||||
const [codes, setCodes] = useState<InviteCode[]>([]);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
const [showSettings, setShowSettings] = useState(false);
|
|
||||||
const [newUrl, setNewUrl] = useState(Settings.getServiceUrl());
|
|
||||||
const [newPassword, setNewPassword] = useState(Settings.getAdminPassword());
|
|
||||||
|
|
||||||
const resetState = () => {
|
|
||||||
setCodes([]);
|
|
||||||
setError(null);
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const sortCodes = (codes: InviteCode[]) => {
|
|
||||||
return [...codes].sort((a, b) => {
|
|
||||||
// First sort by disabled status
|
|
||||||
if (a.disabled && !b.disabled) return 1;
|
|
||||||
if (!a.disabled && b.disabled) return -1;
|
|
||||||
// Then sort by creation date (newest first)
|
|
||||||
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`;
|
|
||||||
console.log("Fetching from:", url);
|
|
||||||
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const saveSettings = () => {
|
|
||||||
resetState();
|
|
||||||
Settings.saveSettings(newUrl, newPassword);
|
|
||||||
setShowSettings(false);
|
|
||||||
fetchCodes();
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
return () => {
|
|
||||||
resetState();
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (typeof window !== "undefined") {
|
|
||||||
fetchCodes();
|
|
||||||
}
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
if (typeof window === "undefined") {
|
|
||||||
return <div></div>;
|
|
||||||
}
|
|
||||||
|
|
||||||
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="flex justify-between items-center mb-4">
|
|
||||||
<h1 className="card-title text-2xl">Invite Codes</h1>
|
|
||||||
<div className="flex gap-2">
|
|
||||||
<button
|
|
||||||
onClick={() => setShowSettings(true)}
|
|
||||||
className="btn btn-ghost btn-sm"
|
|
||||||
>
|
|
||||||
⚙️ Settings
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
onClick={createCode}
|
|
||||||
disabled={loading}
|
|
||||||
className="btn btn-primary"
|
|
||||||
>
|
|
||||||
Create New Code
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</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>
|
|
||||||
|
|
||||||
{/* 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={onLogout} className="btn btn-error">
|
|
||||||
Logout
|
|
||||||
</button>
|
|
||||||
<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>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
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>
|
||||||
|
);
|
||||||
|
}
|
|
@ -1,209 +0,0 @@
|
||||||
---
|
|
||||||
import astroLogo from '../assets/astro.svg';
|
|
||||||
import background from '../assets/background.svg';
|
|
||||||
---
|
|
||||||
|
|
||||||
<div id="container">
|
|
||||||
<img id="background" src={background.src} alt="" fetchpriority="high" />
|
|
||||||
<main>
|
|
||||||
<section id="hero">
|
|
||||||
<a href="https://astro.build"
|
|
||||||
><img src={astroLogo.src} width="115" height="48" alt="Astro Homepage" /></a
|
|
||||||
>
|
|
||||||
<h1>
|
|
||||||
To get started, open the <code><pre>src/pages</pre></code> directory in your project.
|
|
||||||
</h1>
|
|
||||||
<section id="links">
|
|
||||||
<a class="button" href="https://docs.astro.build">Read our docs</a>
|
|
||||||
<a href="https://astro.build/chat"
|
|
||||||
>Join our Discord <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 127.14 96.36"
|
|
||||||
><path
|
|
||||||
fill="currentColor"
|
|
||||||
d="M107.7 8.07A105.15 105.15 0 0 0 81.47 0a72.06 72.06 0 0 0-3.36 6.83 97.68 97.68 0 0 0-29.11 0A72.37 72.37 0 0 0 45.64 0a105.89 105.89 0 0 0-26.25 8.09C2.79 32.65-1.71 56.6.54 80.21a105.73 105.73 0 0 0 32.17 16.15 77.7 77.7 0 0 0 6.89-11.11 68.42 68.42 0 0 1-10.85-5.18c.91-.66 1.8-1.34 2.66-2a75.57 75.57 0 0 0 64.32 0c.87.71 1.76 1.39 2.66 2a68.68 68.68 0 0 1-10.87 5.19 77 77 0 0 0 6.89 11.1 105.25 105.25 0 0 0 32.19-16.14c2.64-27.38-4.51-51.11-18.9-72.15ZM42.45 65.69C36.18 65.69 31 60 31 53s5-12.74 11.43-12.74S54 46 53.89 53s-5.05 12.69-11.44 12.69Zm42.24 0C78.41 65.69 73.25 60 73.25 53s5-12.74 11.44-12.74S96.23 46 96.12 53s-5.04 12.69-11.43 12.69Z"
|
|
||||||
></path></svg
|
|
||||||
>
|
|
||||||
</a>
|
|
||||||
</section>
|
|
||||||
</section>
|
|
||||||
</main>
|
|
||||||
|
|
||||||
<a href="https://astro.build/blog/astro-5/" id="news" class="box">
|
|
||||||
<svg width="32" height="32" fill="none" xmlns="http://www.w3.org/2000/svg"
|
|
||||||
><path
|
|
||||||
d="M24.667 12c1.333 1.414 2 3.192 2 5.334 0 4.62-4.934 5.7-7.334 12C18.444 28.567 18 27.456 18 26c0-4.642 6.667-7.053 6.667-14Zm-5.334-5.333c1.6 1.65 2.4 3.43 2.4 5.333 0 6.602-8.06 7.59-6.4 17.334C13.111 27.787 12 25.564 12 22.666c0-4.434 7.333-8 7.333-16Zm-6-5.333C15.111 3.555 16 5.556 16 7.333c0 8.333-11.333 10.962-5.333 22-3.488-.774-6-4-6-8 0-8.667 8.666-10 8.666-20Z"
|
|
||||||
fill="#111827"></path></svg
|
|
||||||
>
|
|
||||||
<h2>What's New in Astro 5.0?</h2>
|
|
||||||
<p>
|
|
||||||
From content layers to server islands, click to learn more about the new features and
|
|
||||||
improvements in Astro 5.0
|
|
||||||
</p>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
#background {
|
|
||||||
position: fixed;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
z-index: -1;
|
|
||||||
filter: blur(100px);
|
|
||||||
}
|
|
||||||
|
|
||||||
#container {
|
|
||||||
font-family: Inter, Roboto, 'Helvetica Neue', 'Arial Nova', 'Nimbus Sans', Arial, sans-serif;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
main {
|
|
||||||
height: 100%;
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
#hero {
|
|
||||||
display: flex;
|
|
||||||
align-items: start;
|
|
||||||
flex-direction: column;
|
|
||||||
justify-content: center;
|
|
||||||
padding: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1 {
|
|
||||||
font-size: 22px;
|
|
||||||
margin-top: 0.25em;
|
|
||||||
}
|
|
||||||
|
|
||||||
#links {
|
|
||||||
display: flex;
|
|
||||||
gap: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#links a {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
padding: 10px 12px;
|
|
||||||
color: #111827;
|
|
||||||
text-decoration: none;
|
|
||||||
transition: color 0.2s;
|
|
||||||
}
|
|
||||||
|
|
||||||
#links a:hover {
|
|
||||||
color: rgb(78, 80, 86);
|
|
||||||
}
|
|
||||||
|
|
||||||
#links a svg {
|
|
||||||
height: 1em;
|
|
||||||
margin-left: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#links a.button {
|
|
||||||
color: white;
|
|
||||||
background: linear-gradient(83.21deg, #3245ff 0%, #bc52ee 100%);
|
|
||||||
box-shadow:
|
|
||||||
inset 0 0 0 1px rgba(255, 255, 255, 0.12),
|
|
||||||
inset 0 -2px 0 rgba(0, 0, 0, 0.24);
|
|
||||||
border-radius: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#links a.button:hover {
|
|
||||||
color: rgb(230, 230, 230);
|
|
||||||
box-shadow: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
pre {
|
|
||||||
font-family: ui-monospace, 'Cascadia Code', 'Source Code Pro', Menlo, Consolas,
|
|
||||||
'DejaVu Sans Mono', monospace;
|
|
||||||
font-weight: normal;
|
|
||||||
background: linear-gradient(14deg, #d83333 0%, #f041ff 100%);
|
|
||||||
-webkit-background-clip: text;
|
|
||||||
-webkit-text-fill-color: transparent;
|
|
||||||
background-clip: text;
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
h2 {
|
|
||||||
margin: 0 0 1em;
|
|
||||||
font-weight: normal;
|
|
||||||
color: #111827;
|
|
||||||
font-size: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
p {
|
|
||||||
color: #4b5563;
|
|
||||||
font-size: 16px;
|
|
||||||
line-height: 24px;
|
|
||||||
letter-spacing: -0.006em;
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
code {
|
|
||||||
display: inline-block;
|
|
||||||
background:
|
|
||||||
linear-gradient(66.77deg, #f3cddd 0%, #f5cee7 100%) padding-box,
|
|
||||||
linear-gradient(155deg, #d83333 0%, #f041ff 18%, #f5cee7 45%) border-box;
|
|
||||||
border-radius: 8px;
|
|
||||||
border: 1px solid transparent;
|
|
||||||
padding: 6px 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.box {
|
|
||||||
padding: 16px;
|
|
||||||
background: rgba(255, 255, 255, 1);
|
|
||||||
border-radius: 16px;
|
|
||||||
border: 1px solid white;
|
|
||||||
}
|
|
||||||
|
|
||||||
#news {
|
|
||||||
position: absolute;
|
|
||||||
bottom: 16px;
|
|
||||||
right: 16px;
|
|
||||||
max-width: 300px;
|
|
||||||
text-decoration: none;
|
|
||||||
transition: background 0.2s;
|
|
||||||
backdrop-filter: blur(50px);
|
|
||||||
}
|
|
||||||
|
|
||||||
#news:hover {
|
|
||||||
background: rgba(255, 255, 255, 0.55);
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (max-height: 368px) {
|
|
||||||
#news {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (max-width: 768px) {
|
|
||||||
#container {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
#hero {
|
|
||||||
display: block;
|
|
||||||
padding-top: 10%;
|
|
||||||
}
|
|
||||||
|
|
||||||
#links {
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
#links a.button {
|
|
||||||
padding: 14px 18px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#news {
|
|
||||||
right: 16px;
|
|
||||||
left: 16px;
|
|
||||||
bottom: 2.5rem;
|
|
||||||
max-width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1 {
|
|
||||||
line-height: 1.5;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</style>
|
|
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>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
|
@ -1,22 +1,16 @@
|
||||||
<!doctype html>
|
<!doctype html>
|
||||||
<html lang="en">
|
<html data-theme="dark" class="min-h-screen">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
<meta name="viewport" content="width=device-width" />
|
<meta name="viewport" content="width=device-width" />
|
||||||
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
|
<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} />
|
<meta name="generator" content={Astro.generator} />
|
||||||
<title>PDS Manager</title>
|
<title>PDS Manager</title>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body class="min-h-screen bg-base-200">
|
||||||
<slot />
|
<slot />
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
||||||
<style>
|
|
||||||
html,
|
|
||||||
body {
|
|
||||||
margin: 0;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
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);
|
||||||
|
}
|
|
@ -1,8 +1,10 @@
|
||||||
---
|
---
|
||||||
import Layout from "../layouts/Layout.astro";
|
import Layout from "../layouts/Layout.astro";
|
||||||
import App from "../components/App";
|
import AuthWrapper from "../components/auth/AuthWrapper";
|
||||||
---
|
---
|
||||||
|
|
||||||
<Layout>
|
<Layout>
|
||||||
<App client:load />
|
<main class="min-h-screen">
|
||||||
|
<AuthWrapper client:only="react" />
|
||||||
|
</main>
|
||||||
</Layout>
|
</Layout>
|
||||||
|
|
Loading…
Add table
Reference in a new issue