first commit

This commit is contained in:
LeNei
2023-11-25 16:18:54 +01:00
commit 9c5d791d10
31 changed files with 3840 additions and 0 deletions

358
app/dashboard/details.tsx Normal file
View File

@@ -0,0 +1,358 @@
"use client";
import { useOrganization, useSession, useUser } from "@clerk/nextjs";
import classNames from "classnames";
import { useEffect, useState } from "react";
import { CopyIcon, Dot } from "../icons";
import Image from "next/image";
import "./prism.css";
declare global {
interface Window {
Prism: any;
}
}
export function UserDetails() {
const { isLoaded, user } = useUser();
const [jsonOutput, setJsonOutput] = useState(false);
return (
<div
className="bg-white overflow-hidden sm:rounded-lg"
style={{
boxShadow: `0px 20px 24px -4px rgba(16, 24, 40, 0.08)`,
}}
>
<div className="flex p-8">
<h3 className="text-xl leading-6 font-semibold text-gray-900 my-auto">
User
</h3>
<Toggle
checked={jsonOutput}
onChange={() => setJsonOutput(!jsonOutput)}
disabled={!isLoaded}
/>
</div>
{isLoaded && user ? (
jsonOutput ? (
<div className="overflow-scroll max-h-96 pb-6">
<JSONOutput json={user} />
</div>
) : (
<div className="pb-6 max-h-96">
<dl>
<div className="px-8 py-2">
<dt className="text-sm font-semibold">User ID</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2 flex gap-2">
{user.id}
<CopyButton text={user.id} />
</dd>
</div>
{user.firstName && (
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">First Name</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{user.firstName}
</dd>
</div>
)}
{user.lastName && (
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Last Name</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{user.lastName}
</dd>
</div>
)}
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Email addresses</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{user.emailAddresses.map((email) => (
<div key={email.id} className="flex gap-2 mb-1">
{email.emailAddress}
{user.primaryEmailAddressId === email.id && (
<span className="text-xs bg-primary-50 text-primary-700 rounded-2xl px-2 font-medium pt-[2px]">
Primary
</span>
)}
</div>
))}
</dd>
</div>
{user.imageUrl && (
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Profile Image</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
<img
src={user.imageUrl}
className="rounded-full w-12 h-12"
/>
</dd>
</div>
)}
</dl>
</div>
)
) : (
<div className="text-gray-700 px-4 py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
Loading user data...
</div>
)}
</div>
);
}
export function SessionDetails() {
const { isLoaded, session } = useSession();
const [jsonOutput, setJsonOutput] = useState(false);
return (
<div
className="bg-white shadow overflow-hidden sm:rounded-lg"
style={{
boxShadow: `0px 20px 24px -4px rgba(16, 24, 40, 0.08)`,
}}
>
<div className="flex p-8">
<h3 className="text-xl leading-6 font-semibold text-gray-900 my-auto">
Session
</h3>
<Toggle
checked={jsonOutput}
onChange={() => setJsonOutput(!jsonOutput)}
disabled={!isLoaded}
/>
</div>
{isLoaded && session ? (
jsonOutput ? (
<div className="overflow-scroll max-h-96 pb-6">
<JSONOutput
json={{
...session,
user: undefined,
}}
/>
</div>
) : (
<div className="pb-6 max-h-96">
<dl>
<div className="px-8 py-2">
<dt className="text-sm font-semibold">Session ID</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2 flex gap-2">
{session.id}
<CopyButton text={session.id} />
</dd>
</div>
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Status</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{session.status === `active` && (
<span className="text-xs bg-success-50 text-success-700 flex w-min gap-1 px-2 py-[1px] rounded-2xl font-medium">
<div className="m-auto">
<Dot />
</div>
Active
</span>
)}
</dd>
</div>
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Last Active</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{session.lastActiveAt.toLocaleString()}
</dd>
</div>
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Expiry</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{session.expireAt.toLocaleString()}
</dd>
</div>
</dl>
</div>
)
) : (
<div className="text-gray-700 px-4 py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
Loading user data...
</div>
)}
</div>
);
}
export function OrgDetails() {
const { isLoaded, organization } = useOrganization();
const [jsonOutput, setJsonOutput] = useState(false);
return (
<div
className="bg-white shadow overflow-hidden sm:rounded-lg"
style={{
boxShadow: `0px 20px 24px -4px rgba(16, 24, 40, 0.08)`,
}}
>
<div className="flex p-8">
<h3 className="text-xl leading-6 font-semibold text-gray-900 my-auto">
Organization
</h3>
<Toggle
checked={jsonOutput}
onChange={() => setJsonOutput(!jsonOutput)}
disabled={!(isLoaded && organization)}
/>
</div>
{isLoaded ? (
organization ? (
jsonOutput ? (
<div className="overflow-scroll max-h-96 pb-6">
<JSONOutput json={organization} />
</div>
) : (
<div className="pb-6 max-h-96">
<dl>
<div className="px-8 py-2">
<dt className="text-sm font-semibold">Organization ID</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2 flex gap-2">
{organization.id}
<CopyButton text={organization.id} />
</dd>
</div>
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Name</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{organization.name}
</dd>
</div>
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Members</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{organization.membersCount}
</dd>
</div>
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">
Pending invitations
</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
{organization.pendingInvitationsCount}
</dd>
</div>
<div className="px-8 py-2">
<dt className="text-sm font-semibold mb-1">Image</dt>
<dd className="mt-1 text-sm text-gray-600 sm:mt-0 sm:col-span-2">
<Image
className="rounded"
src={organization.imageUrl}
alt={`Logo for ${organization.name}`}
width={48}
height={48}
/>
</dd>
</div>
</dl>
</div>
)
) : (
<div className="text-gray-700 px-8 pb-5 text-sm">
You are currently logged in to your personal workspace.
<br />
Create or switch to an organization to see its details.
</div>
)
) : (
<div className="text-gray-700 px-4 py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
Loading organization data...
</div>
)}
</div>
);
}
function Toggle(props: {
checked: boolean;
onChange: () => void;
disabled: boolean;
}) {
return (
<div className="flex items-center justify-end flex-1">
<button
disabled={props.disabled}
onClick={props.onChange}
className={classNames({
"rounded-l-lg py-2 px-4 border-solid border border-gray-300 transition text-sm font-semibold":
true,
"bg-gray-100": !props.checked,
"bg-gray-50 text-gray-500 cursor-not-allowed": props.disabled,
})}
>
List
</button>
<button
disabled={props.disabled}
onClick={props.onChange}
className={classNames({
"rounded-r-lg py-2 px-4 border-solid border border-gray-300 -ml-[1px] transition text-sm font-semibold":
true,
"bg-gray-100": props.checked,
"bg-gray-50 text-gray-500 cursor-not-allowed": props.disabled,
})}
>
JSON
</button>
</div>
);
}
function CopyButton(props: { text: string }) {
const [tooltipShown, setTooltipShown] = useState(false);
useEffect(() => {
if (tooltipShown) {
const timeout = setTimeout(() => setTooltipShown(false), 2000);
return () => clearTimeout(timeout);
}
}, [tooltipShown]);
return (
<>
<button
onClick={() => {
if (navigator.clipboard) navigator.clipboard.writeText(props.text);
setTooltipShown(true);
}}
>
<CopyIcon />
</button>
<div
className={classNames({
"absolute z-10 bg-gray-900 text-white rounded p-2 text-xs transition-all ease-in-out translate-x-60 shadow-sm shadow-gray-500":
true,
"translate-y-10 opacity-0": !tooltipShown,
"translate-y-6": tooltipShown,
})}
>
Copied!
</div>
</>
);
}
function JSONOutput(props: { json: any }) {
useEffect(() => {
if (window.Prism) {
console.log(`highlighting`);
window.Prism.highlightAll();
}
}, []);
return (
<pre className="px-8 sm:px-6 text-black text-sm">
<code className="language-json">
{JSON.stringify(props.json, null, 2)}
</code>
</pre>
);
}

