aboutsummaryrefslogtreecommitdiff
path: root/src/lib/Tools/CharacterBirthdays.svelte
blob: 936db0c33328bf91b331388622979bb2e67b13b7 (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
121
122
123
124
<script lang="ts">
	import { browser } from '$app/environment';
	import { page } from '$app/stores';
	import { ACDBBirthdays, type ACDBBirthday } from '$lib/Birthday/ACDB';
	import { aniSearchBirthdays, type aniSearchBirthday } from '$lib/Birthday/aniSearch';
	import Error from '$lib/Error.svelte';
	import { clearAllParameters, parseOrDefault } from './tool';

	interface Birthday {
		name: string;
		image: string;
		origin?: 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 anisearchBirthdays: Promise<aniSearchBirthday[]>;
	let acdbBirthdays: Promise<ACDBBirthday[]>;

	$: {
		month = Math.min(month, 12);
		month = Math.max(month, 1);
		day = Math.min(day, new Date(0, month, 0).getDate());
		day = Math.max(day, 1);

		anisearchBirthdays = aniSearchBirthdays(month, day);
		acdbBirthdays = ACDBBirthdays(month, day);

		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()}`);
		}
	}

	const normalizeName = (name: string): string => name.toLowerCase().split(' ').sort().join(' ');

	const combineBirthdaySources = (
		acdb: ACDBBirthday[],
		aniSearch: aniSearchBirthday[]
	): Birthday[] => {
		const nameMap = new Map<string, Birthday>();

		for (const entry of aniSearch.map((entry) => ({
			...entry,
			normalized_name: normalizeName(entry.name)
		})))
			nameMap.set(entry.normalized_name, {
				name: entry.name,
				image: entry.image
			});

		for (const entry of acdb) {
			const normalized_name = normalizeName(entry.name);

			if (!nameMap.has(normalized_name))
				nameMap.set(normalized_name, {
					name: entry.name,
					image: entry.character_image,
					origin: entry.origin
				});
		}

		return Array.from(nameMap.values());
	};
</script>

{#await acdbBirthdays}
	<p>Loading set one ...</p>
{:then acdbBirthdays}
	{#await anisearchBirthdays}
		<p>Loading set two ...</p>
	{:then anisearch}
		{@const birthdays = combineBirthdaySources(acdbBirthdays, anisearch)}

		<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(0, month, 0).getDate() }, (_, i) => i + 1) as day}
					<option value={day}>{day}</option>
				{/each}
			</select>
		</p>

		<div id="characters">
			{#each birthdays as birthday}
				<div>
					<a
						href={`https://anilist.co/search/characters?search=${encodeURIComponent(
							birthday.name
						).replace(/%20/g, '+')}`}
						target="_blank"
					>
						{birthday.name}
						<img src={birthday.image} alt="Character (Large)" class="character-image" />
					</a>
				</div>
			{/each}
		</div>
	{:catch}
		<Error type="Character" />
	{/await}
{:catch}
	<Error type="Character" />
{/await}

<style>
	#characters {
		display: grid;
		grid-template-columns: repeat(auto-fill, minmax(9rem, 1fr));
		grid-gap: 0.25%;
		grid-row-gap: 1rem;
	}
</style>