aboutsummaryrefslogtreecommitdiff
path: root/src/lib/Tools/UmaMusumeBirthdays.svelte
blob: d941cabe01d876a950ff603a8562fc0e9d83d09f (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
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>