47
app/dashboard/page.tsx Normal file
View File

@@ -0,0 +1,47 @@
import { auth, clerkClient, useClerk } from "@clerk/nextjs";
import { redirect, useRouter } from "next/navigation";
import { OrgDetails, SessionDetails, UserDetails } from "./details";
import Link from "next/link";
export default async function DashboardPage() {
const { userId } = auth();
const router = useRouter();
const { signOut } = useClerk();
if (!userId) {
redirect("/");
}
const user = await clerkClient.users.getUser(userId);
return (
<div className="px-8 py-12 sm:py-16 md:px-20">
{user && (
<>
<h1 className="text-3xl font-semibold text-black">
👋 Hi, {user.firstName || `Stranger`}
</h1>
<button onClick={() => signOut(() => router.push("/"))}>
Sign out
</button>
<div className="grid gap-4 mt-8 lg:grid-cols-3">
<UserDetails />
<SessionDetails />
<OrgDetails />
</div>
<h2 className="mt-16 mb-4 text-3xl font-semibold text-black">
What's next?
</h2>
Read the{" "}
<Link
className="font-medium text-primary-600 hover:underline"
href="https://clerk.com/docs?utm_source=vercel-template&utm_medium=template_repos&utm_campaign=nextjs_template"
target="_blank"
>
Clerk Docs -&gt;
</Link>
</>
)}
</div>
);
}

