140 lines
5.7 KiB
Svelte
140 lines
5.7 KiB
Svelte
<script lang="ts">
|
|
import { downloadRisuHub, getRisuHub, hubURL } from "src/ts/characterCards";
|
|
import { DownloadIcon, FlagIcon, MenuIcon, SearchIcon, XIcon } from "lucide-svelte";
|
|
import { alertConfirm, alertInput, alertNormal } from "src/ts/alert";
|
|
|
|
let openedData:null|{
|
|
name:string
|
|
desc: string
|
|
download: number,
|
|
id: string,
|
|
img: string
|
|
} = null
|
|
|
|
let charas:{
|
|
name:string
|
|
desc: string
|
|
download: number,
|
|
id: string,
|
|
img: string
|
|
tags: string[]
|
|
}[] = []
|
|
|
|
let search = ''
|
|
let menuOpen = false
|
|
|
|
async function getHub(){
|
|
charas = await getRisuHub({
|
|
search: search
|
|
})
|
|
}
|
|
|
|
getHub()
|
|
|
|
|
|
</script>
|
|
<div class="w-full flex justify-center mt-4 mb-3">
|
|
<div class="flex w-2xl max-w-full items-center">
|
|
<input class="flex-grow text-xl pl-3 pr-3 rounded-lg bg-darkbg h-16 min-w-0" placeholder="Search" bind:value={search}>
|
|
<button class="bg-darkbg h-14 w-14 min-w-14 rounded-lg ml-2 flex justify-center items-center hover:ring transition-shadow" on:click={getHub}>
|
|
<SearchIcon />
|
|
</button>
|
|
<button class="bg-darkbg h-14 w-14 min-w-14 rounded-lg ml-2 flex justify-center items-center hover:ring transition-shadow" on:click={() => {
|
|
menuOpen = true
|
|
}}>
|
|
<MenuIcon />
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<div class="w-full flex gap-4 p-2 flex-wrap justify-center">
|
|
{#each charas as chara}
|
|
<button class="bg-darkbg rounded-lg p-4 flex flex-col hover:bg-selected transition-colors relative lg:w-96 w-full items-start" on:click={() => {
|
|
openedData = chara
|
|
}}>
|
|
<div class="flex gap-2 w-full">
|
|
<img class="w-20 min-w-20 h-20 sm:h-28 sm:w-28 rounded-md object-top object-cover" alt={chara.name} src={`${hubURL}/resource/` + chara.img}>
|
|
<div class="flex flex-col flex-grow min-w-0">
|
|
<span class="text-white text-lg min-w-0 max-w-full text-ellipsis whitespace-nowrap overflow-hidden text-start">{chara.name}</span>
|
|
<span class="text-gray-400 text-xs min-w-0 max-w-full text-ellipsis break-words max-h-8 whitespace-nowrap overflow-hidden text-start">{chara.desc}</span>
|
|
<div class="flex flex-wrap">
|
|
{#each chara.tags as tag, i}
|
|
{#if i < 4}
|
|
<div class="text-xs p-1 text-blue-400">{tag}</div>
|
|
{:else if i === 4}
|
|
<div class="text-xs p-1 text-blue-400">...</div>
|
|
{/if}
|
|
{/each}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</button>
|
|
{/each}
|
|
</div>
|
|
|
|
|
|
{#if openedData}
|
|
<!-- svelte-ignore a11y-click-events-have-key-events -->
|
|
<div class="top-0 left-0 z-50 fixed w-full h-full bg-black bg-opacity-50 flex justify-center items-center" on:click={() => {
|
|
openedData = null
|
|
}}>
|
|
<div class="p-6 max-w-full bg-darkbg rounded-md flex flex-col gap-4 w-2xl overflow-y-auto">
|
|
<div class="w-full flex gap-4">
|
|
<div class="flex flex-col">
|
|
<img class="h-36 w-36 rounded-md object-top object-cover" alt={openedData.name} src={`${hubURL}/resource/` + openedData.img}>
|
|
<h1 class="text-2xl font-bold max-w-full overflow-hidden whitespace-nowrap text-ellipsis mt-4">{openedData.name}</h1>
|
|
</div>
|
|
<span class="text-gray-400 break-words text-base">{openedData.desc}</span>
|
|
</div>
|
|
<div class="flex flex-row-reverse gap-2">
|
|
<button class="text-gray-400 hover:text-red-500" on:click|stopPropagation={async () => {
|
|
const conf = await alertConfirm('Report this character?')
|
|
if(conf){
|
|
const report = await alertInput('Write a report text that would be sent to the admin')
|
|
const da = await fetch(hubURL + '/hub/report', {
|
|
method: "POST",
|
|
body: JSON.stringify({
|
|
id: openedData.id,
|
|
report: report
|
|
})
|
|
})
|
|
alertNormal(await da.text())
|
|
}
|
|
}}>
|
|
<FlagIcon />
|
|
</button>
|
|
<button class="text-gray-400 hover:text-green-500" on:click={() => {
|
|
downloadRisuHub(openedData.id)
|
|
openedData = null
|
|
}}>
|
|
<DownloadIcon />
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/if}
|
|
|
|
{#if menuOpen}
|
|
<!-- svelte-ignore a11y-click-events-have-key-events -->
|
|
<div class="top-0 left-0 z-50 fixed w-full h-full bg-black bg-opacity-50 flex justify-center items-center" on:click={() => {
|
|
menuOpen = false
|
|
}}>
|
|
<div class="max-w-full bg-darkbg rounded-md flex flex-col gap-4 overflow-y-auto p-4">
|
|
<h1 class="font-bold text-2xl w-full">
|
|
<span>
|
|
Menu
|
|
</span>
|
|
<button class="float-right text-gray-400 hover:text-green-500" on:click={() => {menuOpen = false}}>
|
|
<XIcon />
|
|
</button>
|
|
</h1>
|
|
<div class=" mt-2 w-full border-t-2 border-t-bgcolor"></div>
|
|
<button class="w-full hover:bg-selected p-4" on:click|stopPropagation={async () => {
|
|
menuOpen = false
|
|
const id = await alertInput('Import ID')
|
|
downloadRisuHub(id)
|
|
|
|
}}>Import Character from ID</button>
|
|
</div>
|
|
</div>
|
|
{/if} |