Polish recently played list

This commit is contained in:
Cooper Ransom 2024-03-14 19:47:54 -04:00
parent 44ad56d654
commit 2a5fc9a851
1 changed files with 78 additions and 12 deletions

View File

@ -1,5 +1,8 @@
import classNames from "classnames";
import { ReactNode, useEffect, useState } from "react"; import { ReactNode, useEffect, useState } from "react";
import { useTranslation } from "react-i18next";
import { Icon, Icons } from "@/components/Icon";
import { ThinContainer } from "@/components/layout/ThinContainer"; import { ThinContainer } from "@/components/layout/ThinContainer";
import { Divider } from "@/components/utils/Divider"; import { Divider } from "@/components/utils/Divider";
import { Heading1, Heading2, Paragraph } from "@/components/utils/Text"; import { Heading1, Heading2, Paragraph } from "@/components/utils/Text";
@ -10,6 +13,27 @@ import { WorkerTestPart } from "@/pages/parts/admin/WorkerTestPart";
import { BackendTestPart } from "../parts/admin/BackendTestPart"; import { BackendTestPart } from "../parts/admin/BackendTestPart";
function Button(props: {
className: string;
onClick?: () => void;
children: React.ReactNode;
disabled?: boolean;
}) {
return (
<button
className={classNames(
"font-bold rounded h-10 w-40 scale-95 hover:scale-100 transition-all duration-200",
props.className,
)}
type="button"
onClick={props.onClick}
disabled={props.disabled}
>
{props.children}
</button>
);
}
function ConfigValue(props: { name: string; children?: ReactNode }) { function ConfigValue(props: { name: string; children?: ReactNode }) {
return ( return (
<> <>
@ -30,24 +54,28 @@ async function getRecentPlayedItems() {
/mw_media_watch_count{tmdb_full_id="([^"]+)",provider_id="([^"]+)",title="([^"]+)",success="([^"]+)"} (\d+)/g; /mw_media_watch_count{tmdb_full_id="([^"]+)",provider_id="([^"]+)",title="([^"]+)",success="([^"]+)"} (\d+)/g;
let match; let match;
const loop = true; const loop = true;
const items = []; const items: { [key: string]: any } = {};
while (loop) { while (loop) {
match = regex.exec(text); match = regex.exec(text);
if (match === null) break; if (match === null) break;
const [_, tmdbFullId, providerId, title, success, count] = match; const [_, tmdbFullId, providerId, title, success, count] = match;
items.push({ if (items[tmdbFullId]) {
tmdbFullId, items[tmdbFullId].count += parseInt(count, 10);
providerId, } else {
title, items[tmdbFullId] = {
success: success === "true", tmdbFullId,
count: parseInt(count, 10), providerId,
}); title,
success: success === "true",
count: parseInt(count, 10),
};
}
} }
if (items.length > 0) { if (Object.keys(items).length > 0) {
return items; return Object.values(items);
} }
throw new Error("RECENT_PLAYED_ITEMS not found"); throw new Error("RECENT_PLAYED_ITEMS not found");
} }
@ -55,11 +83,19 @@ async function getRecentPlayedItems() {
export function AdminPage() { export function AdminPage() {
const [recentPlayedItems, setRecentPlayedItems] = useState<any[]>([]); const [recentPlayedItems, setRecentPlayedItems] = useState<any[]>([]);
const [loading, setLoading] = useState(true); const [loading, setLoading] = useState(true);
const [currentPage, setCurrentPage] = useState(1);
const { t } = useTranslation();
const itemsPerPage = 10;
useEffect(() => { useEffect(() => {
getRecentPlayedItems() getRecentPlayedItems()
.then((items) => { .then((items) => {
setRecentPlayedItems(items); const uniqueItems = items.filter(
(item, index, self) =>
index === self.findIndex((t2) => t2.tmdbFullId === item.tmdbFullId),
);
setRecentPlayedItems(uniqueItems);
}) })
.catch((error) => { .catch((error) => {
console.error("Error fetching recent played items:", error); console.error("Error fetching recent played items:", error);
@ -69,6 +105,13 @@ export function AdminPage() {
}); });
}, []); }, []);
function getItemsForCurrentPage() {
const sortedItems = recentPlayedItems.sort((a, b) => b.count - a.count);
const startIndex = (currentPage - 1) * itemsPerPage;
const endIndex = startIndex + itemsPerPage;
return sortedItems.slice(startIndex, endIndex);
}
if (loading) { if (loading) {
return <p>Loading...</p>; return <p>Loading...</p>;
} }
@ -88,7 +131,7 @@ export function AdminPage() {
<p className="mb-8"> <p className="mb-8">
This data is fetched from the current backend deployment. This data is fetched from the current backend deployment.
</p> </p>
{recentPlayedItems.map((item) => { {getItemsForCurrentPage().map((item) => {
const successText = item.success ? "Yes" : "No"; // Convert bool to "Yes" or "No" const successText = item.success ? "Yes" : "No"; // Convert bool to "Yes" or "No"
return ( return (
<ConfigValue key={item.tmdbFullId} name={item.title}> <ConfigValue key={item.tmdbFullId} name={item.title}>
@ -96,6 +139,29 @@ export function AdminPage() {
</ConfigValue> </ConfigValue>
); );
})} })}
<div style={{ display: "flex", justifyContent: "space-between" }}>
<Button
className="py-px box-content bg-buttons-secondary hover:bg-buttons-secondaryHover bg-opacity-90 text-buttons-secondaryText justify-center items-center"
onClick={() => setCurrentPage(currentPage - 1)}
disabled={currentPage === 1}
>
Previous page
</Button>
<div>
Page {currentPage} of{" "}
{Math.ceil(recentPlayedItems.length / itemsPerPage)}
</div>
<Button
className="py-px box-content bg-buttons-secondary hover:bg-buttons-secondaryHover bg-opacity-90 text-buttons-secondaryText justify-center items-center"
onClick={() => setCurrentPage(currentPage + 1)}
disabled={
currentPage ===
Math.ceil(recentPlayedItems.length / itemsPerPage)
}
>
Next page
</Button>
</div>
</div> </div>
</ThinContainer> </ThinContainer>
</SubPageLayout> </SubPageLayout>