mirror of
https://github.com/LeNei/clerk-docker-error.git
synced 2026-02-13 22:56:25 +00:00
first commit
This commit is contained in:
358
app/dashboard/details.tsx
Normal file
358
app/dashboard/details.tsx
Normal 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>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user