144
app/dashboard/prism.css Normal file
View File

@@ -0,0 +1,144 @@
/**
* prism.js default theme for JavaScript, CSS and HTML
* Based on dabblet (http://dabblet.com)
* @author Lea Verou
*/
code[class*='language-'],
pre[class*='language-'] {
color: black;
background: none;
text-shadow: 0 1px white;
font-family: Consolas, Monaco, 'Andale Mono', 'Ubuntu Mono', monospace;
font-size: 1em;
text-align: left;
white-space: pre;
word-spacing: normal;
word-break: normal;
word-wrap: normal;
line-height: 1.5;
-moz-tab-size: 4;
-o-tab-size: 4;
tab-size: 4;
-webkit-hyphens: none;
-moz-hyphens: none;
-ms-hyphens: none;
hyphens: none;
}
pre[class*='language-']::-moz-selection,
pre[class*='language-'] ::-moz-selection,
code[class*='language-']::-moz-selection,
code[class*='language-'] ::-moz-selection {
text-shadow: none;
background: #b3d4fc;
}
pre[class*='language-']::selection,
pre[class*='language-'] ::selection,
code[class*='language-']::selection,
code[class*='language-'] ::selection {
text-shadow: none;
background: #b3d4fc;
}
@media print {
code[class*='language-'],
pre[class*='language-'] {
text-shadow: none;
}
}
/* Code blocks */
pre[class*='language-'] {
padding: 1em;
margin: 0.5em 0;
overflow: auto;
}
/* :not(pre) > code[class*='language-'],
pre[class*='language-'] {
background: #f5f2f0;
} */
/* Inline code */
:not(pre) > code[class*='language-'] {
padding: 0.1em;
border-radius: 0.3em;
white-space: normal;
}
.token.comment,
.token.prolog,
.token.doctype,
.token.cdata {
color: slategray;
}
.token.punctuation {
color: #999;
}
.token.namespace {
opacity: 0.7;
}
.token.property,
.token.tag,
.token.boolean,
.token.number,
.token.constant,
.token.symbol,
.token.deleted {
color: #905;
}
.token.selector,
.token.attr-name,
.token.string,
.token.char,
.token.builtin,
.token.inserted {
color: #690;
}
.token.operator,
.token.entity,
.token.url,
.language-css .token.string,
.style .token.string {
color: #9a6e3a;
/* This background color was intended by the author of this theme. */
background: hsla(0, 0%, 100%, 0.5);
}
.token.atrule,
.token.attr-value,
.token.keyword {
color: #07a;
}
.token.function,
.token.class-name {
color: #dd4a68;
}
.token.regex,
.token.important,
.token.variable {
color: #e90;
}
.token.important,
.token.bold {
font-weight: bold;
}
.token.italic {
font-style: italic;
}
.token.entity {
cursor: help;
}