mirror of
https://github.com/LeNei/clerk-docker-error.git
synced 2026-02-13 14:54:43 +00:00
359 lines
12 KiB
TypeScript
359 lines
12 KiB
TypeScript
"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>
|
|
);
|
|
}
|