fp/services/next/app/profile/page.tsx

242 lines
10 KiB
TypeScript
Raw Normal View History

2024-12-12 07:23:46 +00:00
'use client';
2024-01-20 16:16:14 +00:00
2024-12-12 07:23:46 +00:00
import 'react-loading-skeleton/dist/skeleton.css';
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
import { faSpinner, faCheckCircle, faXmarkCircle } from "@fortawesome/free-solid-svg-icons";
import { useMutation, useQueryClient } from "@tanstack/react-query";
2024-12-16 20:39:23 +00:00
// import { updateIsUsernamePublic } from "./hooks/updateProfile";
import { useQuery } from '@tanstack/react-query';
2024-12-12 07:23:46 +00:00
import ProtectedRoute from "../components/protected-route";
import VibrateTest from "../components/vibrate-test";
import { useSession } from "next-auth/react"
import { LoginButton, LogoutButton } from "../components/auth-buttons";
import Skeleton, { SkeletonTheme } from 'react-loading-skeleton';
2024-12-16 20:39:23 +00:00
import Link from 'next/link';
import { useEffect } from 'react';
import { useState } from 'react';
import { fetchPatreonCurrentlyEntitledTiers } from './hooks/patreonCurrentlyEntitledTiers';
import { Spinner } from '../components/spinner';
import { signIn, signOut } from "next-auth/react";
2024-01-20 16:16:14 +00:00
export default function Page() {
2024-12-12 07:23:46 +00:00
const queryClient = useQueryClient()
// const sessionContext = Session.useSessionContext()
// if (sessionContext.loading) {
// return <Spinner></Spinner>
// }
2024-12-16 20:39:23 +00:00
const { data, status, update } = useSession()
2024-12-12 07:23:46 +00:00
2024-12-16 20:39:23 +00:00
async function renewJwt() {
await signOut({ redirect: false });
await signIn("keycloak", { redirect: false });
}
// const { data: patreonTier, isPending, refetch } = useQuery({
// queryKey: ['user/tier'],
// queryFn: () => fetchPatreonCurrentlyEntitledTiers(),
// })
// const keycloakGroupQuery = useQuery({
// queryKey: ['user/group'],
// queryFn: () => fetchKeycloakGroup()
// })
// const attributesMutation = useMutation({
// mutationFn: updateIsUsernamePublic,
// mutationKey: ['user/attributes'],
// onSuccess: () => queryClient.invalidateQueries({ queryKey: ['user/attributes'] })
// })
2024-12-12 07:23:46 +00:00
// const metadataQuery = useQuery({
// queryKey: ['user/metadata'],
// queryFn: () => fetchMetadata()
// })
// const [ isUsernamePublic, setIsUsernamePublic ] = useState<boolean>()
// const [ username, setUsername ] = useState('')
// useEffect(() => {
// if (metadataQuery.data?.metadata) {
// setUsername(formatPatronUsername(metadataQuery.data.metadata.first_name, metadataQuery.data.metadata.last_name))
// setIsUsernamePublic(metadataQuery.data.metadata.isUsernamePublic)
// }
// }, [metadataQuery.data])
2024-12-16 20:39:23 +00:00
const isUsernamePublic = (data?.profile?.username_visibility === 'public')
const [isGo, setIsGo] = useState<boolean>()
2024-12-12 07:23:46 +00:00
2024-12-16 20:39:23 +00:00
// useEffect(() => {
// const fetchData = async () => {
// const res = await fetch(`${process.env.KEYCLOAK_URL}/protocol/openid-connect/token`, {
// method: 'POST',
// headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
// body: new URLSearchParams({
// client_id: process.env.KEYCLOAK_CLIENT_ID,
// client_secret: process.env.KEYCLOAK_CLIENT_SECRET,
// grant_type: 'refresh_token',
// refresh_token: session.refreshToken,
// }),
// });
// const data = await res.json();
// if (res.ok) {
// // Update session with new token
// session.accessToken = data.access_token;
// session.refreshToken = data.refresh_token;
// res.status(200).json(session);
// } else {
// res.status(400).json({ error: 'Token refresh failed' });
// }
// }
// fetchData();
// }, [isGo]);
2024-12-12 07:23:46 +00:00
2024-01-20 16:16:14 +00:00
return (
2024-12-12 07:23:46 +00:00
<div className="content">
{/* {(metadataQuery.isPending) && <Spinner></Spinner>} */}
<div className="section">
<p className="title is-3">Debug section</p>
2024-12-16 20:39:23 +00:00
{/* {(isPending) ? <span><Spinner></Spinner> Syncronizing Patron entitlements...</span> : <span>Patreon entitlements synced.</span>} */}
2024-12-12 07:23:46 +00:00
{/* <p>metadataQuery as follows</p> */}
{/* <pre><code>{JSON.stringify(metadataQuery.data, null, 2)}</code></pre> */}
<p>session as follows</p>
2024-12-16 20:39:23 +00:00
<pre>{JSON.stringify(data, null, 2)}</pre>
{(status === 'unauthenticated') && <LoginButton></LoginButton>}
{/* <button className='button' onClick={() => refetch()}>Refetch</button> */}
<button className='button' onClick={() => update()}>Update</button>
<button className='button' onClick={() => renewJwt()}>RenewJWT</button>
2024-12-12 07:23:46 +00:00
</div>
2024-12-16 20:39:23 +00:00
{(data?.user) &&
2024-12-12 07:23:46 +00:00
<div className="section">
<h1 className="title is-1 mb-3">Profile</h1>
2024-01-20 16:16:14 +00:00
2024-12-12 07:23:46 +00:00
<LogoutButton></LogoutButton>
2024-01-20 16:16:14 +00:00
2024-12-12 07:23:46 +00:00
<div className="card mt-5">
<div className="card-content">
<div className="media-content">
{/* <p className="title is-4">{username}</p> */}
2024-12-16 20:39:23 +00:00
<p className="subtitle is-6">{data?.user?.name}</p>
2024-12-12 07:23:46 +00:00
</div>
</div>
</div>
2024-01-20 16:16:14 +00:00
2024-12-16 20:39:23 +00:00
{/* <h2>Patronage</h2>
<div className='box'>
<p>You belong to the {patreonTier} tier. Thank you!</p>
</div> */}
2024-12-12 07:23:46 +00:00
<h2 className="title is-2">Patron Perks</h2>
<div className="box">
<h3 className="title is-3">Test Perk (not real)</h3>
<ProtectedRoute featureName="Test Perk" requiredUserRole="default-roles-futureporn">
<p>Hey look at you, you're granted access to this perk!</p>
</ProtectedRoute>
</div>
2024-01-20 16:16:14 +00:00
2024-12-12 07:23:46 +00:00
<div className="box">
<h3 className="title is-3">Website Shoutout</h3>
<ProtectedRoute featureName="Website shoutout" requiredUserRole="patron">
2024-12-16 20:39:23 +00:00
{isUsernamePublic && <span className="mr-2">{data.user.name} will be displayed on the <Link href="/patrons">/patrons page</Link></span>}
{!isUsernamePublic && <span className="mr-2">Your username will be hidden from the <Link href="/patrons">/patrons page</Link> due to your privacy setting.</span>}
{/* Users can update privacy pref at https://keycloak.fp.sbtp.xyz/realms/futureporn/account/ */}
2024-12-12 07:23:46 +00:00
</ProtectedRoute>
</div>
2024-01-20 16:16:14 +00:00
2024-12-12 07:23:46 +00:00
<div className='box'>
<h3 className="title is-3">Premium Content Delivery Network (CDN1)</h3>
<ProtectedRoute featureName="CDN1" requiredUserRole="patron">
2025-01-11 03:10:04 +00:00
<p><FontAwesomeIcon icon={faCheckCircle} className="mr-2 is-success"></FontAwesomeIcon> You have access to Futureporn's Premium CDN for reliable video playback</p>
2024-12-12 07:23:46 +00:00
</ProtectedRoute>
</div>
2024-01-20 16:16:14 +00:00
2024-12-12 07:23:46 +00:00
<div className='box'>
<h3 className="title is-3">Economy Content Delivery Network (CDN2)</h3>
<ProtectedRoute featureName="CDN2" requiredUserRole="default-roles-futureporn">
<p><FontAwesomeIcon icon={faCheckCircle} className="mr-2"></FontAwesomeIcon> You have access to Futureporn's Economy CDN for basic video playback</p>
</ProtectedRoute>
</div>
2024-01-20 16:16:14 +00:00
2024-12-12 07:23:46 +00:00
<div className="box">
<h3 className="title is-3">Vibrate (test)</h3>
<VibrateTest></VibrateTest>
</div>
2024-01-20 16:16:14 +00:00
</div>
2024-12-12 07:23:46 +00:00
}
</div>
2024-01-20 16:16:14 +00:00
)
2024-12-16 20:39:23 +00:00
2024-12-12 07:23:46 +00:00
// if (metadataQuery.isPending) return <Spinner></Spinner>;
// return <pre><code>{JSON.stringify(metadataQuery.data, null, 2)}</code></pre>
// return <p>@todo</p>
// if (!isPending) return <Spinner></Spinner>;
// if (!data) return <p>there is no data</p>;
// if (!data?.metadata) return <p>there is no data.metadata</p>;
// const { first_name, last_name } = data.metadata
// const username = formatPatronUsername(first_name, last_name)
// const { metadata } = data || {}
// const { isUsernamePublic } = metadata
// return (
// <>
// <div className="content">
// <div className="section">
// <h1 className="title is-1 mb-3">Profile</h1>
// <div className="field is-horizontal">
// <div className="field-label">Username</div>
// <p className="field is-expanded">
// {isPending &&
// <SkeletonTheme baseColor="#000" highlightColor="#000">
// <Skeleton count={1} enableAnimation={false} width="10em" />
// </SkeletonTheme>
// }
// {!isPending && <p><i>{username}</i></p>}
// </p>
// </div>
// <div className="field is-horizontal">
// <div className="field-label">User ID</div>
// <div className="field is-expanded">
// <p><i>{sessionContext.userId}</i></p>
// </div>
// </div>
// <h2 className="title is-2">Patron Perks</h2>
// <h3 className="title is-3">Website Shoutout</h3>
// <PatronRoute featureName="Website shoutout" requiredUserRole="patron">
// <div className="box">
// <span className="mr-2">Display {(!!username) ? username : <SkeletonTheme baseColor="#000" highlightColor="#000"><Skeleton count={1} enableAnimation={false} width="10em" /></SkeletonTheme>} publicly?</span>
2024-12-16 20:39:23 +00:00
// <input className="mr-2" type="checkbox" checked={isUsernamePublic} onChange={(evt) => attributesMutation.mutate({ isUsernamePublic: evt.target.checked })}/>
// {(attributesMutation.status === 'success') && <FontAwesomeIcon color="lime" icon={faCheckCircle}></FontAwesomeIcon>}
// {(attributesMutation.status === 'pending') && <FontAwesomeIcon icon={faSpinner}></FontAwesomeIcon>}
// {(attributesMutation.status === 'error') && <FontAwesomeIcon color="red" icon={faXmarkCircle}></FontAwesomeIcon>}
2024-12-12 07:23:46 +00:00
// </div>
// </PatronRoute>
// <h3 className="title is-3">Content Delivery Network (CDN)</h3>
// <PatronRoute featureName="CDN" requiredUserRole="patron">
// <p><FontAwesomeIcon icon={faCheckCircle} className="mr-2"></FontAwesomeIcon> You have access to Futureporn's CDN for reliable video playback</p>
// </PatronRoute>
// <h3 className="title is-3">Vibrate (test)</h3>
// <VibrateTest></VibrateTest>
// </div>
// </div>
// </>
// )
2024-01-20 16:16:14 +00:00
}