aboutsummaryrefslogtreecommitdiff
path: root/src/lib/Tools/HololiveBirthdays.svelte
blob: 769f5d6f75a37a59767b99916da853abd7ed2175 (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
<script lang="ts">
  import { run } from 'svelte/legacy';

  import { browser } from '$app/environment';
  import { page } from '$app/stores';
  import { onMount } from 'svelte';
  import { clearAllParameters, parseOrDefault } from '../Utility/parameters';
  import Message from '$lib/Loading/Message.svelte';
  import locale from '$stores/locale';
  import birthdays from '$lib/Data/Static/hololiveBirthdays.json';

  const urlParameters = browser ? new URLSearchParams(window.location.search) : null;
  let date = new Date();
  let month = $state(parseOrDefault(urlParameters, 'month', date.getMonth() + 1));
  let day = $state(parseOrDefault(urlParameters, 'day', date.getDate()));

  let todaysBirthdays = $derived(birthdays.filter(
    (birthday) => birthday.month === month && birthday.day === day
  ));

  run(() => {
    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) {
      $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>

<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 for ${$locale().dayFormatter(
      new Date(new Date().getFullYear(), month - 1, day)
    )}.`}
    fullscreen={false}
    loader="ripple"
  />
{:else}
  <div class="characters">
    {#each todaysBirthdays as birthday}
      <div class="card card-small">
        <a
          href={`https://anilist.co/search/characters?search=${encodeURIComponent(
            birthday.name
          ).replace(/%20/g, '+')}`}
          target="_blank"
        >
          {birthday.name}
          <img src={birthday.image} alt="Character" class="character-image" />
        </a>
      </div>
    {/each}
  </div>
{/if}

<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>