mirror of
https://github.com/Drop-OSS/drop.git
synced 2025-11-10 04:22:09 +10:00
218 lines
7.6 KiB
Vue
218 lines
7.6 KiB
Vue
<template>
|
|
<div class="space-y-4">
|
|
<div class="mx-auto max-w-2xl lg:mx-0">
|
|
<h2
|
|
class="mt-2 text-xl font-semibold tracking-tight text-zinc-100 sm:text-3xl"
|
|
>
|
|
Library
|
|
</h2>
|
|
<p
|
|
class="mt-2 text-pretty text-sm font-medium text-gray-500 sm:text-md/8"
|
|
>
|
|
As you add folders to your library, Drop will detect it and prompt you
|
|
to import it. Each game needs to be imported before you can import a
|
|
version.
|
|
</p>
|
|
</div>
|
|
<div
|
|
v-if="libraryState.unimportedGames.length > 0"
|
|
class="rounded-md bg-blue-600/10 p-4"
|
|
>
|
|
<div class="flex">
|
|
<div class="flex-shrink-0">
|
|
<InformationCircleIcon
|
|
class="h-5 w-5 text-blue-400"
|
|
aria-hidden="true"
|
|
/>
|
|
</div>
|
|
<div class="ml-3 flex-1 md:flex md:justify-between">
|
|
<p class="text-sm text-blue-400">
|
|
Drop has detected you have new games to import.
|
|
</p>
|
|
<p class="mt-3 text-sm md:ml-6 md:mt-0">
|
|
<NuxtLink
|
|
href="/admin/library/import"
|
|
class="whitespace-nowrap font-medium text-blue-400 hover:text-blue-500"
|
|
>
|
|
Import
|
|
<span aria-hidden="true"> →</span>
|
|
</NuxtLink>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="mt-2 grid grid-cols-1">
|
|
<input
|
|
id="search"
|
|
v-model="searchQuery"
|
|
type="text"
|
|
name="search"
|
|
class="col-start-1 row-start-1 block w-full rounded-md bg-zinc-900 py-1.5 pl-10 pr-3 text-base text-zinc-100 outline outline-1 -outline-offset-1 outline-zinc-700 placeholder:text-gray-400 focus:outline focus:outline-2 focus:-outline-offset-2 focus:outline-blue-600 sm:pl-9 sm:text-sm/6"
|
|
placeholder="Search library..."
|
|
/>
|
|
<MagnifyingGlassIcon
|
|
class="pointer-events-none col-start-1 row-start-1 ml-3 size-5 self-center text-zinc-400 sm:size-4"
|
|
aria-hidden="true"
|
|
/>
|
|
</div>
|
|
<ul
|
|
role="list"
|
|
class="grid grid-cols-1 gap-6 sm:grid-cols-2 md:grid-cols-3 xl:grid-cols-4"
|
|
>
|
|
<li
|
|
v-for="game in filteredLibraryGames"
|
|
:key="game.id"
|
|
class="col-span-1 flex flex-col justify-center divide-y divide-zinc-700 rounded-lg bg-zinc-950/20 text-left shadow"
|
|
>
|
|
<div class="flex flex-1 flex-row p-4 gap-x-4">
|
|
<img
|
|
class="h-16 w-16 flex-shrink-0 rounded-md"
|
|
:src="useObject(game.mIconId)"
|
|
alt=""
|
|
/>
|
|
<div class="flex flex-col">
|
|
<h3 class="text-sm font-medium text-zinc-100 font-display">
|
|
{{ game.mName }}
|
|
<span
|
|
class="ml-2 inline-flex items-center rounded-full bg-blue-600/10 px-2 py-1 text-xs font-medium text-blue-600 ring-1 ring-inset ring-blue-600/20"
|
|
>{{ game.metadataSource }}</span
|
|
>
|
|
</h3>
|
|
<dl class="mt-1 flex flex-col justify-between">
|
|
<dt class="sr-only">Short Description</dt>
|
|
<dd class="text-sm text-zinc-400">
|
|
{{ game.mShortDescription }}
|
|
</dd>
|
|
<dt class="sr-only">Metadata provider</dt>
|
|
</dl>
|
|
<div class="inline-flex gap-x-2 items-center">
|
|
<NuxtLink
|
|
:href="`/admin/library/${game.id}`"
|
|
class="mt-2 w-fit rounded-md bg-blue-600 px-2.5 py-1.5 text-sm font-semibold text-white shadow-sm hover:bg-blue-500 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-blue-600"
|
|
>
|
|
Edit →
|
|
</NuxtLink>
|
|
<button
|
|
class="mt-2 w-fit rounded-md bg-red-600 px-2.5 py-1.5 text-sm font-semibold text-white shadow-sm hover:bg-red-500 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-red-600"
|
|
@click="() => deleteGame(game.id)"
|
|
>
|
|
Delete
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div v-if="game.hasNotifications" class="flex flex-col gap-y-2 p-2">
|
|
<div
|
|
v-if="game.notifications.toImport"
|
|
class="rounded-md bg-blue-600/10 p-4"
|
|
>
|
|
<div class="flex">
|
|
<div class="flex-shrink-0">
|
|
<InformationCircleIcon
|
|
class="h-5 w-5 text-blue-400"
|
|
aria-hidden="true"
|
|
/>
|
|
</div>
|
|
<div class="ml-3 flex-1 md:flex md:justify-between">
|
|
<p class="text-sm text-blue-400">
|
|
Drop has detected you have new verions of this game to import.
|
|
</p>
|
|
<p class="mt-3 text-sm md:ml-6 md:mt-0">
|
|
<NuxtLink
|
|
:href="`/admin/library/${game.id}/import`"
|
|
class="whitespace-nowrap font-medium text-blue-400 hover:text-blue-500"
|
|
>
|
|
Import
|
|
<span aria-hidden="true"> →</span>
|
|
</NuxtLink>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div
|
|
v-if="game.notifications.noVersions"
|
|
class="rounded-md bg-yellow-600/10 p-4"
|
|
>
|
|
<div class="flex">
|
|
<div class="flex-shrink-0">
|
|
<ExclamationTriangleIcon
|
|
class="h-5 w-5 text-yellow-600"
|
|
aria-hidden="true"
|
|
/>
|
|
</div>
|
|
<div class="ml-3">
|
|
<h3 class="text-sm font-medium text-yellow-600">
|
|
You have no versions of this game available.
|
|
</h3>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</li>
|
|
<p
|
|
v-if="filteredLibraryGames.length == 0 && libraryGames.length != 0"
|
|
class="text-zinc-600 text-sm font-display font-bold uppercase text-center col-span-4"
|
|
>
|
|
No results
|
|
</p>
|
|
<p
|
|
v-if="filteredLibraryGames.length == 0 && libraryGames.length == 0"
|
|
class="text-zinc-600 text-sm font-display font-bold uppercase text-center col-span-4"
|
|
>
|
|
No games imported
|
|
</p>
|
|
</ul>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ExclamationTriangleIcon } from "@heroicons/vue/16/solid";
|
|
import { InformationCircleIcon } from "@heroicons/vue/20/solid";
|
|
import { MagnifyingGlassIcon } from "@heroicons/vue/24/outline";
|
|
definePageMeta({
|
|
layout: "admin",
|
|
});
|
|
|
|
useHead({
|
|
title: "Libraries",
|
|
});
|
|
|
|
const searchQuery = ref("");
|
|
|
|
const libraryState = await $dropFetch("/api/v1/admin/library");
|
|
const libraryGames = ref(
|
|
libraryState.games.map((e) => {
|
|
const noVersions = e.status.noVersions;
|
|
const toImport = e.status.unimportedVersions.length > 0;
|
|
|
|
return {
|
|
...e.game,
|
|
notifications: {
|
|
noVersions,
|
|
toImport,
|
|
},
|
|
hasNotifications: noVersions || toImport,
|
|
};
|
|
}),
|
|
);
|
|
|
|
const filteredLibraryGames = computed(() =>
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore excessively deep ts
|
|
libraryGames.value.filter((e) => {
|
|
if (!searchQuery.value) return true;
|
|
const searchQueryLower = searchQuery.value.toLowerCase();
|
|
if (e.mName.toLowerCase().includes(searchQueryLower)) return true;
|
|
if (e.mShortDescription.toLowerCase().includes(searchQueryLower))
|
|
return true;
|
|
return false;
|
|
}),
|
|
);
|
|
|
|
async function deleteGame(id: string) {
|
|
await $dropFetch(`/api/v1/admin/game?id=${id}`, { method: "DELETE" });
|
|
const index = libraryGames.value.findIndex((e) => e.id === id);
|
|
libraryGames.value.splice(index, 1);
|
|
}
|
|
</script>
|