- Animated Svelte component library powered by GSAP and Three.js. Drop-in
+ Animated Svelte component library powered by GSAP and OGL. Drop-in
solutions for motion design, 3D canvases, and interactive animations.
diff --git a/apps/web/src/lib/config/site.ts b/apps/web/src/lib/config/site.ts
index 18ca309..5983b7c 100644
--- a/apps/web/src/lib/config/site.ts
+++ b/apps/web/src/lib/config/site.ts
@@ -11,7 +11,7 @@ export const siteConfig = {
url: "https://motion-core.dev",
/** Default SEO description for the homepage and fallback metadata. */
description:
- "Svelte-native motion toolkit with GSAP and Three.js powered components, demos, and CLI-driven workflows.",
+ "Svelte-native motion toolkit with GSAP and OGL-powered components, demos, and CLI-driven workflows.",
/** Author shown in metadata and structured data. */
author: "Marek Jóźwiak",
/** Primary SEO keywords for indexing and discovery. */
@@ -19,7 +19,7 @@ export const siteConfig = {
"motion core",
"svelte animation",
"gsap",
- "three.js",
+ "ogl",
"component library",
"svelte components",
"motion toolkit",
diff --git a/apps/web/src/routes/docs/globe/+page.svx b/apps/web/src/routes/docs/globe/+page.svx
index 1dfd80d..da32431 100644
--- a/apps/web/src/routes/docs/globe/+page.svx
+++ b/apps/web/src/routes/docs/globe/+page.svx
@@ -87,7 +87,7 @@ import { Globe } from "$lib/motion-core";
},
{
prop: "landPointColor",
- type: "THREE.ColorRepresentation",
+ type: "string | number | readonly [number, number, number] | { r: number; g: number; b: number }",
default: '"#f77114"',
description: "Color of the dots representing land.",
},
@@ -172,13 +172,13 @@ import { Globe } from "$lib/motion-core";
data={[
{
key: "color",
- type: "THREE.ColorRepresentation",
+ type: "string | number | readonly [number, number, number] | { r: number; g: number; b: number }",
default: '"#111113"',
description: "Base color before the rim glow is applied.",
},
{
key: "rimColor",
- type: "THREE.ColorRepresentation",
+ type: "string | number | readonly [number, number, number] | { r: number; g: number; b: number }",
default: '"#FF6900"',
description: "Color tint used for the Fresnel rim.",
},
@@ -205,7 +205,7 @@ import { Globe } from "$lib/motion-core";
data={[
{
key: "color",
- type: "THREE.ColorRepresentation",
+ type: "string | number | readonly [number, number, number] | { r: number; g: number; b: number }",
default: '"#FF6900"',
description: "Color of the atmosphere glow.",
},
diff --git a/apps/web/src/routes/docs/introduction/+page.svx b/apps/web/src/routes/docs/introduction/+page.svx
index ec6e345..2df6af8 100644
--- a/apps/web/src/routes/docs/introduction/+page.svx
+++ b/apps/web/src/routes/docs/introduction/+page.svx
@@ -5,7 +5,7 @@ description: An open-source initiative exploring the intersection of motion and
Motion Core is an open-source initiative born from a simple desire: to bring high-quality, expressive motion to the Svelte ecosystem. It is a curated collection of motion components, heavily inspired by the experimental web, distinct design engineering work, and the creative coding community.
-We are not trying to reinvent the wheel. Instead, we are standing on the shoulders of giants—leveraging the power of **GSAP** for timelines and **Three.js** for WebGL—and wrapping them in the developer experience that Svelte 5 provides.
+We are not trying to reinvent the wheel. Instead, we are standing on the shoulders of giants—leveraging the power of **GSAP** for timelines and **OGL** for WebGL—and wrapping them in the developer experience that Svelte 5 provides.
## The Craft
@@ -21,6 +21,6 @@ Motion Core is, and always will be, free and open source. It is a playground for
- **Svelte 5**: Utilizing runes and snippets for reactive, modern UI logic.
- **Tailwind CSS v4**: For styling that is as fluid as the animations.
-- **GSAP & Three.js**: The industry standards for heavy-lifting motion and 3D.
+- **GSAP & OGL**: The industry standards for heavy-lifting motion and 3D.
This is not just a library; it is a starting point for your own creative journey. Use the CLI to grab what you need, and make it your own.
diff --git a/apps/web/static/registry/components.json b/apps/web/static/registry/components.json
index b2e57e4..6dccacf 100644
--- a/apps/web/static/registry/components.json
+++ b/apps/web/static/registry/components.json
@@ -30,7 +30,7 @@
"components/glitter-cloth/GlitterCloth.svelte": "PHNjcmlwdCBsYW5nPSJ0cyI+CglpbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzIH0gZnJvbSAic3ZlbHRlIjsKCWltcG9ydCBTY2VuZSBmcm9tICIuL0dsaXR0ZXJDbG90aFNjZW5lLnN2ZWx0ZSI7CglpbXBvcnQgeyBjbiB9IGZyb20gIi4uL3V0aWxzL2NuIjsKCgl0eXBlIFNjZW5lUHJvcHMgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgU2NlbmU+OwoKCWludGVyZmFjZSBQcm9wcyB7CgkJLyoqCgkJICogQWRkaXRpb25hbCBDU1MgY2xhc3NlcyBmb3IgdGhlIGNvbnRhaW5lci4KCQkgKi8KCQljbGFzcz86IHN0cmluZzsKCQkvKioKCQkgKiBQcmltYXJ5IGNvbG9yIHVzZWQgdG8gZGVyaXZlIHRoZSBmdWxsIHNoYWRlciBwYWxldHRlLgoJCSAqIEBkZWZhdWx0ICIjRkY2OTAwIgoJCSAqLwoJCWNvbG9yPzogU2NlbmVQcm9wc1siY29sb3IiXTsKCQkvKioKCQkgKiBTcGVlZCBtdWx0aXBsaWVyIGZvciB0aGUgZnVsbCBzaGFkZXIgYW5pbWF0aW9uIHRpbWVsaW5lLgoJCSAqIEBkZWZhdWx0IDEuMAoJCSAqLwoJCXNwZWVkPzogU2NlbmVQcm9wc1sic3BlZWQiXTsKCQkvKioKCQkgKiBHbG9iYWwgaW50ZW5zaXR5IG11bHRpcGxpZXIgZm9yIHRoZSBiYXNlIHNpbGsgY29sb3IuCgkJICogQGRlZmF1bHQgMS4wCgkJICovCgkJYnJpZ2h0bmVzcz86IFNjZW5lUHJvcHNbImJyaWdodG5lc3MiXTsKCQkvKioKCQkgKiBPcGFjaXR5IG9mIHRoZSB2aXZpZC1saWdodCBnbGl0dGVyIGJsZW5kLgoJCSAqIEBkZWZhdWx0IDAuMDIKCQkgKi8KCQlibGVuZFN0cmVuZ3RoPzogU2NlbmVQcm9wc1siYmxlbmRTdHJlbmd0aCJdOwoJCS8qKgoJCSAqIFNwYXRpYWwgc2NhbGUgZm9yIHNpbXBsZXggbm9pc2Ugc2FtcGxpbmcuCgkJICogTG93ZXIgdmFsdWVzIGNyZWF0ZSBmaW5lciBnbGl0dGVyLgoJCSAqIEBkZWZhdWx0IDQuMAoJCSAqLwoJCW5vaXNlU2NhbGU/OiBTY2VuZVByb3BzWyJub2lzZVNjYWxlIl07CgkJLyoqCgkJICogQmFzZSBzdHJlbmd0aCBvZiB2aWduZXR0ZSBmYWxsb2ZmLgoJCSAqIEBkZWZhdWx0IDE1LjAKCQkgKi8KCQl2aWduZXR0ZVN0cmVuZ3RoPzogU2NlbmVQcm9wc1sidmlnbmV0dGVTdHJlbmd0aCJdOwoJCS8qKgoJCSAqIFZpZ25ldHRlIGN1cnZlIGV4cG9uZW50LgoJCSAqIExvd2VyIHZhbHVlcyBwcm9kdWNlIGEgc29mdGVyIHJvbGxvZmYuCgkJICogQGRlZmF1bHQgMC4yNQoJCSAqLwoJCXZpZ25ldHRlUG93ZXI/OiBTY2VuZVByb3BzWyJ2aWduZXR0ZVBvd2VyIl07CgkJLyoqCgkJICogT3BhY2l0eSBvZiB0aGUgdmlnbmV0dGUgZWZmZWN0LgoJCSAqIGAwYCBkaXNhYmxlcyB2aWduZXR0ZSBpbmZsdWVuY2UsIGAxYCBhcHBsaWVzIGZ1bGwgdmlnbmV0dGUuCgkJICogQGRlZmF1bHQgMS4wCgkJICovCgkJdmlnbmV0dGVPcGFjaXR5PzogU2NlbmVQcm9wc1sidmlnbmV0dGVPcGFjaXR5Il07CgkJW2tleTogc3RyaW5nXTogdW5rbm93bjsKCX0KCglsZXQgewoJCWNsYXNzOiBjbGFzc05hbWUgPSAiIiwKCQljb2xvciA9ICIjRkY2OTAwIiwKCQlzcGVlZCA9IDEuMCwKCQlicmlnaHRuZXNzID0gMS4wLAoJCWJsZW5kU3RyZW5ndGggPSAwLjAyLAoJCW5vaXNlU2NhbGUgPSA0LjAsCgkJdmlnbmV0dGVTdHJlbmd0aCA9IDE1LjAsCgkJdmlnbmV0dGVQb3dlciA9IDAuMjUsCgkJdmlnbmV0dGVPcGFjaXR5ID0gMS4wLAoJCS4uLnJlc3QKCX06IFByb3BzID0gJHByb3BzKCk7Cjwvc2NyaXB0PgoKPGRpdiBjbGFzcz17Y24oInJlbGF0aXZlIGgtZnVsbCB3LWZ1bGwgb3ZlcmZsb3ctaGlkZGVuIiwgY2xhc3NOYW1lKX0gey4uLnJlc3R9PgoJPGRpdiBjbGFzcz0iYWJzb2x1dGUgaW5zZXQtMCB6LTAiPgoJCTxTY2VuZQoJCQl7Y29sb3J9CgkJCXtzcGVlZH0KCQkJe2JyaWdodG5lc3N9CgkJCXtibGVuZFN0cmVuZ3RofQoJCQl7bm9pc2VTY2FsZX0KCQkJe3ZpZ25ldHRlU3RyZW5ndGh9CgkJCXt2aWduZXR0ZVBvd2VyfQoJCQl7dmlnbmV0dGVPcGFjaXR5fQoJCS8+Cgk8L2Rpdj4KPC9kaXY+Cg==",
"components/glitter-cloth/GlitterClothScene.svelte": "<script lang="ts">
	import { onMount } from "svelte";
	import {
		Camera,
		Mesh,
		Program,
		Renderer,
		Transform,
		Triangle,
		Vec2,
		Vec3,
	} from "ogl";
	import {
		clamp01,
		type ColorRepresentation,
		srgbToLinear,
		toRgb,
	} from "../helpers/color";

	interface Props {
		/**
		 * Primary color used to derive the full shader palette.
		 * @default "#FF6900"
		 */
		color?: ColorRepresentation;
		/**
		 * Speed multiplier for the full shader animation timeline.
		 * @default 1.0
		 */
		speed?: number;
		/**
		 * Global intensity multiplier for the base silk color.
		 * @default 1.0
		 */
		brightness?: number;
		/**
		 * Opacity of the vivid-light glitter blend.
		 * @default 0.02
		 */
		blendStrength?: number;
		/**
		 * Spatial scale for simplex noise sampling.
		 * Lower values create finer glitter.
		 * @default 4.0
		 */
		noiseScale?: number;
		/**
		 * Base strength of vignette falloff.
		 * @default 15.0
		 */
		vignetteStrength?: number;
		/**
		 * Vignette curve exponent.
		 * Lower values produce a softer rolloff.
		 * @default 0.25
		 */
		vignettePower?: number;
		/**
		 * Opacity of the vignette effect.
		 * `0` disables vignette influence, `1` applies full vignette.
		 * @default 1.0
		 */
		vignetteOpacity?: number;
	}

	let {
		color = "#FF6900",
		speed = 1.0,
		brightness = 1.0,
		blendStrength = 0.02,
		noiseScale = 4.0,
		vignetteStrength = 15.0,
		vignettePower = 0.25,
		vignetteOpacity = 1.0,
	}: Props = $props();

	type UniformState = {
		uTime: { value: number };
		uResolution: { value: Vec2 };
		uPrimaryColor: { value: Vec3 };
		uAccentColor: { value: Vec3 };
		uShadowColor: { value: Vec3 };
		uSpeed: { value: number };
		uBrightness: { value: number };
		uBlendStrength: { value: number };
		uNoiseScale: { value: number };
		uVignetteStrength: { value: number };
		uVignettePower: { value: number };
		uVignetteOpacity: { value: number };
	};

	const DEFAULT_PRIMARY: [number, number, number] = [1, 105 / 255, 0];

	let canvas = $state<HTMLCanvasElement>();
	let uniforms = $state<UniformState>();

	const toLinearTriplet = (
		value: [number, number, number],
	): [number, number, number] => [
		srgbToLinear(value[0]),
		srgbToLinear(value[1]),
		srgbToLinear(value[2]),
	];

	const rgbToHsl = (
		r: number,
		g: number,
		b: number,
	): { h: number; s: number; l: number } => {
		const max = Math.max(r, g, b);
		const min = Math.min(r, g, b);
		const l = (max + min) * 0.5;

		if (max === min) return { h: 0, s: 0, l };

		const d = max - min;
		const s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
		let h = 0;

		switch (max) {
			case r:
				h = (g - b) / d + (g < b ? 6 : 0);
				break;
			case g:
				h = (b - r) / d + 2;
				break;
			default:
				h = (r - g) / d + 4;
				break;
		}

		h /= 6;
		return { h, s, l };
	};

	const hue2rgb = (p: number, q: number, tInput: number) => {
		let t = tInput;
		if (t < 0) t += 1;
		if (t > 1) t -= 1;
		if (t < 1 / 6) return p + (q - p) * 6 * t;
		if (t < 1 / 2) return q;
		if (t < 2 / 3) return p + (q - p) * (2 / 3 - t) * 6;
		return p;
	};

	const hslToRgb = (
		h: number,
		s: number,
		l: number,
	): [number, number, number] => {
		if (s === 0) return [l, l, l];
		const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
		const p = 2 * l - q;
		return [
			hue2rgb(p, q, h + 1 / 3),
			hue2rgb(p, q, h),
			hue2rgb(p, q, h - 1 / 3),
		];
	};

	const derivePalette = (
		primary: [number, number, number],
	): {
		accent: [number, number, number];
		shadow: [number, number, number];
	} => {
		const hsl = rgbToHsl(primary[0], primary[1], primary[2]);
		const accent = hslToRgb(
			(hsl.h + 0.04) % 1,
			clamp01(hsl.s * 1.1),
			clamp01(hsl.l * 1.1 + 0.04),
		);
		const shadow = hslToRgb(
			hsl.h,
			clamp01(hsl.s * 0.55),
			clamp01(hsl.l * 0.45),
		);
		return { accent, shadow };
	};

	const vertexShader = `
		attribute vec2 uv;
		attribute vec2 position;
		varying vec2 vUv;

		void main() {
			vUv = uv;
			gl_Position = vec4(position, 0.0, 1.0);
		}
	`;

	const fragmentShader = `
		precision highp float;
		varying vec2 vUv;

		uniform float uTime;
		uniform vec2 uResolution;
		uniform vec3 uPrimaryColor;
		uniform vec3 uAccentColor;
		uniform vec3 uShadowColor;
		uniform float uSpeed;
		uniform float uBrightness;
		uniform float uBlendStrength;
		uniform float uNoiseScale;
		uniform float uVignetteStrength;
		uniform float uVignettePower;
		uniform float uVignetteOpacity;

		const float noiseSizeCoeff = 0.61;
		const float noiseDensity = 53.0;

		vec3 mod289(vec3 x) {
			return x - floor(x * (1.0 / 289.0)) * 289.0;
		}

		vec4 mod289(vec4 x) {
			return x - floor(x * (1.0 / 289.0)) * 289.0;
		}

		vec4 permute(vec4 x) {
			return mod289(((x * 34.0) + 1.0) * x);
		}

		vec4 taylorInvSqrt(vec4 r) {
			return 1.79284291400159 - 0.85373472095314 * r;
		}

		float snoise(vec3 v) {
			const vec2 C = vec2(1.0 / 6.0, 1.0 / 3.0);
			const vec4 D = vec4(0.0, 0.5, 1.0, 2.0);

			vec3 i = floor(v + dot(v, C.yyy));
			vec3 x0 = v - i + dot(i, C.xxx);

			vec3 g = step(x0.yzx, x0.xyz);
			vec3 l = 1.0 - g;
			vec3 i1 = min(g.xyz, l.zxy);
			vec3 i2 = max(g.xyz, l.zxy);

			vec3 x1 = x0 - i1 + C.xxx;
			vec3 x2 = x0 - i2 + C.yyy;
			vec3 x3 = x0 - D.yyy;

			i = mod289(i);
			vec4 p = permute(permute(permute(i.z + vec4(0.0, i1.z, i2.z, 1.0)) +
				i.y + vec4(0.0, i1.y, i2.y, 1.0)) +
				i.x + vec4(0.0, i1.x, i2.x, 1.0));

			float n_ = 0.142857142857;
			vec3 ns = n_ * D.wyz - D.xzx;

			vec4 j = p - 49.0 * floor(p * ns.z * ns.z);

			vec4 x_ = floor(j * ns.z);
			vec4 y_ = floor(j - 7.0 * x_);

			vec4 x = x_ * ns.x + ns.yyyy;
			vec4 y = y_ * ns.x + ns.yyyy;
			vec4 h = 1.0 - abs(x) - abs(y);

			vec4 b0 = vec4(x.xy, y.xy);
			vec4 b1 = vec4(x.zw, y.zw);

			vec4 s0 = floor(b0) * 2.0 + 1.0;
			vec4 s1 = floor(b1) * 2.0 + 1.0;
			vec4 sh = -step(h, vec4(0.0));

			vec4 a0 = b0.xzyw + s0.xzyw * sh.xxyy;
			vec4 a1 = b1.xzyw + s1.xzyw * sh.zzww;

			vec3 p0 = vec3(a0.xy, h.x);
			vec3 p1 = vec3(a0.zw, h.y);
			vec3 p2 = vec3(a1.xy, h.z);
			vec3 p3 = vec3(a1.zw, h.w);

			vec4 norm = taylorInvSqrt(vec4(dot(p0, p0), dot(p1, p1), dot(p2, p2), dot(p3, p3)));
			p0 *= norm.x;
			p1 *= norm.y;
			p2 *= norm.z;
			p3 *= norm.w;

			vec4 m = max(noiseSizeCoeff - vec4(dot(x0, x0), dot(x1, x1), dot(x2, x2), dot(x3, x3)), 0.0);
			m = m * m;
			return noiseDensity * dot(m * m, vec4(dot(p0, x0), dot(p1, x1), dot(p2, x2), dot(p3, x3)));
		}

		float vividLight(float s, float d) {
			return (s < 0.5) ? 1.0 - (1.0 - d) / (2.0 * s) : d / (2.0 * (1.0 - s));
		}

		vec3 vividLight(vec3 s, vec3 d) {
			vec3 c;
			c.x = vividLight(s.x, d.x);
			c.y = vividLight(s.y, d.y);
			c.z = vividLight(s.z, d.z);
			return c;
		}

		float vignette(vec2 uv, float strength, float power) {
			uv *= 1.0 - uv.yx;
			float vig = uv.x * uv.y * strength;
			vig = pow(vig, power);
			return vig;
		}

		vec3 linearToSrgb(vec3 color) {
			vec3 safe = max(color, vec3(0.0));
			vec3 low = safe * 12.92;
			vec3 high = 1.055 * pow(safe, vec3(1.0 / 2.4)) - 0.055;
			vec3 cutoff = step(vec3(0.0031308), safe);
			return mix(low, high, cutoff);
		}

		void mainImage(out vec4 col, vec2 fragCoord) {
			float time = uTime * uSpeed;
			vec2 resolution = uResolution;

			vec2 uv = fragCoord / resolution.y;
			float t = 0.5 * time;
			uv.y += 0.03 * sin(8.0 * uv.x - t);

			float f = 0.6 + 0.4 * sin(5.0 * (uv.x + uv.y + cos(3.0 * uv.x + 5.0 * uv.y) + 0.02 * t) +
				sin(20.0 * (uv.x + uv.y - 0.1 * t)));

			float b = uBrightness;
			col = vec4(uPrimaryColor * b, 1.0) * vec4(f);

			uv = fragCoord.xy / resolution.xy;
			float vig = vignette(uv, uVignetteStrength, uVignettePower);
			float vignetteMask = mix(1.0, vig, uVignetteOpacity);

			float fadeLR = 0.7 - abs(uv.x - 0.4);
			float fadeTB = 1.1 - uv.y;
			vec3 pos = vec3(uv * vec2(3.0, 1.0) - vec2(0.0, time * 0.00005), time * 0.006);

			float n = fadeLR * fadeTB * smoothstep(0.50, 1.0, snoise(pos * resolution.y / uNoiseScale)) * 8.0;
			vec3 noiseGreyShifted = min((vec3(n) + 1.0) / 3.0 + 0.3, vec3(1.0)) * 0.91;

			vec3 mixed = col.xyz;
			mixed = mix(col.xyz, vividLight(noiseGreyShifted, col.xyz), uBlendStrength);
			col = vec4(mixed, 1.0) * vignetteMask;

			float k = (sin(time / 1.0) + 1.0) / 4.0 + 0.75;
			col.rgb -= uAccentColor * ((1.0 - uv.y) * 0.1 * k);
			col.rgb -= vec3(uShadowColor.r, uShadowColor.g * 0.8, uShadowColor.b) * (uv.y / 8.0);
			col.a = 1.0;
		}

		void main() {
			vec4 fragColor;
			vec2 fragCoord = vUv * uResolution.xy;
			mainImage(fragColor, fragCoord);
			fragColor.rgb = linearToSrgb(fragColor.rgb);
			gl_FragColor = fragColor;
		}
	`;

	$effect(() => {
		if (!uniforms) return;
		const primarySrgb = toRgb(color, DEFAULT_PRIMARY);
		const paletteSrgb = derivePalette(primarySrgb);
		const primary = toLinearTriplet(primarySrgb);
		const accent = toLinearTriplet(paletteSrgb.accent);
		const shadow = toLinearTriplet(paletteSrgb.shadow);
		uniforms.uPrimaryColor.value.set(primary[0], primary[1], primary[2]);
		uniforms.uAccentColor.value.set(accent[0], accent[1], accent[2]);
		uniforms.uShadowColor.value.set(shadow[0], shadow[1], shadow[2]);
		uniforms.uSpeed.value = speed;
		uniforms.uBrightness.value = brightness;
		uniforms.uBlendStrength.value = blendStrength;
		uniforms.uNoiseScale.value = noiseScale;
		uniforms.uVignetteStrength.value = vignetteStrength;
		uniforms.uVignettePower.value = vignettePower;
		uniforms.uVignetteOpacity.value = vignetteOpacity;
	});

	onMount(() => {
		const targetCanvas = canvas;
		if (!targetCanvas) return;

		const renderer = new Renderer({
			canvas: targetCanvas,
			alpha: true,
			dpr: typeof window !== "undefined" ? window.devicePixelRatio : 1,
		});
		const gl = renderer.gl;
		gl.clearColor(0, 0, 0, 0);

		const camera = new Camera(gl);
		camera.position.z = 1;

		const scene = new Transform();
		const geometry = new Triangle(gl);

		const primarySrgb = toRgb(color, DEFAULT_PRIMARY);
		const paletteSrgb = derivePalette(primarySrgb);
		const primary = toLinearTriplet(primarySrgb);
		const accent = toLinearTriplet(paletteSrgb.accent);
		const shadow = toLinearTriplet(paletteSrgb.shadow);
		const localUniforms: UniformState = {
			uTime: { value: 0 },
			uResolution: { value: new Vec2(1, 1) },
			uPrimaryColor: { value: new Vec3(primary[0], primary[1], primary[2]) },
			uAccentColor: {
				value: new Vec3(accent[0], accent[1], accent[2]),
			},
			uShadowColor: {
				value: new Vec3(shadow[0], shadow[1], shadow[2]),
			},
			uSpeed: { value: speed },
			uBrightness: { value: brightness },
			uBlendStrength: { value: blendStrength },
			uNoiseScale: { value: noiseScale },
			uVignetteStrength: { value: vignetteStrength },
			uVignettePower: { value: vignettePower },
			uVignetteOpacity: { value: vignetteOpacity },
		};
		uniforms = localUniforms;

		const program = new Program(gl, {
			vertex: vertexShader,
			fragment: fragmentShader,
			uniforms: localUniforms,
			transparent: true,
			depthTest: false,
			depthWrite: false,
		});

		const mesh = new Mesh(gl, { geometry, program });
		mesh.setParent(scene);

		const resize = () => {
			const host = targetCanvas.parentElement ?? targetCanvas;
			const { width: hostWidth, height: hostHeight } =
				host.getBoundingClientRect();
			const width = Math.max(1, Math.round(hostWidth));
			const height = Math.max(1, Math.round(hostHeight));
			renderer.setSize(width, height);
			localUniforms.uResolution.value.set(width, height);
		};

		resize();
		const observer = new ResizeObserver(resize);
		observer.observe(targetCanvas);
		if (targetCanvas.parentElement)
			observer.observe(targetCanvas.parentElement);

		let raf = 0;
		let previous = 0;
		const tick = (now: number) => {
			const delta = previous ? (now - previous) / 1000 : 0;
			previous = now;
			localUniforms.uTime.value += delta;

			renderer.render({ scene, camera });
			raf = window.requestAnimationFrame(tick);
		};

		raf = window.requestAnimationFrame(tick);

		return () => {
			window.cancelAnimationFrame(raf);
			observer.disconnect();
		};
	});
</script>

<canvas
	bind:this={canvas}
	class="absolute inset-0 block h-full w-full"
	style="width:100%;height:100%;"
	aria-hidden="true"
></canvas>
",
"components/globe/Globe.svelte": "PHNjcmlwdCBsYW5nPSJ0cyI+CglpbXBvcnQgU2NlbmUgZnJvbSAiLi9HbG9iZVNjZW5lLnN2ZWx0ZSI7CglpbXBvcnQgeyBjbiB9IGZyb20gIi4uL3V0aWxzL2NuIjsKCWltcG9ydCB0eXBlIHsgQ29tcG9uZW50UHJvcHMsIFNuaXBwZXQgfSBmcm9tICJzdmVsdGUiOwoJaW1wb3J0IHR5cGUgeyBHbG9iZU1hcmtlciwgR2xvYmVNYXJrZXJUb29sdGlwQ29udGV4dCB9IGZyb20gIi4vdHlwZXMiOwoKCXR5cGUgU2NlbmVQcm9wcyA9IENvbXBvbmVudFByb3BzPHR5cGVvZiBTY2VuZT47CgoJaW50ZXJmYWNlIFByb3BzIHsKCQkvKioKCQkgKiBBZGRpdGlvbmFsIENTUyBjbGFzc2VzIGZvciB0aGUgY29udGFpbmVyLgoJCSAqLwoJCWNsYXNzPzogc3RyaW5nOwoJCS8qKgoJCSAqIFJhZGl1cyBvZiB0aGUgc3BoZXJlLgoJCSAqIEBkZWZhdWx0IDIKCQkgKi8KCQlyYWRpdXM/OiBTY2VuZVByb3BzWyJyYWRpdXMiXTsKCQkvKioKCQkgKiBPcHRpb25hbCBvdmVycmlkZXMgZm9yIHRoZSBGcmVzbmVsIHNoYWRlciB1bmlmb3Jtcy4KCQkgKi8KCQlmcmVzbmVsQ29uZmlnPzogU2NlbmVQcm9wc1siZnJlc25lbENvbmZpZyJdOwoJCS8qKgoJCSAqIE9wdGlvbmFsIGNvbmZpZ3VyYXRpb24gZm9yIHRoZSBhdG1vc3BoZXJpYyBoYWxvLgoJCSAqLwoJCWF0bW9zcGhlcmVDb25maWc/OiBTY2VuZVByb3BzWyJhdG1vc3BoZXJlQ29uZmlnIl07CgkJLyoqCgkJICogTnVtYmVyIG9mIHBvaW50cyByZW5kZXJlZCBvbiB0aGUgc3VyZmFjZS4KCQkgKiBAZGVmYXVsdCAxNTAwMAoJCSAqLwoJCXBvaW50Q291bnQ/OiBTY2VuZVByb3BzWyJwb2ludENvdW50Il07CgkJLyoqCgkJICogQ29sb3IgYXBwbGllZCB0byBwb2ludHMgdGhhdCBmYWxsIG9uIGxhbmQuCgkJICogQGRlZmF1bHQgIiNmNzcxMTQiCgkJICovCgkJbGFuZFBvaW50Q29sb3I/OiBTY2VuZVByb3BzWyJsYW5kUG9pbnRDb2xvciJdOwoJCS8qKgoJCSAqIFNpemUgb2YgZWFjaCBwb2ludCBpbiB3b3JsZCB1bml0cy4KCQkgKiBAZGVmYXVsdCAwLjA1CgkJICovCgkJcG9pbnRTaXplPzogU2NlbmVQcm9wc1sicG9pbnRTaXplIl07CgkJLyoqCgkJICogV2hldGhlciB0aGUgZ2xvYmUgc2hvdWxkIGF1dG8tcm90YXRlLgoJCSAqIEBkZWZhdWx0IHRydWUKCQkgKi8KCQlhdXRvUm90YXRlPzogU2NlbmVQcm9wc1siYXV0b1JvdGF0ZSJdOwoJCS8qKgoJCSAqIFdoZXRoZXIgdG8gbG9jayB0aGUgY2FtZXJhJ3MgcG9sYXIgYW5nbGUgKHZlcnRpY2FsIHJvdGF0aW9uKS4KCQkgKiBJZiB0cnVlLCBsaW1pdHMgdGhlIHZlcnRpY2FsIHZpZXcgdG8gYSBuYXJyb3cgYmFuZC4KCQkgKiBAZGVmYXVsdCB0cnVlCgkJICovCgkJbG9ja2VkUG9sYXJBbmdsZT86IGJvb2xlYW47CgkJLyoqCgkJICogQXJyYXkgb2YgbWFya2VycyB0byBkaXNwbGF5IG9uIHRoZSBnbG9iZS4KCQkgKi8KCQltYXJrZXJzPzogR2xvYmVNYXJrZXJbXTsKCQkvKioKCQkgKiBPcHRpb25hbCBjdXN0b20gdG9vbHRpcCByZW5kZXJlciBmb3IgbWFya2Vycy4KCQkgKiBSZWNlaXZlcyBtYXJrZXIgZGF0YSBhbmQgdmlzaWJpbGl0eSBjb250ZXh0LgoJCSAqLwoJCW1hcmtlclRvb2x0aXA/OiBTbmlwcGV0PFtHbG9iZU1hcmtlclRvb2x0aXBDb250ZXh0XT47CgkJLyoqCgkJICogQ29vcmRpbmF0ZXMgW2xhdCwgbG9uXSB0byBmb2N1cyB0aGUgY2FtZXJhIG9uLgoJCSAqIFdoZW4gc2V0LCBhdXRvLXJvdGF0aW9uIHdpbGwgYmUgZGlzYWJsZWQgdGVtcG9yYXJpbHkuCgkJICovCgkJZm9jdXNPbj86IFtudW1iZXIsIG51bWJlcl0gfCBudWxsOwoKCQlba2V5OiBzdHJpbmddOiB1bmtub3duOwoJfQoKCWxldCB7CgkJY2xhc3M6IGNsYXNzTmFtZSA9ICIiLAoJCXJhZGl1cyA9IDIsCgkJZnJlc25lbENvbmZpZywKCQlhdG1vc3BoZXJlQ29uZmlnLAoJCXBvaW50Q291bnQsCgkJbGFuZFBvaW50Q29sb3IsCgkJcG9pbnRTaXplLAoJCWF1dG9Sb3RhdGUgPSB0cnVlLAoJCWxvY2tlZFBvbGFyQW5nbGUgPSB0cnVlLAoJCW1hcmtlcnMgPSBbXSwKCQltYXJrZXJUb29sdGlwLAoJCWZvY3VzT24gPSBudWxsLAoJCS4uLnJlc3QKCX06IFByb3BzID0gJHByb3BzKCk7Cjwvc2NyaXB0PgoKPGRpdiBjbGFzcz17Y24oInJlbGF0aXZlIGgtZnVsbCB3LWZ1bGwgb3ZlcmZsb3ctaGlkZGVuIiwgY2xhc3NOYW1lKX0gey4uLnJlc3R9PgoJPGRpdiBjbGFzcz0iYWJzb2x1dGUgaW5zZXQtMCB6LTAiPgoJCTxTY2VuZQoJCQl7cmFkaXVzfQoJCQl7ZnJlc25lbENvbmZpZ30KCQkJe2F0bW9zcGhlcmVDb25maWd9CgkJCXtwb2ludENvdW50fQoJCQl7bGFuZFBvaW50Q29sb3J9CgkJCXtwb2ludFNpemV9CgkJCXthdXRvUm90YXRlfQoJCQl7bG9ja2VkUG9sYXJBbmdsZX0KCQkJe21hcmtlcnN9CgkJCXttYXJrZXJUb29sdGlwfQoJCQl7Zm9jdXNPbn0KCQkvPgoJPC9kaXY+CjwvZGl2Pgo=",
- "components/globe/GlobeScene.svelte": "<script lang="ts">
	import { onMount } from "svelte";
	import {
		Camera,
		Mesh,
		Program,
		Renderer,
		Texture,
		Transform,
		Triangle,
		Vec2,
		Vec3,
	} from "ogl";
	import { gsap } from "gsap/dist/gsap";
	import type { Snippet } from "svelte";
	import landTextureUrl from "../assets/land-texture.png";
	import { type ColorRepresentation, toLinearRgb } from "../helpers/color";
	import type { GlobeMarker, GlobeMarkerTooltipContext } from "./types";
	import GlobeMarkerItem from "./GlobeMarkerItem.svelte";

	interface FresnelConfig {
		/**
		 * Base body color for the globe surface.
		 * @default "#111113"
		 */
		color?: ColorRepresentation;
		/**
		 * Accent color applied by the Fresnel rim.
		 * @default "#FF6900"
		 */
		rimColor?: ColorRepresentation;
		/**
		 * Controls how tight the Fresnel rim hug is.
		 * Higher values yield a thinner outline.
		 * @default 6
		 */
		rimPower?: number;
		/**
		 * Intensity multiplier for the Fresnel rim color.
		 * @default 1.5
		 */
		rimIntensity?: number;
	}

	interface AtmosphereConfig {
		/**
		 * Color of the atmosphere glow.
		 * @default "#FF6900"
		 */
		color?: ColorRepresentation;
		/**
		 * Size of the atmosphere relative to the globe radius.
		 * @default 1.1
		 */
		scale?: number;
		/**
		 * Falloff power of the glow. Higher values mean a sharper edge.
		 * @default 12.0
		 */
		power?: number;
		/**
		 * Base coefficient for the intensity calculation.
		 * Controls how far the glow extends inwards.
		 * @default 0.9
		 */
		coefficient?: number;
		/**
		 * Global intensity multiplier.
		 * @default 2.0
		 */
		intensity?: number;
	}

	interface Props {
		/**
		 * Radius of the sphere.
		 * @default 2
		 */
		radius: number;
		/**
		 * Optional overrides for the Fresnel shader uniforms.
		 */
		fresnelConfig?: FresnelConfig;
		/**
		 * Optional configuration for the atmospheric halo.
		 */
		atmosphereConfig?: AtmosphereConfig;
		/**
		 * Number of points rendered along the globe surface.
		 * @default 15000
		 */
		pointCount?: number;
		/**
		 * Size of each point in world units.
		 * @default 0.05
		 */
		pointSize?: number;
		/**
		 * Color applied to points representing land.
		 * @default "#f77114"
		 */
		landPointColor?: ColorRepresentation;
		/**
		 * Whether the globe should auto-rotate.
		 * @default true
		 */
		autoRotate?: boolean;
		/**
		 * Whether to lock the camera's polar angle.
		 * @default true
		 */
		lockedPolarAngle?: boolean;
		/**
		 * Markers to display on the globe.
		 */
		markers?: GlobeMarker[];
		/**
		 * Optional custom tooltip renderer for markers.
		 */
		markerTooltip?: Snippet<[GlobeMarkerTooltipContext]>;
		/**
		 * Coordinates [lat, lon] to focus on.
		 */
		focusOn?: [number, number] | null;
	}

	interface ProjectedMarker {
		marker: GlobeMarker;
		index: number;
		screenX: number;
		screenY: number;
		visibility: number;
		sizePx: number;
	}

	interface UniformUpdaterState {
		radius: number;
		pointCount: number;
		pointSize: number;
		landPointColor: ColorRepresentation;
		fresnelConfig: Required<FresnelConfig>;
		atmosphereConfig: Required<AtmosphereConfig>;
	}

	const PI = Math.PI;
	const DEG2RAD = PI / 180;
	const EPSILON = 1e-6;
	const COBE_GLOBE_RADIUS = 0.8;
	const MARKER_ELEVATION = 0.05;
	const AUTO_ROTATE_SPEED = (2 * PI) / 30;
	const ROTATE_SENSITIVITY = 0.005;
	const SMOOTHING_STRENGTH = 14;
	const LOCKED_POLAR_ANGLE = 1.5;
	const LOCKED_THETA = Math.asin(Math.cos(LOCKED_POLAR_ANGLE));
	const MIN_THETA = -PI * 0.5 + 0.001;
	const MAX_THETA = PI * 0.5 - 0.001;
	const VISIBILITY_MIN_DOT = 0.24;
	const VISIBILITY_MAX_DOT = 0.48;

	const defaultFresnelConfig: Required<FresnelConfig> = {
		color: "#111113",
		rimColor: "#FF6900",
		rimPower: 6,
		rimIntensity: 1.5,
	};

	const defaultAtmosphereConfig: Required<AtmosphereConfig> = {
		color: "#FF6900",
		scale: 1.1,
		power: 12.0,
		coefficient: 0.9,
		intensity: 2.0,
	};

	let {
		radius,
		fresnelConfig = {},
		atmosphereConfig = {},
		pointCount = 15000,
		pointSize = 0.05,
		landPointColor = "#f77114",
		autoRotate = true,
		lockedPolarAngle = true,
		markers = [],
		markerTooltip,
		focusOn = null,
	}: Props = $props();

	let canvas = $state<HTMLCanvasElement>();
	let projectedMarkers = $state<ProjectedMarker[]>([]);

	const resolvedFresnelConfig = $derived({
		...defaultFresnelConfig,
		...fresnelConfig,
	});
	const resolvedAtmosphereConfig = $derived({
		...defaultAtmosphereConfig,
		...atmosphereConfig,
	});

	let updateUniforms = $state<((state: UniformUpdaterState) => void) | null>(
		null,
	);
	let syncFocusTarget = $state<
		((target: [number, number] | null) => void) | null
	>(null);
	let focusTween: gsap.core.Tween | null = null;

	const clamp = (value: number, min: number, max: number) =>
		Math.min(max, Math.max(min, value));

	const clampTheta = (value: number, lockPolar: boolean) =>
		lockPolar ? LOCKED_THETA : clamp(value, MIN_THETA, MAX_THETA);

	const smoothstep = (value: number, edge0: number, edge1: number) => {
		if (Math.abs(edge1 - edge0) <= EPSILON) {
			return value >= edge1 ? 1 : 0;
		}
		const t = clamp((value - edge0) / (edge1 - edge0), 0, 1);
		return t * t * (3 - 2 * t);
	};

	const toScale = (nextRadius: number) => Math.max(0.001, nextRadius / 2);
	const toPointRadius = (nextPointSize: number) =>
		Math.max(0.001, nextPointSize * 0.16);

	function normalizeAngle(value: number): number {
		const wrapped = (((value + PI) % (2 * PI)) + 2 * PI) % (2 * PI);
		return wrapped - PI;
	}

	function shortestAngleTarget(current: number, next: number): number {
		const delta = normalizeAngle(next - current);
		return current + delta;
	}

	function lonLatToCartesian(lon: number, lat: number, r: number) {
		const lonRad = lon * DEG2RAD;
		const latRad = lat * DEG2RAD;

		const y = r * Math.sin(latRad);
		const rXZ = r * Math.cos(latRad);
		const x = rXZ * Math.sin(lonRad);
		const z = rXZ * Math.cos(lonRad);

		return { x, y, z };
	}

	function cartesianToRotation(x: number, y: number, z: number) {
		const length = Math.hypot(x, y, z);
		if (length <= EPSILON) {
			return { phi: 0, theta: 0 };
		}
		const nx = x / length;
		const ny = y / length;
		const nz = z / length;

		return {
			phi: Math.atan2(-nx, nz),
			theta: Math.asin(clamp(ny, -1, 1)),
		};
	}

	function applyRotation(
		x: number,
		y: number,
		z: number,
		phi: number,
		theta: number,
	) {
		const cx = Math.cos(theta);
		const cy = Math.cos(phi);
		const sx = Math.sin(theta);
		const sy = Math.sin(phi);

		return {
			rx: cy * x + sy * z,
			ry: sy * sx * x + cx * y - cy * sx * z,
			rz: -sy * cx * x + sx * y + cy * cx * z,
		};
	}

	function cubicBezierAt(
		t: number,
		p0: number,
		p1: number,
		p2: number,
		p3: number,
	): number {
		const u = 1 - t;
		return (
			u * u * u * p0 + 3 * u * u * t * p1 + 3 * u * t * t * p2 + t * t * t * p3
		);
	}

	function cubicBezierDerivativeAt(
		t: number,
		p0: number,
		p1: number,
		p2: number,
		p3: number,
	): number {
		const u = 1 - t;
		return (
			3 * u * u * (p1 - p0) + 6 * u * t * (p2 - p1) + 3 * t * t * (p3 - p2)
		);
	}

	function dynamicEase(value: number): number {
		const clamped = clamp(value, 0, 1);
		let t = clamped;
		for (let i = 0; i < 5; i++) {
			const x = cubicBezierAt(t, 0, 0.625, 0, 1);
			const dx = cubicBezierDerivativeAt(t, 0, 0.625, 0, 1);
			if (Math.abs(dx) < 1e-6) break;
			t = clamp(t - (x - clamped) / dx, 0, 1);
		}
		return cubicBezierAt(t, 0, 0.05, 1, 1);
	}

	$effect(() => {
		if (!updateUniforms) return;
		updateUniforms({
			radius,
			pointCount,
			pointSize,
			landPointColor,
			fresnelConfig: resolvedFresnelConfig,
			atmosphereConfig: resolvedAtmosphereConfig,
		});
	});

	$effect(() => {
		if (!syncFocusTarget) return;
		syncFocusTarget(focusOn);
	});

	onMount(() => {
		const targetCanvas = canvas;
		if (!targetCanvas) return;

		const renderer = new Renderer({
			canvas: targetCanvas,
			alpha: true,
			antialias: true,
			dpr: typeof window !== "undefined" ? window.devicePixelRatio : 1,
		});
		const gl = renderer.gl;
		gl.clearColor(0, 0, 0, 0);

		const camera = new Camera(gl);
		camera.position.z = 1;

		const scene = new Transform();
		const geometry = new Triangle(gl);
		const landTexture = new Texture(gl, {
			image: new Uint8Array([0, 0, 0, 255]),
			width: 1,
			height: 1,
			format: gl.RGBA,
			type: gl.UNSIGNED_BYTE,
			minFilter: gl.NEAREST,
			magFilter: gl.NEAREST,
			generateMipmaps: false,
			wrapS: gl.REPEAT,
			wrapT: gl.REPEAT,
		});

		const uniforms = {
			uResolution: { value: new Vec2(1, 1) },
			uRotation: { value: new Vec2(0, clampTheta(0, lockedPolarAngle)) },
			uScale: { value: toScale(radius) },
			uDots: { value: Math.max(1, Math.floor(pointCount)) },
			uPointRadius: { value: toPointRadius(pointSize) },
			uBaseColor: { value: new Vec3(0, 0, 0) },
			uRimColor: { value: new Vec3(0, 0, 0) },
			uRimPower: { value: resolvedFresnelConfig.rimPower },
			uRimIntensity: { value: resolvedFresnelConfig.rimIntensity },
			uAtmosphereColor: { value: new Vec3(0, 0, 0) },
			uAtmosphereScale: { value: resolvedAtmosphereConfig.scale },
			uAtmospherePower: { value: resolvedAtmosphereConfig.power },
			uAtmosphereCoefficient: { value: resolvedAtmosphereConfig.coefficient },
			uAtmosphereIntensity: { value: resolvedAtmosphereConfig.intensity },
			uLandPointColor: { value: new Vec3(0, 0, 0) },
			uLandTexture: { value: landTexture },
		};

		const vertexShader = `
			attribute vec2 uv;
			attribute vec2 position;
			varying vec2 vUv;

			void main() {
				vUv = uv;
				gl_Position = vec4(position, 0.0, 1.0);
			}
		`;

		const fragmentShader = `
			precision highp float;

			varying vec2 vUv;

			uniform vec2 uResolution;
			uniform vec2 uRotation;
			uniform float uScale;
			uniform float uDots;
			uniform float uPointRadius;
			uniform vec3 uBaseColor;
			uniform vec3 uRimColor;
			uniform float uRimPower;
			uniform float uRimIntensity;
			uniform vec3 uAtmosphereColor;
			uniform float uAtmosphereScale;
			uniform float uAtmospherePower;
			uniform float uAtmosphereCoefficient;
			uniform float uAtmosphereIntensity;
			uniform vec3 uLandPointColor;
			uniform sampler2D uLandTexture;

			const float kPi = 3.141592653589793;
			const float kTau = 6.283185307179586;
			const float kPhi = 1.618033988749895;
			const float kSqrt5 = 2.23606797749979;
			const float kSphereRadius = 0.8;

			float byDots;

			mat3 rotate(float theta, float phi) {
				float cx = cos(theta);
				float cy = cos(phi);
				float sx = sin(theta);
				float sy = sin(phi);
				return mat3(
					cy, sy * sx, -sy * cx,
					0.0, cx, sx,
					sy, cy * -sx, cy * cx
				);
			}

			vec3 nearestFibonacciLattice(vec3 p, out float m) {
				p = p.xzy;

				float k = max(2.0, floor(log2(kSqrt5 * uDots * kPi * (1.0 - p.z * p.z)) * 0.72021));
				vec2 f = floor(pow(kPhi, k) / kSqrt5 * vec2(1.0, kPhi) + 0.5);
				vec2 br1 = fract((f + 1.0) * (kPhi - 1.0)) * kTau - 3.883222;
				vec2 br2 = -2.0 * f;
				vec2 sp = vec2(atan(p.y, p.x), p.z - 1.0);
				vec2 c = floor(vec2(
					br2.y * sp.x - br1.y * (sp.y * uDots + 1.0),
					-br2.x * sp.x + br1.x * (sp.y * uDots + 1.0)
				) / (br1.x * br2.y - br2.x * br1.y));

				float mindist = kPi;
				vec3 minip = vec3(0.0, 0.0, 1.0);

				for (float s = 0.0; s < 4.0; s += 1.0) {
					vec2 o = vec2(mod(s, 2.0), floor(s * 0.5));
					float idx = dot(f, c + o);
					if (idx > uDots) continue;

					float a = idx;
					float b = 0.0;
					if (a >= 16384.0) a -= 16384.0, b += 0.868872;
					if (a >= 8192.0) a -= 8192.0, b += 0.934436;
					if (a >= 4096.0) a -= 4096.0, b += 0.467218;
					if (a >= 2048.0) a -= 2048.0, b += 0.733609;
					if (a >= 1024.0) a -= 1024.0, b += 0.866804;
					if (a >= 512.0) a -= 512.0, b += 0.433402;
					if (a >= 256.0) a -= 256.0, b += 0.216701;
					if (a >= 128.0) a -= 128.0, b += 0.108351;
					if (a >= 64.0) a -= 64.0, b += 0.554175;
					if (a >= 32.0) a -= 32.0, b += 0.777088;
					if (a >= 16.0) a -= 16.0, b += 0.888544;
					if (a >= 8.0) a -= 8.0, b += 0.944272;
					if (a >= 4.0) a -= 4.0, b += 0.472136;
					if (a >= 2.0) a -= 2.0, b += 0.236068;
					if (a >= 1.0) a -= 1.0, b += 0.618034;

					float theta = fract(b) * kTau;
					float cosphi = 1.0 - 2.0 * idx * byDots;
					float sinphi = sqrt(max(0.0, 1.0 - cosphi * cosphi));
					vec3 samplePoint = vec3(cos(theta) * sinphi, sin(theta) * sinphi, cosphi);

					float dist = length(p - samplePoint);
					if (dist < mindist) {
						mindist = dist;
						minip = samplePoint;
					}
				}

				m = mindist;
				return minip.xzy;
			}

			vec2 pointToMaskUV(vec3 p) {
				float lengthP = length(p);
				if (lengthP <= 0.0) {
					return vec2(0.0, 0.0);
				}

				vec3 n = p / lengthP;

				float nx = n.z;
				float ny = n.y;
				float nz = -n.x;

				float gPhi = asin(clamp(ny, -1.0, 1.0));
				float cosPhi = cos(gPhi);

				float gTheta = 0.0;
				if (abs(cosPhi) > 1e-6) {
					float thetaInput = clamp(-nx / cosPhi, -1.0, 1.0);
					gTheta = acos(thetaInput);
					if (nz < 0.0) {
						gTheta = -gTheta;
					}
				}

				return vec2(
					fract((gTheta * 0.5) / kPi),
					fract(-(gPhi / kPi + 0.5))
				);
			}

			vec3 linearToSrgb(vec3 color) {
				vec3 safe = max(color, vec3(0.0));
				vec3 low = safe * 12.92;
				vec3 high = 1.055 * pow(safe, vec3(1.0 / 2.4)) - 0.055;
				vec3 cutoff = step(vec3(0.0031308), safe);
				return mix(low, high, cutoff);
			}

			void main() {
				byDots = 1.0 / max(1.0, uDots);

				vec2 uv = vUv * 2.0 - 1.0;
				uv.x *= uResolution.x / max(1.0, uResolution.y);
				uv /= max(0.0001, uScale);

				float l = dot(uv, uv);
				float globeR2 = kSphereRadius * kSphereRadius;
				float atmosphereR = kSphereRadius * max(1.0, uAtmosphereScale);
				float atmosphereR2 = atmosphereR * atmosphereR;

				vec3 color = vec3(0.0);
				float alpha = 0.0;

				if (l <= atmosphereR2 && l > globeR2) {
					float d = (sqrt(l) - kSphereRadius) / max(1e-5, atmosphereR - kSphereRadius);
					float outerGlow = pow(clamp(1.0 - d, 0.0, 1.0), uAtmospherePower) * uAtmosphereIntensity;
					color += uAtmosphereColor * outerGlow;
					alpha = max(alpha, outerGlow);
				}

				if (l <= globeR2) {
					float dis;
					vec3 p = normalize(vec3(uv, sqrt(max(0.0, globeR2 - l))));
					mat3 rot = rotate(uRotation.y, uRotation.x);
					vec3 samplePoint = nearestFibonacciLattice(p * rot, dis);
					vec2 mapUv = pointToMaskUV(samplePoint);
					float land = texture2D(uLandTexture, mapUv).r;

					float landDots = step(0.5, land) * smoothstep(uPointRadius, 0.0, dis);

					float dotNV = clamp(p.z / kSphereRadius, 0.0, 1.0);
					float rim = pow(1.0 - dotNV, max(0.0001, uRimPower)) * uRimIntensity;
					float innerAtmosphere = pow(max(0.0, uAtmosphereCoefficient - dotNV), uAtmospherePower) * uAtmosphereIntensity;

					vec3 surface = uBaseColor;
					surface += uRimColor * rim;
					surface += uAtmosphereColor * innerAtmosphere;
					surface += uLandPointColor * landDots;

					color += surface;
					alpha = 1.0;
				}

				gl_FragColor = vec4(linearToSrgb(color), clamp(alpha, 0.0, 1.0));
			}
		`;

		const program = new Program(gl, {
			vertex: vertexShader,
			fragment: fragmentShader,
			uniforms,
			transparent: true,
			depthTest: false,
			depthWrite: false,
		});

		const mesh = new Mesh(gl, {
			geometry,
			program,
			frustumCulled: false,
		});
		mesh.setParent(scene);

		let currentScale = toScale(radius);
		const tempColor = new Vec3();
		const setColor = (
			target: Vec3,
			value: ColorRepresentation,
			fallback: [number, number, number],
		) => {
			const [r, g, b] = toLinearRgb(value, fallback);
			target.set(r, g, b);
		};

		updateUniforms = (state) => {
			currentScale = toScale(state.radius);
			uniforms.uScale.value = currentScale;
			uniforms.uDots.value = Math.max(1, Math.floor(state.pointCount));
			uniforms.uPointRadius.value = toPointRadius(state.pointSize);

			setColor(uniforms.uBaseColor.value, state.fresnelConfig.color, [
				17 / 255,
				17 / 255,
				19 / 255,
			]);
			setColor(uniforms.uRimColor.value, state.fresnelConfig.rimColor, [
				1,
				105 / 255,
				0,
			]);
			uniforms.uRimPower.value = Math.max(0.0001, state.fresnelConfig.rimPower);
			uniforms.uRimIntensity.value = Math.max(
				0,
				state.fresnelConfig.rimIntensity,
			);

			setColor(uniforms.uAtmosphereColor.value, state.atmosphereConfig.color, [
				1,
				105 / 255,
				0,
			]);
			uniforms.uAtmosphereScale.value = Math.max(
				1,
				state.atmosphereConfig.scale,
			);
			uniforms.uAtmospherePower.value = Math.max(
				0.0001,
				state.atmosphereConfig.power,
			);
			uniforms.uAtmosphereCoefficient.value = Math.max(
				0,
				state.atmosphereConfig.coefficient,
			);
			uniforms.uAtmosphereIntensity.value = Math.max(
				0,
				state.atmosphereConfig.intensity,
			);

			setColor(tempColor, state.landPointColor, [
				247 / 255,
				113 / 255,
				20 / 255,
			]);
			uniforms.uLandPointColor.value.set(tempColor.x, tempColor.y, tempColor.z);
		};

		updateUniforms({
			radius,
			pointCount,
			pointSize,
			landPointColor,
			fresnelConfig: resolvedFresnelConfig,
			atmosphereConfig: resolvedAtmosphereConfig,
		});

		let width = 1;
		let height = 1;
		const resize = () => {
			const host = targetCanvas.parentElement ?? targetCanvas;
			const { width: hostWidth, height: hostHeight } =
				host.getBoundingClientRect();
			width = Math.max(1, Math.round(hostWidth));
			height = Math.max(1, Math.round(hostHeight));
			renderer.setSize(width, height);
			uniforms.uResolution.value.set(gl.canvas.width, gl.canvas.height);
		};

		resize();
		const observer = new ResizeObserver(resize);
		observer.observe(targetCanvas);
		if (targetCanvas.parentElement)
			observer.observe(targetCanvas.parentElement);

		const startTheta = clampTheta(0, lockedPolarAngle);
		let phi = 0;
		let theta = startTheta;
		let targetPhi = phi;
		let targetTheta = startTheta;

		const syncMarkers = (
			currentPhi: number,
			currentTheta: number,
			currentScaleValue: number,
		) => {
			const markerRadius = COBE_GLOBE_RADIUS + MARKER_ELEVATION;
			const aspect = width / Math.max(1, height);

			const nextMarkers: ProjectedMarker[] = markers.map((marker, index) => {
				const pos = lonLatToCartesian(
					marker.location[1],
					marker.location[0],
					markerRadius,
				);
				const rotated = applyRotation(
					pos.x,
					pos.y,
					pos.z,
					currentPhi,
					currentTheta,
				);

				const ndcX = (rotated.rx / aspect) * currentScaleValue;
				const ndcY = -rotated.ry * currentScaleValue;
				const screenX = (ndcX + 1) * 0.5;
				const screenY = (ndcY + 1) * 0.5;

				const frontDot = rotated.rz / markerRadius;
				const rawVisibility = smoothstep(
					frontDot,
					VISIBILITY_MIN_DOT,
					VISIBILITY_MAX_DOT,
				);
				const visibility = dynamicEase(rawVisibility);

				return {
					marker,
					index,
					screenX,
					screenY,
					visibility,
					sizePx: Math.max(2, (marker.size ?? 0.05) * 160 * currentScaleValue),
				};
			});

			projectedMarkers = nextMarkers;
		};

		syncFocusTarget = (target) => {
			focusTween?.kill();
			focusTween = null;

			if (!target) return;

			const [lat, lon] = target;
			const nextDirection = lonLatToCartesian(lon, lat, 1);
			const targetRotation = cartesianToRotation(
				nextDirection.x,
				nextDirection.y,
				nextDirection.z,
			);

			const desiredTheta = clampTheta(targetRotation.theta, lockedPolarAngle);
			const desiredPhi = shortestAngleTarget(targetPhi, targetRotation.phi);

			const tweenState = { phi: targetPhi, theta: targetTheta };
			focusTween = gsap.to(tweenState, {
				phi: desiredPhi,
				theta: desiredTheta,
				duration: 1.5,
				ease: "power2.inOut",
				onUpdate: () => {
					targetPhi = tweenState.phi;
					targetTheta = clampTheta(tweenState.theta, lockedPolarAngle);
				},
				overwrite: true,
			});
		};

		syncFocusTarget(focusOn);

		let dragging = false;
		let activePointerId = -1;
		let lastPointerX = 0;
		let lastPointerY = 0;

		const onPointerDown = (event: PointerEvent) => {
			if (event.button !== 0) return;
			dragging = true;
			activePointerId = event.pointerId;
			lastPointerX = event.clientX;
			lastPointerY = event.clientY;
			targetCanvas.setPointerCapture(event.pointerId);
			focusTween?.kill();
			focusTween = null;
		};

		const onPointerMove = (event: PointerEvent) => {
			if (!dragging || event.pointerId !== activePointerId) return;
			const dx = event.clientX - lastPointerX;
			const dy = event.clientY - lastPointerY;
			lastPointerX = event.clientX;
			lastPointerY = event.clientY;

			targetPhi -= dx * ROTATE_SENSITIVITY;
			targetTheta = clampTheta(
				targetTheta + dy * ROTATE_SENSITIVITY,
				lockedPolarAngle,
			);
		};

		const stopDragging = (event: PointerEvent) => {
			if (event.pointerId !== activePointerId) return;
			dragging = false;
			activePointerId = -1;
		};

		targetCanvas.addEventListener("pointerdown", onPointerDown);
		targetCanvas.addEventListener("pointermove", onPointerMove);
		targetCanvas.addEventListener("pointerup", stopDragging);
		targetCanvas.addEventListener("pointercancel", stopDragging);
		targetCanvas.addEventListener("lostpointercapture", stopDragging);

		let disposed = false;
		const image = new Image();
		image.onload = () => {
			if (disposed) return;
			landTexture.image = image;
			landTexture.generateMipmaps = true;
			landTexture.minFilter = gl.NEAREST_MIPMAP_NEAREST;
			landTexture.magFilter = gl.NEAREST;
			landTexture.needsUpdate = true;
		};
		image.onerror = (error) => {
			console.warn("GlobeScene: failed to load land mask texture", error);
		};
		image.src = landTextureUrl;

		let raf = 0;
		let previous = 0;
		const tick = (now: number) => {
			const delta = previous ? (now - previous) / 1000 : 0;
			previous = now;

			if (autoRotate) {
				targetPhi -= AUTO_ROTATE_SPEED * delta;
			}
			targetTheta = clampTheta(targetTheta, lockedPolarAngle);

			const easing = 1 - Math.exp(-delta * SMOOTHING_STRENGTH);
			phi += (targetPhi - phi) * easing;
			theta += (targetTheta - theta) * easing;

			uniforms.uRotation.value.set(phi, theta);

			syncMarkers(phi, theta, currentScale);
			renderer.render({ scene, camera, clear: true });
			raf = window.requestAnimationFrame(tick);
		};

		raf = window.requestAnimationFrame(tick);

		return () => {
			disposed = true;
			focusTween?.kill();
			focusTween = null;
			window.cancelAnimationFrame(raf);
			observer.disconnect();
			targetCanvas.removeEventListener("pointerdown", onPointerDown);
			targetCanvas.removeEventListener("pointermove", onPointerMove);
			targetCanvas.removeEventListener("pointerup", stopDragging);
			targetCanvas.removeEventListener("pointercancel", stopDragging);
			targetCanvas.removeEventListener("lostpointercapture", stopDragging);

			mesh.setParent(null);
			geometry.remove();
			program.remove();
		};
	});
</script>

<canvas
	bind:this={canvas}
	class="absolute inset-0 block h-full w-full"
	style="width:100%;height:100%;touch-action:none;"
	aria-hidden="true"
></canvas>

<div class="pointer-events-none absolute inset-0 overflow-hidden">
	{#each projectedMarkers as projected, i (projected.marker.label || i)}
		<GlobeMarkerItem
			marker={projected.marker}
			index={projected.index}
			screenX={projected.screenX}
			screenY={projected.screenY}
			visibility={projected.visibility}
			sizePx={projected.sizePx}
			tooltip={markerTooltip}
		/>
	{/each}
</div>
",
+ "components/globe/GlobeScene.svelte": "<script lang="ts">
	import { onMount } from "svelte";
	import {
		Camera,
		Mesh,
		Program,
		Renderer,
		Texture,
		Transform,
		Triangle,
		Vec2,
		Vec3,
	} from "ogl";
	import { gsap } from "gsap/dist/gsap";
	import type { Snippet } from "svelte";
	import landTextureUrl from "../assets/land-texture.png";
	import { type ColorRepresentation, toLinearRgb } from "../helpers/color";
	import type { GlobeMarker, GlobeMarkerTooltipContext } from "./types";
	import GlobeMarkerItem from "./GlobeMarkerItem.svelte";

	interface FresnelConfig {
		/**
		 * Base body color for the globe surface.
		 * @default "#111113"
		 */
		color?: ColorRepresentation;
		/**
		 * Accent color applied by the Fresnel rim.
		 * @default "#FF6900"
		 */
		rimColor?: ColorRepresentation;
		/**
		 * Controls how tight the Fresnel rim hug is.
		 * Higher values yield a thinner outline.
		 * @default 6
		 */
		rimPower?: number;
		/**
		 * Intensity multiplier for the Fresnel rim color.
		 * @default 1.5
		 */
		rimIntensity?: number;
	}

	interface AtmosphereConfig {
		/**
		 * Color of the atmosphere glow.
		 * @default "#FF6900"
		 */
		color?: ColorRepresentation;
		/**
		 * Size of the atmosphere relative to the globe radius.
		 * @default 1.1
		 */
		scale?: number;
		/**
		 * Falloff power of the glow. Higher values mean a sharper edge.
		 * @default 12.0
		 */
		power?: number;
		/**
		 * Base coefficient for the intensity calculation.
		 * Controls how far the glow extends inwards.
		 * @default 0.9
		 */
		coefficient?: number;
		/**
		 * Global intensity multiplier.
		 * @default 2.0
		 */
		intensity?: number;
	}

	interface Props {
		/**
		 * Radius of the sphere.
		 * @default 2
		 */
		radius: number;
		/**
		 * Optional overrides for the Fresnel shader uniforms.
		 */
		fresnelConfig?: FresnelConfig;
		/**
		 * Optional configuration for the atmospheric halo.
		 */
		atmosphereConfig?: AtmosphereConfig;
		/**
		 * Number of points rendered along the globe surface.
		 * @default 15000
		 */
		pointCount?: number;
		/**
		 * Size of each point in world units.
		 * @default 0.05
		 */
		pointSize?: number;
		/**
		 * Color applied to points representing land.
		 * @default "#f77114"
		 */
		landPointColor?: ColorRepresentation;
		/**
		 * Whether the globe should auto-rotate.
		 * @default true
		 */
		autoRotate?: boolean;
		/**
		 * Whether to lock the camera's polar angle.
		 * @default true
		 */
		lockedPolarAngle?: boolean;
		/**
		 * Markers to display on the globe.
		 */
		markers?: GlobeMarker[];
		/**
		 * Optional custom tooltip renderer for markers.
		 */
		markerTooltip?: Snippet<[GlobeMarkerTooltipContext]>;
		/**
		 * Coordinates [lat, lon] to focus on.
		 */
		focusOn?: [number, number] | null;
	}

	interface ProjectedMarker {
		marker: GlobeMarker;
		index: number;
		screenX: number;
		screenY: number;
		visibility: number;
		sizePx: number;
	}

	interface UniformUpdaterState {
		radius: number;
		pointCount: number;
		pointSize: number;
		landPointColor: ColorRepresentation;
		fresnelConfig: Required<FresnelConfig>;
		atmosphereConfig: Required<AtmosphereConfig>;
	}

	const PI = Math.PI;
	const DEG2RAD = PI / 180;
	const EPSILON = 1e-6;
	const COBE_GLOBE_RADIUS = 0.8;
	const MARKER_ELEVATION = 0.05;
	const AUTO_ROTATE_SPEED = (2 * PI) / 30;
	const ROTATE_SENSITIVITY = 0.005;
	const SMOOTHING_STRENGTH = 14;
	const LOCKED_POLAR_ANGLE = 1.5;
	const LOCKED_THETA = Math.asin(Math.cos(LOCKED_POLAR_ANGLE));
	const MIN_THETA = -PI * 0.5 + 0.001;
	const MAX_THETA = PI * 0.5 - 0.001;
	const VISIBILITY_MIN_DOT = 0.24;
	const VISIBILITY_MAX_DOT = 0.48;

	const defaultFresnelConfig: Required<FresnelConfig> = {
		color: "#111113",
		rimColor: "#FF6900",
		rimPower: 6,
		rimIntensity: 1.5,
	};

	const defaultAtmosphereConfig: Required<AtmosphereConfig> = {
		color: "#FF6900",
		scale: 1.1,
		power: 12.0,
		coefficient: 0.9,
		intensity: 1.1,
	};

	let {
		radius,
		fresnelConfig = {},
		atmosphereConfig = {},
		pointCount = 15000,
		pointSize = 0.05,
		landPointColor = "#f77114",
		autoRotate = true,
		lockedPolarAngle = true,
		markers = [],
		markerTooltip,
		focusOn = null,
	}: Props = $props();

	let canvas = $state<HTMLCanvasElement>();
	let projectedMarkers = $state<ProjectedMarker[]>([]);

	const resolvedFresnelConfig = $derived({
		...defaultFresnelConfig,
		...fresnelConfig,
	});
	const resolvedAtmosphereConfig = $derived({
		...defaultAtmosphereConfig,
		...atmosphereConfig,
	});

	let updateUniforms = $state<((state: UniformUpdaterState) => void) | null>(
		null,
	);
	let syncFocusTarget = $state<
		((target: [number, number] | null) => void) | null
	>(null);
	let focusTween: gsap.core.Tween | null = null;

	const clamp = (value: number, min: number, max: number) =>
		Math.min(max, Math.max(min, value));

	const clampTheta = (value: number, lockPolar: boolean) =>
		lockPolar ? LOCKED_THETA : clamp(value, MIN_THETA, MAX_THETA);

	const smoothstep = (value: number, edge0: number, edge1: number) => {
		if (Math.abs(edge1 - edge0) <= EPSILON) {
			return value >= edge1 ? 1 : 0;
		}
		const t = clamp((value - edge0) / (edge1 - edge0), 0, 1);
		return t * t * (3 - 2 * t);
	};

	const toScale = (nextRadius: number) => Math.max(0.001, nextRadius / 2);
	const toPointRadius = (nextPointSize: number) =>
		Math.max(0.001, nextPointSize * 0.16);

	function normalizeAngle(value: number): number {
		const wrapped = (((value + PI) % (2 * PI)) + 2 * PI) % (2 * PI);
		return wrapped - PI;
	}

	function shortestAngleTarget(current: number, next: number): number {
		const delta = normalizeAngle(next - current);
		return current + delta;
	}

	function lonLatToCartesian(lon: number, lat: number, r: number) {
		const lonRad = lon * DEG2RAD;
		const latRad = lat * DEG2RAD;

		const y = r * Math.sin(latRad);
		const rXZ = r * Math.cos(latRad);
		const x = rXZ * Math.sin(lonRad);
		const z = rXZ * Math.cos(lonRad);

		return { x, y, z };
	}

	function cartesianToRotation(x: number, y: number, z: number) {
		const length = Math.hypot(x, y, z);
		if (length <= EPSILON) {
			return { phi: 0, theta: 0 };
		}
		const nx = x / length;
		const ny = y / length;
		const nz = z / length;

		return {
			phi: Math.atan2(-nx, nz),
			theta: Math.asin(clamp(ny, -1, 1)),
		};
	}

	function applyRotation(
		x: number,
		y: number,
		z: number,
		phi: number,
		theta: number,
	) {
		const cx = Math.cos(theta);
		const cy = Math.cos(phi);
		const sx = Math.sin(theta);
		const sy = Math.sin(phi);

		return {
			rx: cy * x + sy * z,
			ry: sy * sx * x + cx * y - cy * sx * z,
			rz: -sy * cx * x + sx * y + cy * cx * z,
		};
	}

	function cubicBezierAt(
		t: number,
		p0: number,
		p1: number,
		p2: number,
		p3: number,
	): number {
		const u = 1 - t;
		return (
			u * u * u * p0 + 3 * u * u * t * p1 + 3 * u * t * t * p2 + t * t * t * p3
		);
	}

	function cubicBezierDerivativeAt(
		t: number,
		p0: number,
		p1: number,
		p2: number,
		p3: number,
	): number {
		const u = 1 - t;
		return (
			3 * u * u * (p1 - p0) + 6 * u * t * (p2 - p1) + 3 * t * t * (p3 - p2)
		);
	}

	function dynamicEase(value: number): number {
		const clamped = clamp(value, 0, 1);
		let t = clamped;
		for (let i = 0; i < 5; i++) {
			const x = cubicBezierAt(t, 0, 0.625, 0, 1);
			const dx = cubicBezierDerivativeAt(t, 0, 0.625, 0, 1);
			if (Math.abs(dx) < 1e-6) break;
			t = clamp(t - (x - clamped) / dx, 0, 1);
		}
		return cubicBezierAt(t, 0, 0.05, 1, 1);
	}

	$effect(() => {
		if (!updateUniforms) return;
		updateUniforms({
			radius,
			pointCount,
			pointSize,
			landPointColor,
			fresnelConfig: resolvedFresnelConfig,
			atmosphereConfig: resolvedAtmosphereConfig,
		});
	});

	$effect(() => {
		if (!syncFocusTarget) return;
		syncFocusTarget(focusOn);
	});

	onMount(() => {
		const targetCanvas = canvas;
		if (!targetCanvas) return;

		const renderer = new Renderer({
			canvas: targetCanvas,
			alpha: true,
			antialias: true,
			dpr: typeof window !== "undefined" ? window.devicePixelRatio : 1,
		});
		const gl = renderer.gl;
		gl.clearColor(0, 0, 0, 0);

		const camera = new Camera(gl);
		camera.position.z = 1;

		const globeScene = new Transform();
		const atmosphereScene = new Transform();
		const geometry = new Triangle(gl);
		const landTexture = new Texture(gl, {
			image: new Uint8Array([0, 0, 0, 255]),
			width: 1,
			height: 1,
			format: gl.RGBA,
			type: gl.UNSIGNED_BYTE,
			minFilter: gl.NEAREST,
			magFilter: gl.NEAREST,
			generateMipmaps: false,
			wrapS: gl.REPEAT,
			wrapT: gl.REPEAT,
		});

		const uniforms = {
			uResolution: { value: new Vec2(1, 1) },
			uRotation: { value: new Vec2(0, clampTheta(0, lockedPolarAngle)) },
			uScale: { value: toScale(radius) },
			uDots: { value: Math.max(1, Math.floor(pointCount)) },
			uPointRadius: { value: toPointRadius(pointSize) },
			uBaseColor: { value: new Vec3(0, 0, 0) },
			uRimColor: { value: new Vec3(0, 0, 0) },
			uRimPower: { value: resolvedFresnelConfig.rimPower },
			uRimIntensity: { value: resolvedFresnelConfig.rimIntensity },
			uAtmosphereColor: { value: new Vec3(0, 0, 0) },
			uAtmosphereScale: { value: resolvedAtmosphereConfig.scale },
			uAtmospherePower: { value: resolvedAtmosphereConfig.power },
			uAtmosphereCoefficient: { value: resolvedAtmosphereConfig.coefficient },
			uAtmosphereIntensity: { value: resolvedAtmosphereConfig.intensity },
			uLandPointColor: { value: new Vec3(0, 0, 0) },
			uLandTexture: { value: landTexture },
		};

		const vertexShader = `
			attribute vec2 uv;
			attribute vec2 position;
			varying vec2 vUv;

			void main() {
				vUv = uv;
				gl_Position = vec4(position, 0.0, 1.0);
			}
		`;

		const globeFragmentShader = `
			precision highp float;

			varying vec2 vUv;

			uniform vec2 uResolution;
			uniform vec2 uRotation;
			uniform float uScale;
			uniform float uDots;
			uniform float uPointRadius;
			uniform vec3 uBaseColor;
			uniform vec3 uRimColor;
			uniform float uRimPower;
			uniform float uRimIntensity;
			uniform vec3 uLandPointColor;
			uniform sampler2D uLandTexture;

			const float kPi = 3.141592653589793;
			const float kTau = 6.283185307179586;
			const float kPhi = 1.618033988749895;
			const float kSqrt5 = 2.23606797749979;
			const float kSphereRadius = 0.8;

			float byDots;

			mat3 rotate(float theta, float phi) {
				float cx = cos(theta);
				float cy = cos(phi);
				float sx = sin(theta);
				float sy = sin(phi);
				return mat3(
					cy, sy * sx, -sy * cx,
					0.0, cx, sx,
					sy, cy * -sx, cy * cx
				);
			}

			vec3 nearestFibonacciLattice(vec3 p, out float m) {
				p = p.xzy;

				float k = max(2.0, floor(log2(kSqrt5 * uDots * kPi * (1.0 - p.z * p.z)) * 0.72021));
				vec2 f = floor(pow(kPhi, k) / kSqrt5 * vec2(1.0, kPhi) + 0.5);
				vec2 br1 = fract((f + 1.0) * (kPhi - 1.0)) * kTau - 3.883222;
				vec2 br2 = -2.0 * f;
				vec2 sp = vec2(atan(p.y, p.x), p.z - 1.0);
				vec2 c = floor(vec2(
					br2.y * sp.x - br1.y * (sp.y * uDots + 1.0),
					-br2.x * sp.x + br1.x * (sp.y * uDots + 1.0)
				) / (br1.x * br2.y - br2.x * br1.y));

				float mindist = kPi;
				vec3 minip = vec3(0.0, 0.0, 1.0);

				for (float s = 0.0; s < 4.0; s += 1.0) {
					vec2 o = vec2(mod(s, 2.0), floor(s * 0.5));
					float idx = dot(f, c + o);
					if (idx > uDots) continue;

					float a = idx;
					float b = 0.0;
					if (a >= 16384.0) a -= 16384.0, b += 0.868872;
					if (a >= 8192.0) a -= 8192.0, b += 0.934436;
					if (a >= 4096.0) a -= 4096.0, b += 0.467218;
					if (a >= 2048.0) a -= 2048.0, b += 0.733609;
					if (a >= 1024.0) a -= 1024.0, b += 0.866804;
					if (a >= 512.0) a -= 512.0, b += 0.433402;
					if (a >= 256.0) a -= 256.0, b += 0.216701;
					if (a >= 128.0) a -= 128.0, b += 0.108351;
					if (a >= 64.0) a -= 64.0, b += 0.554175;
					if (a >= 32.0) a -= 32.0, b += 0.777088;
					if (a >= 16.0) a -= 16.0, b += 0.888544;
					if (a >= 8.0) a -= 8.0, b += 0.944272;
					if (a >= 4.0) a -= 4.0, b += 0.472136;
					if (a >= 2.0) a -= 2.0, b += 0.236068;
					if (a >= 1.0) a -= 1.0, b += 0.618034;

					float theta = fract(b) * kTau;
					float cosphi = 1.0 - 2.0 * idx * byDots;
					float sinphi = sqrt(max(0.0, 1.0 - cosphi * cosphi));
					vec3 samplePoint = vec3(cos(theta) * sinphi, sin(theta) * sinphi, cosphi);

					float dist = length(p - samplePoint);
					if (dist < mindist) {
						mindist = dist;
						minip = samplePoint;
					}
				}

				m = mindist;
				return minip.xzy;
			}

			vec2 pointToMaskUV(vec3 p) {
				float lengthP = length(p);
				if (lengthP <= 0.0) {
					return vec2(0.0, 0.0);
				}

				vec3 n = p / lengthP;

				float nx = n.z;
				float ny = n.y;
				float nz = -n.x;

				float gPhi = asin(clamp(ny, -1.0, 1.0));
				float cosPhi = cos(gPhi);

				float gTheta = 0.0;
				if (abs(cosPhi) > 1e-6) {
					float thetaInput = clamp(-nx / cosPhi, -1.0, 1.0);
					gTheta = acos(thetaInput);
					if (nz < 0.0) {
						gTheta = -gTheta;
					}
				}

				return vec2(
					fract((gTheta * 0.5) / kPi),
					fract(gPhi / kPi + 0.5)
				);
			}

			vec3 linearToSrgb(vec3 color) {
				vec3 safe = max(color, vec3(0.0));
				vec3 low = safe * 12.92;
				vec3 high = 1.055 * pow(safe, vec3(1.0 / 2.4)) - 0.055;
				vec3 cutoff = step(vec3(0.0031308), safe);
				return mix(low, high, cutoff);
			}

			void main() {
				byDots = 1.0 / max(1.0, uDots);

				vec2 uv = vUv * 2.0 - 1.0;
				uv.x *= uResolution.x / max(1.0, uResolution.y);
				uv /= max(0.0001, uScale);

				float l = dot(uv, uv);
				float globeR2 = kSphereRadius * kSphereRadius;

				vec3 color = vec3(0.0);
				float alpha = 0.0;

				if (l <= globeR2) {
					float dis;
					vec3 p = normalize(vec3(uv, sqrt(max(0.0, globeR2 - l))));
					mat3 rot = rotate(uRotation.y, uRotation.x);
					vec3 samplePoint = nearestFibonacciLattice(p * rot, dis);
					vec2 mapUv = pointToMaskUV(samplePoint);
					float land = texture2D(uLandTexture, mapUv).r;

					float landDots = step(0.5, land) * smoothstep(uPointRadius, 0.0, dis);

					float dotNV = clamp(p.z / kSphereRadius, 0.0, 1.0);
					float rim = pow(1.0 - dotNV, max(0.0001, uRimPower)) * uRimIntensity;
					// Match the geometric foreshortening from the old instanced point mesh:
					// near the silhouette points should visually fade instead of staying fully crisp.
					float dotFade = smoothstep(0.04, 0.28, dotNV);
					landDots *= dotFade;

					vec3 surface = uBaseColor;
					surface += uRimColor * rim;
					surface += uLandPointColor * landDots;

					color += surface;
					alpha = 1.0;
				}

				gl_FragColor = vec4(linearToSrgb(color), clamp(alpha, 0.0, 1.0));
			}
		`;

		const atmosphereFragmentShader = `
			precision highp float;

			varying vec2 vUv;

			uniform vec2 uResolution;
			uniform float uScale;
			uniform vec3 uAtmosphereColor;
			uniform float uAtmosphereScale;
			uniform float uAtmospherePower;
			uniform float uAtmosphereCoefficient;
			uniform float uAtmosphereIntensity;

			const float kSphereRadius = 0.8;

			vec3 linearToSrgb(vec3 color) {
				vec3 safe = max(color, vec3(0.0));
				vec3 low = safe * 12.92;
				vec3 high = 1.055 * pow(safe, vec3(1.0 / 2.4)) - 0.055;
				vec3 cutoff = step(vec3(0.0031308), safe);
				return mix(low, high, cutoff);
			}

			void main() {
				vec2 uv = vUv * 2.0 - 1.0;
				uv.x *= uResolution.x / max(1.0, uResolution.y);
				uv /= max(0.0001, uScale);

				float globeR = kSphereRadius;
				float atmosphereR = kSphereRadius * max(1.0, uAtmosphereScale);
				float l = dot(uv, uv);
				float radial = sqrt(l);

				if (radial <= globeR) {
					discard;
				}

				float shellWidth = max(1e-5, atmosphereR - globeR);
				float x = (radial - globeR) / shellWidth;
				if (x > 3.0) {
					discard;
				}

				// Smooth outward blur profile (no hard ring cutoff at shell boundary).
				float falloff = exp(-pow(max(0.0, x), 1.2) * max(0.15, uAtmospherePower * 0.09));
				float finalFactor =
					falloff * uAtmosphereIntensity * max(0.0, uAtmosphereCoefficient);

				vec3 finalColor = uAtmosphereColor * finalFactor;
				float alpha = finalFactor;

				gl_FragColor = vec4(linearToSrgb(finalColor), clamp(alpha, 0.0, 1.0));
			}
		`;

		const globeProgram = new Program(gl, {
			vertex: vertexShader,
			fragment: globeFragmentShader,
			uniforms,
			transparent: true,
			depthTest: false,
			depthWrite: false,
		});

		const atmosphereProgram = new Program(gl, {
			vertex: vertexShader,
			fragment: atmosphereFragmentShader,
			uniforms,
			transparent: true,
			depthTest: false,
			depthWrite: false,
		});
		atmosphereProgram.setBlendFunc(gl.SRC_ALPHA, gl.ONE);

		const globeMesh = new Mesh(gl, {
			geometry,
			program: globeProgram,
			frustumCulled: false,
		});
		globeMesh.setParent(globeScene);

		const atmosphereMesh = new Mesh(gl, {
			geometry,
			program: atmosphereProgram,
			frustumCulled: false,
		});
		atmosphereMesh.setParent(atmosphereScene);

		let currentScale = toScale(radius);
		const tempColor = new Vec3();
		const setColor = (
			target: Vec3,
			value: ColorRepresentation,
			fallback: [number, number, number],
		) => {
			const [r, g, b] = toLinearRgb(value, fallback);
			target.set(r, g, b);
		};

		updateUniforms = (state) => {
			currentScale = toScale(state.radius);
			uniforms.uScale.value = currentScale;
			uniforms.uDots.value = Math.max(1, Math.floor(state.pointCount));
			uniforms.uPointRadius.value = toPointRadius(state.pointSize);

			setColor(uniforms.uBaseColor.value, state.fresnelConfig.color, [
				17 / 255,
				17 / 255,
				19 / 255,
			]);
			setColor(uniforms.uRimColor.value, state.fresnelConfig.rimColor, [
				1,
				105 / 255,
				0,
			]);
			uniforms.uRimPower.value = Math.max(0.0001, state.fresnelConfig.rimPower);
			uniforms.uRimIntensity.value = Math.max(
				0,
				state.fresnelConfig.rimIntensity,
			);

			setColor(uniforms.uAtmosphereColor.value, state.atmosphereConfig.color, [
				1,
				105 / 255,
				0,
			]);
			uniforms.uAtmosphereScale.value = Math.max(
				1,
				state.atmosphereConfig.scale,
			);
			uniforms.uAtmospherePower.value = Math.max(
				0.0001,
				state.atmosphereConfig.power,
			);
			uniforms.uAtmosphereCoefficient.value = Math.max(
				0,
				state.atmosphereConfig.coefficient,
			);
			uniforms.uAtmosphereIntensity.value = Math.max(
				0,
				state.atmosphereConfig.intensity,
			);

			setColor(tempColor, state.landPointColor, [
				247 / 255,
				113 / 255,
				20 / 255,
			]);
			uniforms.uLandPointColor.value.set(tempColor.x, tempColor.y, tempColor.z);
		};

		updateUniforms({
			radius,
			pointCount,
			pointSize,
			landPointColor,
			fresnelConfig: resolvedFresnelConfig,
			atmosphereConfig: resolvedAtmosphereConfig,
		});

		let width = 1;
		let height = 1;
		const resize = () => {
			const host = targetCanvas.parentElement ?? targetCanvas;
			const { width: hostWidth, height: hostHeight } =
				host.getBoundingClientRect();
			width = Math.max(1, Math.round(hostWidth));
			height = Math.max(1, Math.round(hostHeight));
			renderer.setSize(width, height);
			uniforms.uResolution.value.set(gl.canvas.width, gl.canvas.height);
		};

		resize();
		const observer = new ResizeObserver(resize);
		observer.observe(targetCanvas);
		if (targetCanvas.parentElement)
			observer.observe(targetCanvas.parentElement);

		const startTheta = clampTheta(0, lockedPolarAngle);
		let phi = 0;
		let theta = startTheta;
		let targetPhi = phi;
		let targetTheta = startTheta;

		const syncMarkers = (
			currentPhi: number,
			currentTheta: number,
			currentScaleValue: number,
		) => {
			const markerRadius = COBE_GLOBE_RADIUS + MARKER_ELEVATION;
			const aspect = width / Math.max(1, height);

			const nextMarkers: ProjectedMarker[] = markers.map((marker, index) => {
				const pos = lonLatToCartesian(
					marker.location[1],
					marker.location[0],
					markerRadius,
				);
				const rotated = applyRotation(
					pos.x,
					pos.y,
					pos.z,
					currentPhi,
					currentTheta,
				);

				const ndcX = (rotated.rx / aspect) * currentScaleValue;
				const ndcY = -rotated.ry * currentScaleValue;
				const screenX = (ndcX + 1) * 0.5;
				const screenY = (ndcY + 1) * 0.5;

				const frontDot = rotated.rz / markerRadius;
				const rawVisibility = smoothstep(
					frontDot,
					VISIBILITY_MIN_DOT,
					VISIBILITY_MAX_DOT,
				);
				const visibility = dynamicEase(rawVisibility);

				return {
					marker,
					index,
					screenX,
					screenY,
					visibility,
					sizePx: Math.max(2, (marker.size ?? 0.05) * 160 * currentScaleValue),
				};
			});

			projectedMarkers = nextMarkers;
		};

		syncFocusTarget = (target) => {
			focusTween?.kill();
			focusTween = null;

			if (!target) return;

			const [lat, lon] = target;
			const nextDirection = lonLatToCartesian(lon, lat, 1);
			const targetRotation = cartesianToRotation(
				nextDirection.x,
				nextDirection.y,
				nextDirection.z,
			);

			const desiredTheta = clampTheta(targetRotation.theta, lockedPolarAngle);
			const desiredPhi = shortestAngleTarget(targetPhi, targetRotation.phi);

			const tweenState = { phi: targetPhi, theta: targetTheta };
			focusTween = gsap.to(tweenState, {
				phi: desiredPhi,
				theta: desiredTheta,
				duration: 1.5,
				ease: "power2.inOut",
				onUpdate: () => {
					targetPhi = tweenState.phi;
					targetTheta = clampTheta(tweenState.theta, lockedPolarAngle);
				},
				overwrite: true,
			});
		};

		syncFocusTarget(focusOn);

		let dragging = false;
		let activePointerId = -1;
		let lastPointerX = 0;
		let lastPointerY = 0;

		const onPointerDown = (event: PointerEvent) => {
			if (event.button !== 0) return;
			dragging = true;
			activePointerId = event.pointerId;
			lastPointerX = event.clientX;
			lastPointerY = event.clientY;
			targetCanvas.setPointerCapture(event.pointerId);
			focusTween?.kill();
			focusTween = null;
		};

		const onPointerMove = (event: PointerEvent) => {
			if (!dragging || event.pointerId !== activePointerId) return;
			const dx = event.clientX - lastPointerX;
			const dy = event.clientY - lastPointerY;
			lastPointerX = event.clientX;
			lastPointerY = event.clientY;

			targetPhi += dx * ROTATE_SENSITIVITY;
			targetTheta = clampTheta(
				targetTheta + dy * ROTATE_SENSITIVITY,
				lockedPolarAngle,
			);
		};

		const stopDragging = (event: PointerEvent) => {
			if (event.pointerId !== activePointerId) return;
			dragging = false;
			activePointerId = -1;
		};

		targetCanvas.addEventListener("pointerdown", onPointerDown);
		targetCanvas.addEventListener("pointermove", onPointerMove);
		targetCanvas.addEventListener("pointerup", stopDragging);
		targetCanvas.addEventListener("pointercancel", stopDragging);
		targetCanvas.addEventListener("lostpointercapture", stopDragging);

		let disposed = false;
		const image = new Image();
		image.onload = () => {
			if (disposed) return;
			landTexture.image = image;
			landTexture.generateMipmaps = true;
			landTexture.minFilter = gl.NEAREST_MIPMAP_NEAREST;
			landTexture.magFilter = gl.NEAREST;
			landTexture.needsUpdate = true;
		};
		image.onerror = (error) => {
			console.warn("GlobeScene: failed to load land mask texture", error);
		};
		image.src = landTextureUrl;

		let raf = 0;
		let previous = 0;
		const tick = (now: number) => {
			const delta = previous ? (now - previous) / 1000 : 0;
			previous = now;

			if (autoRotate) {
				targetPhi -= AUTO_ROTATE_SPEED * delta;
			}
			targetTheta = clampTheta(targetTheta, lockedPolarAngle);

			const easing = 1 - Math.exp(-delta * SMOOTHING_STRENGTH);
			phi += (targetPhi - phi) * easing;
			theta += (targetTheta - theta) * easing;

			uniforms.uRotation.value.set(phi, theta);

			syncMarkers(phi, theta, currentScale);
			renderer.render({ scene: globeScene, camera, clear: true });
			renderer.render({ scene: atmosphereScene, camera, clear: false });
			raf = window.requestAnimationFrame(tick);
		};

		raf = window.requestAnimationFrame(tick);

		return () => {
			disposed = true;
			focusTween?.kill();
			focusTween = null;
			window.cancelAnimationFrame(raf);
			observer.disconnect();
			targetCanvas.removeEventListener("pointerdown", onPointerDown);
			targetCanvas.removeEventListener("pointermove", onPointerMove);
			targetCanvas.removeEventListener("pointerup", stopDragging);
			targetCanvas.removeEventListener("pointercancel", stopDragging);
			targetCanvas.removeEventListener("lostpointercapture", stopDragging);

			globeMesh.setParent(null);
			atmosphereMesh.setParent(null);
			geometry.remove();
			globeProgram.remove();
			atmosphereProgram.remove();
		};
	});
</script>

<canvas
	bind:this={canvas}
	class="absolute inset-0 block h-full w-full"
	style="width:100%;height:100%;touch-action:none;"
	aria-hidden="true"
></canvas>

<div class="pointer-events-none absolute inset-0 overflow-hidden">
	{#each projectedMarkers as projected, i (projected.marker.label || i)}
		<GlobeMarkerItem
			marker={projected.marker}
			index={projected.index}
			screenX={projected.screenX}
			screenY={projected.screenY}
			visibility={projected.visibility}
			sizePx={projected.sizePx}
			tooltip={markerTooltip}
		/>
	{/each}
</div>
",
"components/globe/GlobeMarkerItem.svelte": "PHNjcmlwdCBsYW5nPSJ0cyI+CglpbXBvcnQgdHlwZSB7IFNuaXBwZXQgfSBmcm9tICJzdmVsdGUiOwoJaW1wb3J0IHR5cGUgeyBHbG9iZU1hcmtlciwgR2xvYmVNYXJrZXJUb29sdGlwQ29udGV4dCB9IGZyb20gIi4vdHlwZXMiOwoKCWludGVyZmFjZSBQcm9wcyB7CgkJLyoqCgkJICogVGhlIG1hcmtlciBkYXRhIG9iamVjdCBjb250YWluaW5nIGxvY2F0aW9uLCBjb2xvciwgc2l6ZSwgZXRjLgoJCSAqLwoJCW1hcmtlcjogR2xvYmVNYXJrZXI7CgkJLyoqCgkJICogTWFya2VyIGluZGV4IGluIHRoZSBtYXJrZXJzIGFycmF5LgoJCSAqLwoJCWluZGV4OiBudW1iZXI7CgkJLyoqCgkJICogSG9yaXpvbnRhbCBtYXJrZXIgcG9zaXRpb24gaW4gbm9ybWFsaXplZCBbMCwgMV0gdmlld3BvcnQgc3BhY2UuCgkJICovCgkJc2NyZWVuWDogbnVtYmVyOwoJCS8qKgoJCSAqIFZlcnRpY2FsIG1hcmtlciBwb3NpdGlvbiBpbiBub3JtYWxpemVkIFswLCAxXSB2aWV3cG9ydCBzcGFjZS4KCQkgKi8KCQlzY3JlZW5ZOiBudW1iZXI7CgkJLyoqCgkJICogTWFya2VyIHZpc2liaWxpdHkgZmFjdG9yIGluIHJhbmdlIFswLCAxXS4KCQkgKi8KCQl2aXNpYmlsaXR5OiBudW1iZXI7CgkJLyoqCgkJICogTWFya2VyIHZpc3VhbCBzaXplIGluIHBpeGVscy4KCQkgKi8KCQlzaXplUHg6IG51bWJlcjsKCQkvKioKCQkgKiBPcHRpb25hbCBjdXN0b20gdG9vbHRpcCBzbmlwcGV0LgoJCSAqLwoJCXRvb2x0aXA/OiBTbmlwcGV0PFtHbG9iZU1hcmtlclRvb2x0aXBDb250ZXh0XT47Cgl9CgoJbGV0IHsgbWFya2VyLCBpbmRleCwgc2NyZWVuWCwgc2NyZWVuWSwgdmlzaWJpbGl0eSwgc2l6ZVB4LCB0b29sdGlwIH06IFByb3BzID0KCQkkcHJvcHMoKTsKCgljb25zdCBNQVhfVE9PTFRJUF9CTFVSID0gODsKCglsZXQgdG9vbHRpcEJsdXIgPSAkZGVyaXZlZCgoMSAtIHZpc2liaWxpdHkpICogTUFYX1RPT0xUSVBfQkxVUik7CglsZXQgdG9vbHRpcENvbnRleHQgPSAkZGVyaXZlZDxHbG9iZU1hcmtlclRvb2x0aXBDb250ZXh0Pih7CgkJbWFya2VyLAoJCWluZGV4LAoJCXZpc2liaWxpdHksCgl9KTsKCWxldCBtYXJrZXJDb2xvciA9ICRkZXJpdmVkKG1hcmtlci5jb2xvciA/PyAiI2ZmZmZmZiIpOwoJbGV0IG1hcmtlck9wYWNpdHkgPSAkZGVyaXZlZCh2aXNpYmlsaXR5KTsKCWxldCBjbGFtcGVkU2l6ZVB4ID0gJGRlcml2ZWQoTWF0aC5tYXgoMiwgc2l6ZVB4KSk7Cjwvc2NyaXB0PgoKPGRpdgoJY2xhc3M9InBvaW50ZXItZXZlbnRzLW5vbmUgYWJzb2x1dGUiCglzdHlsZTpsZWZ0PXtgJHtzY3JlZW5YICogMTAwfSVgfQoJc3R5bGU6dG9wPXtgJHtzY3JlZW5ZICogMTAwfSVgfQoJc3R5bGU6dHJhbnNmb3JtPSJ0cmFuc2xhdGUoLTUwJSwgLTUwJSkiCj4KCTxkaXYKCQljbGFzcz0icm91bmRlZC1mdWxsIgoJCXN0eWxlOndpZHRoPXtgJHtjbGFtcGVkU2l6ZVB4fXB4YH0KCQlzdHlsZTpoZWlnaHQ9e2Ake2NsYW1wZWRTaXplUHh9cHhgfQoJCXN0eWxlOmJhY2tncm91bmQ9e21hcmtlckNvbG9yfQoJCXN0eWxlOm9wYWNpdHk9e21hcmtlck9wYWNpdHl9Cgk+PC9kaXY+CgoJeyNpZiB0b29sdGlwIHx8IG1hcmtlci5sYWJlbH0KCQk8ZGl2CgkJCWNsYXNzPSJwb2ludGVyLWV2ZW50cy1ub25lIGFic29sdXRlIHRvcC0wIGxlZnQtMS8yIGlubGluZS1mbGV4IC10cmFuc2xhdGUteC0xLzIgLXRyYW5zbGF0ZS15LTYgZmxleC1jb2wgaXRlbXMtY2VudGVyIHRyYW5zaXRpb24tW29wYWNpdHksZmlsdGVyXSBkdXJhdGlvbi0yMDAgZWFzZS1vdXQiCgkJCXN0eWxlOm9wYWNpdHk9e3Zpc2liaWxpdHl9CgkJCXN0eWxlOmZpbHRlcj17YGJsdXIoJHt0b29sdGlwQmx1cn1weClgfQoJCT4KCQkJeyNpZiB0b29sdGlwfQoJCQkJe0ByZW5kZXIgdG9vbHRpcCh0b29sdGlwQ29udGV4dCl9CgkJCXs6ZWxzZX0KCQkJCTxkaXYKCQkJCQljbGFzcz0iYmctZml4ZWQtZGFyay84MCByb3VuZGVkLXhzIHB4LTIgcHktMSB0ZXh0LXhzIHdoaXRlc3BhY2Utbm93cmFwIHRleHQtZml4ZWQtbGlnaHQgYmFja2Ryb3AtYmx1ci1zbSIKCQkJCT4KCQkJCQl7bWFya2VyLmxhYmVsfQoJCQkJPC9kaXY+CgkJCXsvaWZ9CgkJPC9kaXY+Cgl7L2lmfQo8L2Rpdj4K",
"components/globe/types.ts": "ZXhwb3J0IGludGVyZmFjZSBHbG9iZU1hcmtlciB7CgkvKioKCSAqIExhdGl0dWRlIGFuZCBMb25naXR1ZGUgY29vcmRpbmF0ZXMgW2xhdCwgbG9uXS4KCSAqLwoJbG9jYXRpb246IFtudW1iZXIsIG51bWJlcl07CgkvKioKCSAqIFNpemUgb2YgdGhlIG1hcmtlciBpbiB3b3JsZCB1bml0cy4KCSAqIEBkZWZhdWx0IDAuMDUKCSAqLwoJc2l6ZT86IG51bWJlcjsKCS8qKgoJICogQ29sb3Igb2YgdGhlIG1hcmtlci4KCSAqIEBkZWZhdWx0ICIjZmZmZmZmIgoJICovCgljb2xvcj86IHN0cmluZzsKCS8qKgoJICogT3B0aW9uYWwgZmFsbGJhY2sgdG9vbHRpcCB0ZXh0ICh1c2VkIHdoZW4gbm8gY3VzdG9tIHRvb2x0aXAgcmVuZGVyZXIgaXMgcHJvdmlkZWQpLgoJICovCglsYWJlbD86IHN0cmluZzsKfQoKZXhwb3J0IGludGVyZmFjZSBHbG9iZU1hcmtlclRvb2x0aXBDb250ZXh0IHsKCS8qKgoJICogTWFya2VyIGN1cnJlbnRseSBiZWluZyByZW5kZXJlZC4KCSAqLwoJbWFya2VyOiBHbG9iZU1hcmtlcjsKCS8qKgoJICogTWFya2VyIGluZGV4IGluIHRoZSBtYXJrZXJzIGFycmF5LgoJICovCglpbmRleDogbnVtYmVyOwoJLyoqCgkgKiBNYXJrZXIgdmlzaWJpbGl0eSBmYWN0b3IgaW4gcmFuZ2UgWzAsIDFdLgoJICogMCBtZWFucyBmdWxseSBoaWRkZW4gYmVoaW5kIHRoZSBnbG9iZSwgMSBtZWFucyBmdWxseSB2aXNpYmxlLgoJICovCgl2aXNpYmlsaXR5OiBudW1iZXI7Cn0K",
"assets/land-texture.png": "iVBORw0KGgoAAAANSUhEUgAAAQAAAACAAQAAAADMzoqnAAAAAXNSR0IArs4c6QAABA5JREFUeNrV179uHEUAx/Hf3JpbF+E2VASBsmVKTBcpKJs3SMEDcDwBiVJAAewYEBUivIHT0uUBIt0YCovKD0CRjUC4QfHYh8hYXu+P25vZ2Zm9c66gMd/GJ/tz82d3bk8GN4SrByYF2366FNTACIAkivVAAazQdnf3MvAlbNUQfOPAdQDvSAimMWhwy4I2g4SU+Kp04ISLpPBAKLxPyic3O/CCi+Y7rUJbiodcpDOFY7CgxCEXmdYD2EYK2s5lApOx5pEDDYCUwM1XdJUwBV11QQMg59kePSCaPAASQMEL2hwo6TJFgxpg+TgC2ymXPbuvc40awr3D1QCFfbH9kcoqAOkZozpQo0aqAGQRKCog/+tjkgbNFEtg2FffBvBGlSxHoAaAa1u6X4PBAwDiR8FFsrQgeUhfJTSALaB9jy5NCybJPn1SVFiWk7ywN+KzhH1aKAuydhGkbEF4lWohLXDXavlyFgHY7LBnLRdlAP6BS5Cc8RfVDXbkwN/oIvmY+6obbNeBP0JwTuMGu9gTzy1Q4RS/cWpfzszeYwd+CAFrtBW/Hur0gLbJGlD+/OjVwe/drfBxkbbg63dndEDfiEBlAd7ac0BPe1D6Jd8dfbLH+RI0OzseFB5s01/M+gMdAeluLOCAuaUA9Lezo/vSgXoCX9rtEiXnp7Q1W/CNyWcd8DXoS6jH/YZ5vAJEWY2dXFQe2TUgaFaNejCzJ98g6HnlVrsE58sDcYqg+9XY75fPqdoh/kRQWiXKg8MWlJQxUFMPjqnyujhFBE7UxIMjyszk0QwQlFsezImsyvUYYYVED2pk6m0Tg8T04Fwjk2kdAwSACqlM6gRRt3vQYAFGX0Ah7Ebx1H+MDRI5ui0QldH4j7FGcm90XdxD2Jg1AOEAVAKhEFXSn4cKUELurIAKwJ3MArypPscQaLhJFICJ0ohjDySAdH8AhDtCiTuMycH8CXzhH9jUACAO5uMhoAwA5i+T6WAKmmAqnLy80wxHqIPFYpqCwxGaYLt4Dyievg5kEoVEUAhs6pqKgFtDQYOuaXypaWKQfIuwwoGSZgfLsu/XAtI8cGN+h7Cc1A5oLOMhwlIPXuhu48AIvsSBkvtV9wsJRKCyYLfq5lTrQMFd1a262oqBck9K1V0YjQg0iEYYgpS1A9GlXQV5cykwm4A7BzVsxQqo7E+zCegO7Ma7yKgsuOcfKbMBwLC8wvVNYDsANYalEpOAa6zpWjTeMKGwEwC1CiQewJc5EKfgy7GmRAZA4vUVGwE2dPM/g0xuAInE/yG5aZ8ISxWGfYigUVbdyBElTHh2uCwGdfCkOLGgQVBh3Ewp+/QK4CDlR5Ws/Zf7yhCf8pH7vinWAvoVCQ6zz0NX5V/6GkAVV+2/5qsJ/gU8bsxpM8IeAQAAAABJRU5ErkJggg==",
diff --git a/apps/web/static/registry/registry.json b/apps/web/static/registry/registry.json
index 47c5491..909e932 100644
--- a/apps/web/static/registry/registry.json
+++ b/apps/web/static/registry/registry.json
@@ -11,8 +11,7 @@
"tailwind-merge": "^3.4.0"
},
"baseDevDependencies": {
- "@types/gsap": "^3.0.0",
- "@types/three": "^0.182.0"
+ "@types/gsap": "^3.0.0"
},
"components": {
"ascii-renderer": {
diff --git a/bun.lock b/bun.lock
index 3e63815..f3b2ea8 100644
--- a/bun.lock
+++ b/bun.lock
@@ -29,15 +29,13 @@
"@fontsource/inter": "^5.2.8",
"@mediapipe/tasks-vision": "^0.10.22-rc.20250304",
"@takumi-rs/image-response": "1.0.0-rc.12",
- "@threlte/core": "^8.3.1",
- "@threlte/extras": "^9.7.1",
"carbon-icons-svelte": "^13.9.0",
"class-variance-authority": "^0.7.1",
"clsx": "^2.1.1",
"gsap": "^3.14.2",
"motion-core": "workspace:*",
+ "ogl": "^1.0.11",
"tailwind-merge": "^3.5.0",
- "three": "^0.182.0",
},
"devDependencies": {
"@eslint/js": "^10.0.1",
@@ -45,7 +43,6 @@
"@sveltejs/kit": "^2.53.0",
"@sveltejs/vite-plugin-svelte": "^6.2.4",
"@tailwindcss/vite": "^4.1.17",
- "@types/three": "^0.182.0",
"@typescript-eslint/eslint-plugin": "^8.56.1",
"@typescript-eslint/parser": "^8.56.1",
"eslint": "^10.0.2",
@@ -76,22 +73,16 @@
},
"devDependencies": {
"@mediapipe/tasks-vision": "^0.10.22-rc.20250304",
- "@threlte/core": "^8.3.1",
- "@threlte/extras": "^9.7.1",
"@types/gsap": "^3.0.0",
- "@types/three": "^0.182.0",
"ogl": "^1.0.11",
"tailwindcss": "^4.1.18",
- "three": "^0.182.0",
},
"peerDependencies": {
"@mediapipe/tasks-vision": "^0.10.22-rc.20250304",
- "@threlte/core": "^8.0.0",
- "@threlte/extras": "^9.0.0",
"gsap": "^3.14.2",
+ "ogl": "^1.0.11",
"svelte": "^5.0.0",
"tailwindcss": "^4.1.18",
- "three": "^0.182.0",
},
},
},
@@ -123,8 +114,6 @@
"@cspotcode/source-map-support": ["@cspotcode/source-map-support@0.8.1", "", { "dependencies": { "@jridgewell/trace-mapping": "0.3.9" } }, "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw=="],
- "@dimforge/rapier3d-compat": ["@dimforge/rapier3d-compat@0.12.0", "", {}, "sha512-uekIGetywIgopfD97oDL5PfeezkFpNhwlzlaEYNOA0N6ghdsOvh/HYjSMek5Q2O1PYvRSDFcqFVJl4r4ZBwOow=="],
-
"@emnapi/core": ["@emnapi/core@1.7.1", "", { "dependencies": { "@emnapi/wasi-threads": "1.1.0", "tslib": "^2.4.0" } }, "sha512-o1uhUASyo921r2XtHYOHy7gdkGLge8ghBEQHMWmyJFoXlpU58kIrhhN3w26lpQb6dspetweapMn2CSNwQ8I4wg=="],
"@emnapi/runtime": ["@emnapi/runtime@1.7.1", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-PVtJr5CmLwYAU9PZDMITZoR5iAOShYREoR45EyyLrbntV50mdePTgUn4AmOw90Ifcj+x2kRjdzr1HP3RrNiHGA=="],
@@ -447,14 +436,6 @@
"@takumi-rs/wasm": ["@takumi-rs/wasm@1.0.0-rc.12", "", { "dependencies": { "@takumi-rs/helpers": "1.0.0-rc.12" } }, "sha512-F4cid0g3At2jPcO/LfX10NjiJD3RvuBjsVpWv6LGeN+cYP65h9F5YGifwqSpTQ2IdPXLL1nctVGWjHe6CKN9tA=="],
- "@threejs-kit/instanced-sprite-mesh": ["@threejs-kit/instanced-sprite-mesh@2.5.1", "", { "dependencies": { "diet-sprite": "^0.0.1", "earcut": "^2.2.4", "maath": "^0.10.7", "three-instanced-uniforms-mesh": "^0.52.4", "troika-three-utils": "^0.52.4" }, "peerDependencies": { "three": ">=0.170.0" } }, "sha512-pmt1ALRhbHhCJQTj2FuthH6PeLIeaM4hOuS2JO3kWSwlnvx/9xuUkjFR3JOi/myMqsH7pSsLIROSaBxDfttjeA=="],
-
- "@threlte/core": ["@threlte/core@8.3.1", "", { "dependencies": { "mitt": "^3.0.1" }, "peerDependencies": { "svelte": ">=5", "three": ">=0.160" } }, "sha512-qKjjNCQ+40hyeBcfOMh/8ef5x/j5PG5Wmo/L9Ye0aDCcdD6fCewWxfp7tV/J3CxPzX1dEp1JGK7sjyc7ntZSrg=="],
-
- "@threlte/extras": ["@threlte/extras@9.7.1", "", { "dependencies": { "@threejs-kit/instanced-sprite-mesh": "^2.5.1", "camera-controls": "^3.1.2", "three-mesh-bvh": "^0.9.1", "three-perf": "^1.0.11", "three-viewport-gizmo": "^2.2.0", "troika-three-text": "^0.52.4" }, "peerDependencies": { "svelte": ">=5", "three": ">=0.160" } }, "sha512-SGm59HDCdHxADFHuweHfFDknwubkCPodyK0pbfsVtOWWOX26gE2xfK7aKolh6YFDiPAjWjGxN0jIgkNbbr1ohg=="],
-
- "@tweenjs/tween.js": ["@tweenjs/tween.js@23.1.3", "", {}, "sha512-vJmvvwFxYuGnF2axRtPYocag6Clbb5YS7kLL+SO/TeVFzHqDIWrNKYtcsPMibjDx9O+bu+psAy9NKfWklassUA=="],
-
"@tybys/wasm-util": ["@tybys/wasm-util@0.9.0", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-6+7nlbMVX/PVDCwaIQ8nTOPveOcFLSt8GcXdx8hD0bt39uWxYT88uXzqTd4fTvqta7oeUJqudepapKNt2DYJFw=="],
"@types/chai": ["@types/chai@5.2.3", "", { "dependencies": { "@types/deep-eql": "*", "assertion-error": "^2.0.1" } }, "sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA=="],
@@ -477,16 +458,10 @@
"@types/node": ["@types/node@25.3.0", "", { "dependencies": { "undici-types": "~7.18.0" } }, "sha512-4K3bqJpXpqfg2XKGK9bpDTc6xO/xoUP/RBWS7AtRMug6zZFaRekiLzjVtAoZMquxoAbzBvy5nxQ7veS5eYzf8A=="],
- "@types/stats.js": ["@types/stats.js@0.17.4", "", {}, "sha512-jIBvWWShCvlBqBNIZt0KAshWpvSjhkwkEu4ZUcASoAvhmrgAUI2t1dXrjSL4xXVLB4FznPrIsX3nKXFl/Dt4vA=="],
-
- "@types/three": ["@types/three@0.182.0", "", { "dependencies": { "@dimforge/rapier3d-compat": "~0.12.0", "@tweenjs/tween.js": "~23.1.3", "@types/stats.js": "*", "@types/webxr": ">=0.5.17", "@webgpu/types": "*", "fflate": "~0.8.2", "meshoptimizer": "~0.22.0" } }, "sha512-WByN9V3Sbwbe2OkWuSGyoqQO8Du6yhYaXtXLoA5FkKTUJorZ+yOHBZ35zUUPQXlAKABZmbYp5oAqpA4RBjtJ/Q=="],
-
"@types/trusted-types": ["@types/trusted-types@2.0.7", "", {}, "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw=="],
"@types/unist": ["@types/unist@2.0.11", "", {}, "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA=="],
- "@types/webxr": ["@types/webxr@0.5.24", "", {}, "sha512-h8fgEd/DpoS9CBrjEQXR+dIDraopAEfu4wYVNY2tEPwk60stPWhvZMf4Foo5FakuQ7HFZoa8WceaWFervK2Ovg=="],
-
"@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.57.0", "", { "dependencies": { "@eslint-community/regexpp": "^4.12.2", "@typescript-eslint/scope-manager": "8.57.0", "@typescript-eslint/type-utils": "8.57.0", "@typescript-eslint/utils": "8.57.0", "@typescript-eslint/visitor-keys": "8.57.0", "ignore": "^7.0.5", "natural-compare": "^1.4.0", "ts-api-utils": "^2.4.0" }, "peerDependencies": { "@typescript-eslint/parser": "^8.57.0", "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-qeu4rTHR3/IaFORbD16gmjq9+rEs9fGKdX0kF6BKSfi+gCuG3RCKLlSBYzn/bGsY9Tj7KE/DAQStbp8AHJGHEQ=="],
"@typescript-eslint/parser": ["@typescript-eslint/parser@8.57.0", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.57.0", "@typescript-eslint/types": "8.57.0", "@typescript-eslint/typescript-estree": "8.57.0", "@typescript-eslint/visitor-keys": "8.57.0", "debug": "^4.4.3" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-XZzOmihLIr8AD1b9hL9ccNMzEMWt/dE2u7NyTY9jJG6YNiNthaD5XtUHVF2uCXZ15ng+z2hT3MVuxnUYhq6k1g=="],
@@ -523,8 +498,6 @@
"@vitest/utils": ["@vitest/utils@4.1.0", "", { "dependencies": { "@vitest/pretty-format": "4.1.0", "convert-source-map": "^2.0.0", "tinyrainbow": "^3.0.3" } }, "sha512-XfPXT6a8TZY3dcGY8EdwsBulFCIw+BeeX0RZn2x/BtiY/75YGh8FeWGG8QISN/WhaqSrE2OrlDgtF8q5uhOTmw=="],
- "@webgpu/types": ["@webgpu/types@0.1.68", "", {}, "sha512-3ab1B59Ojb6RwjOspYLsTpCzbNB3ZaamIAxBMmvnNkiDoLTZUOBXZ9p5nAYVEkQlDdf6qAZWi1pqj9+ypiqznA=="],
-
"@yarnpkg/lockfile": ["@yarnpkg/lockfile@1.1.0", "", {}, "sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ=="],
"@yarnpkg/parsers": ["@yarnpkg/parsers@3.0.2", "", { "dependencies": { "js-yaml": "^3.10.0", "tslib": "^2.4.0" } }, "sha512-/HcYgtUSiJiot/XWGLOlGxPYUG65+/31V8oqk17vZLW1xlCoR4PampyePljOxY2n8/3jz9+tIFzICsyGujJZoA=="],
@@ -563,8 +536,6 @@
"base64-js": ["base64-js@1.5.1", "", {}, "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA=="],
- "bidi-js": ["bidi-js@1.0.3", "", { "dependencies": { "require-from-string": "^2.0.2" } }, "sha512-RKshQI1R3YQ+n9YJz2QQ147P66ELpa1FQEg20Dk8oW9t2KgLbpDLLp9aGZ7y8WHSshDknG0bknqGw5/tyCs5tw=="],
-
"bl": ["bl@4.1.0", "", { "dependencies": { "buffer": "^5.5.0", "inherits": "^2.0.4", "readable-stream": "^3.4.0" } }, "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w=="],
"blake3-wasm": ["blake3-wasm@2.1.5", "", {}, "sha512-F1+K8EbfOZE49dtoPtmxUQrpXaBIl3ICvasLh+nJta0xkz+9kF/7uet9fLnwKqhDrmj6g+6K3Tw9yQPUg2ka5g=="],
@@ -577,8 +548,6 @@
"callsites": ["callsites@3.1.0", "", {}, "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ=="],
- "camera-controls": ["camera-controls@3.1.2", "", { "peerDependencies": { "three": ">=0.126.1" } }, "sha512-xkxfpG2ECZ6Ww5/9+kf4mfg1VEYAoe9aDSY+IwF0UEs7qEzwy0aVRfs2grImIECs/PoBtWFrh7RXsQkwG922JA=="],
-
"carbon-icons-svelte": ["carbon-icons-svelte@13.9.0", "", {}, "sha512-+5bd/94B1mRt0sdCTADLfy5ArJRxtz5gZdIZBF3EYJPHnWbJUD5/y/S7Q/QoOSzco+B7MSsgezqIQYpmvBYK5w=="],
"ccount": ["ccount@2.0.1", "", {}, "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg=="],
@@ -641,16 +610,12 @@
"devlop": ["devlop@1.1.0", "", { "dependencies": { "dequal": "^2.0.0" } }, "sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA=="],
- "diet-sprite": ["diet-sprite@0.0.1", "", {}, "sha512-zSHI2WDAn1wJqJYxcmjWfJv3Iw8oL9reQIbEyx2x2/EZ4/qmUTIo8/5qOCurnAcq61EwtJJaZ0XTy2NRYqpB5A=="],
-
"dotenv": ["dotenv@16.4.7", "", {}, "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ=="],
"dotenv-expand": ["dotenv-expand@11.0.7", "", { "dependencies": { "dotenv": "^16.4.5" } }, "sha512-zIHwmZPRshsCdpMDyVsqGmgyP0yT8GAgXUnkdAoJisxvf33k7yO6OuoKmcTGuXPWSsm8Oh88nZicRLA9Y0rUeA=="],
"dunder-proto": ["dunder-proto@1.0.1", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.1", "es-errors": "^1.3.0", "gopd": "^1.2.0" } }, "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A=="],
- "earcut": ["earcut@2.2.4", "", {}, "sha512-/pjZsA1b4RPHbeWZQn66SWS8nZZWLQQ23oE3Eam7aroEFGEvwKAsJfZ9ytiEMycfzXWpca4FA9QIOehf7PocBQ=="],
-
"ejs": ["ejs@3.1.10", "", { "dependencies": { "jake": "^10.8.5" }, "bin": { "ejs": "bin/cli.js" } }, "sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA=="],
"emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="],
@@ -721,8 +686,6 @@
"fdir": ["fdir@6.5.0", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg=="],
- "fflate": ["fflate@0.8.2", "", {}, "sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A=="],
-
"figures": ["figures@3.2.0", "", { "dependencies": { "escape-string-regexp": "^1.0.5" } }, "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg=="],
"file-entry-cache": ["file-entry-cache@8.0.0", "", { "dependencies": { "flat-cache": "^4.0.0" } }, "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ=="],
@@ -885,8 +848,6 @@
"log-symbols": ["log-symbols@4.1.0", "", { "dependencies": { "chalk": "^4.1.0", "is-unicode-supported": "^0.1.0" } }, "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg=="],
- "maath": ["maath@0.10.8", "", { "peerDependencies": { "@types/three": ">=0.134.0", "three": ">=0.134.0" } }, "sha512-tRvbDF0Pgqz+9XUa4jjfgAQ8/aPKmQdWXilFu2tMy4GWj4NOsx99HlULO4IeREfbO3a0sA145DZYyvXPkybm0g=="],
-
"magic-string": ["magic-string@0.30.21", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ=="],
"math-intrinsics": ["math-intrinsics@1.1.0", "", {}, "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g=="],
@@ -895,8 +856,6 @@
"mdsvex": ["mdsvex@0.12.6", "", { "dependencies": { "@types/mdast": "^4.0.4", "@types/unist": "^2.0.3", "prism-svelte": "^0.4.7", "prismjs": "^1.17.1", "unist-util-visit": "^2.0.1", "vfile-message": "^2.0.4" }, "peerDependencies": { "svelte": "^3.56.0 || ^4.0.0 || ^5.0.0-next.120" } }, "sha512-pupx2gzWh3hDtm/iDW4WuCpljmyHbHi34r7ktOqpPGvyiM4MyfNgdJ3qMizXdgCErmvYC9Nn/qyjePy+4ss9Wg=="],
- "meshoptimizer": ["meshoptimizer@0.22.0", "", {}, "sha512-IebiK79sqIy+E4EgOr+CAw+Ke8hAspXKzBd0JdgEmPHiAwmvEj2S4h1rfvo+o/BnfEYd/jAOg5IeeIjzlzSnDg=="],
-
"micromark-util-character": ["micromark-util-character@2.1.1", "", { "dependencies": { "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-wv8tdUTJ3thSFFFJKtpYKOYiGP2+v96Hvk4Tu8KpCAsTMs6yi+nVmGh1syvSCsaxz45J6Jbw+9DD6g97+NV67Q=="],
"micromark-util-encode": ["micromark-util-encode@2.0.1", "", {}, "sha512-c3cVx2y4KqUnwopcO9b/SCdo2O67LwJJ/UyqGfbigahfegL9myoEFoDYZgkT7f36T0bLrM9hZTAaAyH+PCAXjw=="],
@@ -919,8 +878,6 @@
"minimist": ["minimist@1.2.8", "", {}, "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA=="],
- "mitt": ["mitt@3.0.1", "", {}, "sha512-vKivATfr97l2/QBCYAkXYDbrIWPM2IIKEl7YPhjCvKlG3kE2gm+uBo6nEXK3M5/Ffh/FLpKExzOQ3JJoJGFKBw=="],
-
"motion-core": ["motion-core@workspace:packages/motion-core"],
"mri": ["mri@1.2.0", "", {}, "sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA=="],
@@ -1031,8 +988,6 @@
"require-directory": ["require-directory@2.1.1", "", {}, "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q=="],
- "require-from-string": ["require-from-string@2.0.2", "", {}, "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw=="],
-
"resolve-from": ["resolve-from@4.0.0", "", {}, "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g=="],
"resolve.exports": ["resolve.exports@2.0.3", "", {}, "sha512-OcXjMsGdhL4XnbShKpAcSqPMzQoYkYyhbEaeSko47MjRP9NfEQMhZkXL1DoFlt9LWQn4YttrdnV6X2OiyzBi+A=="],
@@ -1103,16 +1058,6 @@
"tar-stream": ["tar-stream@2.2.0", "", { "dependencies": { "bl": "^4.0.3", "end-of-stream": "^1.4.1", "fs-constants": "^1.0.0", "inherits": "^2.0.3", "readable-stream": "^3.1.1" } }, "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ=="],
- "three": ["three@0.182.0", "", {}, "sha512-GbHabT+Irv+ihI1/f5kIIsZ+Ef9Sl5A1Y7imvS5RQjWgtTPfPnZ43JmlYI7NtCRDK9zir20lQpfg8/9Yd02OvQ=="],
-
- "three-instanced-uniforms-mesh": ["three-instanced-uniforms-mesh@0.52.4", "", { "dependencies": { "troika-three-utils": "^0.52.4" }, "peerDependencies": { "three": ">=0.125.0" } }, "sha512-YwDBy05hfKZQtU+Rp0KyDf9yH4GxfhxMbVt9OYruxdgLfPwmDG5oAbGoW0DrKtKZSM3BfFcCiejiOHCjFBTeng=="],
-
- "three-mesh-bvh": ["three-mesh-bvh@0.9.4", "", { "peerDependencies": { "three": ">= 0.159.0" } }, "sha512-+y6xLS6k5LWkNNhYsTgKXBC2D9r/z0swiehVHYhZZ8AOhaKDRCWKsN94ctV5Xy7xA4Xbnv4LKYzf7epRLPT6oQ=="],
-
- "three-perf": ["three-perf@1.0.11", "", { "dependencies": { "troika-three-text": "^0.52.0", "tweakpane": "^3.1.10" }, "peerDependencies": { "three": ">=0.170" } }, "sha512-OgBpZjwL+csQKGKZjpkH/QHdbGFMxqngMbSEJeSnVNfXDYd6On7WHNv/GhUZH4YxIpNMwMahBWrNnsJvnbSJHQ=="],
-
- "three-viewport-gizmo": ["three-viewport-gizmo@2.2.0", "", { "peerDependencies": { "three": ">=0.162.0 <1.0.0" } }, "sha512-Jo9Liur1rUmdKk75FZumLU/+hbF+RtJHi1qsKZpntjKlCYScK6tjbYoqvJ9M+IJphrlQJF5oReFW7Sambh0N4Q=="],
-
"tinybench": ["tinybench@2.9.0", "", {}, "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg=="],
"tinyexec": ["tinyexec@1.0.4", "", {}, "sha512-u9r3uZC0bdpGOXtlxUIdwf9pkmvhqJdrVCH9fapQtgy/OeTTMZ1nqH7agtvEfmGui6e1XxjcdrlxvxJvc3sMqw=="],
@@ -1129,12 +1074,6 @@
"trim-lines": ["trim-lines@3.0.1", "", {}, "sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg=="],
- "troika-three-text": ["troika-three-text@0.52.4", "", { "dependencies": { "bidi-js": "^1.0.2", "troika-three-utils": "^0.52.4", "troika-worker-utils": "^0.52.0", "webgl-sdf-generator": "1.1.1" }, "peerDependencies": { "three": ">=0.125.0" } }, "sha512-V50EwcYGruV5rUZ9F4aNsrytGdKcXKALjEtQXIOBfhVoZU9VAqZNIoGQ3TMiooVqFAbR1w15T+f+8gkzoFzawg=="],
-
- "troika-three-utils": ["troika-three-utils@0.52.4", "", { "peerDependencies": { "three": ">=0.125.0" } }, "sha512-NORAStSVa/BDiG52Mfudk4j1FG4jC4ILutB3foPnfGbOeIs9+G5vZLa0pnmnaftZUGm4UwSoqEpWdqvC7zms3A=="],
-
- "troika-worker-utils": ["troika-worker-utils@0.52.0", "", {}, "sha512-W1CpvTHykaPH5brv5VHLfQo9D1OYuo0cSBEUQFFT/nBUzM8iD6Lq2/tgG/f1OelbAS1WtaTPQzE5uM49egnngw=="],
-
"trough": ["trough@2.2.0", "", {}, "sha512-tmMpK00BjZiUyVyvrBK7knerNgmgvcV/KLVyuma/SC+TQN167GrMRciANTz09+k3zW8L8t60jWO1GpfkZdjTaw=="],
"ts-api-utils": ["ts-api-utils@2.4.0", "", { "peerDependencies": { "typescript": ">=4.8.4" } }, "sha512-3TaVTaAv2gTiMB35i3FiGJaRfwb3Pyn/j3m/bfAvGe8FB7CF6u+LMYqYlDh7reQf7UNvoTvdfAqHGmPGOSsPmA=="],
@@ -1143,8 +1082,6 @@
"tslib": ["tslib@2.8.1", "", {}, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
- "tweakpane": ["tweakpane@3.1.10", "", {}, "sha512-rqwnl/pUa7+inhI2E9ayGTqqP0EPOOn/wVvSWjZsRbZUItzNShny7pzwL3hVlaN4m9t/aZhsP0aFQ9U5VVR2VQ=="],
-
"type-check": ["type-check@0.4.0", "", { "dependencies": { "prelude-ls": "^1.2.1" } }, "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew=="],
"typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="],
@@ -1189,8 +1126,6 @@
"web-namespaces": ["web-namespaces@2.0.1", "", {}, "sha512-bKr1DkiNa2krS7qxNtdrtHAmzuYGFQLiQ13TsorsdT6ULTkPLKuu5+GsFpDlg6JFjUTwX2DyhMPG2be8uPrqsQ=="],
- "webgl-sdf-generator": ["webgl-sdf-generator@1.1.1", "", {}, "sha512-9Z0JcMTFxeE+b2x1LJTdnaT8rT8aEp7MVxkNwoycNmJWwPdzoXzMh0BjJSh/AEFP+KPYZUli814h8bJZFIZ2jA=="],
-
"which": ["which@2.0.2", "", { "dependencies": { "isexe": "^2.0.0" }, "bin": { "node-which": "./bin/node-which" } }, "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA=="],
"why-is-node-running": ["why-is-node-running@2.3.0", "", { "dependencies": { "siginfo": "^2.0.0", "stackback": "0.0.2" }, "bin": { "why-is-node-running": "cli.js" } }, "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w=="],
diff --git a/packages/motion-core/package.json b/packages/motion-core/package.json
index 7afc0de..ba4d09a 100644
--- a/packages/motion-core/package.json
+++ b/packages/motion-core/package.json
@@ -15,21 +15,14 @@
"svelte": "^5.0.0",
"gsap": "^3.14.2",
"tailwindcss": "^4.1.18",
- "three": "^0.182.0",
- "@threlte/core": "^8.0.0",
- "@threlte/extras": "^9.0.0",
"@mediapipe/tasks-vision": "^0.10.22-rc.20250304",
"ogl": "^1.0.11"
},
"devDependencies": {
"@mediapipe/tasks-vision": "^0.10.22-rc.20250304",
- "@threlte/core": "^8.3.1",
- "@threlte/extras": "^9.7.1",
"@types/gsap": "^3.0.0",
- "@types/three": "^0.182.0",
"ogl": "^1.0.11",
- "tailwindcss": "^4.1.18",
- "three": "^0.182.0"
+ "tailwindcss": "^4.1.18"
},
"dependencies": {
"clsx": "^2.1.1",
From 04bc40abaf8d8b5013aecf7c6ca35b14858d0aa6 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Marek=20J=C3=B3=C5=BAwiak?=
<168720167+66HEX@users.noreply.github.com>
Date: Thu, 16 Apr 2026 20:43:00 +0200
Subject: [PATCH 15/24] chore(globe): update default base color
---
apps/web/src/routes/docs/globe/+page.svx | 2 +-
packages/motion-core/src/lib/components/globe/GlobeScene.svelte | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/apps/web/src/routes/docs/globe/+page.svx b/apps/web/src/routes/docs/globe/+page.svx
index da32431..692556b 100644
--- a/apps/web/src/routes/docs/globe/+page.svx
+++ b/apps/web/src/routes/docs/globe/+page.svx
@@ -173,7 +173,7 @@ import { Globe } from "$lib/motion-core";
{
key: "color",
type: "string | number | readonly [number, number, number] | { r: number; g: number; b: number }",
- default: '"#111113"',
+ default: '"#17181A"',
description: "Base color before the rim glow is applied.",
},
{
diff --git a/packages/motion-core/src/lib/components/globe/GlobeScene.svelte b/packages/motion-core/src/lib/components/globe/GlobeScene.svelte
index 12187ea..f800e53 100644
--- a/packages/motion-core/src/lib/components/globe/GlobeScene.svelte
+++ b/packages/motion-core/src/lib/components/globe/GlobeScene.svelte
@@ -158,7 +158,7 @@
const VISIBILITY_MAX_DOT = 0.48;
const defaultFresnelConfig: Required
= {
- color: "#111113",
+ color: "#17181A",
rimColor: "#FF6900",
rimPower: 6,
rimIntensity: 1.5,
From 76da7950068b4977f794cf29c171fd2ce7ab6d5e Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Marek=20J=C3=B3=C5=BAwiak?=
<168720167+66HEX@users.noreply.github.com>
Date: Thu, 16 Apr 2026 20:47:24 +0200
Subject: [PATCH 16/24] perf(globe): add will-change transform to marker
tooltip
---
.../motion-core/src/lib/components/globe/GlobeMarkerItem.svelte | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte b/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte
index f74a17f..91ef9a9 100644
--- a/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte
+++ b/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte
@@ -65,7 +65,7 @@
{#if tooltip || marker.label}
From fdd825cbb64bd2da40939691d9302144cceb089d Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Marek=20J=C3=B3=C5=BAwiak?=
<168720167+66HEX@users.noreply.github.com>
Date: Thu, 16 Apr 2026 21:00:15 +0200
Subject: [PATCH 17/24] refactor(globe): render markers in shader and keep html
tooltips
---
.../components/globe/GlobeMarkerItem.svelte | 19 +---
.../lib/components/globe/GlobeScene.svelte | 87 ++++++++++++++++---
2 files changed, 77 insertions(+), 29 deletions(-)
diff --git a/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte b/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte
index 91ef9a9..9a1a414 100644
--- a/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte
+++ b/packages/motion-core/src/lib/components/globe/GlobeMarkerItem.svelte
@@ -23,17 +23,13 @@
* Marker visibility factor in range [0, 1].
*/
visibility: number;
- /**
- * Marker visual size in pixels.
- */
- sizePx: number;
/**
* Optional custom tooltip snippet.
*/
tooltip?: Snippet<[GlobeMarkerTooltipContext]>;
}
- let { marker, index, screenX, screenY, visibility, sizePx, tooltip }: Props =
+ let { marker, index, screenX, screenY, visibility, tooltip }: Props =
$props();
const MAX_TOOLTIP_BLUR = 8;
@@ -44,9 +40,6 @@
index,
visibility,
});
- let markerColor = $derived(marker.color ?? "#ffffff");
- let markerOpacity = $derived(visibility);
- let clampedSizePx = $derived(Math.max(2, sizePx));
-
-
{#if tooltip || marker.label}