style: reduce frontend visual effects
This commit is contained in:
@@ -1,95 +1,36 @@
|
||||
"use client";
|
||||
|
||||
import { motion, useReducedMotion } from "framer-motion";
|
||||
import type { SynopStation, WeatherMood } from "@/types/imgw";
|
||||
|
||||
const windLines = Array.from({ length: 7 }, (_, index) => ({
|
||||
top: `${18 + index * 11}%`,
|
||||
delay: index * 0.22,
|
||||
width: 70 + (index % 3) * 34,
|
||||
}));
|
||||
|
||||
const stars = Array.from({ length: 16 }, (_, index) => ({
|
||||
left: `${(index * 37 + 11) % 96}%`,
|
||||
top: `${(index * 23 + 8) % 72}%`,
|
||||
delay: (index % 6) * 0.35,
|
||||
}));
|
||||
|
||||
const rainDrops = Array.from({ length: 22 }, (_, index) => ({
|
||||
const rainDrops = Array.from({ length: 12 }, (_, index) => ({
|
||||
left: `${(index * 43 + 7) % 101}%`,
|
||||
delay: (index % 9) * 0.18,
|
||||
duration: 0.8 + (index % 4) * 0.14,
|
||||
duration: 1.1 + (index % 4) * 0.18,
|
||||
}));
|
||||
|
||||
export function WeatherEffects({ station, mood, precipitation10m, thunderstorm = false }: { station: SynopStation; mood: WeatherMood; precipitation10m?: number | null; thunderstorm?: boolean }) {
|
||||
export function WeatherEffects({ precipitation10m, thunderstorm = false }: { precipitation10m?: number | null; thunderstorm?: boolean }) {
|
||||
const reduceMotion = useReducedMotion();
|
||||
const isWindy = (station.windSpeed ?? 0) >= 8;
|
||||
const isRaining = (precipitation10m ?? 0) > 0;
|
||||
|
||||
return (
|
||||
<div aria-hidden="true" className="pointer-events-none absolute inset-0 z-[1] overflow-hidden">
|
||||
{mood === "cloudy" && (
|
||||
<>
|
||||
<motion.div
|
||||
animate={reduceMotion ? undefined : { x: ["-4%", "5%", "-4%"], y: [0, 8, 0], scale: [1, 1.05, 1] }}
|
||||
transition={{ duration: 18, repeat: Infinity, ease: "easeInOut" }}
|
||||
className="absolute -left-24 -top-20 h-52 w-[78%] rounded-full bg-slate-100/25 blur-2xl"
|
||||
/>
|
||||
<motion.div
|
||||
animate={reduceMotion ? undefined : { x: ["8%", "-5%", "8%"], y: [0, -6, 0], scale: [1.04, 1, 1.04] }}
|
||||
transition={{ duration: 22, repeat: Infinity, ease: "easeInOut" }}
|
||||
className="absolute -right-24 top-0 h-60 w-[82%] rounded-full bg-slate-300/20 blur-2xl"
|
||||
/>
|
||||
<div className="absolute inset-x-0 bottom-0 h-44 bg-gradient-to-t from-slate-100/25 via-slate-200/10 to-transparent" />
|
||||
<div className="absolute inset-x-0 top-0 h-32 bg-gradient-to-b from-slate-950/15 to-transparent" />
|
||||
</>
|
||||
)}
|
||||
{mood === "warm" && (
|
||||
<motion.div
|
||||
animate={reduceMotion ? undefined : { scale: [1, 1.08, 1], opacity: [0.4, 0.58, 0.4] }}
|
||||
transition={{ duration: 7, repeat: Infinity, ease: "easeInOut" }}
|
||||
className="absolute -right-12 -top-16 size-52 rounded-full bg-amber-200/25 blur-2xl"
|
||||
/>
|
||||
)}
|
||||
{mood === "night" && stars.map((star, index) => (
|
||||
<motion.span
|
||||
key={index}
|
||||
animate={reduceMotion ? undefined : { opacity: [0.2, 0.75, 0.2] }}
|
||||
transition={{ duration: 2.4, delay: star.delay, repeat: Infinity, ease: "easeInOut" }}
|
||||
className="absolute size-1 rounded-full bg-white/75"
|
||||
style={{ left: star.left, top: star.top }}
|
||||
/>
|
||||
))}
|
||||
{isWindy && windLines.map((line, index) => (
|
||||
<motion.span
|
||||
key={index}
|
||||
initial={{ x: "-30%", opacity: 0 }}
|
||||
animate={reduceMotion ? { opacity: 0.28 } : { x: ["-30%", "135%"], opacity: [0, 0.35, 0] }}
|
||||
transition={{ duration: 3.4, delay: line.delay, repeat: Infinity, ease: "linear" }}
|
||||
className="absolute h-px rounded-full bg-white/60"
|
||||
style={{ top: line.top, width: line.width }}
|
||||
/>
|
||||
))}
|
||||
{isRaining && rainDrops.map((drop, index) => (
|
||||
<motion.span
|
||||
key={`rain-${index}`}
|
||||
initial={{ y: "-12vh", opacity: 0 }}
|
||||
animate={reduceMotion ? { opacity: 0.36 } : { y: ["-12vh", "115vh"], opacity: [0, 0.55, 0] }}
|
||||
animate={reduceMotion ? { opacity: 0.18 } : { y: ["-12vh", "115vh"], opacity: [0, 0.22, 0] }}
|
||||
transition={{ duration: drop.duration, delay: drop.delay, repeat: Infinity, ease: "linear" }}
|
||||
className="absolute -top-8 h-14 w-px rotate-[8deg] rounded-full bg-gradient-to-b from-transparent via-white/55 to-transparent blur-[0.35px]"
|
||||
className="absolute -top-8 h-10 w-px rotate-[8deg] rounded-full bg-foreground/35"
|
||||
style={{ left: drop.left }}
|
||||
/>
|
||||
))}
|
||||
{thunderstorm && (
|
||||
<motion.div
|
||||
animate={reduceMotion ? { opacity: 0.12 } : { opacity: [0, 0, 0.34, 0, 0.18, 0] }}
|
||||
animate={reduceMotion ? { opacity: 0.08 } : { opacity: [0, 0, 0.16, 0, 0.08, 0] }}
|
||||
transition={{ duration: 6, repeat: Infinity, repeatDelay: 2.5 }}
|
||||
className="absolute inset-0 bg-white"
|
||||
className="absolute inset-0 bg-foreground"
|
||||
/>
|
||||
)}
|
||||
{mood === "cold" && (
|
||||
<div className="absolute inset-x-0 bottom-0 h-28 bg-gradient-to-t from-blue-100/20 to-transparent" />
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -12,7 +12,7 @@ import {
|
||||
formatWind,
|
||||
getWeatherDescription,
|
||||
getWeatherMoodFromData,
|
||||
moodGradient,
|
||||
moodAccentClass,
|
||||
} from "@/lib/weather-utils";
|
||||
import type { SynopStation } from "@/types/imgw";
|
||||
import type { ImgwCurrentWeather } from "@/types/imgw-current";
|
||||
@@ -37,6 +37,7 @@ export function WeatherHero({ station, currentWeather, currentWeatherLoading = f
|
||||
rainfall: currentWeather.precipitation10m ?? station.rainfall,
|
||||
} : station;
|
||||
const mood = getWeatherMoodFromData(displayedStation);
|
||||
const moodAccent = moodAccentClass(mood);
|
||||
const feelsLike = currentWeather?.feelsLike ?? calculateFeelsLike(displayedStation.temperature, displayedStation.humidity, displayedStation.windSpeed);
|
||||
const metrics = [
|
||||
{ icon: Droplets, label: t("weather.humidity"), value: formatHumidity(displayedStation.humidity, language) },
|
||||
@@ -50,13 +51,18 @@ export function WeatherHero({ station, currentWeather, currentWeatherLoading = f
|
||||
initial={{ opacity: 0, y: 18 }}
|
||||
animate={{ opacity: 1, y: 0 }}
|
||||
transition={{ duration: 0.55, ease: "easeOut" }}
|
||||
className={`relative isolate overflow-hidden rounded-panel bg-gradient-to-br ${moodGradient(mood)} px-5 py-6 text-white shadow-card sm:px-8 sm:py-8 lg:px-10`}
|
||||
className="relative isolate overflow-hidden rounded-panel border border-border/70 bg-surface-raised px-5 py-6 shadow-card sm:px-8 sm:py-8 lg:px-10"
|
||||
>
|
||||
<WeatherEffects station={displayedStation} mood={mood} precipitation10m={currentWeather?.precipitation10m} thunderstorm={currentWeather?.condition === "thunderstorm"} />
|
||||
<WeatherEffects precipitation10m={currentWeather?.precipitation10m} thunderstorm={currentWeather?.condition === "thunderstorm"} />
|
||||
<div className="relative z-10">
|
||||
<div>
|
||||
<span className="flex items-center gap-1.5 text-sm font-medium text-white/85"><MapPin className="size-4" />{displayedLocationName}</span>
|
||||
<div className="mt-1.5 space-y-1 text-xs text-white/65">
|
||||
<div className="flex flex-wrap items-center gap-2">
|
||||
<span className="flex items-center gap-1.5 text-sm font-medium text-muted"><MapPin className="size-4" />{displayedLocationName}</span>
|
||||
<span className={`rounded-control border px-2.5 py-1 text-[0.68rem] font-semibold uppercase tracking-[0.14em] ${moodAccent}`}>
|
||||
{getWeatherDescription(displayedStation, language, currentWeather?.condition)}
|
||||
</span>
|
||||
</div>
|
||||
<div className="mt-2 space-y-1 text-xs text-muted">
|
||||
<p>{currentWeatherLoading
|
||||
? t("location.heroHybridLoading", { station: station.name })
|
||||
: hasFullHybridAnalysis
|
||||
@@ -67,7 +73,7 @@ export function WeatherHero({ station, currentWeather, currentWeatherLoading = f
|
||||
? t("location.heroStationFallbackWithDistance", { station: station.name, distance: distanceKm ?? 0 })
|
||||
: t("location.heroStationFallback", { station: station.name })}</p>
|
||||
{hasFullHybridAnalysis && locationName && <p>{t("location.heroNearestStation", { station: station.name, distance: distanceKm ?? 0 })}</p>}
|
||||
{hasDistantFallback && <p className="flex items-start gap-1.5 text-amber-100"><AlertTriangle className="mt-0.5 size-3.5 shrink-0" />{t("location.heroDistantFallback")}</p>}
|
||||
{hasDistantFallback && <p className="flex items-start gap-1.5 text-warning"><AlertTriangle className="mt-0.5 size-3.5 shrink-0" />{t("location.heroDistantFallback")}</p>}
|
||||
</div>
|
||||
</div>
|
||||
<div className="mt-8 flex items-end justify-between gap-3 sm:mt-10">
|
||||
@@ -76,20 +82,20 @@ export function WeatherHero({ station, currentWeather, currentWeatherLoading = f
|
||||
{formatTemperature(displayedStation.temperature, language)}
|
||||
</div>
|
||||
<p className="mt-5 text-xl font-medium tracking-tight sm:text-2xl">{getWeatherDescription(displayedStation, language, currentWeather?.condition)}</p>
|
||||
<p className="mt-1 text-sm text-white/75">{t("weather.feelsLike")} {formatTemperature(feelsLike, language)} · {t("weather.measurement")} {formatDateTime(displayedStation.measuredAt, language)}</p>
|
||||
<p className="mt-1 text-sm text-muted">{t("weather.feelsLike")} {formatTemperature(feelsLike, language)} · {t("weather.measurement")} {formatDateTime(displayedStation.measuredAt, language)}</p>
|
||||
</div>
|
||||
<WeatherIcon mood={mood} condition={currentWeather?.condition} className="mb-4 size-20 text-white/80 sm:size-28" />
|
||||
<WeatherIcon mood={mood} condition={currentWeather?.condition} className="mb-4 size-20 text-accent sm:size-28" />
|
||||
</div>
|
||||
<div className="mt-8 grid grid-cols-2 gap-2.5 sm:mt-10 lg:grid-cols-4">
|
||||
{metrics.map(({ icon: Icon, label, value }) => (
|
||||
<div key={label} className="rounded-card border border-white/20 bg-white/10 p-3.5 backdrop-blur-xl">
|
||||
<div className="flex items-center gap-2 text-xs text-white/70"><Icon className="size-3.5" />{label}</div>
|
||||
<div key={label} className="rounded-card border border-border/60 bg-surface-muted p-3.5">
|
||||
<div className="flex items-center gap-2 text-xs text-muted"><Icon className="size-3.5 text-accent" />{label}</div>
|
||||
<p className="mt-2 text-base font-semibold">{value}</p>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
{displayedStation.windDirection !== null && (
|
||||
<p className="mt-4 flex items-center gap-1.5 text-xs text-white/70">
|
||||
<p className="mt-4 flex items-center gap-1.5 text-xs text-muted">
|
||||
<Navigation className="size-3.5" style={{ transform: `rotate(${displayedStation.windDirection}deg)` }} />
|
||||
{t("weather.windDirection")}: {displayedStation.windDirection}°
|
||||
</p>
|
||||
|
||||
Reference in New Issue
Block a user