fp/services/next/app/components/stream-page.tsx

204 lines
8.1 KiB
TypeScript
Raw Normal View History

2024-07-15 16:07:04 +00:00
import { IStream } from "@futureporn/types";
2024-07-10 22:11:18 +00:00
import { IVod } from "@/app/lib/vods";
2024-01-20 16:16:14 +00:00
import Link from "next/link";
2024-06-13 02:54:44 +00:00
import Image from "next/legacy/image";
2024-01-20 16:16:14 +00:00
import { LocalizedDate } from "./localized-date";
import { FontAwesomeIcon, FontAwesomeIconProps } from "@fortawesome/react-fontawesome";
import { faTriangleExclamation, faCircleInfo, faThumbsUp, IconDefinition, faO, faX, faExternalLinkAlt } from "@fortawesome/free-solid-svg-icons";
import { faXTwitter } from "@fortawesome/free-brands-svg-icons";
2024-07-06 08:49:51 +00:00
import { notFound } from "next/navigation";
2025-01-11 03:10:04 +00:00
import ProtectedRoute from "./protected-route";
2024-01-20 16:16:14 +00:00
export interface IStreamProps {
stream: IStream;
}
type Status = 'missing' | 'issue' | 'good';
interface StyleDef {
heading: string;
icon: IconDefinition;
2025-01-11 03:10:04 +00:00
description: string;
prompt: string;
2024-01-20 16:16:14 +00:00
}
function capitalizeFirstLetter(string: string): string {
return string.charAt(0).toUpperCase() + string.slice(1);
}
function hasNote(vod: IVod) {
2024-12-16 20:39:23 +00:00
if (!!vod?.note) return true;
2024-01-20 16:16:14 +00:00
else return false;
}
function determineStatus(stream: IStream): Status {
2024-12-16 20:39:23 +00:00
if (stream.vods.length < 1) {
2024-01-20 16:16:14 +00:00
return 'missing'
} else {
2024-12-16 20:39:23 +00:00
if (stream.vods.some((vod: IVod) => !hasNote(vod))) {
2024-01-20 16:16:14 +00:00
return 'good';
} else {
return 'issue';
}
}
}
export default function StreamPage({ stream }: IStreamProps) {
2024-05-27 22:20:58 +00:00
console.log('StreamPage function has been invoked! stream as follows')
console.log(stream)
2024-07-06 08:49:51 +00:00
if (!stream) notFound()
2024-12-16 20:39:23 +00:00
const displayName = stream.vtuber.display_name;
const date = new Date(stream.date);
2025-01-11 03:10:04 +00:00
const selectedStatus = determineStatus(stream);
2024-01-20 16:16:14 +00:00
const styleMap: Record<Status, StyleDef> = {
'missing': {
heading: 'is-danger',
icon: faTriangleExclamation,
2025-01-11 03:10:04 +00:00
description: "We don't have a VOD for this stream.",
prompt: 'Know someone who does?'
2024-01-20 16:16:14 +00:00
},
'issue': {
heading: 'is-warning',
icon: faCircleInfo,
2025-01-11 03:10:04 +00:00
description: "We have a VOD for this stream, but it's not full quality.",
prompt: 'Have a better copy?'
2024-01-20 16:16:14 +00:00
},
'good': {
heading: 'is-success',
icon: faThumbsUp,
2025-01-11 03:10:04 +00:00
description: "We have a VOD for this stream, and we think it's the best quality possible.",
prompt: "Have one that's even better?"
2024-01-20 16:16:14 +00:00
}
};
2025-01-11 03:10:04 +00:00
const { heading, icon, description, prompt } = styleMap[selectedStatus] || {};
2024-01-20 16:16:14 +00:00
2024-03-29 07:28:02 +00:00
if (!stream) return <p>NotFound</p>
// <NotFound></NotFound>
2024-01-20 16:16:14 +00:00
// return <p>
// <pre>
// <code>
// {JSON.stringify(stream, null, 2)}
// </code>
// </pre>
// </p>
2024-05-27 22:20:58 +00:00
// const platformsList = [
// stream.attributes.isChaturbateStream ? 'Chaturbate' : null,
// stream.attributes.isFanslyStream ? 'Fansly' : null
// ].filter(Boolean).join(', ');
// platformsList = platformsArray.length > 0 ? platformsArray.join(', ') : 'None';
// const platformsList = [
// (stream.attributes.isChaturbateStream && 'CB'),
// (stream.attributes.isFanslyStream && 'Fansly')
// ].filter(Boolean).join(', ')
const platformsList = [
2024-12-16 20:39:23 +00:00
(stream.is_chaturbate_stream && 'CB'),
(stream.is_fansly_stream && 'Fansly')
2024-05-27 22:20:58 +00:00
].filter(Boolean).join(', ') || '!!!';
2024-01-20 16:16:14 +00:00
return (
<>
<div className="content">
<div className="section">
<h1 className="title"><LocalizedDate date={date} /> {displayName} Stream Archive</h1>
</div>
<div className="section columns is-multiline">
<div className="column is-half">
<div className="box">
<div className="columns is-multiline">
<div className="column is-full">
2024-05-27 22:20:58 +00:00
<table className="table">
<thead>
<tr>
<th className="is-family-sans-serif">Description</th>
<th className="is-family-sans-serif">Details</th>
</tr>
</thead>
<tbody>
<tr>
<td>Platform</td>
<td>{platformsList}</td>
</tr>
<tr>
<td>UTC Datetime</td>
<td><time dateTime={date.toISOString()}>{date.toISOString()}</time></td>
</tr>
<tr>
<td>Local Datetime</td>
<td>{date.toLocaleDateString()} {date.toLocaleTimeString()}</td>
</tr>
</tbody>
</table>
2024-01-20 16:16:14 +00:00
</div>
</div>
</div>
</div>
<div className="column is-half">
<article className={`message ${heading}`}>
<div className="message-header">
<span>VOD {capitalizeFirstLetter(selectedStatus)}</span>
</div>
<div className="message-body has-text-centered">
<span className="title is-1"><FontAwesomeIcon icon={icon}></FontAwesomeIcon></span>
2025-01-11 03:10:04 +00:00
<p className="mt-3">{description}</p>
<ProtectedRoute requiredUserRole="patron" accessDenied={<span></span>} >
<p className="mt-5">{prompt}<br />
<Link href={`/upload?uuid=${stream.uuid}`}>Upload it here.</Link></p>
</ProtectedRoute>
2024-01-20 16:16:14 +00:00
</div>
</article>
</div>
</div>
2024-12-16 20:39:23 +00:00
{stream.vods.length !== 0 &&
2024-01-20 16:16:14 +00:00
<div className="section">
<h1 className="title">VODs</h1>
<table className="table">
<thead>
<tr>
<th>ID</th>
<th>Upload Date</th>
{/* <th>Thumbnail</th>
<th>Duration</th> */}
<th>Tags</th>
<th>Timestamps</th>
<th>Note</th>
</tr>
</thead>
<tbody>
2024-12-16 20:39:23 +00:00
{stream.vods.map((vod: IVod) => (
2024-01-20 16:16:14 +00:00
<tr key={vod.id}>
{/* <p>{JSON.stringify(vod, null, 2)}</p> */}
2024-12-16 20:39:23 +00:00
<td><Link href={`/vt/${stream.vtuber.slug}/vod/${vod.uuid}`}>{vod.uuid}</Link></td>
<td>{vod.publishedAt}</td>
2025-01-11 03:10:04 +00:00
{/* <td>{(!!vod?.attributes?.thumbnail?.data?.attributes?.cdnUrl) ? <Image alt="" src={vod.thumbnail.cdn_url}></Image> : <FontAwesomeIcon icon={faX} />}</td>
<td>{(!!vod?.attributes?.duration) ? vod.duration : <FontAwesomeIcon icon={faX} />}</td> */}
2024-12-16 20:39:23 +00:00
<td>{vod.tagVodRelations.length}</td>
<td>{vod.timestamps.length}</td>
<td>{(!!vod.note) ? <FontAwesomeIcon icon={faO} /> : <FontAwesomeIcon icon={faX} />}</td>
2024-01-20 16:16:14 +00:00
</tr>
))}
</tbody>
</table>
2024-05-27 22:20:58 +00:00
</div>}
2024-01-20 16:16:14 +00:00
</div>
</>
)
}