aboutsummaryrefslogtreecommitdiff
path: root/src/lib/Tools
diff options
context:
space:
mode:
authorFuwn <[email protected]>2024-03-16 22:06:56 -0700
committerFuwn <[email protected]>2024-03-16 22:06:56 -0700
commite9cb29bf27ca4992740ccc5fd634cd3bd5ae5bf9 (patch)
tree9e0675a831fbab5c4a47d4b22e276bb9a2669ea3 /src/lib/Tools
parentfeat(schedule): fix current day episode (diff)
downloaddue.moe-e9cb29bf27ca4992740ccc5fd634cd3bd5ae5bf9.tar.xz
due.moe-e9cb29bf27ca4992740ccc5fd634cd3bd5ae5bf9.zip
feat(tools): uma musume birthdays
Diffstat (limited to 'src/lib/Tools')
-rw-r--r--src/lib/Tools/UmaMusumeBirthdays.svelte120
-rw-r--r--src/lib/Tools/tools.ts8
2 files changed, 128 insertions, 0 deletions
diff --git a/src/lib/Tools/UmaMusumeBirthdays.svelte b/src/lib/Tools/UmaMusumeBirthdays.svelte
new file mode 100644
index 00000000..d941cabe
--- /dev/null
+++ b/src/lib/Tools/UmaMusumeBirthdays.svelte
@@ -0,0 +1,120 @@
+<script lang="ts">
+ import { browser } from '$app/environment';
+ import { page } from '$app/stores';
+ import Error from '$lib/Error/RateLimited.svelte';
+ import { onMount } from 'svelte';
+ import { clearAllParameters, parseOrDefault } from '../Utility/parameters';
+ import Skeleton from '$lib/Loading/Skeleton.svelte';
+ import Message from '$lib/Loading/Message.svelte';
+ import tooltip from '$lib/Tooltip/tooltip';
+ import settings from '$stores/settings';
+
+ interface Birthday {
+ birth_day: number;
+ birth_month: number;
+ game_id: number;
+ id: number;
+ name_en: string;
+ name_jp: string;
+ preferred_url: string;
+ sns_icon: string;
+ }
+
+ const urlParameters = browser ? new URLSearchParams(window.location.search) : null;
+ let date = new Date();
+ let month = parseOrDefault(urlParameters, 'month', date.getMonth() + 1);
+ let day = parseOrDefault(urlParameters, 'day', date.getDate());
+ let umapyoi: Promise<Birthday[]>;
+
+ $: {
+ month = Math.min(month, 12);
+ month = Math.max(month, 1);
+ day = Math.min(day, new Date(new Date().getFullYear(), month, 0).getDate());
+ day = Math.max(day, 1);
+
+ if (browser)
+ umapyoi = fetch('https://umapyoi.net/api/v1/character/birthday').then((r) => r.json());
+
+ if (browser) {
+ $page.url.searchParams.set('month', month.toString());
+ $page.url.searchParams.set('day', day.toString());
+ clearAllParameters(['month', 'day']);
+ history.replaceState(null, '', `?${$page.url.searchParams.toString()}`);
+ }
+ }
+
+ onMount(() => clearAllParameters(['month', 'day']));
+</script>
+
+{#await umapyoi}
+ <Message message="Loading birthdays ..." />
+
+ <Skeleton grid={true} count={100} width="150px" height="170px" />
+{:then birthdays}
+ {@const todaysBirthdays = birthdays.filter(
+ (birthday) => birthday.birth_month === month && birthday.birth_day === day
+ )}
+
+ <p>
+ <select bind:value={month}>
+ {#each Array.from({ length: 12 }, (_, i) => i + 1) as month}
+ <option value={month}>
+ {new Date(0, month - 1).toLocaleString('default', { month: 'long' })}
+ </option>
+ {/each}
+ </select>
+
+ <select bind:value={day}>
+ {#each Array.from({ length: new Date(new Date().getFullYear(), month, 0).getDate() }, (_, i) => i + 1) as day}
+ <option value={day}>{day}</option>
+ {/each}
+ </select>
+ </p>
+
+ {#if todaysBirthdays.length === 0}
+ <Message message="No birthdays today." fullscreen={false} loader="ripple" />
+ {:else}
+ <div class="characters">
+ {#each todaysBirthdays as birthday}
+ {@const name = $settings.displayLanguage === 'en' ? birthday.name_en : birthday.name_jp}
+ {@const nameOther =
+ $settings.displayLanguage === 'ja' ? birthday.name_en : birthday.name_jp}
+
+ <div class="card card-small">
+ <a
+ href={`https://anilist.co/search/characters?search=${encodeURIComponent(
+ birthday.name_en
+ ).replace(/%20/g, '+')}`}
+ target="_blank"
+ title={nameOther}
+ use:tooltip
+ >
+ {name}
+ <img src={birthday.sns_icon} alt="Character" class="character-image" />
+ </a>
+ </div>
+ {/each}
+ </div>
+ {/if}
+{:catch}
+ <Error type="Character" card />
+{/await}
+
+<style lang="scss">
+ .characters {
+ display: grid;
+ grid-template-columns: repeat(auto-fill, minmax(9rem, 1fr));
+ gap: 1rem;
+ grid-row-gap: 1rem;
+ align-items: start;
+
+ img {
+ width: 100%;
+ height: auto;
+ object-fit: cover;
+ border-radius: 8px;
+ margin-top: 0.5rem;
+ box-shadow: 0 4px 30px var(--base01);
+ }
+ }
+</style>
diff --git a/src/lib/Tools/tools.ts b/src/lib/Tools/tools.ts
index 27a5f2b6..2b7d27f2 100644
--- a/src/lib/Tools/tools.ts
+++ b/src/lib/Tools/tools.ts
@@ -35,6 +35,14 @@ export const tools: {
"Find media with prequels you haven't seen yet for any given simulcast season",
id: 'sequel_spy'
},
+ uma_musume_birthdays: {
+ name: () => {
+ return 'Uma Musume: Pretty Derby Character Birthdays';
+ },
+ description: () =>
+ 'Find and display the birthdays of all Uma Musume characters for today, or any other day of the year',
+ id: 'uma_musume_birthdays'
+ },
discussions: {
name: () => 'Episode Discussion Collector',
description: () => 'Find and display all episode discussions created by a given user',