Compare commits
22 Commits
06e5fe5593
...
feature/ux
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
2226a483c5 | ||
|
|
93636b6968 | ||
|
|
5718bca963 | ||
|
|
6f41f6e53e | ||
|
|
d157055ab7 | ||
|
|
76d7e815c3 | ||
|
|
c7cf053105 | ||
|
|
a7079c1f18 | ||
|
|
cf9717149f | ||
|
|
ec4eea0977 | ||
|
|
92fae28383 | ||
|
|
b3c5fbe3dd | ||
|
|
8e9b40cc69 | ||
|
|
d389b496b4 | ||
|
|
bf2700efb7 | ||
| 0a823948df | |||
|
|
d2fb40f692 | ||
| 36f0c25721 | |||
| d09946161f | |||
| 87993ae9c6 | |||
| 65248fd082 | |||
| e9662754c4 |
BIN
public/airplane.png
Normal file
|
After Width: | Height: | Size: 1.1 KiB |
|
Before Width: | Height: | Size: 8.1 KiB After Width: | Height: | Size: 8.1 KiB |
|
Before Width: | Height: | Size: 36 KiB |
@@ -6,13 +6,14 @@
|
|||||||
import WorldMap from './lib/world-map/WorldMap.svelte';
|
import WorldMap from './lib/world-map/WorldMap.svelte';
|
||||||
import JourneyView from './lib/world-map/JourneyView.svelte';
|
import JourneyView from './lib/world-map/JourneyView.svelte';
|
||||||
import StatsPanel from './lib/world-map/StatsPanel.svelte';
|
import StatsPanel from './lib/world-map/StatsPanel.svelte';
|
||||||
import TimelineView from './lib/timeline/TimelineView.svelte';
|
import TimelineView from './lib/timeline/view/TimelineView.svelte';
|
||||||
|
|
||||||
let screen = $state('worldmap');
|
let screen = $state('worldmap');
|
||||||
let journeyActive = $state(false);
|
let journeyActive = $state(false);
|
||||||
let journeyProgress = $state(null);
|
let journeyProgress = $state(null);
|
||||||
let inDetail = $state(false);
|
let inDetail = $state(false);
|
||||||
let pendingCountry = $state('');
|
let pendingCountry = $state('');
|
||||||
|
let journeyMode = $state('map');
|
||||||
|
|
||||||
function onNavigate(s) {
|
function onNavigate(s) {
|
||||||
screen = s;
|
screen = s;
|
||||||
@@ -56,19 +57,20 @@
|
|||||||
<div class="worldmap-page">
|
<div class="worldmap-page">
|
||||||
<div class="map-area">
|
<div class="map-area">
|
||||||
{#if journeyActive}
|
{#if journeyActive}
|
||||||
<JourneyView onclose={endJourney} onprogress={onJourneyProgress} />
|
<JourneyView onclose={endJourney} onprogress={onJourneyProgress} mode={journeyMode} onmodechange={(m) => journeyMode = m} />
|
||||||
{:else}
|
{:else}
|
||||||
<WorldMap onCountryClick={handleCountryClick} />
|
<WorldMap onCountryClick={handleCountryClick} />
|
||||||
<button class="journey-play-btn" onclick={startJourney}>▶</button>
|
<button class="journey-play-btn" onclick={startJourney}>▶ Replay My Trips</button>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
<StatsPanel />
|
{#if !journeyActive}<StatsPanel />{/if}
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
<TimelineView
|
<TimelineView
|
||||||
onDetailChange={(v) => (inDetail = v)}
|
onDetailChange={(v) => (inDetail = v)}
|
||||||
{pendingCountry}
|
{pendingCountry}
|
||||||
onNewEntryClear={() => (pendingCountry = '')}
|
onNewEntryClear={() => (pendingCountry = '')}
|
||||||
|
onGoToMap={() => { screen = 'worldmap'; }}
|
||||||
/>
|
/>
|
||||||
{/if}
|
{/if}
|
||||||
</Layout>
|
</Layout>
|
||||||
@@ -115,14 +117,14 @@
|
|||||||
bottom: 24px;
|
bottom: 24px;
|
||||||
right: 24px;
|
right: 24px;
|
||||||
z-index: 10;
|
z-index: 10;
|
||||||
width: 44px;
|
padding: 12px 28px;
|
||||||
height: 44px;
|
border-radius: 24px;
|
||||||
border-radius: 50%;
|
|
||||||
border: none;
|
border: none;
|
||||||
background: #8b5cf6;
|
background: #8b5cf6;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
font-size: 20px;
|
font-size: 15px;
|
||||||
line-height: 1;
|
font-weight: 600;
|
||||||
|
gap: 6px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -139,4 +141,5 @@
|
|||||||
.journey-play-btn:active {
|
.journey-play-btn:active {
|
||||||
transform: scale(0.92);
|
transform: scale(0.92);
|
||||||
}
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
BIN
src/assets/home.png
Normal file
|
After Width: | Height: | Size: 436 KiB |
BIN
src/assets/logo-1-cursor.png
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
src/assets/logo-1.png
Normal file
|
After Width: | Height: | Size: 102 KiB |
BIN
src/assets/logo-2.png
Normal file
|
After Width: | Height: | Size: 290 KiB |
BIN
src/assets/logo-cursor.png
Normal file
|
After Width: | Height: | Size: 2.1 KiB |
BIN
src/assets/logo.png
Normal file
|
After Width: | Height: | Size: 112 KiB |
@@ -1,93 +1,72 @@
|
|||||||
<script>
|
<script>
|
||||||
import { getUser, getUserProfile, setHomeCountry } from './userStore.svelte.js';
|
import { getUser, getUserProfile, setHomeCountry } from './userStore.svelte.js';
|
||||||
import worldData from 'world-atlas/countries-50m.json';
|
import { countryNames } from '../shared/countries.js';
|
||||||
|
import homeImg from '../../assets/home.png';
|
||||||
|
|
||||||
let user = $derived(getUser());
|
let user = $derived(getUser());
|
||||||
let profile = $derived(getUserProfile());
|
let profile = $derived(getUserProfile());
|
||||||
|
|
||||||
const countries = $derived.by(() => {
|
|
||||||
if (!worldData?.objects?.countries?.geometries) return [];
|
|
||||||
return worldData.objects.countries.geometries
|
|
||||||
.map(g => ({ name: g.properties?.name, code: g.id }))
|
|
||||||
.filter(c => c.name && c.code)
|
|
||||||
.sort((a, b) => a.name.localeCompare(b.name));
|
|
||||||
});
|
|
||||||
|
|
||||||
let search = $state('');
|
let search = $state('');
|
||||||
let selectedCountry = $state(null);
|
let selectedCountry = $state('');
|
||||||
|
|
||||||
let filtered = $derived(
|
|
||||||
search
|
|
||||||
? countries.filter(c => c.name.toLowerCase().includes(search.toLowerCase()))
|
|
||||||
: countries
|
|
||||||
);
|
|
||||||
|
|
||||||
let open = $state(false);
|
let open = $state(false);
|
||||||
|
|
||||||
function select(c) {
|
let filtered = $derived(
|
||||||
selectedCountry = c;
|
search.trim()
|
||||||
search = c.name;
|
? countryNames.filter(c => c.toLowerCase().includes(search.toLowerCase()))
|
||||||
|
: countryNames
|
||||||
|
);
|
||||||
|
|
||||||
|
function select(name) {
|
||||||
|
selectedCountry = name;
|
||||||
|
search = name;
|
||||||
open = false;
|
open = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleSubmit() {
|
function handleSubmit() {
|
||||||
if (selectedCountry) {
|
if (selectedCountry) {
|
||||||
setHomeCountry(selectedCountry.name, selectedCountry.code);
|
setHomeCountry(selectedCountry, selectedCountry);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleKeydown(e) {
|
function handleKeydown(e) {
|
||||||
if (e.key === 'Enter' && selectedCountry) {
|
if (e.key === 'Enter' && selectedCountry) handleSubmit();
|
||||||
handleSubmit();
|
if (e.key === 'Escape') open = false;
|
||||||
}
|
|
||||||
if (e.key === 'Escape') {
|
|
||||||
open = false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="overlay">
|
<div class="overlay">
|
||||||
<div class="card">
|
<div class="card">
|
||||||
<h1 class="heading">Welcome, {profile?.displayName || 'Traveler'}!</h1>
|
<img src={homeImg} alt="home" class="home-img" />
|
||||||
<p class="subtitle">Select your home country to get started</p>
|
<h1 class="title">Welcome, {profile?.displayName?.split(' ')[0] || 'Traveler'}!</h1>
|
||||||
|
<p class="subtitle">Where do you call home?</p>
|
||||||
|
|
||||||
<div class="dropdown" class:open>
|
<div class="dropdown">
|
||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
placeholder="Search for a country..."
|
placeholder="Search country..."
|
||||||
bind:value={search}
|
bind:value={search}
|
||||||
onfocus={() => { open = true; }}
|
onfocus={() => { open = true; }}
|
||||||
oninput={() => { open = true; selectedCountry = null; }}
|
oninput={() => { open = true; selectedCountry = ''; }}
|
||||||
onkeydown={handleKeydown}
|
onkeydown={handleKeydown}
|
||||||
class="search-input"
|
class="search-input"
|
||||||
/>
|
/>
|
||||||
{#if open}
|
{#if open && filtered.length > 0}
|
||||||
<ul class="list" role="listbox">
|
<ul class="list" role="listbox">
|
||||||
{#each filtered as country}
|
{#each filtered as name}
|
||||||
<li
|
<li
|
||||||
role="option"
|
role="option"
|
||||||
aria-selected={selectedCountry?.name === country.name}
|
aria-selected={selectedCountry === name}
|
||||||
class:selected={selectedCountry?.name === country.name}
|
class:selected={selectedCountry === name}
|
||||||
onclick={() => select(country)}
|
onmousedown={() => select(name)}
|
||||||
onkeydown={(e) => { if (e.key === 'Enter') select(country); }}
|
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
>
|
>{name}</li>
|
||||||
{country.name}
|
|
||||||
</li>
|
|
||||||
{/each}
|
{/each}
|
||||||
{#if filtered.length === 0}
|
|
||||||
<li class="no-results">No countries found</li>
|
|
||||||
{/if}
|
|
||||||
</ul>
|
</ul>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button
|
<button class="continue-btn" disabled={!selectedCountry} onclick={handleSubmit}>
|
||||||
class="continue-btn"
|
Set home country
|
||||||
disabled={!selectedCountry}
|
|
||||||
onclick={handleSubmit}
|
|
||||||
>
|
|
||||||
Continue
|
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -96,113 +75,116 @@
|
|||||||
.overlay {
|
.overlay {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
inset: 0;
|
inset: 0;
|
||||||
background: rgba(15, 23, 42, 0.85);
|
background: var(--bg);
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
z-index: 100;
|
z-index: 100;
|
||||||
backdrop-filter: blur(4px);
|
padding-bottom: 20vh;
|
||||||
}
|
}
|
||||||
|
|
||||||
.card {
|
.card {
|
||||||
background: #1e2937;
|
|
||||||
border-radius: 16px;
|
|
||||||
padding: 40px 36px;
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
box-shadow: 0 20px 60px rgba(0, 0, 0, 0.5);
|
max-width: 360px;
|
||||||
max-width: 420px;
|
|
||||||
width: 90%;
|
width: 90%;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.heading {
|
.home-img {
|
||||||
font: 700 24px/1.3 sans-serif;
|
width: 200px;
|
||||||
color: #f1f5f9;
|
height: 200px;
|
||||||
margin-bottom: 6px;
|
object-fit: contain;
|
||||||
|
margin-bottom: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-family: var(--heading);
|
||||||
|
font-size: 24px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-h);
|
||||||
|
letter-spacing: -0.5px;
|
||||||
|
margin: 0 0 6px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.subtitle {
|
.subtitle {
|
||||||
font: 400 15px/1.4 sans-serif;
|
font-family: var(--sans);
|
||||||
color: #94a3b8;
|
font-size: 14px;
|
||||||
margin-bottom: 28px;
|
font-weight: 300;
|
||||||
|
color: var(--text);
|
||||||
|
margin: 0 0 24px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.dropdown {
|
.dropdown {
|
||||||
position: relative;
|
position: relative;
|
||||||
margin-bottom: 24px;
|
width: 100%;
|
||||||
|
margin-bottom: 16px;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-input {
|
.search-input {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 12px 16px;
|
padding: 10px 14px;
|
||||||
border: 1px solid #475569;
|
border: 1px solid var(--border);
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
background: #0f172a;
|
background: var(--bg-subtle);
|
||||||
color: #f1f5f9;
|
color: var(--text-h);
|
||||||
font: 400 15px/1.4 sans-serif;
|
font-family: var(--sans);
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 300;
|
||||||
outline: none;
|
outline: none;
|
||||||
transition: border-color 0.2s;
|
transition: border-color 0.15s;
|
||||||
}
|
box-sizing: border-box;
|
||||||
|
|
||||||
.search-input:focus {
|
|
||||||
border-color: #3b82f6;
|
|
||||||
}
|
|
||||||
|
|
||||||
.search-input::placeholder {
|
|
||||||
color: #64748b;
|
|
||||||
}
|
}
|
||||||
|
.search-input:focus { border-color: var(--accent-border); }
|
||||||
|
.search-input::placeholder { color: var(--text-sub); }
|
||||||
|
|
||||||
.list {
|
.list {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: calc(100% + 4px);
|
top: calc(100% + 4px);
|
||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
max-height: 240px;
|
max-height: 220px;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
background: #0f172a;
|
background: var(--bg);
|
||||||
border: 1px solid #475569;
|
border: 1px solid var(--border);
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
list-style: none;
|
list-style: none;
|
||||||
z-index: 10;
|
z-index: 10;
|
||||||
|
padding: 4px;
|
||||||
|
box-shadow: 0 4px 16px rgba(0,0,0,0.1);
|
||||||
}
|
}
|
||||||
|
|
||||||
.list li {
|
.list li {
|
||||||
padding: 10px 16px;
|
padding: 8px 12px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
color: #cbd5e1;
|
color: var(--text);
|
||||||
font: 400 14px/1.4 sans-serif;
|
font-family: var(--sans);
|
||||||
transition: background 0.15s;
|
font-size: 13px;
|
||||||
|
font-weight: 300;
|
||||||
|
border-radius: 6px;
|
||||||
|
transition: background 0.1s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.list li:hover,
|
.list li:hover, .list li.selected {
|
||||||
.list li.selected {
|
background: var(--accent-bg);
|
||||||
background: #1e3a5f;
|
color: var(--accent);
|
||||||
color: #f1f5f9;
|
|
||||||
}
|
|
||||||
|
|
||||||
.no-results {
|
|
||||||
color: #64748b;
|
|
||||||
cursor: default;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.continue-btn {
|
.continue-btn {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 12px 24px;
|
padding: 11px 24px;
|
||||||
border: none;
|
border: 1px solid var(--border);
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
background: #3b82f6;
|
background: var(--accent);
|
||||||
color: #fff;
|
color: #fff;
|
||||||
font: 600 16px/1.4 sans-serif;
|
font-family: var(--sans);
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: background 0.2s, opacity 0.2s;
|
transition: background 0.15s, opacity 0.15s;
|
||||||
}
|
|
||||||
|
|
||||||
.continue-btn:hover:not(:disabled) {
|
|
||||||
background: #2563eb;
|
|
||||||
}
|
|
||||||
|
|
||||||
.continue-btn:disabled {
|
|
||||||
opacity: 0.4;
|
|
||||||
cursor: default;
|
|
||||||
}
|
}
|
||||||
|
.continue-btn:hover:not(:disabled) { background: var(--accent-dark); }
|
||||||
|
.continue-btn:disabled { opacity: 0.4; cursor: default; }
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,12 +1,13 @@
|
|||||||
<script>
|
<script>
|
||||||
import { signInWithGoogle } from './userStore.svelte.js';
|
import { signInWithGoogle } from './userStore.svelte.js';
|
||||||
|
import logoImg from '../../assets/logo.png';
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="overlay">
|
<div class="overlay">
|
||||||
<div class="card">
|
<div class="card">
|
||||||
<img src="/logo.png" alt="Map Journal" class="logo" />
|
<img src={logoImg} alt="Journi" class="logo" />
|
||||||
<h1 class="title">Map Journal</h1>
|
<h1 class="title">Journi</h1>
|
||||||
<p class="subtitle">Sign in to start your journey</p>
|
<p class="subtitle">Collect Colors Along the Way</p>
|
||||||
<button class="google-btn" onclick={signInWithGoogle}>
|
<button class="google-btn" onclick={signInWithGoogle}>
|
||||||
<svg class="google-icon" viewBox="0 0 48 48">
|
<svg class="google-icon" viewBox="0 0 48 48">
|
||||||
<path fill="#EA4335" d="M24 9.5c3.54 0 6.71 1.22 9.21 3.6l6.85-6.85C35.9 2.38 30.47 0 24 0 14.62 0 6.51 5.38 2.56 13.22l7.98 6.19C12.43 13.72 17.74 9.5 24 9.5z"/>
|
<path fill="#EA4335" d="M24 9.5c3.54 0 6.71 1.22 9.21 3.6l6.85-6.85C35.9 2.38 30.47 0 24 0 14.62 0 6.51 5.38 2.56 13.22l7.98 6.19C12.43 13.72 17.74 9.5 24 9.5z"/>
|
||||||
@@ -24,64 +25,90 @@
|
|||||||
.overlay {
|
.overlay {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
inset: 0;
|
inset: 0;
|
||||||
background: rgba(15, 23, 42, 0.85);
|
background: var(--bg);
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
padding-bottom: 20vh;
|
||||||
z-index: 100;
|
z-index: 100;
|
||||||
backdrop-filter: blur(4px);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.card {
|
.card {
|
||||||
background: #1e2937;
|
|
||||||
border-radius: 16px;
|
|
||||||
padding: 48px 40px;
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
box-shadow: 0 20px 60px rgba(0, 0, 0, 0.5);
|
max-width: 360px;
|
||||||
max-width: 400px;
|
|
||||||
width: 90%;
|
width: 90%;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.logo {
|
.logo {
|
||||||
width: 80px;
|
width: 216px;
|
||||||
height: 80px;
|
height: 216px;
|
||||||
border-radius: 12px;
|
object-fit: contain;
|
||||||
margin-bottom: 16px;
|
margin-bottom: 16px;
|
||||||
|
animation: jitter 1.4s steps(1, end) 1 forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes jitter {
|
||||||
|
0% { transform: scale(0.7) rotate(0deg); opacity: 0.5; }
|
||||||
|
8% { transform: scale(0.85) rotate(-16deg); opacity: 1; }
|
||||||
|
16% { transform: scale(1.0) rotate(16deg); }
|
||||||
|
24% { transform: scale(1.06) rotate(-16deg); }
|
||||||
|
32% { transform: scale(1.12) rotate(16deg); }
|
||||||
|
40% { transform: scale(1.16) rotate(-16deg); }
|
||||||
|
48% { transform: scale(1.2) rotate(16deg); }
|
||||||
|
56% { transform: scale(1.2) rotate(-16deg); }
|
||||||
|
64% { transform: scale(1.2) rotate(16deg); }
|
||||||
|
72% { transform: scale(1.2) rotate(-10deg); }
|
||||||
|
80% { transform: scale(1.2) rotate(10deg); }
|
||||||
|
88% { transform: scale(1.2) rotate(-4deg); }
|
||||||
|
94% { transform: scale(1.2) rotate(4deg); }
|
||||||
|
100% { transform: scale(1.2) rotate(0deg); }
|
||||||
}
|
}
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
font: 700 28px/1.2 sans-serif;
|
font-family: var(--heading);
|
||||||
color: #f1f5f9;
|
font-size: 28px;
|
||||||
margin-bottom: 8px;
|
font-weight: 600;
|
||||||
|
color: var(--text-h);
|
||||||
|
letter-spacing: -0.5px;
|
||||||
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.subtitle {
|
.subtitle {
|
||||||
font: 400 15px/1.4 sans-serif;
|
font-family: var(--sans);
|
||||||
color: #94a3b8;
|
font-size: 14px;
|
||||||
margin-bottom: 32px;
|
font-weight: 300;
|
||||||
|
color: var(--text);
|
||||||
|
margin: 0 0 32px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.google-btn {
|
.google-btn {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 12px;
|
gap: 10px;
|
||||||
padding: 12px 28px;
|
padding: 10px 24px;
|
||||||
border: 1px solid rgba(255,255,255,0.15);
|
border: 1px solid var(--border);
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
background: #334155;
|
background: var(--bg-subtle);
|
||||||
color: #f1f5f9;
|
color: var(--text-h);
|
||||||
font: 500 16px/1.4 sans-serif;
|
font-family: var(--sans);
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 400;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: background 0.2s;
|
transition: background 0.15s, border-color 0.15s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.google-btn:hover {
|
.google-btn:hover {
|
||||||
background: #475569;
|
background: var(--bg);
|
||||||
|
border-color: var(--accent-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
.google-icon {
|
.google-icon {
|
||||||
width: 22px;
|
width: 20px;
|
||||||
height: 22px;
|
height: 20px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import { auth, db, googleProvider } from '../firebase.js';
|
import { auth, db, googleProvider } from '../firebase.js';
|
||||||
import { onAuthStateChanged, signInWithPopup, signOut as fbSignOut } from 'firebase/auth';
|
import { onAuthStateChanged, signInWithPopup, signOut as fbSignOut } from 'firebase/auth';
|
||||||
import { doc, getDoc, setDoc, serverTimestamp } from 'firebase/firestore';
|
import { doc, getDoc, setDoc, serverTimestamp } from 'firebase/firestore';
|
||||||
import { initSelectionListener } from '../layout/selection.svelte.js';
|
|
||||||
import { initEntriesListener } from '../stores/entriesStore.svelte.js';
|
import { initEntriesListener } from '../stores/entriesStore.svelte.js';
|
||||||
|
|
||||||
let _initialized = false;
|
let _initialized = false;
|
||||||
@@ -48,7 +47,6 @@ export function initAuth() {
|
|||||||
onAuthStateChanged(auth, async (fbUser) => {
|
onAuthStateChanged(auth, async (fbUser) => {
|
||||||
if (fbUser) {
|
if (fbUser) {
|
||||||
user = fbUser;
|
user = fbUser;
|
||||||
initSelectionListener(fbUser.uid);
|
|
||||||
initEntriesListener(fbUser.uid);
|
initEntriesListener(fbUser.uid);
|
||||||
const docRef = doc(db, 'users', fbUser.uid);
|
const docRef = doc(db, 'users', fbUser.uid);
|
||||||
const docSnap = await getDoc(docRef);
|
const docSnap = await getDoc(docRef);
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import { getUser, getUserProfile, signOut } from '../auth/userStore.svelte.js';
|
import { getUser, getUserProfile, signOut } from '../auth/userStore.svelte.js';
|
||||||
|
|
||||||
let { screen, onNavigate } = $props();
|
let { screen, onNavigate } = $props();
|
||||||
|
|
||||||
let user = $derived(getUser());
|
let user = $derived(getUser());
|
||||||
@@ -20,7 +19,9 @@
|
|||||||
|
|
||||||
<div class="topbar">
|
<div class="topbar">
|
||||||
<div class="left">
|
<div class="left">
|
||||||
<span class="app-name">Journi</span>
|
<div class="brand">
|
||||||
|
<span class="app-name">Journi</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="center">
|
<div class="center">
|
||||||
@@ -29,8 +30,8 @@
|
|||||||
class="slider"
|
class="slider"
|
||||||
style="transform: translateX({screen === 'worldmap' ? 0 : 100}%);"
|
style="transform: translateX({screen === 'worldmap' ? 0 : 100}%);"
|
||||||
></div>
|
></div>
|
||||||
<button onclick={() => onNavigate('worldmap')}>Worldmap</button>
|
<button class:active={screen === 'worldmap'} onclick={() => onNavigate('worldmap')}>Worldmap</button>
|
||||||
<button onclick={() => onNavigate('timeline')}>Timeline</button>
|
<button class:active={screen === 'timeline'} onclick={() => onNavigate('timeline')}>Journal</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -83,6 +84,12 @@
|
|||||||
gap: 10px;
|
gap: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.brand {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
.app-name {
|
.app-name {
|
||||||
font-family: var(--heading);
|
font-family: var(--heading);
|
||||||
font-size: 22px;
|
font-size: 22px;
|
||||||
@@ -103,18 +110,18 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
background: var(--bg-subtle);
|
background: var(--bg-subtle);
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
border-radius: 8px;
|
border-radius: 9999px;
|
||||||
padding: 3px;
|
padding: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.slider {
|
.slider {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 3px;
|
top: 4px;
|
||||||
left: 3px;
|
left: 4px;
|
||||||
width: calc(50% - 3px);
|
width: calc(50% - 4px);
|
||||||
height: calc(100% - 6px);
|
height: calc(100% - 8px);
|
||||||
background: var(--bg);
|
background: var(--accent);
|
||||||
border-radius: 6px;
|
border-radius: 9999px;
|
||||||
box-shadow: 0 1px 4px rgba(0,0,0,0.08);
|
box-shadow: 0 1px 4px rgba(0,0,0,0.08);
|
||||||
transition: transform 0.25s ease;
|
transition: transform 0.25s ease;
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
@@ -124,15 +131,20 @@
|
|||||||
position: relative;
|
position: relative;
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
flex: 1;
|
flex: 1;
|
||||||
padding: 4px 18px;
|
padding: 6px 24px;
|
||||||
border: none;
|
border: none;
|
||||||
background: none;
|
background: none;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-family: var(--sans);
|
font-family: var(--sans);
|
||||||
font-size: 13px;
|
font-size: 14px;
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
letter-spacing: 0.01em;
|
letter-spacing: 0.01em;
|
||||||
|
transition: color 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.segmented button.active {
|
||||||
|
color: #fff;
|
||||||
}
|
}
|
||||||
.right {
|
.right {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@@ -1,60 +1,19 @@
|
|||||||
import { db } from '../firebase.js';
|
import { journals } from '../stores/journalStore.js';
|
||||||
import { doc, onSnapshot, setDoc, updateDoc, arrayUnion, arrayRemove } from 'firebase/firestore';
|
import { nameToId } from '../shared/countries.js';
|
||||||
|
|
||||||
let selected = $state(new Set());
|
let selected = $state(new Set());
|
||||||
let totalCountries = $state(0);
|
let totalCountries = $state(0);
|
||||||
let homeCountryCode = $state(null);
|
let flashing = $state(new Set());
|
||||||
let _uid = null;
|
|
||||||
let _unsubscribe = null;
|
|
||||||
|
|
||||||
export function initSelectionListener(uid) {
|
journals.subscribe((entries) => {
|
||||||
if (_unsubscribe) _unsubscribe();
|
const ids = new Set();
|
||||||
_uid = uid;
|
for (const e of entries) {
|
||||||
const userRef = doc(db, 'users', uid);
|
const id = nameToId[e.location?.country];
|
||||||
_unsubscribe = onSnapshot(userRef, (snap) => {
|
if (id) ids.add(id);
|
||||||
if (snap.exists()) {
|
|
||||||
const codes = snap.data().visitedCountries || [];
|
|
||||||
selected = new Set(codes);
|
|
||||||
homeCountryCode = snap.data().homeCountryCode || null;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const visitedRef = doc(db, 'visited', 'countries');
|
|
||||||
|
|
||||||
onSnapshot(visitedRef, (snap) => {
|
|
||||||
if (snap.exists()) {
|
|
||||||
selected = new Set(snap.data().ids ?? []);
|
|
||||||
}
|
}
|
||||||
|
selected = ids;
|
||||||
});
|
});
|
||||||
|
|
||||||
function persist() {
|
|
||||||
setDoc(visitedRef, { ids: [...selected] });
|
|
||||||
}
|
|
||||||
|
|
||||||
export function toggle(id) {
|
|
||||||
const was = selected.has(id);
|
|
||||||
const next = new Set(selected);
|
|
||||||
if (was) next.delete(id);
|
|
||||||
else next.add(id);
|
|
||||||
selected = next;
|
|
||||||
persist();
|
|
||||||
if (_uid) {
|
|
||||||
const userRef = doc(db, 'users', _uid);
|
|
||||||
if (was) updateDoc(userRef, { visitedCountries: arrayRemove(id) });
|
|
||||||
else updateDoc(userRef, { visitedCountries: arrayUnion(id) });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function clearAll() {
|
|
||||||
selected = new Set();
|
|
||||||
persist();
|
|
||||||
if (_uid) {
|
|
||||||
const userRef = doc(db, 'users', _uid);
|
|
||||||
updateDoc(userRef, { visitedCountries: [] });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getSelected() {
|
export function getSelected() {
|
||||||
return selected;
|
return selected;
|
||||||
}
|
}
|
||||||
@@ -67,6 +26,15 @@ export function getTotalCount() {
|
|||||||
return totalCountries;
|
return totalCountries;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getHomeCountryCode() {
|
export function getFlashing() {
|
||||||
return homeCountryCode;
|
return flashing;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function flashCountry(countryName) {
|
||||||
|
const id = nameToId[countryName];
|
||||||
|
if (!id) return;
|
||||||
|
flashing = new Set([...flashing, id]);
|
||||||
|
setTimeout(() => {
|
||||||
|
flashing = new Set([...flashing].filter(x => x !== id));
|
||||||
|
}, 1600);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
* Searchable combobox input.
|
* Searchable combobox input.
|
||||||
* @type {{ id?: string, value: string, options: string[], placeholder?: string, required?: boolean, onchange?: (v: string) => void }}
|
* @type {{ id?: string, value: string, options: string[], placeholder?: string, required?: boolean, onchange?: (v: string) => void }}
|
||||||
*/
|
*/
|
||||||
let { id, value = $bindable(), options, placeholder = '', required = false, onselect } = $props();
|
let { id, value = $bindable(), options, placeholder = '', required = false, onselect, onblurcommit } = $props();
|
||||||
|
|
||||||
let query = $state(value);
|
let query = $state(value);
|
||||||
let open = $state(false);
|
let open = $state(false);
|
||||||
@@ -39,7 +39,11 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
function onBlur() {
|
function onBlur() {
|
||||||
setTimeout(() => { open = false; focused = -1; }, 150);
|
setTimeout(() => {
|
||||||
|
open = false;
|
||||||
|
focused = -1;
|
||||||
|
if (onblurcommit && query.trim()) onblurcommit(query.trim());
|
||||||
|
}, 150);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Keep query in sync if value is changed externally
|
// Keep query in sync if value is changed externally
|
||||||
|
|||||||
209
src/lib/shared/cities.js
Normal file
@@ -0,0 +1,209 @@
|
|||||||
|
export const ALL_CITIES = {
|
||||||
|
'Afghanistan': ['Kabul', 'Herat', 'Kandahar', 'Mazar-i-Sharif', 'Jalalabad'],
|
||||||
|
'Albania': ['Tirana', 'Durrës', 'Vlorë', 'Shkodër', 'Sarandë'],
|
||||||
|
'Algeria': ['Algiers', 'Oran', 'Constantine', 'Annaba', 'Tlemcen'],
|
||||||
|
'Angola': ['Luanda', 'Huambo', 'Benguela', 'Lubango', 'Malanje'],
|
||||||
|
'Argentina': ['Buenos Aires', 'Córdoba', 'Rosario', 'Mendoza', 'Bariloche', 'Salta', 'Ushuaia', 'Mar del Plata', 'Iguazú'],
|
||||||
|
'Armenia': ['Yerevan', 'Gyumri', 'Vanadzor', 'Vagharshapat'],
|
||||||
|
'Australia': ['Sydney', 'Melbourne', 'Brisbane', 'Perth', 'Adelaide', 'Gold Coast', 'Cairns', 'Hobart', 'Darwin', 'Canberra', 'Newcastle'],
|
||||||
|
'Austria': ['Vienna', 'Salzburg', 'Innsbruck', 'Graz', 'Linz', 'Hallstatt', 'Zell am See'],
|
||||||
|
'Azerbaijan': ['Baku', 'Ganja', 'Sumqayit', 'Mingachevir', 'Nakhchivan'],
|
||||||
|
'Bahamas': ['Nassau', 'Freeport', 'Marsh Harbour', 'George Town'],
|
||||||
|
'Bahrain': ['Manama', 'Muharraq', 'Riffa', 'Hamad Town'],
|
||||||
|
'Bangladesh': ['Dhaka', 'Chittagong', 'Sylhet', 'Cox\'s Bazar', 'Rajshahi', 'Khulna'],
|
||||||
|
'Barbados': ['Bridgetown', 'Speightstown', 'Oistins', 'Holetown'],
|
||||||
|
'Belarus': ['Minsk', 'Brest', 'Grodno', 'Vitebsk', 'Gomel'],
|
||||||
|
'Belgium': ['Brussels', 'Antwerp', 'Ghent', 'Bruges', 'Leuven', 'Liège', 'Namur'],
|
||||||
|
'Belize': ['Belize City', 'San Ignacio', 'Belmopan', 'Placencia', 'Caye Caulker'],
|
||||||
|
'Benin': ['Porto-Novo', 'Cotonou', 'Parakou', 'Abomey'],
|
||||||
|
'Bhutan': ['Thimphu', 'Paro', 'Punakha', 'Jakar'],
|
||||||
|
'Bolivia': ['La Paz', 'Sucre', 'Santa Cruz', 'Cochabamba', 'Uyuni', 'Potosí'],
|
||||||
|
'Bosnia and Herz.': ['Sarajevo', 'Mostar', 'Banja Luka', 'Tuzla', 'Zenica'],
|
||||||
|
'Botswana': ['Gaborone', 'Maun', 'Kasane', 'Francistown'],
|
||||||
|
'Brazil': ['Rio de Janeiro', 'São Paulo', 'Brasília', 'Salvador', 'Fortaleza', 'Recife', 'Porto Alegre', 'Curitiba', 'Manaus', 'Florianópolis', 'Belo Horizonte', 'Iguaçu Falls', 'Paraty', 'Bonito'],
|
||||||
|
'Brunei': ['Bandar Seri Begawan', 'Kuala Belait', 'Seria', 'Tutong'],
|
||||||
|
'Bulgaria': ['Sofia', 'Plovdiv', 'Varna', 'Burgas', 'Ruse', 'Bansko'],
|
||||||
|
'Burkina Faso': ['Ouagadougou', 'Bobo-Dioulasso', 'Koudougou', 'Banfora'],
|
||||||
|
'Burundi': ['Bujumbura', 'Gitega', 'Ngozi', 'Ruyigi'],
|
||||||
|
'Cabo Verde': ['Praia', 'Mindelo', 'Santa Maria', 'Sal Rei'],
|
||||||
|
'Cambodia': ['Phnom Penh', 'Siem Reap', 'Sihanoukville', 'Battambang', 'Kampot'],
|
||||||
|
'Cameroon': ['Yaoundé', 'Douala', 'Bamenda', 'Garoua', 'Kribi'],
|
||||||
|
'Canada': ['Toronto', 'Vancouver', 'Montreal', 'Calgary', 'Ottawa', 'Quebec City', 'Halifax', 'Whistler', 'Banff', 'Victoria', 'Edmonton'],
|
||||||
|
'Chad': ['N\'Djamena', 'Moundou', 'Sarh', 'Abéché'],
|
||||||
|
'Chile': ['Santiago', 'Valparaíso', 'Viña del Mar', 'Puerto Varas', 'San Pedro de Atacama', 'Punta Arenas', 'Easter Island (Rapa Nui)', 'Concepción', 'La Serena'],
|
||||||
|
'China': ['Beijing', 'Shanghai', 'Guangzhou', 'Shenzhen', 'Chengdu', 'Hangzhou', 'Xi\'an', 'Kunming', 'Zhangjiajie', 'Guilin', 'Hong Kong', 'Macao', 'Lhasa', 'Suzhou', 'Nanjing', 'Chongqing', 'Wuhan', 'Harbin'],
|
||||||
|
'Colombia': ['Bogotá', 'Medellín', 'Cartagena', 'Cali', 'Santa Marta', 'Bucaramanga', 'San Andrés', 'Leticia', 'Tayrona'],
|
||||||
|
'Congo': ['Brazzaville', 'Pointe-Noire', 'Dolisie', 'Ouésso'],
|
||||||
|
'Costa Rica': ['San José', 'Liberia', 'Puerto Viejo', 'La Fortuna', 'Monteverde', 'Manuel Antonio', 'Tamarindo'],
|
||||||
|
'Croatia': ['Zagreb', 'Dubrovnik', 'Split', 'Zadar', 'Rovinj', 'Pula', 'Hvar', 'Šibenik', 'Trogir'],
|
||||||
|
'Cuba': ['Havana', 'Varadero', 'Trinidad', 'Viñales', 'Santiago de Cuba', 'Cienfuegos'],
|
||||||
|
'Curaçao': ['Willemstad', 'Westpunt', 'Sint Willibrordus'],
|
||||||
|
'Cyprus': ['Nicosia', 'Limassol', 'Paphos', 'Larnaca', 'Ayia Napa'],
|
||||||
|
'Czechia': ['Prague', 'Brno', 'Český Krumlov', 'Karlovy Vary', 'Plzeň', 'Olomouc', 'Ostrava', 'Liberec'],
|
||||||
|
'Dem. Rep. Congo': ['Kinshasa', 'Lubumbashi', 'Goma', 'Bukavu', 'Kisangani'],
|
||||||
|
'Denmark': ['Copenhagen', 'Aarhus', 'Odense', 'Aalborg', 'Ribe', 'Skagen', 'Bornholm', 'Møns Klint'],
|
||||||
|
'Djibouti': ['Djibouti City', 'Tadjoura', 'Obock', 'Ali Sabieh'],
|
||||||
|
'Dominican Rep.': ['Santo Domingo', 'Punta Cana', 'Puerto Plata', 'La Romana', 'Samaná', 'Sosúa'],
|
||||||
|
'Ecuador': ['Quito', 'Guayaquil', 'Cuenca', 'Baños', 'Galápagos Islands', 'Otavalo', 'Montañita'],
|
||||||
|
'Egypt': ['Cairo', 'Alexandria', 'Luxor', 'Aswan', 'Hurghada', 'Sharm el-Sheikh', 'Giza', 'Dahab'],
|
||||||
|
'El Salvador': ['San Salvador', 'Santa Ana', 'San Miguel', 'La Libertad', 'Suchitoto'],
|
||||||
|
'Eq. Guinea': ['Malabo', 'Bata', 'Ebebiyín'],
|
||||||
|
'Eritrea': ['Asmara', 'Massawa', 'Keren', 'Assab'],
|
||||||
|
'Estonia': ['Tallinn', 'Tartu', 'Pärnu', 'Kuressaare', 'Narva'],
|
||||||
|
'Eswatini': ['Mbabane', 'Manzini', 'Big Bend', 'Mhlume'],
|
||||||
|
'Ethiopia': ['Addis Ababa', 'Lalibela', 'Gondar', 'Axum', 'Bahir Dar', 'Harar'],
|
||||||
|
'Faeroe Is.': ['Tórshavn', 'Klaksvík', 'Runavík', 'Vestmanna'],
|
||||||
|
'Fiji': ['Suva', 'Nadi', 'Lautoka', 'Denarau', 'Coral Coast'],
|
||||||
|
'Finland': ['Helsinki', 'Rovaniemi', 'Tampere', 'Turku', 'Levi', 'Savonlinna', 'Porvoo'],
|
||||||
|
'France': ['Paris', 'Nice', 'Marseille', 'Lyon', 'Bordeaux', 'Toulouse', 'Strasbourg', 'Lille', 'Montpellier', 'Avignon', 'Arles', 'Cannes', 'Saint-Tropez', 'Annecy', 'Chamonix', 'Biarritz', 'Colmar'],
|
||||||
|
'Gabon': ['Libreville', 'Port-Gentil', 'Franceville', 'Oyem'],
|
||||||
|
'Gambia': ['Banjul', 'Serrekunda', 'Brikama', 'Bakau'],
|
||||||
|
'Georgia': ['Tbilisi', 'Batumi', 'Kutaisi', 'Stepantsminda', 'Sighnaghi', 'Telavi', 'Mestia'],
|
||||||
|
'Germany': ['Berlin', 'Munich', 'Hamburg', 'Frankfurt', 'Cologne', 'Stuttgart', 'Düsseldorf', 'Dresden', 'Leipzig', 'Nuremberg', 'Heidelberg', 'Freiburg', 'Hannover', 'Bremen', 'Bonn', 'Rothenburg ob der Tauber', 'Neuschwanstein'],
|
||||||
|
'Ghana': ['Accra', 'Kumasi', 'Cape Coast', 'Tamale', 'Elmina', 'Takoradi'],
|
||||||
|
'Greece': ['Athens', 'Santorini', 'Mykonos', 'Crete', 'Thessaloniki', 'Corfu', 'Rhodes', 'Naxos', 'Paros', 'Milos', 'Delphi', 'Meteora', 'Olympia', 'Zakynthos'],
|
||||||
|
'Greenland': ['Nuuk', 'Ilulissat', 'Kangerlussuaq', 'Sisimiut'],
|
||||||
|
'Grenada': ['St. George\'s', 'Gouyave', 'Grenville', 'Sauteurs'],
|
||||||
|
'Guatemala': ['Guatemala City', 'Antigua', 'Lake Atitlán', 'Flores', 'Chichicastenango', 'Quetzaltenango', 'Semuc Champey'],
|
||||||
|
'Guinea': ['Conakry', 'Kindia', 'Kankan', 'N\'Zérékoré', 'Labé'],
|
||||||
|
'Guinea-Bissau': ['Bissau', 'Bafatá', 'Gabú', 'Cacheu'],
|
||||||
|
'Guyana': ['Georgetown', 'Linden', 'New Amsterdam', 'Bartica'],
|
||||||
|
'Haiti': ['Port-au-Prince', 'Cap-Haïtien', 'Jacmel', 'Les Cayes', 'Gonaïves'],
|
||||||
|
'Honduras': ['Tegucigalpa', 'San Pedro Sula', 'La Ceiba', 'Roatán', 'Copán Ruinas'],
|
||||||
|
'Hungary': ['Budapest', 'Debrecen', 'Szeged', 'Pécs', 'Eger', 'Siófok (Lake Balaton)', 'Visegrád', 'Hévíz'],
|
||||||
|
'Iceland': ['Reykjavík', 'Akureyri', 'Vík', 'Höfn', 'Ísafjörður', 'Blue Lagoon', 'Thingvellir'],
|
||||||
|
'India': ['Mumbai', 'Delhi', 'Jaipur', 'Agra', 'Varanasi', 'Goa', 'Kerala', 'Bangalore', 'Chennai', 'Kolkata', 'Hyderabad', 'Udaipur', 'Jaisalmer', 'Rishikesh', 'Darjeeling', 'Amritsar', 'Leh', 'Hampi', 'Mysore', 'Pondicherry'],
|
||||||
|
'Indonesia': ['Bali (Denpasar)', 'Jakarta', 'Yogyakarta', 'Lombok', 'Komodo', 'Surabaya', 'Bandung', 'Medan', 'Makassar', 'Labuan Bajo', 'Raja Ampat', 'Gili Islands'],
|
||||||
|
'Iran': ['Tehran', 'Isfahan', 'Shiraz', 'Mashhad', 'Tabriz', 'Yazd', 'Kashan'],
|
||||||
|
'Iraq': ['Baghdad', 'Erbil', 'Basra', 'Najaf', 'Karbala', 'Sulaymaniyah'],
|
||||||
|
'Ireland': ['Dublin', 'Galway', 'Cork', 'Killarney', 'Dingle', 'Cliffs of Moher', 'Kilkenny', 'Belfast (NI)', 'Ring of Kerry'],
|
||||||
|
'Israel': ['Tel Aviv', 'Jerusalem', 'Haifa', 'Eilat', 'Dead Sea', 'Nazareth', 'Tiberias', 'Caesarea', 'Akko'],
|
||||||
|
'Italy': ['Rome', 'Florence', 'Venice', 'Milan', 'Naples', 'Cinque Terre', 'Amalfi Coast', 'Positano', 'Capri', 'Verona', 'Bologna', 'Turin', 'Siena', 'Lake Como', 'Pisa', 'Palermo', 'Catania', 'Matera', 'Tuscany', 'Dolomites'],
|
||||||
|
'Jamaica': ['Kingston', 'Montego Bay', 'Negril', 'Ocho Rios', 'Port Antonio'],
|
||||||
|
'Japan': ['Tokyo', 'Osaka', 'Kyoto', 'Sapporo', 'Fukuoka', 'Hiroshima', 'Nara', 'Kanazawa', 'Nagoya', 'Yokohama', 'Kobe', 'Hakone', 'Nikko', 'Miyajima', 'Takayama', 'Okinawa', 'Kamakura', 'Fuji Five Lakes'],
|
||||||
|
'Jordan': ['Amman', 'Petra', 'Wadi Rum', 'Dead Sea', 'Aqaba', 'Madaba', 'Jerash'],
|
||||||
|
'Kazakhstan': ['Almaty', 'Nur-Sultan', 'Shymkent', 'Aktau', 'Karaganda'],
|
||||||
|
'Kenya': ['Nairobi', 'Mombasa', 'Masai Mara', 'Diani Beach', 'Amboseli', 'Lake Nakuru', 'Tsavo', 'Nanyuki'],
|
||||||
|
'Kiribati': ['Tarawa', 'Kiritimati (Christmas Island)'],
|
||||||
|
'Kosovo': ['Pristina', 'Prizren', 'Peja', 'Gjakova', 'Mitrovica'],
|
||||||
|
'Kuwait': ['Kuwait City', 'Salmiya', 'Hawally', 'Ahmadi', 'Jahra'],
|
||||||
|
'Kyrgyzstan': ['Bishkek', 'Osh', 'Karakol', 'Jalal-Abad', 'Talas'],
|
||||||
|
'Laos': ['Vientiane', 'Luang Prabang', 'Vang Vieng', 'Pakse', 'Savannakhet', 'Si Phan Don (4000 Islands)'],
|
||||||
|
'Latvia': ['Riga', 'Jūrmala', 'Liepāja', 'Cēsis', 'Sigulda', 'Daugavpils'],
|
||||||
|
'Lebanon': ['Beirut', 'Byblos', 'Baalbek', 'Tripoli', 'Sidon', 'Tyre', 'Jounieh'],
|
||||||
|
'Lesotho': ['Maseru', 'Teyateyaneng', 'Mafeteng', 'Hlotse'],
|
||||||
|
'Liberia': ['Monrovia', 'Buchanan', 'Ganta', 'Harper', 'Robertsport'],
|
||||||
|
'Libya': ['Tripoli', 'Benghazi', 'Misrata', 'Sabratha', 'Leptis Magna'],
|
||||||
|
'Liechtenstein': ['Vaduz', 'Schaan', 'Balzers', 'Triesenberg'],
|
||||||
|
'Lithuania': ['Vilnius', 'Kaunas', 'Klaipėda', 'Šiauliai', 'Trakai', 'Palanga', 'Nida'],
|
||||||
|
'Luxembourg': ['Luxembourg City', 'Echternach', 'Vianden', 'Ettelbruck'],
|
||||||
|
'Madagascar': ['Antananarivo', 'Nosy Be', 'Morondava', 'Fianarantsoa', 'Isalo', 'Tôlanaro'],
|
||||||
|
'Malawi': ['Lilongwe', 'Blantyre', 'Mzuzu', 'Lake Malawi', 'Zomba'],
|
||||||
|
'Malaysia': ['Kuala Lumpur', 'Penang (George Town)', 'Langkawi', 'Borneo (Kota Kinabalu)', 'Malacca', 'Cameron Highlands', 'Johor Bahru', 'Kuching', 'Sipadan Island'],
|
||||||
|
'Maldives': ['Malé', 'Ari Atoll', 'Baa Atoll', 'South Male Atoll', 'Addu City'],
|
||||||
|
'Mali': ['Bamako', 'Timbuktu', 'Ségou', 'Mopti', 'Djenné'],
|
||||||
|
'Malta': ['Valletta', 'Sliema', 'Gozo', 'Mellieħa', 'Mdina', 'St. Julian\'s', 'Comino'],
|
||||||
|
'Marshall Is.': ['Majuro', 'Kwajalein', 'Ebeye'],
|
||||||
|
'Mauritania': ['Nouakchott', 'Nouadhibou', 'Atar', 'Chinguetti', 'Ouadane'],
|
||||||
|
'Mauritius': ['Port Louis', 'Grand Baie', 'Flic en Flac', 'Belle Mare', 'Le Morne', 'Chamarel'],
|
||||||
|
'Mexico': ['Mexico City', 'Cancún', 'Playa del Carmen', 'Tulum', 'Guadalajara', 'Monterrey', 'Puerto Vallarta', 'Oaxaca', 'San Miguel de Allende', 'Mérida', 'Cabo San Lucas', 'Guanajuato', 'Chichen Itza', 'Palenque', 'Cuernavaca', 'Puebla'],
|
||||||
|
'Micronesia': ['Palikir', 'Chuuk', 'Pohnpei', 'Yap', 'Kosrae'],
|
||||||
|
'Moldova': ['Chișinău', 'Bălți', 'Tiraspol', 'Cahul', 'Orhei'],
|
||||||
|
'Monaco': ['Monaco City', 'Monte Carlo', 'La Condamine', 'Fontvieille'],
|
||||||
|
'Mongolia': ['Ulaanbaatar', 'Karakorum', 'Gobi Desert', 'Lake Khövsgöl', 'Altai', 'Erdenet'],
|
||||||
|
'Montenegro': ['Podgorica', 'Kotor', 'Budva', 'Bar', 'Ulcinj', 'Žabljak', 'Perast'],
|
||||||
|
'Morocco': ['Marrakech', 'Fes', 'Casablanca', 'Rabat', 'Tangier', 'Chefchaouen', 'Essaouira', 'Ouarzazate', 'Agadir', 'Meknes', 'Merzouga (Sahara)'],
|
||||||
|
'Mozambique': ['Maputo', 'Beira', 'Tofo (Inhambane)', 'Vilankulo', 'Bazaruto Archipelago', 'Nampula'],
|
||||||
|
'Myanmar': ['Yangon', 'Mandalay', 'Bagan', 'Inle Lake', 'Hpa-An', 'Ngapali Beach'],
|
||||||
|
'Namibia': ['Windhoek', 'Swakopmund', 'Sossusvlei', 'Etosha National Park', 'Fish River Canyon', 'Walvis Bay'],
|
||||||
|
'Nauru': ['Yaren', 'Boe', 'Aiwo'],
|
||||||
|
'Nepal': ['Kathmandu', 'Pokhara', 'Chitwan', 'Lumbini', 'Everest Base Camp', 'Nagarkot', 'Bandipur'],
|
||||||
|
'Netherlands': ['Amsterdam', 'Rotterdam', 'The Hague', 'Utrecht', 'Maastricht', 'Groningen', 'Leiden', 'Delft', 'Giethoorn', 'Haarlem', 'Zaanse Schans', 'Keukenhof'],
|
||||||
|
'New Zealand': ['Auckland', 'Queenstown', 'Wellington', 'Christchurch', 'Rotorua', 'Milford Sound', 'Wanaka', 'Taupō', 'Dunedin', 'Tongariro', 'Abel Tasman', 'Bay of Islands'],
|
||||||
|
'Nicaragua': ['Managua', 'Granada', 'León', 'San Juan del Sur', 'Ometepe Island', 'Corn Islands'],
|
||||||
|
'Niger': ['Niamey', 'Agadez', 'Zinder', 'Maradi', 'Tahoua'],
|
||||||
|
'Nigeria': ['Lagos', 'Abuja', 'Port Harcourt', 'Calabar', 'Ibadan', 'Kano', 'Enugu', 'Jos'],
|
||||||
|
'North Korea': ['Pyongyang', 'Kaesong', 'Chongjin', 'Nampo', 'Wonsan'],
|
||||||
|
'North Macedonia': ['Skopje', 'Ohrid', 'Bitola', 'Tetovo', 'Struga'],
|
||||||
|
'Norway': ['Oslo', 'Bergen', 'Tromsø', 'Stavanger', 'Trondheim', 'Lofoten Islands', 'Geirangerfjord', 'Flåm', 'Alesund', 'Preikestolen', 'Nordkapp'],
|
||||||
|
'Oman': ['Muscat', 'Salalah', 'Nizwa', 'Sur', 'Wahiba Sands', 'Khasab', 'Sohar'],
|
||||||
|
'Pakistan': ['Islamabad', 'Karachi', 'Lahore', 'Hunza Valley', 'Skardu', 'Faisalabad', 'Multan', 'Swat Valley'],
|
||||||
|
'Palau': ['Ngerulmud', 'Koror', 'Peleliu'],
|
||||||
|
'Palestine': ['Ramallah', 'Bethlehem', 'Hebron', 'Nablus', 'Jericho', 'Gaza'],
|
||||||
|
'Panama': ['Panama City', 'Bocas del Toro', 'Boquete', 'San Blas Islands', 'El Valle de Antón'],
|
||||||
|
'Papua New Guinea': ['Port Moresby', 'Lae', 'Mount Hagen', 'Kokopo', 'Alotau', 'Tufi'],
|
||||||
|
'Paraguay': ['Asunción', 'Ciudad del Este', 'Encarnación', 'San Bernardino', 'Filadelfia'],
|
||||||
|
'Peru': ['Lima', 'Cusco', 'Arequipa', 'Machu Picchu', 'Sacred Valley', 'Lake Titicaca', 'Iquitos (Amazon)', 'Paracas', 'Huaraz', 'Nazca', 'Máncora', 'Trujillo'],
|
||||||
|
'Philippines': ['Manila', 'Cebu', 'Palawan (El Nido)', 'Siargao', 'Boracay', 'Davao', 'Bohol (Panglao)', 'Banaue Rice Terraces', 'Coron', 'Baguio', 'Puerto Princesa'],
|
||||||
|
'Poland': ['Warsaw', 'Kraków', 'Gdańsk', 'Wrocław', 'Poznań', 'Zakopane', 'Gdynia', 'Łódź', 'Toruń', 'Szczecin', 'Lublin', 'Malbork', 'Morskie Oko'],
|
||||||
|
'Portugal': ['Lisbon', 'Porto', 'Algarve (Faro)', 'Sintra', 'Madeira', 'Coimbra', 'Azores', 'Braga', 'Évora', 'Cascais', 'Douro Valley'],
|
||||||
|
'Puerto Rico': ['San Juan', 'Ponce', 'Mayagüez', 'Culebra', 'Vieques', 'Rincón'],
|
||||||
|
'Qatar': ['Doha', 'Al Wakrah', 'Al Khor', 'Mesaieed', 'Katara'],
|
||||||
|
'Romania': ['Bucharest', 'Cluj-Napoca', 'Brașov', 'Sibiu', 'Sighișoara', 'Timișoara', 'Iași', 'Constanța', 'Transfăgărășan', 'Mamaia'],
|
||||||
|
'Russia': ['Moscow', 'Saint Petersburg', 'Moscow', 'Sochi', 'Vladivostok', 'Kazan', 'Novosibirsk', 'Yekaterinburg', 'Irkutsk', 'Lake Baikal', 'Murmansk', 'Kaliningrad', 'Kamchatka', 'Krasnodar', 'Nizhny Novgorod', 'Rostov-on-Don'],
|
||||||
|
'Rwanda': ['Kigali', 'Butare', 'Gisenyi', 'Volcanoes National Park', 'Akagera', 'Nyungwe Forest'],
|
||||||
|
'S. Sudan': ['Juba', 'Malakal', 'Wau', 'Bor', 'Yei'],
|
||||||
|
'Samoa': ['Apia', 'Salelologa', 'Lalomanu', 'Safua'],
|
||||||
|
'San Marino': ['San Marino City', 'Borgo Maggiore', 'Serravalle'],
|
||||||
|
'São Tomé and Principe': ['São Tomé', 'Santo António', 'Neves'],
|
||||||
|
'Saudi Arabia': ['Riyadh', 'Jeddah', 'Mecca', 'Medina', 'Dammam', 'AlUla', 'Abha', 'Tabuk', 'Neom'],
|
||||||
|
'Senegal': ['Dakar', 'Saint-Louis', 'Gorée Island', 'Sine-Saloum Delta', 'Pink Lake (Lac Rose)', 'Cap Skirring'],
|
||||||
|
'Serbia': ['Belgrade', 'Novi Sad', 'Niš', 'Subotica', 'Kragujevac', 'Zlatibor', 'Kopaonik'],
|
||||||
|
'Seychelles': ['Mahé (Victoria)', 'Praslin', 'La Digue', 'Silhouette Island'],
|
||||||
|
'Sierra Leone': ['Freetown', 'Bo', 'Kenema', 'Makeni', 'Bunce Island'],
|
||||||
|
'Singapore': ['Singapore'],
|
||||||
|
'Slovakia': ['Bratislava', 'Košice', 'Tatras (High Tatras)', 'Banská Štiavnica', 'Levoča', 'Žilina', 'Poprad'],
|
||||||
|
'Slovenia': ['Ljubljana', 'Lake Bled', 'Piran', 'Maribor', 'Postojna Cave', 'Triglav National Park', 'Celje'],
|
||||||
|
'Solomon Is.': ['Honiara', 'Gizo', 'Auki', 'Munda'],
|
||||||
|
'Somalia': ['Mogadishu', 'Hargeisa', 'Kismayo', 'Baidoa', 'Berbera'],
|
||||||
|
'South Africa': ['Cape Town', 'Johannesburg', 'Durban', 'Kruger National Park', 'Garden Route', 'Cape Winelands (Stellenbosch)', 'Port Elizabeth', 'Hermanus', 'Blyde River Canyon', 'Drakensberg', 'Pretoria', 'Soweto', 'Knysna'],
|
||||||
|
'South Korea': ['Seoul', 'Busan', 'Jeju Island', 'Gyeongju', 'Incheon', 'Daegu', 'Daejeon', 'Gwangju', 'Jeonju', 'Seoraksan National Park', 'Andong', 'Suwon', 'Sokcho', 'Pyeongchang'],
|
||||||
|
'Spain': ['Barcelona', 'Madrid', 'Seville', 'Granada', 'Valencia', 'Bilbao', 'San Sebastián', 'Mallorca', 'Ibiza', 'Tenerife', 'Córdoba', 'Málaga', 'Santiago de Compostela', 'Toledo', 'Ronda', 'Salamanca', 'Marbella', 'Costa Brava', 'Alhambra', 'Picos de Europa'],
|
||||||
|
'Sri Lanka': ['Colombo', 'Kandy', 'Galle', 'Sigiriya', 'Ella', 'Mirissa', 'Anuradhapura', 'Polonnaruwa', 'Nuwara Eliya', 'Yala National Park'],
|
||||||
|
'St. Kitts and Nevis': ['Basseterre', 'Charlestown', 'Frigate Bay', 'Nevis'],
|
||||||
|
'St. Lucia': ['Castries', 'Soufrière', 'Gros Islet', 'Vieux Fort', 'Marigot Bay'],
|
||||||
|
'St. Pierre and Miquelon': ['Saint-Pierre', 'Miquelon'],
|
||||||
|
'St. Vin. and Gren.': ['Kingstown', 'Bequia', 'Mustique', 'Canouan', 'Union Island'],
|
||||||
|
'Sudan': ['Khartoum', 'Omdurman', 'Port Sudan', 'Kassala', 'Nyala'],
|
||||||
|
'Suriname': ['Paramaribo', 'Lelydorp', 'Brokopondo', 'Nieuw Nickerie'],
|
||||||
|
'Sweden': ['Stockholm', 'Gothenburg', 'Malmö', 'Kiruna', 'Visby', 'Uppsala', 'Lund', 'Abisko National Park', 'Icehotel (Jukkasjärvi)', 'Smögen'],
|
||||||
|
'Switzerland': ['Zurich', 'Geneva', 'Lucerne', 'Zermatt', 'Interlaken', 'Lugano', 'Lausanne', 'Bern', 'Grindelwald', 'St. Moritz', 'Jungfraujoch', 'Montreux', 'Matterhorn Glacier Paradise'],
|
||||||
|
'Syria': ['Damascus', 'Aleppo', 'Palmyra', 'Homs', 'Latakia', 'Maaloula'],
|
||||||
|
'Taiwan': ['Taipei', 'Taichung', 'Kaohsiung', 'Tainan', 'Taroko Gorge', 'Sun Moon Lake', 'Alishan', 'Jiufen', 'Kenting', 'Yilan', 'Taoyuan', 'Hualien'],
|
||||||
|
'Tajikistan': ['Dushanbe', 'Khujand', 'Pamir Mountains', 'Khorog', 'Bokhtar'],
|
||||||
|
'Tanzania': ['Dar es Salaam', 'Zanzibar City', 'Arusha', 'Serengeti National Park', 'Kilimanjaro', 'Ngorongoro Crater', 'Mwanza', 'Mbeya', 'Mafia Island', 'Lake Manyara', 'Selous'],
|
||||||
|
'Thailand': ['Bangkok', 'Chiang Mai', 'Phuket', 'Krabi', 'Pattaya', 'Koh Samui', 'Koh Phi Phi', 'Koh Tao', 'Ayutthaya', 'Chiang Rai', 'Hua Hin', 'Khao Sok', 'Pai', 'Kanchanaburi', 'Sukhothai', 'Koh Lanta', 'Railay Beach', 'Erawan National Park'],
|
||||||
|
'Timor-Leste': ['Dili', 'Baucau', 'Same', 'Atauro Island', 'Jaco Island'],
|
||||||
|
'Togo': ['Lomé', 'Kpalimé', 'Sokodé', 'Kara', 'Aneho'],
|
||||||
|
'Tonga': ['Nuku\'alofa', 'Neiafu', 'Pangai', 'Ha\'apai', 'Eua'],
|
||||||
|
'Trinidad and Tobago': ['Port of Spain', 'San Fernando', 'Tobago (Scarborough)', 'Chaguanas', 'Maracas Bay'],
|
||||||
|
'Tunisia': ['Tunis', 'Sousse', 'Hammamet', 'Djerba', 'Sfax', 'Carthage', 'Douz (Sahara)', 'Matmata', 'Bizerte', 'El Jem'],
|
||||||
|
'Turkey': ['Istanbul', 'Cappadocia (Göreme)', 'Antalya', 'Izmir', 'Bodrum', 'Fethiye', 'Pamukkale', 'Ephesus', 'Marmaris', 'Alanya', 'Ankara', 'Trabzon', 'Kas', 'Olympos', 'Gallipoli', 'Konya', 'Mardin', 'Butterfly Valley'],
|
||||||
|
'Turkmenistan': ['Ashgabat', 'Mary', 'Turkmenbashi', 'Dashoguz', 'Köneürgenç'],
|
||||||
|
'Tuvalu': ['Funafuti', 'Nanumea', 'Nukulaelae'],
|
||||||
|
'U.S. Virgin Is.': ['Charlotte Amalie', 'Christiansted', 'Frederiksted', 'Cruz Bay'],
|
||||||
|
'Uganda': ['Kampala', 'Jinja', 'Murchison Falls', 'Queen Elizabeth National Park', 'Bwindi Impenetrable Forest', 'Kibale', 'Lake Bunyonyi', 'Entebbe'],
|
||||||
|
'Ukraine': ['Kyiv', 'Lviv', 'Odesa', 'Kharkiv', 'Carpathian Mountains', 'Dnipro', 'Chernivtsi', 'Kamianets-Podilskyi', 'Zaporizhzhia', 'Lutsk'],
|
||||||
|
'United Arab Emirates': ['Dubai', 'Abu Dhabi', 'Sharjah', 'Fujairah', 'Ras Al Khaimah', 'Ajman', 'Hatta'],
|
||||||
|
'United Kingdom': ['London', 'Edinburgh', 'Bath', 'York', 'Liverpool', 'Manchester', 'Birmingham', 'Cambridge', 'Oxford', 'Brighton', 'Cornwall', 'Bristol', 'Cardiff', 'Glasgow', 'Inverness', 'Belfast', 'Lake District', 'Scottish Highlands', 'St. Ives', 'Canterbury', 'Dover', 'Stratford-upon-Avon'],
|
||||||
|
'United States of America': ['New York', 'Los Angeles', 'Chicago', 'San Francisco', 'Las Vegas', 'Miami', 'Orlando', 'Washington DC', 'Boston', 'Seattle', 'Portland', 'Denver', 'New Orleans', 'Nashville', 'Austin', 'San Diego', 'Honolulu', 'Grand Canyon', 'Yellowstone', 'Yosemite', 'Houston', 'Dallas', 'Atlanta', 'Philadelphia', 'Phoenix', 'San Antonio', 'Savannah', 'Charleston', 'Santa Fe', 'Anchorage', 'Maui', 'Kauai', 'Moab', 'Portland (ME)', 'Asheville', 'Sedona', 'Napa Valley'],
|
||||||
|
'Uruguay': ['Montevideo', 'Punta del Este', 'Colonia del Sacramento', 'Piriapolis', 'Cabo Polonio', 'Rocha'],
|
||||||
|
'Uzbekistan': ['Tashkent', 'Samarkand', 'Bukhara', 'Khiva', 'Shakhrisabz', 'Fergana', 'Nukus', 'Termez'],
|
||||||
|
'Vanuatu': ['Port Vila', 'Luganville', 'Tanna Island', 'Pentecost', 'Espiritu Santo'],
|
||||||
|
'Vatican': ['Vatican City'],
|
||||||
|
'Venezuela': ['Caracas', 'Angel Falls (Canaima)', 'Margarita Island', 'Los Roques', 'Mérida', 'Roraima', 'Maracaibo', 'Valencia'],
|
||||||
|
'Vietnam': ['Hanoi', 'Ho Chi Minh City (Saigon)', 'Ha Long Bay', 'Hoi An', 'Da Nang', 'Hue', 'Nha Trang', 'Phong Nha', 'Da Lat', 'Sapa', 'Phu Quoc', 'Mui Ne', 'Con Dao', 'Mekong Delta (Can Tho)', 'Ninh Binh', 'Son Doong Cave'],
|
||||||
|
'W. Sahara': ['Laayoune', 'Dakhla', 'Smara', 'Boujdour'],
|
||||||
|
'Yemen': ['Sana\'a', 'Aden', 'Socotra', 'Taiz', 'Mukalla', 'Shibam'],
|
||||||
|
'Zambia': ['Lusaka', 'Victoria Falls', 'Livingstone', 'South Luangwa National Park', 'Kitwe', 'Ndola', 'Kasama'],
|
||||||
|
'Zimbabwe': ['Harare', 'Victoria Falls', 'Bulawayo', 'Hwange National Park', 'Mutare', 'Gweru', 'Masvingo (Great Zimbabwe)'],
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get curated city suggestions for a given country name.
|
||||||
|
* @param {string} countryName
|
||||||
|
* @returns {string[]}
|
||||||
|
*/
|
||||||
|
export function getCitiesForCountry(countryName) {
|
||||||
|
return ALL_CITIES[countryName] || [];
|
||||||
|
}
|
||||||
@@ -63,8 +63,18 @@ const nameToAlpha2 = {
|
|||||||
'eSwatini':'SZ','Åland':'AX',
|
'eSwatini':'SZ','Åland':'AX',
|
||||||
};
|
};
|
||||||
|
|
||||||
export const countryNames = feature(worldData, worldData.objects.countries)
|
const _features = feature(worldData, worldData.objects.countries).features;
|
||||||
.features.map(f => f.properties?.name).filter(Boolean).sort();
|
|
||||||
|
export const countryNames = _features
|
||||||
|
.map(f => f.properties?.name).filter(Boolean).sort();
|
||||||
|
|
||||||
|
// country name → topojson numeric ID (e.g. 'Japan' → '392')
|
||||||
|
export const nameToId = Object.fromEntries(
|
||||||
|
_features
|
||||||
|
.filter(f => f.properties?.name && f.id)
|
||||||
|
.map(f => [f.properties.name, String(f.id)])
|
||||||
|
);
|
||||||
|
nameToId['Kosovo'] = 'XK';
|
||||||
|
|
||||||
/** @param {string} country */
|
/** @param {string} country */
|
||||||
export function flagEmoji(country) {
|
export function flagEmoji(country) {
|
||||||
|
|||||||
184
src/lib/shared/countryCities.js
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
export const countryCities = {
|
||||||
|
'Afghanistan': ['Kabul','Kandahar','Herat','Mazar-i-Sharif','Jalalabad'],
|
||||||
|
'Albania': ['Tirana','Durrës','Vlorë','Shkodër','Elbasan'],
|
||||||
|
'Algeria': ['Algiers','Oran','Constantine','Annaba','Blida'],
|
||||||
|
'Andorra': ['Andorra la Vella','Escaldes-Engordany','Encamp'],
|
||||||
|
'Angola': ['Luanda','Huambo','Lobito','Benguela','Lubango'],
|
||||||
|
'Argentina': ['Buenos Aires','Córdoba','Rosario','Mendoza','Bariloche','Salta','Mar del Plata'],
|
||||||
|
'Armenia': ['Yerevan','Gyumri','Vanadzor'],
|
||||||
|
'Australia': ['Sydney','Melbourne','Brisbane','Perth','Adelaide','Gold Coast','Cairns','Darwin','Hobart','Canberra'],
|
||||||
|
'Austria': ['Vienna','Salzburg','Graz','Innsbruck','Linz','Hallstatt'],
|
||||||
|
'Azerbaijan': ['Baku','Ganja','Sumqayit'],
|
||||||
|
'Bahamas': ['Nassau','Freeport'],
|
||||||
|
'Bahrain': ['Manama','Riffa','Muharraq'],
|
||||||
|
'Bangladesh': ['Dhaka','Chittagong','Sylhet','Rajshahi','Khulna'],
|
||||||
|
'Barbados': ['Bridgetown'],
|
||||||
|
'Belarus': ['Minsk','Gomel','Brest','Grodno'],
|
||||||
|
'Belgium': ['Brussels','Bruges','Ghent','Antwerp','Liège'],
|
||||||
|
'Belize': ['Belize City','San Ignacio','Placencia'],
|
||||||
|
'Benin': ['Cotonou','Porto-Novo','Abomey'],
|
||||||
|
'Bhutan': ['Thimphu','Paro','Punakha'],
|
||||||
|
'Bolivia': ['La Paz','Santa Cruz','Cochabamba','Sucre','Uyuni'],
|
||||||
|
'Bosnia and Herz.': ['Sarajevo','Mostar','Banja Luka'],
|
||||||
|
'Botswana': ['Gaborone','Francistown','Maun'],
|
||||||
|
'Brazil': ['São Paulo','Rio de Janeiro','Brasília','Salvador','Fortaleza','Manaus','Recife','Florianópolis','Foz do Iguaçu'],
|
||||||
|
'Brunei': ['Bandar Seri Begawan'],
|
||||||
|
'Bulgaria': ['Sofia','Plovdiv','Varna','Burgas','Ruse'],
|
||||||
|
'Burkina Faso': ['Ouagadougou','Bobo-Dioulasso'],
|
||||||
|
'Burundi': ['Bujumbura','Gitega'],
|
||||||
|
'Cabo Verde': ['Praia','Mindelo'],
|
||||||
|
'Cambodia': ['Phnom Penh','Siem Reap','Sihanoukville','Battambang'],
|
||||||
|
'Cameroon': ['Yaoundé','Douala','Bafoussam'],
|
||||||
|
'Canada': ['Toronto','Vancouver','Montreal','Calgary','Ottawa','Edmonton','Quebec City','Whistler','Banff','Niagara Falls','Halifax'],
|
||||||
|
'Central African Rep.': ['Bangui'],
|
||||||
|
'Chad': ["N'Djamena",'Moundou'],
|
||||||
|
'Chile': ['Santiago','Valparaíso','Atacama','Puerto Natales','Punta Arenas','Viña del Mar','San Pedro de Atacama'],
|
||||||
|
'China': ['Beijing','Shanghai','Guangzhou','Shenzhen','Chengdu','Xi\'an','Hangzhou','Chongqing','Guilin','Zhangjiajie','Lijiang','Hong Kong','Macau'],
|
||||||
|
'Colombia': ['Bogotá','Medellín','Cartagena','Cali','Santa Marta','Barranquilla'],
|
||||||
|
'Comoros': ['Moroni'],
|
||||||
|
'Congo': ['Brazzaville','Pointe-Noire'],
|
||||||
|
'Costa Rica': ['San José','Manuel Antonio','Tamarindo','Arenal','Monteverde'],
|
||||||
|
'Croatia': ['Zagreb','Dubrovnik','Split','Hvar','Zadar','Pula','Rijeka'],
|
||||||
|
'Cuba': ['Havana','Trinidad','Varadero','Santiago de Cuba','Cienfuegos'],
|
||||||
|
'Cyprus': ['Nicosia','Limassol','Paphos','Larnaca','Ayia Napa'],
|
||||||
|
'Czechia': ['Prague','Brno','Český Krumlov','Karlovy Vary','Olomouc'],
|
||||||
|
"Côte d'Ivoire": ['Abidjan','Yamoussoukro','Bouaké'],
|
||||||
|
'Dem. Rep. Congo': ['Kinshasa','Lubumbashi','Goma','Kisangani'],
|
||||||
|
'Denmark': ['Copenhagen','Aarhus','Odense','Aalborg'],
|
||||||
|
'Djibouti': ['Djibouti City'],
|
||||||
|
'Dominican Rep.': ['Santo Domingo','Punta Cana','Santiago','La Romana'],
|
||||||
|
'Ecuador': ['Quito','Guayaquil','Cuenca','Baños','Galápagos Islands'],
|
||||||
|
'Egypt': ['Cairo','Alexandria','Luxor','Aswan','Sharm el-Sheikh','Hurghada','Giza'],
|
||||||
|
'El Salvador': ['San Salvador','Santa Ana','San Miguel'],
|
||||||
|
'Eq. Guinea': ['Malabo','Bata'],
|
||||||
|
'Eritrea': ['Asmara','Massawa'],
|
||||||
|
'Estonia': ['Tallinn','Tartu','Pärnu'],
|
||||||
|
'Ethiopia': ['Addis Ababa','Lalibela','Gondar','Axum','Dire Dawa'],
|
||||||
|
'Fiji': ['Suva','Nadi','Mamanuca Islands'],
|
||||||
|
'Finland': ['Helsinki','Rovaniemi','Tampere','Turku','Oulu'],
|
||||||
|
'Fr. Polynesia': ['Papeete','Bora Bora','Moorea'],
|
||||||
|
'France': ['Paris','Nice','Lyon','Marseille','Bordeaux','Strasbourg','Toulouse','Cannes','Monaco','Mont Saint-Michel','Versailles'],
|
||||||
|
'Gabon': ['Libreville','Port-Gentil'],
|
||||||
|
'Gambia': ['Banjul','Serekunda'],
|
||||||
|
'Georgia': ['Tbilisi','Batumi','Kutaisi','Sighnaghi'],
|
||||||
|
'Germany': ['Berlin','Munich','Hamburg','Frankfurt','Cologne','Dresden','Heidelberg','Rothenburg ob der Tauber','Neuschwanstein','Stuttgart'],
|
||||||
|
'Ghana': ['Accra','Kumasi','Cape Coast','Tamale'],
|
||||||
|
'Greece': ['Athens','Santorini','Mykonos','Rhodes','Thessaloniki','Crete','Corfu','Meteora'],
|
||||||
|
'Greenland': ['Nuuk','Ilulissat'],
|
||||||
|
'Grenada': ["St. George's"],
|
||||||
|
'Guatemala': ['Guatemala City','Antigua','Lake Atitlán','Tikal','Quetzaltenango'],
|
||||||
|
'Guinea': ['Conakry'],
|
||||||
|
'Guyana': ['Georgetown'],
|
||||||
|
'Haiti': ['Port-au-Prince','Cap-Haïtien'],
|
||||||
|
'Honduras': ['Tegucigalpa','San Pedro Sula','Roatán'],
|
||||||
|
'Hong Kong': ['Hong Kong'],
|
||||||
|
'Hungary': ['Budapest','Debrecen','Pécs','Eger','Győr'],
|
||||||
|
'Iceland': ['Reykjavik','Akureyri','Blue Lagoon','Golden Circle'],
|
||||||
|
'India': ['Mumbai','Delhi','Jaipur','Agra','Bangalore','Chennai','Kolkata','Goa','Varanasi','Udaipur','Kerala','Leh','Shimla'],
|
||||||
|
'Indonesia': ['Jakarta','Bali','Yogyakarta','Lombok','Medan','Komodo','Raja Ampat','Surabaya'],
|
||||||
|
'Iran': ['Tehran','Isfahan','Shiraz','Persepolis','Yazd'],
|
||||||
|
'Iraq': ['Baghdad','Erbil','Basra','Najaf'],
|
||||||
|
'Ireland': ['Dublin','Cork','Galway','Killarney','Limerick'],
|
||||||
|
'Israel': ['Jerusalem','Tel Aviv','Haifa','Eilat','Dead Sea'],
|
||||||
|
'Italy': ['Rome','Florence','Venice','Milan','Naples','Amalfi','Sicily','Cinque Terre','Bologna','Turin'],
|
||||||
|
'Jamaica': ['Kingston','Montego Bay','Negril','Ocho Rios'],
|
||||||
|
'Japan': ['Tokyo','Kyoto','Osaka','Hiroshima','Nara','Sapporo','Hakone','Nikko','Kanazawa','Okinawa','Fukuoka'],
|
||||||
|
'Jordan': ['Amman','Petra','Wadi Rum','Aqaba','Jerash'],
|
||||||
|
'Kazakhstan': ['Almaty','Nur-Sultan','Shymkent'],
|
||||||
|
'Kenya': ['Nairobi','Mombasa','Masai Mara','Amboseli','Zanzibar'],
|
||||||
|
'Kosovo': ['Pristina','Prizren'],
|
||||||
|
'Kuwait': ['Kuwait City'],
|
||||||
|
'Kyrgyzstan': ['Bishkek','Osh','Karakol'],
|
||||||
|
'Laos': ['Vientiane','Luang Prabang','Vang Vieng'],
|
||||||
|
'Latvia': ['Riga','Jūrmala','Sigulda'],
|
||||||
|
'Lebanon': ['Beirut','Byblos','Baalbek','Sidon'],
|
||||||
|
'Libya': ['Tripoli','Benghazi','Leptis Magna'],
|
||||||
|
'Liechtenstein': ['Vaduz'],
|
||||||
|
'Lithuania': ['Vilnius','Kaunas','Trakai','Klaipėda'],
|
||||||
|
'Luxembourg': ['Luxembourg City','Vianden'],
|
||||||
|
'Madagascar': ['Antananarivo','Nosy Be','Morondava'],
|
||||||
|
'Malawi': ['Lilongwe','Blantyre','Lake Malawi'],
|
||||||
|
'Malaysia': ['Kuala Lumpur','Penang','Langkawi','Kota Kinabalu','Malacca','George Town'],
|
||||||
|
'Maldives': ['Malé','Maafushi'],
|
||||||
|
'Mali': ['Bamako','Timbuktu','Djenné'],
|
||||||
|
'Malta': ['Valletta','Mdina','Gozo'],
|
||||||
|
'Mauritania': ['Nouakchott'],
|
||||||
|
'Mauritius': ['Port Louis','Grand Baie','Flic en Flac'],
|
||||||
|
'Mexico': ['Mexico City','Cancún','Guadalajara','Oaxaca','Tulum','Playa del Carmen','San Miguel de Allende','Monterrey','Chichen Itza'],
|
||||||
|
'Moldova': ['Chișinău'],
|
||||||
|
'Monaco': ['Monaco'],
|
||||||
|
'Mongolia': ['Ulaanbaatar','Gobi Desert'],
|
||||||
|
'Montenegro': ['Podgorica','Kotor','Budva','Bar'],
|
||||||
|
'Morocco': ['Marrakech','Fes','Casablanca','Rabat','Chefchaouen','Essaouira','Sahara Desert'],
|
||||||
|
'Mozambique': ['Maputo','Beira','Pemba'],
|
||||||
|
'Myanmar': ['Yangon','Bagan','Mandalay','Inle Lake'],
|
||||||
|
'Namibia': ['Windhoek','Swakopmund','Etosha','Sossusvlei'],
|
||||||
|
'Nepal': ['Kathmandu','Pokhara','Everest Base Camp','Chitwan','Lumbini'],
|
||||||
|
'Netherlands': ['Amsterdam','Rotterdam','The Hague','Utrecht','Delft','Eindhoven'],
|
||||||
|
'New Zealand': ['Auckland','Queenstown','Wellington','Christchurch','Rotorua','Milford Sound'],
|
||||||
|
'Nicaragua': ['Managua','Granada','León'],
|
||||||
|
'Niger': ['Niamey','Agadez'],
|
||||||
|
'Nigeria': ['Lagos','Abuja','Kano','Ibadan'],
|
||||||
|
'North Korea': ['Pyongyang'],
|
||||||
|
'North Macedonia': ['Skopje','Ohrid'],
|
||||||
|
'Norway': ['Oslo','Bergen','Tromsø','Flåm','Ålesund','Stavanger'],
|
||||||
|
'Oman': ['Muscat','Nizwa','Salalah','Wahiba Sands'],
|
||||||
|
'Pakistan': ['Karachi','Lahore','Islamabad','Peshawar','Gilgit'],
|
||||||
|
'Palestine': ['Ramallah','Bethlehem','Jericho','Hebron'],
|
||||||
|
'Panama': ['Panama City','Bocas del Toro','Boquete'],
|
||||||
|
'Papua New Guinea': ['Port Moresby'],
|
||||||
|
'Paraguay': ['Asunción','Ciudad del Este'],
|
||||||
|
'Peru': ['Lima','Cusco','Machu Picchu','Arequipa','Puno','Iquitos'],
|
||||||
|
'Philippines': ['Manila','Cebu','Palawan','Boracay','Davao','Siargao'],
|
||||||
|
'Poland': ['Warsaw','Kraków','Gdańsk','Wrocław','Poznań','Zakopane'],
|
||||||
|
'Portugal': ['Lisbon','Porto','Algarve','Sintra','Madeira','Azores','Évora'],
|
||||||
|
'Puerto Rico': ['San Juan','Ponce','Rincon'],
|
||||||
|
'Qatar': ['Doha'],
|
||||||
|
'Romania': ['Bucharest','Transylvania','Cluj-Napoca','Sibiu','Brașov','Sinaia'],
|
||||||
|
'Russia': ['Moscow','St. Petersburg','Irkutsk','Vladivostok','Sochi','Kazan','Novosibirsk'],
|
||||||
|
'Rwanda': ['Kigali','Volcanoes National Park'],
|
||||||
|
'S. Sudan': ['Juba'],
|
||||||
|
'Saint Lucia': ['Castries','Soufrière'],
|
||||||
|
'Saudi Arabia': ['Riyadh','Jeddah','Mecca','Medina','AlUla','NEOM'],
|
||||||
|
'Senegal': ['Dakar','Saint-Louis','Ziguinchor'],
|
||||||
|
'Serbia': ['Belgrade','Novi Sad','Niš'],
|
||||||
|
'Seychelles': ['Victoria','La Digue','Praslin','Mahé'],
|
||||||
|
'Sierra Leone': ['Freetown'],
|
||||||
|
'Singapore': ['Singapore'],
|
||||||
|
'Slovakia': ['Bratislava','Košice','Banská Bystrica'],
|
||||||
|
'Slovenia': ['Ljubljana','Bled','Piran','Maribor'],
|
||||||
|
'Solomon Is.': ['Honiara'],
|
||||||
|
'Somalia': ['Mogadishu'],
|
||||||
|
'South Africa': ['Cape Town','Johannesburg','Durban','Stellenbosch','Kruger','Garden Route','Pretoria'],
|
||||||
|
'South Korea': ['Seoul','Busan','Jeju','Gyeongju','Incheon','Suwon'],
|
||||||
|
'Spain': ['Barcelona','Madrid','Seville','Granada','Valencia','Bilbao','Toledo','San Sebastián','Ibiza','Mallorca'],
|
||||||
|
'Sri Lanka': ['Colombo','Kandy','Galle','Ella','Sigiriya','Mirissa'],
|
||||||
|
'Sudan': ['Khartoum','Omdurman'],
|
||||||
|
'Suriname': ['Paramaribo'],
|
||||||
|
'Sweden': ['Stockholm','Gothenburg','Malmö','Uppsala','Kiruna'],
|
||||||
|
'Switzerland': ['Zurich','Geneva','Bern','Interlaken','Lucerne','Zermatt','Lugano','Grindelwald'],
|
||||||
|
'Syria': ['Damascus','Aleppo','Palmyra'],
|
||||||
|
'São Tomé and Príncipe': ['São Tomé'],
|
||||||
|
'Taiwan': ['Taipei','Kaohsiung','Tainan','Taichung'],
|
||||||
|
'Tajikistan': ['Dushanbe','Khujand'],
|
||||||
|
'Tanzania': ['Dar es Salaam','Zanzibar','Serengeti','Arusha','Kilimanjaro'],
|
||||||
|
'Thailand': ['Bangkok','Chiang Mai','Phuket','Koh Samui','Koh Phi Phi','Ayutthaya','Pai','Krabi'],
|
||||||
|
'Timor-Leste': ['Dili'],
|
||||||
|
'Togo': ['Lomé'],
|
||||||
|
'Trinidad and Tobago': ['Port of Spain'],
|
||||||
|
'Tunisia': ['Tunis','Carthage','Sousse','Hammamet','Djerba'],
|
||||||
|
'Turkey': ['Istanbul','Cappadocia','Antalya','Bodrum','Ankara','Ephesus','Pamukkale','Trabzon'],
|
||||||
|
'Turkmenistan': ['Ashgabat','Merv'],
|
||||||
|
'Uganda': ['Kampala','Bwindi','Jinja'],
|
||||||
|
'Ukraine': ['Kyiv','Lviv','Odessa','Kharkiv'],
|
||||||
|
'United Arab Emirates': ['Dubai','Abu Dhabi','Sharjah'],
|
||||||
|
'United Kingdom': ['London','Edinburgh','Manchester','Liverpool','Oxford','Cambridge','Bath','York','Brighton','Glasgow','Dublin'],
|
||||||
|
'United States of America': ['New York','Los Angeles','Chicago','Miami','San Francisco','Las Vegas','New Orleans','Seattle','Boston','Washington D.C.','Nashville','Denver','Honolulu','Anchorage','Portland'],
|
||||||
|
'Uruguay': ['Montevideo','Punta del Este','Colonia del Sacramento'],
|
||||||
|
'Uzbekistan': ['Tashkent','Samarkand','Bukhara','Khiva'],
|
||||||
|
'Venezuela': ['Caracas','Medellín','Canaima','Los Roques'],
|
||||||
|
'Vietnam': ['Hanoi','Ho Chi Minh City','Hoi An','Da Nang','Ha Long Bay','Hue','Sapa','Phu Quoc'],
|
||||||
|
'Yemen': ["Sana'a",'Aden'],
|
||||||
|
'Zambia': ['Lusaka','Livingstone','Victoria Falls'],
|
||||||
|
'Zimbabwe': ['Harare','Bulawayo','Victoria Falls'],
|
||||||
|
};
|
||||||
15
src/lib/shared/types.js
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
/**
|
||||||
|
* @typedef {{
|
||||||
|
* id: string,
|
||||||
|
* title: string,
|
||||||
|
* date: string,
|
||||||
|
* location: { country: string, cities: string[] },
|
||||||
|
* photos: string[],
|
||||||
|
* transport: 'flight' | 'train' | 'bus' | 'car' | 'ship' | 'walk',
|
||||||
|
* tripType: 'solo' | 'friends' | 'family',
|
||||||
|
* days: number,
|
||||||
|
* memo: string
|
||||||
|
* }} JournalEntry
|
||||||
|
*/
|
||||||
|
|
||||||
|
export {};
|
||||||
@@ -1,10 +1,13 @@
|
|||||||
import { db } from '../firebase.js';
|
import { db } from '../firebase.js';
|
||||||
import { collection, doc, onSnapshot, query, orderBy, addDoc, updateDoc, deleteDoc, serverTimestamp } from 'firebase/firestore';
|
import { collection, doc, onSnapshot, query, orderBy, addDoc, updateDoc, deleteDoc, serverTimestamp } from 'firebase/firestore';
|
||||||
|
import { writable } from 'svelte/store';
|
||||||
|
|
||||||
let entries = $state([]);
|
let entries = $state([]);
|
||||||
let _uid = null;
|
let _uid = null;
|
||||||
let _unsubscribe = null;
|
let _unsubscribe = null;
|
||||||
|
|
||||||
|
export const journals = writable([]);
|
||||||
|
|
||||||
export function getEntries() {
|
export function getEntries() {
|
||||||
return entries;
|
return entries;
|
||||||
}
|
}
|
||||||
@@ -17,12 +20,14 @@ export function initEntriesListener(uid) {
|
|||||||
orderBy('createdAt', 'desc')
|
orderBy('createdAt', 'desc')
|
||||||
);
|
);
|
||||||
_unsubscribe = onSnapshot(q, (snap) => {
|
_unsubscribe = onSnapshot(q, (snap) => {
|
||||||
entries = snap.docs.map((d) => ({ id: d.id, ...d.data() }));
|
const data = snap.docs.map((d) => ({ id: d.id, ...d.data() }));
|
||||||
|
entries = data;
|
||||||
|
journals.set(data);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function addEntry(data) {
|
export async function addEntry(data) {
|
||||||
if (!_uid) return null;
|
if (!_uid) throw new Error('Not logged in');
|
||||||
const ref = await addDoc(collection(db, 'users', _uid, 'entries'), {
|
const ref = await addDoc(collection(db, 'users', _uid, 'entries'), {
|
||||||
...data,
|
...data,
|
||||||
createdAt: serverTimestamp(),
|
createdAt: serverTimestamp(),
|
||||||
|
|||||||
@@ -1,45 +1,2 @@
|
|||||||
import { writable } from 'svelte/store';
|
export { journals } from './entriesStore.svelte.js';
|
||||||
import { db } from '../firebase.js';
|
export { addEntry as addJournal } from './entriesStore.svelte.js';
|
||||||
import {
|
|
||||||
collection, onSnapshot, addDoc, updateDoc, deleteDoc, doc, serverTimestamp
|
|
||||||
} from 'firebase/firestore';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @typedef {{
|
|
||||||
* id: string,
|
|
||||||
* title: string,
|
|
||||||
* date: string,
|
|
||||||
* location: { country: string, cities: string[] },
|
|
||||||
* photos: string[],
|
|
||||||
* transport: 'flight' | 'train' | 'bus' | 'car' | 'ship' | 'walk',
|
|
||||||
* tripType: 'solo' | 'friends' | 'family',
|
|
||||||
* days: number,
|
|
||||||
* memo: string
|
|
||||||
* }} JournalEntry
|
|
||||||
*/
|
|
||||||
|
|
||||||
export const journals = writable(/** @type {JournalEntry[]} */([]));
|
|
||||||
export const journalsLoading = writable(true);
|
|
||||||
|
|
||||||
const entriesRef = collection(db, 'entries');
|
|
||||||
|
|
||||||
onSnapshot(entriesRef, (snap) => {
|
|
||||||
journals.set(snap.docs.map(d => ({ id: d.id, ...d.data() })));
|
|
||||||
journalsLoading.set(false);
|
|
||||||
});
|
|
||||||
|
|
||||||
/** @param {Omit<JournalEntry, 'id'>} entry */
|
|
||||||
export async function addJournal(entry) {
|
|
||||||
await addDoc(entriesRef, { ...entry, createdAt: serverTimestamp() });
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @param {string} id */
|
|
||||||
export async function removeJournal(id) {
|
|
||||||
await deleteDoc(doc(db, 'entries', id));
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @param {JournalEntry} updated */
|
|
||||||
export async function updateJournal(updated) {
|
|
||||||
const { id, ...data } = updated;
|
|
||||||
await updateDoc(doc(db, 'entries', id), data);
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,14 +1,15 @@
|
|||||||
<script>
|
<script>
|
||||||
import { get } from 'svelte/store';
|
import { getEntries } from '../../stores/entriesStore.svelte.js';
|
||||||
import { journals, addJournal, updateJournal } from '../stores/journalStore.js';
|
import { addEntry, updateEntry } from '../../stores/entriesStore.svelte.js';
|
||||||
import { countryNames } from '../shared/countries.js';
|
import { countryNames } from '../../shared/countries.js';
|
||||||
import SearchInput from '../shared/SearchInput.svelte';
|
import { getCitiesForCountry, ALL_CITIES } from '../../shared/cities.js';
|
||||||
|
import SearchInput from '../../shared/SearchInput.svelte';
|
||||||
import PhotoEditor from './PhotoEditor.svelte';
|
import PhotoEditor from './PhotoEditor.svelte';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* entry = null → "new entry" mode
|
* entry = null → "new entry" mode
|
||||||
* entry = {...} → "edit" mode
|
* entry = {...} → "edit" mode
|
||||||
* @type {{ entry?: import('../stores/journalStore.js').JournalEntry | null, initialCountry?: string, onBack: () => void }}
|
* @type {{ entry?: import('../shared/types.js').JournalEntry | null, initialCountry?: string, onBack: () => void }}
|
||||||
*/
|
*/
|
||||||
let { entry = null, initialCountry = '', onBack } = $props();
|
let { entry = null, initialCountry = '', onBack } = $props();
|
||||||
|
|
||||||
@@ -58,10 +59,11 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Suggest cities — when a country is selected show only cities from that country.
|
// Suggest cities — when a country is selected show only cities from that country.
|
||||||
|
let allEntries = $derived(getEntries());
|
||||||
let cityOptions = $derived(
|
let cityOptions = $derived(
|
||||||
country.trim()
|
country.trim()
|
||||||
? [...new Set(get(journals).filter(j => (j.location.country || '').toLowerCase() === country.trim().toLowerCase()).flatMap(e => e.location.cities))].sort()
|
? [...new Set([...getCitiesForCountry(country), ...allEntries.filter(j => (j.location.country || '').toLowerCase() === country.trim().toLowerCase()).flatMap(e => e.location.cities)])].sort()
|
||||||
: [...new Set(get(journals).flatMap(e => e.location.cities))].sort()
|
: [...new Set([...Object.values(ALL_CITIES).flat(), ...allEntries.flatMap(e => e.location.cities)])].sort()
|
||||||
);
|
);
|
||||||
|
|
||||||
function addCity(val) {
|
function addCity(val) {
|
||||||
@@ -89,7 +91,7 @@
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
if (isNew) {
|
if (isNew) {
|
||||||
await addJournal({
|
await addEntry({
|
||||||
title: `${cities.join(', ')}, ${country}`,
|
title: `${cities.join(', ')}, ${country}`,
|
||||||
date,
|
date,
|
||||||
days: Number(days),
|
days: Number(days),
|
||||||
@@ -100,8 +102,7 @@
|
|||||||
location: { cities, country },
|
location: { cities, country },
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
await updateJournal({
|
await updateEntry(entry.id, {
|
||||||
...entry,
|
|
||||||
date,
|
date,
|
||||||
days: Number(days),
|
days: Number(days),
|
||||||
tripType,
|
tripType,
|
||||||
@@ -113,7 +114,7 @@
|
|||||||
}
|
}
|
||||||
onBack();
|
onBack();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showToast('Failed to save. Please try again.');
|
console.error('Save failed:', err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -240,7 +241,7 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
padding: 0 20px;
|
padding: 0 20px;
|
||||||
height: 52px;
|
height: 60px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
border-bottom: 1px solid var(--border);
|
border-bottom: 1px solid var(--border);
|
||||||
background: var(--bg);
|
background: var(--bg);
|
||||||
@@ -255,8 +256,8 @@
|
|||||||
.topbar-right { justify-content: flex-end; }
|
.topbar-right { justify-content: flex-end; }
|
||||||
|
|
||||||
.topbar-title {
|
.topbar-title {
|
||||||
font-size: 14px;
|
font-size: 16px;
|
||||||
font-weight: 400;
|
font-weight: 500;
|
||||||
color: var(--text-h);
|
color: var(--text-h);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -265,13 +266,13 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 6px;
|
gap: 6px;
|
||||||
font-family: var(--sans);
|
font-family: var(--sans);
|
||||||
font-size: 13px;
|
font-size: 15px;
|
||||||
font-weight: 300;
|
font-weight: 400;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
background: none;
|
background: none;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
border-radius: 8px;
|
border-radius: 10px;
|
||||||
padding: 6px 12px;
|
padding: 8px 14px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: background 0.15s, color 0.15s, border-color 0.15s;
|
transition: background 0.15s, color 0.15s, border-color 0.15s;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
@@ -1,15 +1,15 @@
|
|||||||
<script>
|
<script>
|
||||||
import { removeJournal } from '../stores/journalStore.js';
|
import { removeEntry } from '../../stores/entriesStore.svelte.js';
|
||||||
import { flagEmoji } from '../shared/countries.js';
|
import { flagEmoji } from '../../shared/countries.js';
|
||||||
import DeleteConfirm from './DeleteConfirm.svelte';
|
import DeleteConfirm from './DeleteConfirm.svelte';
|
||||||
|
|
||||||
/** @type {{ entry: import('../stores/journalStore.js').JournalEntry, onBack: () => void, onEdit: () => void }} */
|
/** @type {{ entry: import('../shared/types.js').JournalEntry, onBack: () => void, onEdit: () => void }} */
|
||||||
let { entry, onBack, onEdit } = $props();
|
let { entry, onBack, onEdit } = $props();
|
||||||
|
|
||||||
let showDeleteConfirm = $state(false);
|
let showDeleteConfirm = $state(false);
|
||||||
|
|
||||||
function handleDelete() {
|
function handleDelete() {
|
||||||
removeJournal(entry.id);
|
removeEntry(entry.id);
|
||||||
onBack();
|
onBack();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -149,7 +149,7 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
padding: 0 20px;
|
padding: 0 20px;
|
||||||
height: 52px;
|
height: 60px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
background: var(--bg);
|
background: var(--bg);
|
||||||
border-bottom: 1px solid var(--border);
|
border-bottom: 1px solid var(--border);
|
||||||
@@ -196,13 +196,13 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 6px;
|
gap: 6px;
|
||||||
font-family: var(--sans);
|
font-family: var(--sans);
|
||||||
font-size: 13px;
|
font-size: 15px;
|
||||||
font-weight: 300;
|
font-weight: 400;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
background: none;
|
background: none;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
border-radius: 8px;
|
border-radius: 10px;
|
||||||
padding: 6px 12px;
|
padding: 8px 14px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: background 0.15s, color 0.15s, border-color 0.15s;
|
transition: background 0.15s, color 0.15s, border-color 0.15s;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
/** @type {{ entries: import('../stores/journalStore.js').JournalEntry[] }} */
|
/** @type {{ entries: import('../shared/types.js').JournalEntry[] }} */
|
||||||
let { entries } = $props();
|
let { entries } = $props();
|
||||||
|
|
||||||
let stats = $derived.by(() => {
|
let stats = $derived.by(() => {
|
||||||
@@ -1,17 +1,26 @@
|
|||||||
<script>
|
<script>
|
||||||
|
import { journals, addJournal } from '../../stores/journalStore.js';
|
||||||
import { get } from 'svelte/store';
|
import { get } from 'svelte/store';
|
||||||
import { journals, addJournal } from '../stores/journalStore.js';
|
import { flashCountry } from '../../layout/selection.svelte.js';
|
||||||
import { countryNames } from '../shared/countries.js';
|
import { countryNames } from '../../shared/countries.js';
|
||||||
import SearchInput from '../shared/SearchInput.svelte';
|
import { countryCities } from '../../shared/countryCities.js';
|
||||||
|
import SearchInput from '../../shared/SearchInput.svelte';
|
||||||
import PhotoEditor from './PhotoEditor.svelte';
|
import PhotoEditor from './PhotoEditor.svelte';
|
||||||
import airplaneImg from '../../assets/airplane.png';
|
import airplaneImg from '../../../assets/airplane.png';
|
||||||
import trainImg from '../../assets/train.png';
|
import trainImg from '../../../assets/train.png';
|
||||||
import busImg from '../../assets/bus.png';
|
import busImg from '../../../assets/bus.png';
|
||||||
import carImg from '../../assets/car.png';
|
import carImg from '../../../assets/car.png';
|
||||||
import shipImg from '../../assets/ship.png';
|
import shipImg from '../../../assets/ship.png';
|
||||||
import walkImg from '../../assets/walk.png';
|
import walkImg from '../../../assets/walk.png';
|
||||||
|
|
||||||
let { initialCountry = '', onBack } = $props();
|
let { initialCountry = '', onBack, onSaved = onBack } = $props();
|
||||||
|
|
||||||
|
// ── Journal store (reactive) ────────────────────────────────────────
|
||||||
|
let journalEntries = $state(get(journals));
|
||||||
|
$effect(() => {
|
||||||
|
const unsub = journals.subscribe(v => { journalEntries = v; });
|
||||||
|
return unsub;
|
||||||
|
});
|
||||||
|
|
||||||
// ── Fields ─────────────────────────────────────────────────────────
|
// ── Fields ─────────────────────────────────────────────────────────
|
||||||
let cities = $state([]);
|
let cities = $state([]);
|
||||||
@@ -55,8 +64,11 @@
|
|||||||
// otherwise show all known cities.
|
// otherwise show all known cities.
|
||||||
let cityOptions = $derived(
|
let cityOptions = $derived(
|
||||||
country.trim()
|
country.trim()
|
||||||
? [...new Set(get(journals).filter(j => (j.location.country || '').toLowerCase() === country.trim().toLowerCase()).flatMap(e => e.location.cities))].sort()
|
? [...new Set([
|
||||||
: [...new Set(get(journals).flatMap(e => e.location.cities))].sort()
|
...(countryCities[country.trim()] ?? []),
|
||||||
|
...journalEntries.filter(j => (j.location?.country || '').toLowerCase() === country.trim().toLowerCase()).flatMap(e => e.location?.cities ?? []),
|
||||||
|
])]
|
||||||
|
: []
|
||||||
);
|
);
|
||||||
|
|
||||||
function addCity(val) {
|
function addCity(val) {
|
||||||
@@ -106,9 +118,11 @@
|
|||||||
|
|
||||||
// ── Save ───────────────────────────────────────────────────────────
|
// ── Save ───────────────────────────────────────────────────────────
|
||||||
let saving = $state(false);
|
let saving = $state(false);
|
||||||
|
let saveError = $state('');
|
||||||
|
|
||||||
async function save() {
|
async function save() {
|
||||||
saving = true;
|
saving = true;
|
||||||
|
saveError = '';
|
||||||
const memo = questions
|
const memo = questions
|
||||||
.map((q, i) => answers[i].trim() ? `Q: ${q.split('\n')[0]}\nA: ${answers[i].trim()}` : '')
|
.map((q, i) => answers[i].trim() ? `Q: ${q.split('\n')[0]}\nA: ${answers[i].trim()}` : '')
|
||||||
.filter(Boolean)
|
.filter(Boolean)
|
||||||
@@ -124,9 +138,11 @@
|
|||||||
photos,
|
photos,
|
||||||
location: { cities, country },
|
location: { cities, country },
|
||||||
});
|
});
|
||||||
onBack();
|
flashCountry(country);
|
||||||
} catch {
|
onSaved();
|
||||||
|
} catch (e) {
|
||||||
saving = false;
|
saving = false;
|
||||||
|
saveError = e?.message ?? 'Failed to save. Please try again.';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -153,6 +169,7 @@
|
|||||||
<button class="save-btn" onclick={save} disabled={saving}>
|
<button class="save-btn" onclick={save} disabled={saving}>
|
||||||
{saving ? 'Saving…' : 'Save trip'}
|
{saving ? 'Saving…' : 'Save trip'}
|
||||||
</button>
|
</button>
|
||||||
|
{#if saveError}<span class="save-err">{saveError}</span>{/if}
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
@@ -166,13 +183,13 @@
|
|||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="field">
|
<div class="field">
|
||||||
<label class="label" for="nc-country">Country <span class="req">*</span></label>
|
<label class="label" for="nc-country">Where did you go? <span class="req">*</span></label>
|
||||||
<SearchInput id="nc-country" bind:value={country} options={countryNames} />
|
<SearchInput id="nc-country" bind:value={country} options={countryNames} />
|
||||||
{#if errors.country}<span class="ferr">{errors.country}</span>{/if}
|
{#if errors.country}<span class="ferr">{errors.country}</span>{/if}
|
||||||
</div>
|
</div>
|
||||||
<div class="field">
|
<div class="field">
|
||||||
<label class="label" for="nc-city">Cities <span class="req">*</span></label>
|
<label class="label" for="nc-city">Which cities did you visit? <span class="req">*</span></label>
|
||||||
<SearchInput id="nc-city" bind:value={cityInput} options={cityOptions} onselect={addCity} />
|
<SearchInput id="nc-city" bind:value={cityInput} options={cityOptions} onselect={addCity} onblurcommit={addCity} />
|
||||||
{#if errors.cities}<span class="ferr">{errors.cities}</span>{/if}
|
{#if errors.cities}<span class="ferr">{errors.cities}</span>{/if}
|
||||||
{#if cities.length > 0}
|
{#if cities.length > 0}
|
||||||
<div class="tags">
|
<div class="tags">
|
||||||
@@ -186,19 +203,19 @@
|
|||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="field">
|
<div class="field">
|
||||||
<label class="label" for="nc-date">Date <span class="req">*</span></label>
|
<label class="label" for="nc-date">When did you travel? <span class="req">*</span></label>
|
||||||
<input id="nc-date" class="input" type="date" bind:value={date} />
|
<input id="nc-date" class="input" type="date" bind:value={date} />
|
||||||
{#if errors.date}<span class="ferr">{errors.date}</span>{/if}
|
{#if errors.date}<span class="ferr">{errors.date}</span>{/if}
|
||||||
</div>
|
</div>
|
||||||
<div class="field">
|
<div class="field">
|
||||||
<label class="label" for="nc-days">Days <span class="req">*</span></label>
|
<label class="label" for="nc-days">How long was the trip? <span class="req">*</span></label>
|
||||||
<input id="nc-days" class="input" type="number" min="1" bind:value={days} />
|
<input id="nc-days" class="input" type="number" min="1" bind:value={days} />
|
||||||
{#if errors.days}<span class="ferr">{errors.days}</span>{/if}
|
{#if errors.days}<span class="ferr">{errors.days}</span>{/if}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="field">
|
<div class="field">
|
||||||
<label class="label">Trip type <span class="req">*</span></label>
|
<label class="label">Who did you travel with? <span class="req">*</span></label>
|
||||||
<div class="toggle-row">
|
<div class="toggle-row">
|
||||||
{#each ['solo','friends','family'] as t}
|
{#each ['solo','friends','family'] as t}
|
||||||
<label class="toggle-opt" class:active={tripType === t}>
|
<label class="toggle-opt" class:active={tripType === t}>
|
||||||
@@ -326,6 +343,7 @@
|
|||||||
}
|
}
|
||||||
.save-btn:hover { background: var(--accent-dark); border-color: var(--accent-dark); }
|
.save-btn:hover { background: var(--accent-dark); border-color: var(--accent-dark); }
|
||||||
.save-btn:disabled { opacity: 0.6; cursor: not-allowed; }
|
.save-btn:disabled { opacity: 0.6; cursor: not-allowed; }
|
||||||
|
.save-err { font-size: 12px; color: #ef4444; margin-top: 4px; display: block; text-align: right; }
|
||||||
|
|
||||||
/* scroll + form */
|
/* scroll + form */
|
||||||
.scroll { flex: 1; overflow-y: auto; }
|
.scroll { flex: 1; overflow-y: auto; }
|
||||||
@@ -369,6 +387,25 @@
|
|||||||
|
|
||||||
.ferr { font-size: 11px; color: #dc2626; }
|
.ferr { font-size: 11px; color: #dc2626; }
|
||||||
|
|
||||||
|
.combo-select, .city-text {
|
||||||
|
font-family: var(--sans);
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 300;
|
||||||
|
color: var(--text-h);
|
||||||
|
background: var(--bg-subtle);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 8px 12px;
|
||||||
|
outline: none;
|
||||||
|
transition: border-color 0.15s;
|
||||||
|
width: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.combo-select:focus, .city-text:focus { border-color: var(--accent-border); }
|
||||||
|
.combo-select { margin-bottom: 6px; cursor: pointer; }
|
||||||
|
.city-text { margin-top: 0; }
|
||||||
|
|
||||||
.input {
|
.input {
|
||||||
font-family: var(--sans);
|
font-family: var(--sans);
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import { storage } from '../firebase.js';
|
import { storage } from '../../firebase.js';
|
||||||
import { ref, uploadBytes, getDownloadURL } from 'firebase/storage';
|
import { ref, uploadBytes, getDownloadURL } from 'firebase/storage';
|
||||||
|
|
||||||
/** @type {{ photos: string[], onchange: (photos: string[]) => void }} */
|
/** @type {{ photos: string[], onchange: (photos: string[]) => void }} */
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
<script>
|
<script>
|
||||||
import { toPng } from 'html-to-image';
|
import { toPng } from 'html-to-image';
|
||||||
|
|
||||||
/** @type {{ entries: import('../stores/journalStore.js').JournalEntry[], onClose: () => void }} */
|
/** @type {{ entries: import('../shared/types.js').JournalEntry[], onClose: () => void }} */
|
||||||
let { entries, onClose } = $props();
|
let { entries, onClose } = $props();
|
||||||
|
|
||||||
let cardEl = $state(null);
|
let cardEl = $state(null);
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
/** @type {{ entries: import('../stores/journalStore.js').JournalEntry[], onClick: () => void }} */
|
/** @type {{ entries: import('../shared/types.js').JournalEntry[], onClick: () => void }} */
|
||||||
let { entries, onClick } = $props();
|
let { entries, onClick } = $props();
|
||||||
|
|
||||||
const continentMap = {
|
const continentMap = {
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
<script>
|
<script>
|
||||||
import { flagEmoji } from '../shared/countries.js';
|
import { flagEmoji } from '../../shared/countries.js';
|
||||||
|
|
||||||
/** @type {{ entry: import('../stores/journalStore.js').JournalEntry, onClick: () => void }} */
|
/** @type {{ entry: import('../shared/types.js').JournalEntry, onClick: () => void }} */
|
||||||
let { entry, onClick } = $props();
|
let { entry, onClick } = $props();
|
||||||
|
|
||||||
function formatDate(/** @type {string} */ iso) {
|
function formatDate(/** @type {string} */ iso) {
|
||||||
@@ -1,15 +1,14 @@
|
|||||||
<script>
|
<script>
|
||||||
import { get } from 'svelte/store';
|
import { getEntries } from '../../stores/entriesStore.svelte.js';
|
||||||
import { journals } from '../stores/journalStore.js';
|
|
||||||
import TimelineToolbar from './TimelineToolbar.svelte';
|
import TimelineToolbar from './TimelineToolbar.svelte';
|
||||||
import TimelineCard from './TimelineCard.svelte';
|
import TimelineCard from './TimelineCard.svelte';
|
||||||
import JournalDetail from './JournalDetail.svelte';
|
import JournalDetail from '../detail/JournalDetail.svelte';
|
||||||
import EditForm from './EditForm.svelte';
|
import EditForm from '../detail/EditForm.svelte';
|
||||||
import NewEntryForm from './NewEntryForm.svelte';
|
import NewEntryForm from '../detail/NewEntryForm.svelte';
|
||||||
import ShareCard from './ShareCard.svelte';
|
import ShareCard from './ShareCard.svelte';
|
||||||
import SharePreview from './SharePreview.svelte';
|
import SharePreview from './SharePreview.svelte';
|
||||||
|
|
||||||
let { onDetailChange = () => {}, pendingCountry = '', onNewEntryClear = () => {} } = $props();
|
let { onDetailChange = () => {}, pendingCountry = '', onNewEntryClear = () => {}, onGoToMap = () => {} } = $props();
|
||||||
let selectedId = $state(/** @type {string|null} */(null));
|
let selectedId = $state(/** @type {string|null} */(null));
|
||||||
let view = $state(/** @type {'list'|'detail'|'edit'|'new'} */('list'));
|
let view = $state(/** @type {'list'|'detail'|'edit'|'new'} */('list'));
|
||||||
let showShare = $state(false);
|
let showShare = $state(false);
|
||||||
@@ -26,11 +25,7 @@
|
|||||||
});
|
});
|
||||||
let selected = $derived(selectedId ? (entries.find(e => e.id === selectedId) ?? null) : null);
|
let selected = $derived(selectedId ? (entries.find(e => e.id === selectedId) ?? null) : null);
|
||||||
|
|
||||||
let entries = $state(get(journals));
|
let entries = $derived(getEntries());
|
||||||
$effect(() => {
|
|
||||||
const unsub = journals.subscribe((v) => { entries = v; });
|
|
||||||
return unsub;
|
|
||||||
});
|
|
||||||
|
|
||||||
let sortKey = $state('date-desc');
|
let sortKey = $state('date-desc');
|
||||||
|
|
||||||
@@ -55,7 +50,7 @@
|
|||||||
|
|
||||||
{#if view === 'new'}
|
{#if view === 'new'}
|
||||||
<div class="detail-scroll">
|
<div class="detail-scroll">
|
||||||
<NewEntryForm initialCountry={newEntryCountry} onBack={() => { view = 'list'; newEntryCountry = ''; onDetailChange(false); }} />
|
<NewEntryForm initialCountry={newEntryCountry} onBack={() => { view = 'list'; newEntryCountry = ''; onDetailChange(false); }} onSaved={() => { onGoToMap(); }} />
|
||||||
</div>
|
</div>
|
||||||
{:else if view === 'edit' && selected}
|
{:else if view === 'edit' && selected}
|
||||||
<div class="detail-scroll">
|
<div class="detail-scroll">
|
||||||
@@ -70,53 +65,55 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
<div class="right-panel">
|
<div class="list-view">
|
||||||
<div class="center-col">
|
<div class="page-header">
|
||||||
<div class="page-header">
|
<h1 class="page-title">My Journey</h1>
|
||||||
<h1 class="page-title">My Journey</h1>
|
<button class="new-btn" onclick={() => { view = 'new'; }}>
|
||||||
<button class="new-btn" onclick={() => { view = 'new'; }}>
|
<svg width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round">
|
||||||
<svg width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.2" stroke-linecap="round">
|
<path d="M12 5v14M5 12h14"/>
|
||||||
<path d="M12 5v14M5 12h14"/>
|
</svg>
|
||||||
</svg>
|
Add trip
|
||||||
Add trip
|
</button>
|
||||||
</button>
|
</div>
|
||||||
|
|
||||||
|
<div class="two-col">
|
||||||
|
<div class="left-col">
|
||||||
|
<TimelineToolbar {sortKey} onSort={(k) => (sortKey = k)} />
|
||||||
|
|
||||||
|
{#if sortedEntries.length === 0}
|
||||||
|
<p class="empty">No journal entries yet.</p>
|
||||||
|
{:else}
|
||||||
|
<div class="sort-row">
|
||||||
|
<span class="sort-label">Sort by</span>
|
||||||
|
<select class="sort-select" onchange={(e) => (sortKey = e.currentTarget.value)}>
|
||||||
|
<option value="date-desc" selected={sortKey === 'date-desc'}>Newest first</option>
|
||||||
|
<option value="date-asc" selected={sortKey === 'date-asc'}>Oldest first</option>
|
||||||
|
<option value="country-asc" selected={sortKey === 'country-asc'}>Country A → Z</option>
|
||||||
|
<option value="country-desc" selected={sortKey === 'country-desc'}>Country Z → A</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<ol class="v-list">
|
||||||
|
{#each sortedEntries as entry, i (entry.id)}
|
||||||
|
{#if i === 0 || getYear(entry.date) !== getYear(sortedEntries[i - 1].date)}
|
||||||
|
<li class="year-marker" aria-hidden="true">
|
||||||
|
<span class="year-label">{getYear(entry.date)}</span>
|
||||||
|
</li>
|
||||||
|
{/if}
|
||||||
|
<TimelineCard {entry} onClick={() => { selectedId = entry.id; view = 'detail'; onDetailChange(true); }} />
|
||||||
|
{/each}
|
||||||
|
</ol>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<footer class="page-footer">
|
||||||
|
{sortedEntries.length} {sortedEntries.length === 1 ? 'trip' : 'trips'}
|
||||||
|
</footer>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#if sortedEntries.length > 0}
|
{#if sortedEntries.length > 0}
|
||||||
<div class="share-row">
|
<div class="right-col">
|
||||||
<SharePreview entries={sortedEntries} onClick={() => (showShare = true)} />
|
<SharePreview entries={sortedEntries} onClick={() => (showShare = true)} />
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<TimelineToolbar {sortKey} onSort={(k) => (sortKey = k)} />
|
|
||||||
|
|
||||||
{#if sortedEntries.length === 0}
|
|
||||||
<p class="empty">No journal entries yet.</p>
|
|
||||||
{:else}
|
|
||||||
<div class="sort-row">
|
|
||||||
<span class="sort-label">Sort by</span>
|
|
||||||
<select class="sort-select" onchange={(e) => (sortKey = e.currentTarget.value)}>
|
|
||||||
<option value="date-desc" selected={sortKey === 'date-desc'}>Newest first</option>
|
|
||||||
<option value="date-asc" selected={sortKey === 'date-asc'}>Oldest first</option>
|
|
||||||
<option value="country-asc" selected={sortKey === 'country-asc'}>Country A → Z</option>
|
|
||||||
<option value="country-desc" selected={sortKey === 'country-desc'}>Country Z → A</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
<ol class="v-list">
|
|
||||||
{#each sortedEntries as entry, i (entry.id)}
|
|
||||||
{#if i === 0 || getYear(entry.date) !== getYear(sortedEntries[i - 1].date)}
|
|
||||||
<li class="year-marker" aria-hidden="true">
|
|
||||||
<span class="year-label">{getYear(entry.date)}</span>
|
|
||||||
</li>
|
|
||||||
{/if}
|
|
||||||
<TimelineCard {entry} onClick={() => { selectedId = entry.id; view = 'detail'; onDetailChange(true); }} />
|
|
||||||
{/each}
|
|
||||||
</ol>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<footer class="page-footer">
|
|
||||||
{sortedEntries.length} {sortedEntries.length === 1 ? 'trip' : 'trips'}
|
|
||||||
</footer>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
@@ -137,31 +134,51 @@
|
|||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ── Right panel ── */
|
/* ── List view wrapper (scrollable) ── */
|
||||||
.right-panel {
|
.list-view {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: 48px 0 80px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-header,
|
||||||
|
.two-col {
|
||||||
|
max-width: 960px;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
padding-left: 48px;
|
||||||
|
padding-right: 48px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ── Two-column below header ── */
|
||||||
|
.two-col {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
gap: 32px;
|
||||||
|
align-items: flex-start;
|
||||||
|
margin-top: 24px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-col {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
overflow-y: auto;
|
|
||||||
background: var(--bg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ── Centered single column ── */
|
.right-col {
|
||||||
.center-col {
|
width: 260px;
|
||||||
max-width: 680px;
|
flex-shrink: 0;
|
||||||
width: 100%;
|
position: sticky;
|
||||||
margin: 0 auto;
|
top: 0;
|
||||||
padding: 48px 48px 80px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.share-row {
|
@media (max-width: 900px) {
|
||||||
margin-bottom: 24px;
|
.right-col { display: none; }
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 760px) {
|
@media (max-width: 760px) {
|
||||||
.center-col {
|
.list-view { padding: 32px 24px 60px; }
|
||||||
padding: 32px 24px 60px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ── Detail view ── */
|
/* ── Detail view ── */
|
||||||
@@ -275,38 +292,4 @@
|
|||||||
}
|
}
|
||||||
.new-btn:hover { background: var(--accent-dark); border-color: var(--accent-dark); }
|
.new-btn:hover { background: var(--accent-dark); border-color: var(--accent-dark); }
|
||||||
|
|
||||||
.share-nudge {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
width: 100%;
|
|
||||||
padding: 10px 14px;
|
|
||||||
margin-bottom: 12px;
|
|
||||||
border-radius: 10px;
|
|
||||||
border: 1px dashed var(--border-bright);
|
|
||||||
background: var(--bg-subtle);
|
|
||||||
cursor: pointer;
|
|
||||||
transition: border-color 0.15s, background 0.15s;
|
|
||||||
font-family: var(--sans);
|
|
||||||
text-align: left;
|
|
||||||
}
|
|
||||||
.share-nudge:hover {
|
|
||||||
border-color: var(--accent-light);
|
|
||||||
background: var(--accent-bg);
|
|
||||||
}
|
|
||||||
.nudge-left {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 7px;
|
|
||||||
font-size: 13px;
|
|
||||||
font-weight: 400;
|
|
||||||
color: var(--text-h);
|
|
||||||
}
|
|
||||||
.nudge-right {
|
|
||||||
font-size: 11px;
|
|
||||||
font-weight: 300;
|
|
||||||
color: var(--text-sub);
|
|
||||||
letter-spacing: 0.02em;
|
|
||||||
}
|
|
||||||
.share-nudge:hover .nudge-right { color: var(--accent); }
|
|
||||||
</style>
|
</style>
|
||||||
@@ -3,25 +3,32 @@
|
|||||||
import * as d3 from 'd3';
|
import * as d3 from 'd3';
|
||||||
import { feature } from 'topojson-client';
|
import { feature } from 'topojson-client';
|
||||||
import worldData from 'world-atlas/countries-50m.json';
|
import worldData from 'world-atlas/countries-50m.json';
|
||||||
|
import { get } from 'svelte/store';
|
||||||
|
import { journals } from '../stores/journalStore.js';
|
||||||
|
import airplaneImg from '../../assets/airplane.png';
|
||||||
|
import trainImg from '../../assets/train.png';
|
||||||
|
import busImg from '../../assets/bus.png';
|
||||||
|
import carImg from '../../assets/car.png';
|
||||||
|
import shipImg from '../../assets/ship.png';
|
||||||
|
import walkImg from '../../assets/walk.png';
|
||||||
|
|
||||||
let { onclose, onprogress } = $props();
|
let { onclose, onprogress, mode = 'map', onmodechange } = $props();
|
||||||
|
|
||||||
const HOME_CODE = '203';
|
const HOME_CODE = '203';
|
||||||
|
|
||||||
const MOCK_TRIPS = [
|
const TRANSPORT_IMG = {
|
||||||
{ countryName: 'Japan', countryCode: '392', date: '2024-03-15', city: 'Tokyo' },
|
flight: airplaneImg,
|
||||||
{ countryName: 'France', countryCode: '191', date: '2024-06-20', city: 'Paris' },
|
train: trainImg,
|
||||||
{ countryName: 'Spain', countryCode: '724', date: '2024-09-10', city: 'Barcelona' },
|
bus: busImg,
|
||||||
{ countryName: 'United States', countryCode: '840', date: '2025-01-05', city: 'New York' },
|
car: carImg,
|
||||||
{ countryName: 'Thailand', countryCode: '764', date: '2025-04-18', city: 'Bangkok' },
|
ship: shipImg,
|
||||||
{ countryName: 'Australia', countryCode: '036', date: '2025-08-22', city: 'Sydney' },
|
walk: walkImg,
|
||||||
];
|
};
|
||||||
|
const PLANE_SIZE = 28;
|
||||||
|
|
||||||
const HOME_COLOR = '#8b5cf6';
|
const HOME_COLOR = '#8b5cf6';
|
||||||
const VISITED_COLOR = '#22c55e';
|
const VISITED_COLOR = '#22c55e';
|
||||||
const ARC_COLOR = '#000000';
|
const ARC_COLOR = '#666666';
|
||||||
const PLANE_COLOR = '#7c3aed';
|
|
||||||
const PLANE_PATH = 'M14,0 L4,-3 L0,-7 L-3,-5 L0,-2 L-5,-1 L-9,-5 L-11,-4 L-7,0 L-11,4 L-9,5 L-5,1 L0,2 L-3,5 L0,7 L4,3 Z';
|
|
||||||
const UNVISITED = '#ffffff';
|
const UNVISITED = '#ffffff';
|
||||||
|
|
||||||
const TERRITORY_PARENT = {
|
const TERRITORY_PARENT = {
|
||||||
@@ -35,357 +42,386 @@
|
|||||||
'850': '840', '876': '250',
|
'850': '840', '876': '250',
|
||||||
};
|
};
|
||||||
|
|
||||||
function effId(d) {
|
function effId(d) { return TERRITORY_PARENT[d.id] || d.id; }
|
||||||
return TERRITORY_PARENT[d.id] || d.id;
|
|
||||||
}
|
|
||||||
|
|
||||||
let frameEl;
|
let frameEl;
|
||||||
let svg, g, pathFn, projection;
|
let svg, gBase, gCountries, gAnim, pathFn, projection;
|
||||||
let countryPaths;
|
let countryPaths;
|
||||||
let homeFeature;
|
let homeFeature;
|
||||||
let featuresById = {};
|
let featuresById = {};
|
||||||
|
let countriesData = [];
|
||||||
let isCancelled = false;
|
let isCancelled = false;
|
||||||
let isPlaying = $state(false);
|
let isPlaying = $state(false);
|
||||||
let isFinished = $state(false);
|
let isFinished = $state(false);
|
||||||
|
let visitedCodes = new Set();
|
||||||
|
let animId = 0;
|
||||||
|
let currentDateLabel = $state('');
|
||||||
|
|
||||||
function fitProjection(proj, w, h) {
|
function formatDateLabel(dateStr) {
|
||||||
proj.fitSize([w, h], { type: 'Sphere' });
|
const d = new Date(dateStr);
|
||||||
const s = proj.scale() * 1.5;
|
const months = ['January','February','March','April','May','June','July','August','September','October','November','December'];
|
||||||
proj.scale(s).translate([w / 2, h * 0.70]);
|
return `${months[d.getMonth()]} ${d.getFullYear()}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
function computeArc(p1, p2) {
|
function computeArc(p1, p2) {
|
||||||
const interp = d3.geoInterpolate(p1, p2);
|
const interp = d3.geoInterpolate(p1, p2);
|
||||||
const steps = 80;
|
|
||||||
const raw = [];
|
const raw = [];
|
||||||
|
for (let i = 0; i <= 80; i++) {
|
||||||
for (let i = 0; i <= steps; i++) {
|
const t = i / 80;
|
||||||
const t = i / steps;
|
const pt = projection(interp(t));
|
||||||
const geo = interp(t);
|
|
||||||
const pt = projection(geo);
|
|
||||||
if (!pt) continue;
|
if (!pt) continue;
|
||||||
raw.push({ t, x: pt[0], y: pt[1] });
|
raw.push({ t, x: pt[0], y: pt[1] });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (raw.length < 2) return [];
|
if (raw.length < 2) return [];
|
||||||
|
const first = raw[0], last = raw[raw.length - 1];
|
||||||
const first = raw[0];
|
const dist = Math.sqrt((last.x-first.x)**2 + (last.y-first.y)**2);
|
||||||
const last = raw[raw.length - 1];
|
|
||||||
const dx = last.x - first.x;
|
|
||||||
const dy = last.y - first.y;
|
|
||||||
const dist = Math.sqrt(dx * dx + dy * dy);
|
|
||||||
const arcH = Math.max(40, Math.min(200, dist * 0.22));
|
const arcH = Math.max(40, Math.min(200, dist * 0.22));
|
||||||
|
|
||||||
return raw.map(p => [p.x, p.y - arcH * Math.sin(Math.PI * p.t)]);
|
return raw.map(p => [p.x, p.y - arcH * Math.sin(Math.PI * p.t)]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function getAngleAtLength(node, len) {
|
function planeTransform(x, y, angle, flip) {
|
||||||
const d = 0.5;
|
return `translate(${x},${y}) rotate(${angle})${flip ? ' scale(1,-1)' : ''}`;
|
||||||
const total = node.getTotalLength();
|
|
||||||
const p1 = node.getPointAtLength(Math.max(0, len - d));
|
|
||||||
const p2 = node.getPointAtLength(Math.min(total, len + d));
|
|
||||||
return Math.atan2(p2.y - p1.y, p2.x - p1.x) * 180 / Math.PI;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function animateStroke(pathEl, tipEl, startOffset, endOffset, duration) {
|
function delay(ms) {
|
||||||
return new Promise((resolve) => {
|
return new Promise(resolve => { if (isCancelled) { resolve(); return; } setTimeout(resolve, ms); });
|
||||||
const node = pathEl.node();
|
}
|
||||||
if (!node) { resolve(); return; }
|
|
||||||
const totalLength = node.getTotalLength();
|
|
||||||
|
|
||||||
if (totalLength === 0) { resolve(); return; }
|
function setupProjection(width, height) {
|
||||||
|
if (mode === 'map') {
|
||||||
|
projection = d3.geoMercator();
|
||||||
|
projection.fitSize([width, height], { type: 'Sphere' });
|
||||||
|
const s = projection.scale() * 1.5;
|
||||||
|
projection.scale(s).translate([width / 2, height * 0.70]);
|
||||||
|
} else {
|
||||||
|
const size = Math.min(width, height) * 0.92;
|
||||||
|
projection = d3.geoOrthographic()
|
||||||
|
.rotate([0, 0])
|
||||||
|
.fitSize([size, size], { type: 'Sphere' })
|
||||||
|
.translate([width / 2, height / 2]);
|
||||||
|
}
|
||||||
|
pathFn = d3.geoPath().projection(projection);
|
||||||
|
}
|
||||||
|
|
||||||
d3.timer(elapsed => {
|
function renderMap() {
|
||||||
if (isCancelled) { resolve(); return true; }
|
gBase.selectAll('*').remove();
|
||||||
|
gCountries.selectAll('*').remove();
|
||||||
|
gAnim.selectAll('*').remove();
|
||||||
|
|
||||||
|
const fillFn = d => {
|
||||||
|
const id = effId(d);
|
||||||
|
if (id === HOME_CODE) return visitedCodes.has(id) ? VISITED_COLOR : HOME_COLOR;
|
||||||
|
if (visitedCodes.has(id)) return VISITED_COLOR;
|
||||||
|
return UNVISITED;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (mode === 'globe') {
|
||||||
|
gBase.append('path').attr('class', 'sphere').datum({ type: 'Sphere' })
|
||||||
|
.attr('d', pathFn).attr('fill', '#a4c8e0').attr('stroke', '#8b9bb0').attr('stroke-width', 1.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
countryPaths = gCountries.selectAll('path')
|
||||||
|
.data(countriesData, d => effId(d)).join('path')
|
||||||
|
.attr('d', pathFn).attr('fill', fillFn)
|
||||||
|
.attr('stroke', mode === 'globe' ? '#4a6a8c' : '#d4d4d4')
|
||||||
|
.attr('stroke-width', mode === 'globe' ? 0.3 : 0.5);
|
||||||
|
|
||||||
|
if (mode === 'map') renderMicrostates();
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderMicrostates() {
|
||||||
|
gBase.selectAll('.micro-state-j').remove();
|
||||||
|
const fillFn = d => {
|
||||||
|
const id = effId(d);
|
||||||
|
if (id === HOME_CODE) return visitedCodes.has(id) ? VISITED_COLOR : HOME_COLOR;
|
||||||
|
if (visitedCodes.has(id)) return VISITED_COLOR;
|
||||||
|
return UNVISITED;
|
||||||
|
};
|
||||||
|
countryPaths.each(function(d) {
|
||||||
|
if (effId(d) !== d.id) return;
|
||||||
|
const { width, height } = this.getBBox();
|
||||||
|
if (width < 4 && height < 4) {
|
||||||
|
const [cx, cy] = pathFn.centroid(d);
|
||||||
|
gBase.append('circle').attr('class', 'micro-state-j').datum(d)
|
||||||
|
.attr('cx', cx).attr('cy', cy).attr('r', 2).attr('fill', fillFn(d))
|
||||||
|
.attr('stroke', '#94a3b8').attr('stroke-width', 0.5);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function redrawBase() {
|
||||||
|
countryPaths.attr('d', pathFn);
|
||||||
|
if (mode === 'globe') gBase.select('.sphere').attr('d', pathFn);
|
||||||
|
}
|
||||||
|
|
||||||
|
function rotateGlobeTo(lon, lat, duration = 1500) {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
if (isCancelled) { resolve(); return; }
|
||||||
|
const current = projection.rotate();
|
||||||
|
const interp = d3.geoInterpolate([-current[0], -current[1]], [lon, lat]);
|
||||||
|
const timer = d3.timer(elapsed => {
|
||||||
|
if (isCancelled) { timer.stop(); resolve(); return true; }
|
||||||
const t = Math.min(elapsed / duration, 1);
|
const t = Math.min(elapsed / duration, 1);
|
||||||
const offset = startOffset + (endOffset - startOffset) * t;
|
const point = interp(t);
|
||||||
|
projection.rotate([-point[0], -point[1]]);
|
||||||
pathEl.attr('stroke-dashoffset', offset);
|
redrawBase();
|
||||||
|
if (t >= 1) { timer.stop(); resolve(); return true; }
|
||||||
const drawn = totalLength - offset;
|
|
||||||
const clamped = Math.max(0, Math.min(drawn, totalLength));
|
|
||||||
try {
|
|
||||||
const pt = node.getPointAtLength(clamped);
|
|
||||||
const angle = getAngleAtLength(node, clamped);
|
|
||||||
tipEl.attr('transform', `translate(${pt.x}, ${pt.y}) rotate(${angle}) scale(1.4)`).attr('opacity', 1);
|
|
||||||
} catch (e) {
|
|
||||||
// ignore SVG errors
|
|
||||||
}
|
|
||||||
|
|
||||||
if (t >= 1) {
|
|
||||||
resolve();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function delay(ms) {
|
function createArcEl(iconSrc) {
|
||||||
|
const el = gAnim.append('path')
|
||||||
|
.attr('fill', 'none').attr('stroke', ARC_COLOR)
|
||||||
|
.attr('stroke-width', 2.5).attr('stroke-opacity', 0.8)
|
||||||
|
.attr('stroke-linecap', 'round').attr('stroke-dasharray', '10, 6');
|
||||||
|
const tip = gAnim.append('image')
|
||||||
|
.attr('href', iconSrc).attr('width', PLANE_SIZE).attr('height', PLANE_SIZE)
|
||||||
|
.attr('x', -PLANE_SIZE / 2).attr('y', -PLANE_SIZE / 2)
|
||||||
|
.attr('preserveAspectRatio', 'xMidYMid meet').attr('opacity', 0);
|
||||||
|
return { el, tip };
|
||||||
|
}
|
||||||
|
|
||||||
|
function animateIncrementalPath(el, tip, pts, duration, flip = false) {
|
||||||
return new Promise(resolve => {
|
return new Promise(resolve => {
|
||||||
if (isCancelled) { resolve(); return; }
|
const lineGen = d3.line().curve(d3.curveBasis);
|
||||||
const id = setTimeout(resolve, ms);
|
d3.timer(elapsed => {
|
||||||
|
if (isCancelled) { resolve(); return true; }
|
||||||
|
const t = Math.min(elapsed / duration, 1);
|
||||||
|
const count = Math.max(2, Math.floor(t * (pts.length - 1)) + 1);
|
||||||
|
const visible = pts.slice(0, count);
|
||||||
|
if (visible.length >= 2) el.attr('d', lineGen(visible));
|
||||||
|
if (visible.length > 0) {
|
||||||
|
const last = visible[visible.length - 1];
|
||||||
|
let angle = 0;
|
||||||
|
if (visible.length >= 2) {
|
||||||
|
const prev = visible[visible.length - 2];
|
||||||
|
angle = Math.atan2(last[1] - prev[1], last[0] - prev[0]) * 180 / Math.PI;
|
||||||
|
}
|
||||||
|
tip.attr('transform', planeTransform(last[0], last[1], angle, flip)).attr('opacity', 1);
|
||||||
|
}
|
||||||
|
if (t >= 1) { resolve(); return true; }
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async function animateTrip(destCode, destFeature) {
|
function animateReprojectingArc(el, tip, geoPts, lineGen, duration, flip = false) {
|
||||||
if (!homeFeature || !destFeature) return;
|
return new Promise(resolve => {
|
||||||
|
const timer = d3.timer(elapsed => {
|
||||||
|
if (isCancelled) { timer.stop(); resolve(); return true; }
|
||||||
|
const t = Math.min(elapsed / duration, 1);
|
||||||
|
const count = Math.max(2, Math.floor(t * (geoPts.length - 1)) + 1);
|
||||||
|
const screenPts = geoPts.slice(0, count).map(p => projection(p)).filter(Boolean);
|
||||||
|
if (screenPts.length >= 2) el.attr('d', lineGen(screenPts));
|
||||||
|
if (screenPts.length > 0) {
|
||||||
|
const last = screenPts[screenPts.length - 1];
|
||||||
|
let angle = 0;
|
||||||
|
if (screenPts.length >= 2) {
|
||||||
|
const prev = screenPts[screenPts.length - 2];
|
||||||
|
angle = Math.atan2(last[1] - prev[1], last[0] - prev[0]) * 180 / Math.PI;
|
||||||
|
}
|
||||||
|
tip.attr('transform', planeTransform(last[0], last[1], angle, flip)).attr('opacity', 1);
|
||||||
|
}
|
||||||
|
if (t >= 1) { timer.stop(); resolve(); return true; }
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function animateTrip(destCode, destFeature, transport = 'flight') {
|
||||||
|
if (!homeFeature || !destFeature) return;
|
||||||
|
const iconSrc = TRANSPORT_IMG[transport] ?? airplaneImg;
|
||||||
const homeCentroid = d3.geoCentroid(homeFeature);
|
const homeCentroid = d3.geoCentroid(homeFeature);
|
||||||
const destCentroid = d3.geoCentroid(destFeature);
|
const destCentroid = d3.geoCentroid(destFeature);
|
||||||
|
if (mode === 'map') {
|
||||||
|
await animateMapTrip(homeCentroid, destCentroid, destCode, iconSrc);
|
||||||
|
} else {
|
||||||
|
await animateGlobeTrip(homeCentroid, destCentroid, destCode, iconSrc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function animateMapTrip(homeCentroid, destCentroid, destCode, iconSrc) {
|
||||||
const pts = computeArc(homeCentroid, destCentroid);
|
const pts = computeArc(homeCentroid, destCentroid);
|
||||||
if (pts.length < 2) return;
|
if (pts.length < 2) return;
|
||||||
|
const { el: outEl, tip: outTip } = createArcEl(iconSrc);
|
||||||
const lineGen = d3.line().curve(d3.curveBasis);
|
await animateIncrementalPath(outEl, outTip, pts, 2500, pts[pts.length-1][0] < pts[0][0]);
|
||||||
const pathData = lineGen(pts);
|
|
||||||
|
|
||||||
if (!pathData) return;
|
|
||||||
|
|
||||||
function createArc(pathData) {
|
|
||||||
const el = g.append('path')
|
|
||||||
.attr('d', pathData)
|
|
||||||
.attr('fill', 'none')
|
|
||||||
.attr('stroke', ARC_COLOR)
|
|
||||||
.attr('stroke-width', 2.5)
|
|
||||||
.attr('stroke-opacity', 0.8)
|
|
||||||
.attr('stroke-linecap', 'round');
|
|
||||||
const tip = g.append('path')
|
|
||||||
.attr('d', PLANE_PATH)
|
|
||||||
.attr('fill', PLANE_COLOR)
|
|
||||||
.attr('opacity', 0);
|
|
||||||
return { el, tip };
|
|
||||||
}
|
|
||||||
|
|
||||||
// --- Outbound: home -> dest ---
|
|
||||||
let { el: outEl, tip: outTip } = createArc(pathData);
|
|
||||||
const outLen = outEl.node().getTotalLength();
|
|
||||||
outEl.attr('stroke-dasharray', outLen).attr('stroke-dashoffset', outLen);
|
|
||||||
|
|
||||||
const homeDot = g.append('circle')
|
|
||||||
.attr('r', 4)
|
|
||||||
.attr('fill', PLANE_COLOR)
|
|
||||||
.attr('cx', pts[0][0])
|
|
||||||
.attr('cy', pts[0][1])
|
|
||||||
.attr('opacity', 1);
|
|
||||||
|
|
||||||
await animateStroke(outEl, outTip, outLen, 0, 2500);
|
|
||||||
if (isCancelled) return;
|
if (isCancelled) return;
|
||||||
|
outEl.remove(); outTip.remove();
|
||||||
outEl.remove();
|
countryPaths.filter(d => effId(d) === destCode).transition().duration(500).attr('fill', VISITED_COLOR);
|
||||||
outTip.remove();
|
visitedCodes.add(destCode);
|
||||||
homeDot.remove();
|
gBase.selectAll('.micro-state-j').filter(d => effId(d) === destCode).transition().duration(500).attr('fill', VISITED_COLOR);
|
||||||
|
|
||||||
// Color the destination country
|
|
||||||
const targetPath = countryPaths.filter(d => effId(d) === destCode);
|
|
||||||
targetPath.transition().duration(500).attr('fill', VISITED_COLOR);
|
|
||||||
g.selectAll('.micro-state-j')
|
|
||||||
.filter(d => effId(d) === destCode)
|
|
||||||
.transition().duration(500)
|
|
||||||
.attr('fill', VISITED_COLOR);
|
|
||||||
|
|
||||||
await delay(800);
|
await delay(800);
|
||||||
if (isCancelled) return;
|
if (isCancelled) return;
|
||||||
|
|
||||||
// --- Return: dest -> home ---
|
|
||||||
const revPts = [...pts].reverse();
|
const revPts = [...pts].reverse();
|
||||||
const revData = d3.line().curve(d3.curveBasis)(revPts);
|
const { el: retEl, tip: retTip } = createArcEl(iconSrc);
|
||||||
let { el: retEl, tip: retTip } = createArc(revData);
|
await animateIncrementalPath(retEl, retTip, revPts, 2200, revPts[revPts.length-1][0] < revPts[0][0]);
|
||||||
const retLen = retEl.node().getTotalLength();
|
|
||||||
retEl.attr('stroke-dasharray', retLen).attr('stroke-dashoffset', retLen);
|
|
||||||
|
|
||||||
const destDot = g.append('circle')
|
|
||||||
.attr('r', 4)
|
|
||||||
.attr('fill', PLANE_COLOR)
|
|
||||||
.attr('cx', revPts[0][0])
|
|
||||||
.attr('cy', revPts[0][1])
|
|
||||||
.attr('opacity', 1);
|
|
||||||
|
|
||||||
await animateStroke(retEl, retTip, retLen, 0, 2200);
|
|
||||||
if (isCancelled) return;
|
if (isCancelled) return;
|
||||||
|
retEl.remove(); retTip.remove();
|
||||||
|
await delay(300);
|
||||||
|
}
|
||||||
|
|
||||||
retEl.remove();
|
async function animateGlobeTrip(homeCentroid, destCentroid, destCode, iconSrc) {
|
||||||
retTip.remove();
|
const interp = d3.geoInterpolate(homeCentroid, destCentroid);
|
||||||
destDot.remove();
|
const geoPts = Array.from({ length: 81 }, (_, i) => interp(i / 80));
|
||||||
|
const dur = Math.round(1500 + d3.geoDistance(homeCentroid, destCentroid) * 2500);
|
||||||
|
const lineGen = d3.line().curve(d3.curveBasis);
|
||||||
|
const { el: outEl, tip: outTip } = createArcEl(iconSrc);
|
||||||
|
const outGcs = geoPts.map(p => projection(p)).filter(Boolean);
|
||||||
|
await Promise.all([
|
||||||
|
rotateGlobeTo(destCentroid[0], destCentroid[1], dur),
|
||||||
|
animateReprojectingArc(outEl, outTip, geoPts, lineGen, dur, outGcs.length >= 2 && outGcs[outGcs.length-1][0] < outGcs[0][0]),
|
||||||
|
]);
|
||||||
|
if (isCancelled) return;
|
||||||
|
outEl.remove(); outTip.remove();
|
||||||
|
countryPaths.filter(d => effId(d) === destCode).transition().duration(500).attr('fill', VISITED_COLOR);
|
||||||
|
visitedCodes.add(destCode);
|
||||||
|
await delay(600);
|
||||||
|
if (isCancelled) return;
|
||||||
|
const revGeoPts = [...geoPts].reverse();
|
||||||
|
const { el: retEl, tip: retTip } = createArcEl(iconSrc);
|
||||||
|
const retGcs = revGeoPts.map(p => projection(p)).filter(Boolean);
|
||||||
|
await Promise.all([
|
||||||
|
rotateGlobeTo(homeCentroid[0], homeCentroid[1], dur),
|
||||||
|
animateReprojectingArc(retEl, retTip, revGeoPts, lineGen, dur, retGcs.length >= 2 && retGcs[retGcs.length-1][0] < retGcs[0][0]),
|
||||||
|
]);
|
||||||
|
if (isCancelled) return;
|
||||||
|
retEl.remove(); retTip.remove();
|
||||||
await delay(300);
|
await delay(300);
|
||||||
}
|
}
|
||||||
|
|
||||||
async function startJourney() {
|
async function startJourney() {
|
||||||
isPlaying = true;
|
const myId = ++animId;
|
||||||
isFinished = false;
|
isPlaying = true; isFinished = false; isCancelled = false; visitedCodes = new Set();
|
||||||
isCancelled = false;
|
|
||||||
|
|
||||||
const trips = MOCK_TRIPS;
|
const width = frameEl.clientWidth, height = frameEl.clientHeight;
|
||||||
const total = trips.length;
|
svg.selectAll('*').remove();
|
||||||
|
gBase = svg.append('g'); gCountries = svg.append('g'); gAnim = svg.append('g');
|
||||||
|
setupProjection(width, height);
|
||||||
|
if (mode === 'globe' && homeFeature) {
|
||||||
|
const c = d3.geoCentroid(homeFeature);
|
||||||
|
projection.rotate([-c[0], -c[1]]);
|
||||||
|
pathFn = d3.geoPath().projection(projection);
|
||||||
|
}
|
||||||
|
renderMap();
|
||||||
|
|
||||||
for (let i = 0; i < total; i++) {
|
const nameToId = Object.fromEntries(Object.entries(featuresById).filter(([,f]) => f.properties?.name).map(([id, f]) => [f.properties.name, id]));
|
||||||
if (isCancelled) break;
|
const entries = get(journals).slice().sort((a, b) => a.date.localeCompare(b.date));
|
||||||
|
const trips = entries.length > 0
|
||||||
|
? entries.map(e => ({
|
||||||
|
countryName: e.location.country,
|
||||||
|
countryCode: nameToId[e.location.country] ?? null,
|
||||||
|
city: e.location.cities?.[0] ?? e.location.country,
|
||||||
|
transport: e.transport ?? 'flight',
|
||||||
|
date: e.date,
|
||||||
|
})).filter(t => t.countryCode)
|
||||||
|
: [
|
||||||
|
{ countryName: 'Japan', countryCode: '392', city: 'Tokyo', transport: 'flight', date: '2024-03-15' },
|
||||||
|
{ countryName: 'France', countryCode: '250', city: 'Paris', transport: 'flight', date: '2024-06-20' },
|
||||||
|
{ countryName: 'Spain', countryCode: '724', city: 'Barcelona', transport: 'flight', date: '2024-09-10' },
|
||||||
|
{ countryName: 'United States of America', countryCode: '840', city: 'New York', transport: 'flight', date: '2025-01-05' },
|
||||||
|
{ countryName: 'Thailand', countryCode: '764', city: 'Bangkok', transport: 'flight', date: '2025-04-18' },
|
||||||
|
{ countryName: 'Australia', countryCode: '036', city: 'Sydney', transport: 'flight', date: '2025-08-22' },
|
||||||
|
];
|
||||||
|
|
||||||
|
for (let i = 0; i < trips.length; i++) {
|
||||||
|
if (isCancelled || myId !== animId) break;
|
||||||
const trip = trips[i];
|
const trip = trips[i];
|
||||||
|
if (trip.date) currentDateLabel = formatDateLabel(trip.date);
|
||||||
const destFeature = featuresById[trip.countryCode];
|
const destFeature = featuresById[trip.countryCode];
|
||||||
if (!destFeature) continue;
|
if (!destFeature) continue;
|
||||||
|
if (onprogress) onprogress({ index: i + 1, total: trips.length, label: `${trip.city}, ${trip.countryName}` });
|
||||||
const label = `${trip.city}, ${trip.countryName}`;
|
await animateTrip(trip.countryCode, destFeature, trip.transport);
|
||||||
if (onprogress) onprogress({ index: i + 1, total, label });
|
|
||||||
|
|
||||||
await animateTrip(trip.countryCode, destFeature);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isCancelled) {
|
if (!isCancelled && myId === animId) {
|
||||||
isFinished = true;
|
isFinished = true; isPlaying = false;
|
||||||
isPlaying = false;
|
|
||||||
if (onprogress) onprogress({ index: trips.length, total: trips.length, label: 'Journey complete!' });
|
if (onprogress) onprogress({ index: trips.length, total: trips.length, label: 'Journey complete!' });
|
||||||
} else {
|
} else if (myId === animId) { isPlaying = false; }
|
||||||
isPlaying = false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function stopJourney() {
|
function stopJourney() { isCancelled = true; isPlaying = false; }
|
||||||
isCancelled = true;
|
function replay() { stopJourney(); setTimeout(() => startJourney(), 100); }
|
||||||
isPlaying = false;
|
function switchMode(target) {
|
||||||
|
if (target === mode) return;
|
||||||
|
onmodechange?.(target);
|
||||||
|
stopJourney();
|
||||||
|
setTimeout(() => startJourney(), 100);
|
||||||
}
|
}
|
||||||
|
function close() { stopJourney(); onclose?.(); }
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
const width = frameEl.clientWidth;
|
const width = frameEl.clientWidth, height = frameEl.clientHeight;
|
||||||
const height = frameEl.clientHeight;
|
setupProjection(width, height);
|
||||||
|
|
||||||
projection = d3.geoMercator();
|
countriesData = feature(worldData, worldData.objects.countries)
|
||||||
fitProjection(projection, width, height);
|
|
||||||
|
|
||||||
pathFn = d3.geoPath().projection(projection);
|
|
||||||
|
|
||||||
const countries = feature(worldData, worldData.objects.countries)
|
|
||||||
.features.filter(f => (f.id || f.properties.name === 'Kosovo') && f.id !== '010');
|
.features.filter(f => (f.id || f.properties.name === 'Kosovo') && f.id !== '010');
|
||||||
|
countriesData.forEach(f => { if (!f.id) f.id = 'XK'; });
|
||||||
countries.forEach(f => {
|
for (const f of countriesData) featuresById[effId(f)] = f;
|
||||||
if (!f.id) f.id = 'XK';
|
|
||||||
});
|
|
||||||
|
|
||||||
for (const f of countries) {
|
|
||||||
featuresById[effId(f)] = f;
|
|
||||||
}
|
|
||||||
|
|
||||||
homeFeature = featuresById[HOME_CODE];
|
homeFeature = featuresById[HOME_CODE];
|
||||||
|
|
||||||
svg = d3.select(frameEl)
|
svg = d3.select(frameEl).append('svg').attr('width', width).attr('height', height).style('cursor', 'default');
|
||||||
.append('svg')
|
gBase = svg.append('g'); gCountries = svg.append('g'); gAnim = svg.append('g');
|
||||||
.attr('width', width)
|
renderMap();
|
||||||
.attr('height', height)
|
|
||||||
.style('cursor', 'default');
|
|
||||||
|
|
||||||
g = svg.append('g');
|
|
||||||
|
|
||||||
countryPaths = g.selectAll('path')
|
|
||||||
.data(countries)
|
|
||||||
.join('path')
|
|
||||||
.attr('d', pathFn)
|
|
||||||
.attr('fill', d => effId(d) === HOME_CODE ? HOME_COLOR : UNVISITED)
|
|
||||||
.attr('stroke', '#d4d4d4')
|
|
||||||
.attr('stroke-width', 0.5);
|
|
||||||
|
|
||||||
function renderMicrostates() {
|
|
||||||
g.selectAll('.micro-state-j').remove();
|
|
||||||
const threshold = 4;
|
|
||||||
countryPaths.each(function (d) {
|
|
||||||
if (effId(d) !== d.id) return;
|
|
||||||
const { width, height } = this.getBBox();
|
|
||||||
if (width < threshold && height < threshold) {
|
|
||||||
const [cx, cy] = pathFn.centroid(d);
|
|
||||||
g.append('circle')
|
|
||||||
.attr('class', 'micro-state-j')
|
|
||||||
.datum(d)
|
|
||||||
.attr('cx', cx)
|
|
||||||
.attr('cy', cy)
|
|
||||||
.attr('r', 2)
|
|
||||||
.attr('fill', effId(d) === HOME_CODE ? HOME_COLOR : UNVISITED)
|
|
||||||
.attr('stroke', '#94a3b8')
|
|
||||||
.attr('stroke-width', 0.5);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
renderMicrostates();
|
|
||||||
|
|
||||||
const observer = new ResizeObserver((entries) => {
|
const observer = new ResizeObserver((entries) => {
|
||||||
for (const entry of entries) {
|
for (const entry of entries) {
|
||||||
const { width, height } = entry.contentRect;
|
const { width, height } = entry.contentRect;
|
||||||
svg.attr('width', width).attr('height', height);
|
svg.attr('width', width).attr('height', height);
|
||||||
fitProjection(projection, width, height);
|
const prevRotate = mode === 'globe' ? projection.rotate() : null;
|
||||||
countryPaths.attr('d', pathFn);
|
if (mode === 'map') {
|
||||||
renderMicrostates();
|
projection = d3.geoMercator();
|
||||||
|
projection.fitSize([width, height], { type: 'Sphere' });
|
||||||
|
projection.scale(projection.scale() * 1.5).translate([width / 2, height * 0.70]);
|
||||||
|
} else {
|
||||||
|
const size = Math.min(width, height) * 0.92;
|
||||||
|
projection = d3.geoOrthographic().rotate(prevRotate).fitSize([size, size], { type: 'Sphere' }).translate([width / 2, height / 2]);
|
||||||
|
}
|
||||||
|
pathFn = d3.geoPath().projection(projection);
|
||||||
|
redrawBase();
|
||||||
|
if (mode === 'map') renderMicrostates();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
observer.observe(frameEl);
|
observer.observe(frameEl);
|
||||||
|
|
||||||
startJourney();
|
startJourney();
|
||||||
|
return () => { stopJourney(); observer.disconnect(); if (svg) svg.remove(); };
|
||||||
return () => {
|
|
||||||
stopJourney();
|
|
||||||
observer.disconnect();
|
|
||||||
if (svg) svg.remove();
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div bind:this={frameEl} class="journey-frame">
|
<div bind:this={frameEl} class="journey-frame" class:globe-mode={mode === 'globe'}>
|
||||||
<button class="close-btn" onclick={() => { stopJourney(); onclose?.(); }}>✕</button>
|
<div class="top-label">
|
||||||
{#if isFinished}
|
{#if isFinished}Journey complete!{:else if currentDateLabel}{currentDateLabel}{/if}
|
||||||
<div class="done-badge">Journey complete!</div>
|
</div>
|
||||||
{/if}
|
<div class="control-bar">
|
||||||
|
<button class="control-btn" onclick={replay}>⟳ Replay</button>
|
||||||
|
<button class="control-btn" onclick={() => switchMode(mode === 'map' ? 'globe' : 'map')}>
|
||||||
|
{mode === 'map' ? '🌍 Globe view' : '🗺 Map view'}
|
||||||
|
</button>
|
||||||
|
<button class="control-btn" onclick={close}>✕ Close</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.journey-frame {
|
.journey-frame { width: 100%; height: 100%; overflow: hidden; position: relative; background: #a4c8e0; }
|
||||||
width: 100%;
|
.journey-frame.globe-mode { background: #ffffff; }
|
||||||
height: 100%;
|
.journey-frame :global(svg) { display: block; }
|
||||||
overflow: hidden;
|
|
||||||
position: relative;
|
.top-label {
|
||||||
background: #a4c8e0;
|
position: absolute; top: 16px; left: 16px; z-index: 10;
|
||||||
|
background: rgba(0,0,0,0.65); color: #fff;
|
||||||
|
font-family: var(--heading, sans-serif); font-size: 16px; font-weight: 600;
|
||||||
|
padding: 10px 24px; border-radius: 24px; white-space: nowrap;
|
||||||
|
letter-spacing: 0.04em; min-width: 200px; text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.journey-frame :global(svg) {
|
.control-bar {
|
||||||
display: block;
|
position: absolute; bottom: 24px; right: 24px; z-index: 10;
|
||||||
|
display: flex; flex-direction: column; gap: 8px; align-items: flex-end;
|
||||||
}
|
}
|
||||||
|
|
||||||
.close-btn {
|
.control-btn {
|
||||||
position: absolute;
|
padding: 10px 24px; border: none; border-radius: 24px;
|
||||||
top: 12px;
|
background: #8b5cf6; color: #fff; font-size: 14px; font-weight: 600;
|
||||||
right: 12px;
|
cursor: pointer; transition: background 0.15s ease; white-space: nowrap; font-family: inherit;
|
||||||
z-index: 10;
|
|
||||||
width: 36px;
|
|
||||||
height: 36px;
|
|
||||||
border-radius: 50%;
|
|
||||||
border: none;
|
|
||||||
background: rgba(0,0,0,0.55);
|
|
||||||
color: #fff;
|
|
||||||
font-size: 18px;
|
|
||||||
line-height: 1;
|
|
||||||
cursor: pointer;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
transition: background 0.15s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.close-btn:hover {
|
|
||||||
background: rgba(0,0,0,0.75);
|
|
||||||
}
|
|
||||||
|
|
||||||
.done-badge {
|
|
||||||
position: absolute;
|
|
||||||
bottom: 24px;
|
|
||||||
left: 50%;
|
|
||||||
transform: translateX(-50%);
|
|
||||||
z-index: 10;
|
|
||||||
background: rgba(0,0,0,0.65);
|
|
||||||
color: #fff;
|
|
||||||
font-family: var(--heading, sans-serif);
|
|
||||||
font-size: 16px;
|
|
||||||
padding: 10px 24px;
|
|
||||||
border-radius: 24px;
|
|
||||||
white-space: nowrap;
|
|
||||||
letter-spacing: 0.04em;
|
|
||||||
}
|
}
|
||||||
|
.control-btn:hover { background: #7c3aed; }
|
||||||
|
.control-btn:active { transform: scale(0.96); }
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,18 +1,46 @@
|
|||||||
<script>
|
<script>
|
||||||
import { CONTINENTS, getContinent, continentTotals } from './continents.js';
|
import { CONTINENTS, getContinent, continentTotals } from './continents.js';
|
||||||
import { getSelected } from '../layout/selection.svelte.js';
|
import { getSelected, getTotalCount } from '../layout/selection.svelte.js';
|
||||||
|
import worldData from 'world-atlas/countries-50m.json';
|
||||||
|
|
||||||
let hoveredSeg = $state(null);
|
let collapsed = $state(false);
|
||||||
|
|
||||||
const continentColors = {
|
const continentColors = {
|
||||||
'Europe': '#6366f1',
|
'Europe': '#3b82f6',
|
||||||
'Asia': '#f43f5e',
|
'Asia': '#ef4444',
|
||||||
'Africa': '#fb923c',
|
'Africa': '#f97316',
|
||||||
'N. America': '#06b6d4',
|
'N. America': '#ec4899',
|
||||||
'S. America': '#f59e0b',
|
'S. America': '#eab308',
|
||||||
'Oceania': '#8b5cf6'
|
'Oceania': '#a16207'
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const countryNameById = $derived.by(() => {
|
||||||
|
const map = { XK: 'Kosovo' };
|
||||||
|
for (const g of worldData.objects.countries.geometries) {
|
||||||
|
map[g.id] = g.properties?.name || g.id;
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
});
|
||||||
|
|
||||||
|
let visitedCountries = $derived(
|
||||||
|
[...getSelected()].map(id => countryNameById[id]).filter(Boolean).sort()
|
||||||
|
);
|
||||||
|
|
||||||
|
let visitedByContinent = $derived.by(() => {
|
||||||
|
const map = {};
|
||||||
|
for (const id of getSelected()) {
|
||||||
|
const cont = getContinent(id);
|
||||||
|
if (cont) {
|
||||||
|
if (!map[cont]) map[cont] = [];
|
||||||
|
map[cont].push(countryNameById[id] || id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (const cont of Object.keys(map)) {
|
||||||
|
map[cont].sort();
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
});
|
||||||
|
|
||||||
let counts = $derived.by(() => {
|
let counts = $derived.by(() => {
|
||||||
const c = {};
|
const c = {};
|
||||||
for (const cont of CONTINENTS) c[cont] = 0;
|
for (const cont of CONTINENTS) c[cont] = 0;
|
||||||
@@ -36,9 +64,11 @@
|
|||||||
if (angle > 0) {
|
if (angle > 0) {
|
||||||
const startDeg = deg;
|
const startDeg = deg;
|
||||||
const endDeg = deg + angle;
|
const endDeg = deg + angle;
|
||||||
|
const midDeg = (startDeg + endDeg) / 2;
|
||||||
|
const rad = (midDeg - 90) * Math.PI / 180;
|
||||||
const sr = (startDeg - 90) * Math.PI / 180;
|
const sr = (startDeg - 90) * Math.PI / 180;
|
||||||
const er = (endDeg - 90) * Math.PI / 180;
|
const er = (endDeg - 90) * Math.PI / 180;
|
||||||
const cx = 50, cy = 50, outerR = 44, innerR = 22;
|
const cx = 90, cy = 90, outerR = 65, innerR = 30;
|
||||||
const x1 = cx + outerR * Math.cos(sr);
|
const x1 = cx + outerR * Math.cos(sr);
|
||||||
const y1 = cy + outerR * Math.sin(sr);
|
const y1 = cy + outerR * Math.sin(sr);
|
||||||
const x2 = cx + outerR * Math.cos(er);
|
const x2 = cx + outerR * Math.cos(er);
|
||||||
@@ -49,7 +79,9 @@
|
|||||||
const y4 = cy + innerR * Math.sin(sr);
|
const y4 = cy + innerR * Math.sin(sr);
|
||||||
const largeArc = angle > 180 ? 1 : 0;
|
const largeArc = angle > 180 ? 1 : 0;
|
||||||
const path = `M ${x1} ${y1} A ${outerR} ${outerR} 0 ${largeArc} 1 ${x2} ${y2} L ${x3} ${y3} A ${innerR} ${innerR} 0 ${largeArc} 0 ${x4} ${y4} Z`;
|
const path = `M ${x1} ${y1} A ${outerR} ${outerR} 0 ${largeArc} 1 ${x2} ${y2} L ${x3} ${y3} A ${innerR} ${innerR} 0 ${largeArc} 0 ${x4} ${y4} Z`;
|
||||||
segs.push({ cont, color: continentColors[cont], path, angle });
|
const lx = cx + 82 * Math.cos(rad);
|
||||||
|
const ly = cy + 82 * Math.sin(rad);
|
||||||
|
segs.push({ cont, color: continentColors[cont], path, lx, ly, angle });
|
||||||
deg += angle;
|
deg += angle;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -57,199 +89,296 @@
|
|||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="card">
|
<div class="panel" class:collapsed>
|
||||||
<!-- count -->
|
<button class="collapse-btn" onclick={() => collapsed = !collapsed} data-tip={collapsed ? 'see statistics' : 'close statistics'}>
|
||||||
<div class="stat-block">
|
{collapsed ? '◀' : '▶'}
|
||||||
<span class="big-num">{total}</span>
|
</button>
|
||||||
<span class="stat-sub">countries visited</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="vdivider"></div>
|
{#if !collapsed}
|
||||||
|
<div class="panel-content">
|
||||||
|
<h2 class="headline">your statistics</h2>
|
||||||
|
|
||||||
<!-- world % -->
|
<div class="total-bar-bg">
|
||||||
<div class="stat-block">
|
<div class="total-bar-fill" style="width: {pct}%"></div>
|
||||||
<span class="big-num accent">{pct}%</span>
|
<span class="bar-pct">{pct}%</span>
|
||||||
<span class="stat-sub">of the world</span>
|
</div>
|
||||||
</div>
|
<span class="total-bar-text">{total} / {grandTotal} countries visited</span>
|
||||||
|
|
||||||
<div class="vdivider"></div>
|
<div class="divider"></div>
|
||||||
|
|
||||||
<!-- donut -->
|
<span class="bar-label">by continent</span>
|
||||||
<div class="donut-block">
|
{#each CONTINENTS as continent}
|
||||||
<svg viewBox="0 0 100 100" class="donut-svg">
|
{@const contTotal = continentTotals[continent]}
|
||||||
{#if segments.length > 0}
|
<div class="row tooltip-wrap">
|
||||||
{#each segments as seg}
|
<span class="dot" style="background: {continentColors[continent]}"></span>
|
||||||
<g class="seg-group"
|
<span class="label">{continent}</span>
|
||||||
onmouseenter={() => hoveredSeg = seg}
|
<span class="value">{counts[continent]}<span class="total">/{contTotal}</span></span>
|
||||||
onmouseleave={() => hoveredSeg = null}>
|
{#if visitedByContinent[continent]?.length > 0}
|
||||||
<path d={seg.path} fill={seg.color} />
|
<div class="tooltip-list">
|
||||||
</g>
|
{#each visitedByContinent[continent].slice(0, 10) as country}
|
||||||
{/each}
|
<span class="tooltip-item">{country}</span>
|
||||||
<circle cx="50" cy="50" r="22" fill="#fff" />
|
{/each}
|
||||||
{:else}
|
{#if visitedByContinent[continent].length > 10}
|
||||||
<circle cx="50" cy="50" r="44" fill="#f1f5f9" />
|
<span class="tooltip-item tooltip-more">...</span>
|
||||||
<circle cx="50" cy="50" r="22" fill="#fff" />
|
{/if}
|
||||||
{/if}
|
</div>
|
||||||
</svg>
|
{/if}
|
||||||
|
|
||||||
<div class="donut-info">
|
|
||||||
<span class="section-label">by continent</span>
|
|
||||||
{#if hoveredSeg}
|
|
||||||
<div class="tooltip" style="--dot:{hoveredSeg.color}">
|
|
||||||
<span class="tt-name">{hoveredSeg.cont}</span>
|
|
||||||
<span class="tt-val">{counts[hoveredSeg.cont]} / {continentTotals[hoveredSeg.cont]}</span>
|
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{/each}
|
||||||
<span class="hint">hover a slice</span>
|
|
||||||
{/if}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="vdivider"></div>
|
<div class="donut-wrap">
|
||||||
|
{#if segments.length > 0}
|
||||||
|
<svg viewBox="-25 -25 230 230" class="donut-svg">
|
||||||
|
{#each segments as seg}
|
||||||
|
<g class="seg-group">
|
||||||
|
<path d={seg.path} fill={seg.color} />
|
||||||
|
<text x={seg.lx} y={seg.ly} text-anchor="middle" dominant-baseline="middle" class="donut-label" style="font-size: {seg.angle < 20 ? 12 : 15}px">{seg.cont}</text>
|
||||||
|
</g>
|
||||||
|
{/each}
|
||||||
|
<circle cx="90" cy="90" r="30" fill="var(--bg-raised)" />
|
||||||
|
</svg>
|
||||||
|
{:else}
|
||||||
|
<svg viewBox="-25 -25 230 230" class="donut-svg">
|
||||||
|
<circle cx="90" cy="90" r="65" fill="var(--border)" />
|
||||||
|
<circle cx="90" cy="90" r="30" fill="var(--bg-raised)" />
|
||||||
|
</svg>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
<!-- progress bar -->
|
<div class="divider"></div>
|
||||||
<div class="bar-block">
|
|
||||||
<span class="section-label" style="margin-bottom:6px">world coverage</span>
|
<div class="disclaimer">Contains all UN countries, Kosovo, Hong Kong and Taiwan</div>
|
||||||
<div class="bar-bg">
|
|
||||||
<div class="bar-fill" style="width:{pct}%"></div>
|
|
||||||
</div>
|
</div>
|
||||||
<span class="disclaimer">All UN countries · Kosovo · HK · Taiwan</span>
|
{/if}
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.card {
|
.panel {
|
||||||
position: absolute;
|
flex: 0 0 min(360px, 25vw);
|
||||||
top: 16px;
|
background: var(--bg-raised);
|
||||||
left: 50%;
|
border-left: 1px solid var(--border);
|
||||||
transform: translateX(-50%);
|
|
||||||
background: #fff;
|
|
||||||
border-radius: 16px;
|
|
||||||
box-shadow: 0 4px 20px rgba(0,0,0,0.10), 0 1px 4px rgba(0,0,0,0.06);
|
|
||||||
border: 1px solid rgba(0,0,0,0.06);
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: center;
|
|
||||||
gap: 0;
|
|
||||||
padding: 0 4px;
|
|
||||||
height: 110px;
|
|
||||||
z-index: 10;
|
|
||||||
font-family: var(--sans);
|
font-family: var(--sans);
|
||||||
white-space: nowrap;
|
transition: flex-basis 0.25s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.stat-block {
|
.panel.collapsed {
|
||||||
display: flex;
|
flex: 0 0 28px;
|
||||||
flex-direction: column;
|
border-left: none;
|
||||||
align-items: center;
|
|
||||||
gap: 4px;
|
|
||||||
padding: 0 36px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.big-num {
|
.panel-content {
|
||||||
font-size: 40px;
|
flex: 1;
|
||||||
font-weight: 300;
|
padding: 24px 28px;
|
||||||
letter-spacing: -2px;
|
overflow-y: auto;
|
||||||
color: var(--text-h);
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collapse-btn {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
align-self: flex-start;
|
||||||
|
background: var(--accent-bg);
|
||||||
|
border: none;
|
||||||
|
border-radius: 0 8px 8px 0;
|
||||||
|
padding: 14px 5px;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 16px;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
|
color: var(--accent);
|
||||||
|
transition: background 0.15s ease, padding 0.15s ease;
|
||||||
|
margin-top: 24px;
|
||||||
|
position: relative;
|
||||||
}
|
}
|
||||||
.big-num.accent { color: var(--accent); }
|
|
||||||
|
|
||||||
.stat-sub {
|
.collapse-btn:hover {
|
||||||
|
background: var(--lavender-bg);
|
||||||
|
padding-right: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collapse-btn::after {
|
||||||
|
content: attr(data-tip);
|
||||||
|
position: absolute;
|
||||||
|
right: calc(100% + 8px);
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
background: var(--text-h);
|
||||||
|
color: var(--bg-raised);
|
||||||
|
font-family: var(--sans);
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
color: var(--text-sub);
|
padding: 6px 12px;
|
||||||
letter-spacing: 0.03em;
|
border-radius: 6px;
|
||||||
|
white-space: nowrap;
|
||||||
|
pointer-events: none;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.15s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vdivider {
|
.collapse-btn:hover::after {
|
||||||
width: 1px;
|
opacity: 1;
|
||||||
height: 56px;
|
|
||||||
background: var(--border);
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* donut */
|
.headline {
|
||||||
.donut-block {
|
font-family: var(--heading);
|
||||||
display: flex;
|
font-size: var(--text-sm);
|
||||||
flex-direction: row;
|
font-weight: 400;
|
||||||
align-items: center;
|
|
||||||
gap: 14px;
|
|
||||||
padding: 0 28px;
|
|
||||||
}
|
|
||||||
.donut-svg {
|
|
||||||
width: 72px;
|
|
||||||
height: 72px;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
.seg-group { cursor: pointer; }
|
|
||||||
.seg-group:hover path { opacity: 0.8; }
|
|
||||||
|
|
||||||
.donut-info {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 6px;
|
|
||||||
min-width: 130px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tooltip {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 7px;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
.tooltip::before {
|
|
||||||
content: '';
|
|
||||||
width: 8px;
|
|
||||||
height: 8px;
|
|
||||||
border-radius: 50%;
|
|
||||||
background: var(--dot);
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
.tt-name { font-weight: 400; color: var(--text-h); }
|
|
||||||
.tt-val { font-weight: 300; color: var(--text-sub); }
|
|
||||||
|
|
||||||
.section-label {
|
|
||||||
font-size: 10px;
|
|
||||||
font-weight: 500;
|
|
||||||
letter-spacing: 0.14em;
|
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
color: var(--text-sub);
|
letter-spacing: 0.1em;
|
||||||
|
color: var(--accent);
|
||||||
|
margin: 0 0 20px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.hint {
|
.bar-label {
|
||||||
font-size: 12px;
|
font-family: var(--sans);
|
||||||
|
font-size: var(--text-xs);
|
||||||
|
font-weight: 400;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
color: var(--text-sub);
|
color: var(--text-sub);
|
||||||
opacity: 0.45;
|
display: block;
|
||||||
|
margin-bottom: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* bar */
|
.total-bar-bg {
|
||||||
.bar-block {
|
position: relative;
|
||||||
display: flex;
|
height: 28px;
|
||||||
flex-direction: column;
|
background: var(--accent-bg);
|
||||||
padding: 0 28px;
|
border-radius: 10px;
|
||||||
gap: 0;
|
|
||||||
min-width: 160px;
|
|
||||||
}
|
|
||||||
.bar-bg {
|
|
||||||
width: 100%;
|
|
||||||
height: 5px;
|
|
||||||
background: var(--bg-subtle);
|
|
||||||
border-radius: 4px;
|
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
margin-bottom: 10px;
|
|
||||||
}
|
}
|
||||||
.bar-fill {
|
|
||||||
|
.total-bar-fill {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
background: linear-gradient(90deg, var(--accent), #a78bfa);
|
background: linear-gradient(90deg, var(--accent-dark), var(--lavender));
|
||||||
border-radius: 4px;
|
border-radius: 10px;
|
||||||
transition: width 0.3s ease;
|
transition: width 0.3s ease;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
}
|
}
|
||||||
.disclaimer {
|
|
||||||
font-size: 11px;
|
.bar-pct {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: var(--text-sm);
|
||||||
|
font-weight: 500;
|
||||||
|
color: var(--text-h);
|
||||||
|
}
|
||||||
|
|
||||||
|
.total-bar-text {
|
||||||
|
display: block;
|
||||||
|
text-align: center;
|
||||||
|
font-size: var(--text-sm);
|
||||||
|
font-weight: 400;
|
||||||
|
color: var(--text-h);
|
||||||
|
margin-top: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.divider {
|
||||||
|
height: 1px;
|
||||||
|
background: var(--border);
|
||||||
|
margin: 16px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
padding: 6px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dot {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 50%;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.label {
|
||||||
|
flex: 1;
|
||||||
|
font-size: var(--text-sm);
|
||||||
|
font-weight: 300;
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
|
||||||
|
.value {
|
||||||
|
font-size: var(--text-sm);
|
||||||
|
font-weight: 400;
|
||||||
|
color: var(--text-h);
|
||||||
|
}
|
||||||
|
|
||||||
|
.total {
|
||||||
|
font-weight: 400;
|
||||||
color: var(--text-sub);
|
color: var(--text-sub);
|
||||||
opacity: 0.5;
|
font-size: var(--text-xs);
|
||||||
letter-spacing: 0.02em;
|
}
|
||||||
|
|
||||||
|
.donut-wrap {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
margin: 8px 0 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.donut-svg {
|
||||||
|
width: 180px;
|
||||||
|
height: 180px;
|
||||||
|
filter: drop-shadow(0 2px 8px rgba(99,102,241,0.15));
|
||||||
|
}
|
||||||
|
|
||||||
|
.donut-label {
|
||||||
|
fill: var(--text-h);
|
||||||
|
font-family: var(--sans);
|
||||||
|
font-weight: 300;
|
||||||
|
pointer-events: none;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.15s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.seg-group:hover .donut-label {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip-wrap {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip-list {
|
||||||
|
display: none;
|
||||||
|
position: absolute;
|
||||||
|
top: calc(100% + 6px);
|
||||||
|
left: 0;
|
||||||
|
background: var(--text-h);
|
||||||
|
color: var(--bg-raised);
|
||||||
|
font-family: var(--sans);
|
||||||
|
font-size: 12px;
|
||||||
|
line-height: 1.5;
|
||||||
|
padding: 8px 12px;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: var(--shadow);
|
||||||
|
z-index: 20;
|
||||||
|
white-space: nowrap;
|
||||||
|
min-width: 120px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip-wrap:hover .tooltip-list {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip-item {
|
||||||
|
display: block;
|
||||||
|
padding: 2px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip-item + .tooltip-item {
|
||||||
|
border-top: 1px solid rgba(255, 255, 255, 0.06);
|
||||||
|
}
|
||||||
|
|
||||||
|
.disclaimer {
|
||||||
|
font-size: var(--text-xs);
|
||||||
|
color: var(--text-sub);
|
||||||
|
line-height: 1.5;
|
||||||
|
text-align: center;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -3,7 +3,10 @@
|
|||||||
import * as d3 from 'd3';
|
import * as d3 from 'd3';
|
||||||
import { feature } from 'topojson-client';
|
import { feature } from 'topojson-client';
|
||||||
import worldData from 'world-atlas/countries-50m.json';
|
import worldData from 'world-atlas/countries-50m.json';
|
||||||
import { getSelected, toggle, setTotalCount, getHomeCountryCode } from '../layout/selection.svelte.js';
|
import { getSelected, setTotalCount, getFlashing } from '../layout/selection.svelte.js';
|
||||||
|
import { getUserProfile } from '../auth/userStore.svelte.js';
|
||||||
|
import homeIconUrl from '../../assets/home.png';
|
||||||
|
import crayonCursorUrl from '../../assets/logo-cursor.png';
|
||||||
|
|
||||||
let { onCountryClick = (_name) => {} } = $props();
|
let { onCountryClick = (_name) => {} } = $props();
|
||||||
|
|
||||||
@@ -73,8 +76,34 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
let frameEl;
|
let frameEl;
|
||||||
let _paths = null;
|
let _paths = $state(null);
|
||||||
let _g = null;
|
let _g = null;
|
||||||
|
let _pathFn = null;
|
||||||
|
let _countries = null;
|
||||||
|
|
||||||
|
function updateHomeMarker(homeCountryName) {
|
||||||
|
if (!_g || !_pathFn || !_countries) return;
|
||||||
|
_g.selectAll('.home-marker').remove();
|
||||||
|
if (!homeCountryName) return;
|
||||||
|
const found = _countries.find(f => f.properties.name === homeCountryName);
|
||||||
|
if (!found) return;
|
||||||
|
const [cx, cy] = _pathFn.centroid(found);
|
||||||
|
if (isNaN(cx) || isNaN(cy)) return;
|
||||||
|
const SIZE = 24;
|
||||||
|
_g.append('image')
|
||||||
|
.attr('class', 'home-marker')
|
||||||
|
.attr('href', homeIconUrl)
|
||||||
|
.attr('x', cx - SIZE / 2)
|
||||||
|
.attr('y', cy - SIZE / 2)
|
||||||
|
.attr('width', SIZE)
|
||||||
|
.attr('height', SIZE)
|
||||||
|
.style('pointer-events', 'none');
|
||||||
|
}
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
const homeCountry = getUserProfile()?.homeCountry ?? null;
|
||||||
|
updateHomeMarker(homeCountry);
|
||||||
|
});
|
||||||
|
|
||||||
function fitProjection(proj, w, h) {
|
function fitProjection(proj, w, h) {
|
||||||
proj.fitSize([w, h], { type: 'Sphere' });
|
proj.fitSize([w, h], { type: 'Sphere' });
|
||||||
@@ -84,14 +113,30 @@
|
|||||||
|
|
||||||
function updateAllFills() {
|
function updateAllFills() {
|
||||||
const sel = getSelected();
|
const sel = getSelected();
|
||||||
const hc = getHomeCountryCode();
|
|
||||||
if (!_paths || !_g) return;
|
if (!_paths || !_g) return;
|
||||||
_paths.attr('fill', d => countryColor(d, sel, hc));
|
_paths.attr('fill', d => countryColor(d, sel, null));
|
||||||
_g.selectAll('.micro-state').attr('fill', d => countryColor(d, sel, hc));
|
_g.selectAll('.micro-state').attr('fill', d => countryColor(d, sel, null));
|
||||||
}
|
}
|
||||||
|
|
||||||
$effect(updateAllFills);
|
$effect(updateAllFills);
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
const flashSet = getFlashing();
|
||||||
|
const paths = _paths; // reactive read so effect re-runs when _paths is set
|
||||||
|
if (!paths || flashSet.size === 0) return;
|
||||||
|
paths
|
||||||
|
.filter(d => flashSet.has(effId(d)))
|
||||||
|
.each(function() {
|
||||||
|
d3.select(this).interrupt()
|
||||||
|
.transition().duration(200).attr('fill', '#facc15')
|
||||||
|
.transition().duration(200).attr('fill', '#fb923c')
|
||||||
|
.transition().duration(200).attr('fill', '#facc15')
|
||||||
|
.transition().duration(200).attr('fill', '#fb923c')
|
||||||
|
.transition().duration(200).attr('fill', '#facc15')
|
||||||
|
.transition().duration(400).attr('fill', VISITED_COLOR);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
const width = frameEl.clientWidth;
|
const width = frameEl.clientWidth;
|
||||||
const height = frameEl.clientHeight;
|
const height = frameEl.clientHeight;
|
||||||
@@ -108,6 +153,9 @@
|
|||||||
if (!f.id) f.id = 'XK';
|
if (!f.id) f.id = 'XK';
|
||||||
});
|
});
|
||||||
|
|
||||||
|
_pathFn = path;
|
||||||
|
_countries = countries;
|
||||||
|
|
||||||
const sovereignIds = new Set(countries.map(f => effId(f)));
|
const sovereignIds = new Set(countries.map(f => effId(f)));
|
||||||
setTotalCount(sovereignIds.size);
|
setTotalCount(sovereignIds.size);
|
||||||
|
|
||||||
@@ -123,24 +171,14 @@
|
|||||||
.attr('class', 'tooltip')
|
.attr('class', 'tooltip')
|
||||||
.style('display', 'none');
|
.style('display', 'none');
|
||||||
|
|
||||||
function updateFill(sel) {
|
|
||||||
const s = getSelected();
|
|
||||||
const hc = getHomeCountryCode();
|
|
||||||
sel.attr('fill', d => countryColor(d, s, hc));
|
|
||||||
_g.selectAll('.micro-state').attr('fill', d => countryColor(d, s, hc));
|
|
||||||
}
|
|
||||||
|
|
||||||
function attachEvents(sel) {
|
function attachEvents(sel) {
|
||||||
sel
|
sel
|
||||||
.on('click', (event, d) => {
|
.on('click', (event, d) => {
|
||||||
toggle(effId(d));
|
|
||||||
updateFill(d3.select(event.currentTarget));
|
|
||||||
onCountryClick(d.properties.name);
|
onCountryClick(d.properties.name);
|
||||||
})
|
})
|
||||||
.on('mouseenter', (event, d) => {
|
.on('mouseenter', (event, d) => {
|
||||||
const s = getSelected();
|
const s = getSelected();
|
||||||
const hc = getHomeCountryCode();
|
d3.select(event.currentTarget).attr('fill', countryHoverColor(d, s, null));
|
||||||
d3.select(event.currentTarget).attr('fill', countryHoverColor(d, s, hc));
|
|
||||||
tooltip.style('display', 'block').text(d.properties.name);
|
tooltip.style('display', 'block').text(d.properties.name);
|
||||||
})
|
})
|
||||||
.on('mousemove', (event) => {
|
.on('mousemove', (event) => {
|
||||||
@@ -149,8 +187,7 @@
|
|||||||
})
|
})
|
||||||
.on('mouseleave', (event, d) => {
|
.on('mouseleave', (event, d) => {
|
||||||
const s = getSelected();
|
const s = getSelected();
|
||||||
const hc = getHomeCountryCode();
|
d3.select(event.currentTarget).attr('fill', countryColor(d, s, null));
|
||||||
d3.select(event.currentTarget).attr('fill', countryColor(d, s, hc));
|
|
||||||
tooltip.style('display', 'none');
|
tooltip.style('display', 'none');
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -172,14 +209,13 @@
|
|||||||
const { width, height } = this.getBBox();
|
const { width, height } = this.getBBox();
|
||||||
if (width < threshold && height < threshold) {
|
if (width < threshold && height < threshold) {
|
||||||
const [cx, cy] = path.centroid(d);
|
const [cx, cy] = path.centroid(d);
|
||||||
const hc = getHomeCountryCode();
|
|
||||||
const c = _g.append('circle')
|
const c = _g.append('circle')
|
||||||
.attr('class', 'micro-state')
|
.attr('class', 'micro-state')
|
||||||
.datum(d)
|
.datum(d)
|
||||||
.attr('cx', cx)
|
.attr('cx', cx)
|
||||||
.attr('cy', cy)
|
.attr('cy', cy)
|
||||||
.attr('r', 2)
|
.attr('r', 2)
|
||||||
.attr('fill', countryColor(d, getSelected(), hc))
|
.attr('fill', countryColor(d, getSelected(), null))
|
||||||
.attr('stroke', '#94a3b8')
|
.attr('stroke', '#94a3b8')
|
||||||
.attr('stroke-width', 0.5);
|
.attr('stroke-width', 0.5);
|
||||||
attachEvents(c);
|
attachEvents(c);
|
||||||
@@ -188,6 +224,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
renderMicrostates();
|
renderMicrostates();
|
||||||
|
updateHomeMarker(getUserProfile()?.homeCountry ?? null);
|
||||||
|
|
||||||
const zoom = d3.zoom()
|
const zoom = d3.zoom()
|
||||||
.scaleExtent([1, 32])
|
.scaleExtent([1, 32])
|
||||||
@@ -225,7 +262,7 @@
|
|||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div bind:this={frameEl} class="map-frame"></div>
|
<div bind:this={frameEl} class="map-frame" style="cursor: url({crayonCursorUrl}) 4 28, crosshair;"></div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.map-frame {
|
.map-frame {
|
||||||
@@ -238,15 +275,15 @@
|
|||||||
|
|
||||||
.map-frame :global(svg) {
|
.map-frame :global(svg) {
|
||||||
display: block;
|
display: block;
|
||||||
cursor: grab;
|
cursor: inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
.map-frame :global(svg:active) {
|
.map-frame :global(svg:active) {
|
||||||
cursor: grabbing;
|
cursor: inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
.map-frame :global(svg path) {
|
.map-frame :global(svg path) {
|
||||||
cursor: pointer;
|
cursor: inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
.map-frame :global(.tooltip) {
|
.map-frame :global(.tooltip) {
|
||||||
|
|||||||