mirror of
https://github.com/ajnart/homarr.git
synced 2025-11-07 22:15:49 +01:00
191 lines
5.3 KiB
TypeScript
191 lines
5.3 KiB
TypeScript
import { Group, Space, Title, Tooltip, Skeleton } from '@mantine/core';
|
|
import axios from 'axios';
|
|
import { useEffect, useState } from 'react';
|
|
import {
|
|
IconArrowDownRight as ArrowDownRight,
|
|
IconArrowUpRight as ArrowUpRight,
|
|
IconCloud as Cloud,
|
|
IconCloudFog as CloudFog,
|
|
IconCloudRain as CloudRain,
|
|
IconCloudSnow as CloudSnow,
|
|
IconCloudStorm as CloudStorm,
|
|
IconQuestionMark as QuestionMark,
|
|
IconSnowflake as Snowflake,
|
|
IconSun as Sun,
|
|
} from '@tabler/icons';
|
|
import { useConfig } from '../../../tools/state';
|
|
import { IModule } from '../modules';
|
|
import { WeatherResponse } from './WeatherInterface';
|
|
|
|
export const WeatherModule: IModule = {
|
|
title: 'Weather',
|
|
description: 'Look up the current weather in your location',
|
|
icon: Sun,
|
|
component: WeatherComponent,
|
|
options: {
|
|
freedomunit: {
|
|
name: 'Display in Fahrenheit',
|
|
value: false,
|
|
},
|
|
location: {
|
|
name: 'Current location',
|
|
value: 'Paris',
|
|
},
|
|
},
|
|
};
|
|
|
|
// 0 Clear sky
|
|
// 1, 2, 3 Mainly clear, partly cloudy, and overcast
|
|
// 45, 48 Fog and depositing rime fog
|
|
// 51, 53, 55 Drizzle: Light, moderate, and dense intensity
|
|
// 56, 57 Freezing Drizzle: Light and dense intensity
|
|
// 61, 63, 65 Rain: Slight, moderate and heavy intensity
|
|
// 66, 67 Freezing Rain: Light and heavy intensity
|
|
// 71, 73, 75 Snow fall: Slight, moderate, and heavy intensity
|
|
// 77 Snow grains
|
|
// 80, 81, 82 Rain showers: Slight, moderate, and violent
|
|
// 85, 86Snow showers slight and heavy
|
|
// 95 *Thunderstorm: Slight or moderate
|
|
// 96, 99 *Thunderstorm with slight and heavy hail
|
|
export function WeatherIcon(props: any) {
|
|
const { code } = props;
|
|
let data: { icon: any; name: string };
|
|
switch (code) {
|
|
case 0: {
|
|
data = { icon: Sun, name: 'Clear' };
|
|
break;
|
|
}
|
|
case 1:
|
|
case 2:
|
|
case 3: {
|
|
data = { icon: Cloud, name: 'Mainly clear' };
|
|
break;
|
|
}
|
|
case 45:
|
|
case 48: {
|
|
data = { icon: CloudFog, name: 'Fog' };
|
|
break;
|
|
}
|
|
case 51:
|
|
case 53:
|
|
case 55: {
|
|
data = { icon: Cloud, name: 'Drizzle' };
|
|
break;
|
|
}
|
|
case 56:
|
|
case 57: {
|
|
data = { icon: Snowflake, name: 'Freezing drizzle' };
|
|
break;
|
|
}
|
|
case 61:
|
|
case 63:
|
|
case 65: {
|
|
data = { icon: CloudRain, name: 'Rain' };
|
|
break;
|
|
}
|
|
case 66:
|
|
case 67: {
|
|
data = { icon: CloudRain, name: 'Freezing rain' };
|
|
break;
|
|
}
|
|
case 71:
|
|
case 73:
|
|
case 75: {
|
|
data = { icon: CloudSnow, name: 'Snow fall' };
|
|
break;
|
|
}
|
|
case 77: {
|
|
data = { icon: CloudSnow, name: 'Snow grains' };
|
|
break;
|
|
}
|
|
case 80:
|
|
case 81:
|
|
case 82: {
|
|
data = { icon: CloudRain, name: 'Rain showers' };
|
|
|
|
break;
|
|
}
|
|
case 85:
|
|
case 86: {
|
|
data = { icon: CloudSnow, name: 'Snow showers' };
|
|
break;
|
|
}
|
|
case 95: {
|
|
data = { icon: CloudStorm, name: 'Thunderstorm' };
|
|
break;
|
|
}
|
|
case 96:
|
|
case 99: {
|
|
data = { icon: CloudStorm, name: 'Thunderstorm with hail' };
|
|
break;
|
|
}
|
|
default: {
|
|
data = { icon: QuestionMark, name: 'Unknown' };
|
|
}
|
|
}
|
|
return (
|
|
<Tooltip label={data.name}>
|
|
<data.icon size={50} />
|
|
</Tooltip>
|
|
);
|
|
}
|
|
|
|
export default function WeatherComponent(props: any) {
|
|
// Get location from browser
|
|
const { config } = useConfig();
|
|
const [weather, setWeather] = useState({} as WeatherResponse);
|
|
const cityInput: string =
|
|
(config?.modules?.[WeatherModule.title]?.options?.location?.value as string) ?? 'Paris';
|
|
const isFahrenheit: boolean =
|
|
(config?.modules?.[WeatherModule.title]?.options?.freedomunit?.value as boolean) ?? false;
|
|
|
|
useEffect(() => {
|
|
axios
|
|
.get(`https://geocoding-api.open-meteo.com/v1/search?name=${cityInput}`)
|
|
.then((response) => {
|
|
// Check if results exists
|
|
const { latitude, longitude } = response.data.results
|
|
? response.data.results[0]
|
|
: { latitude: 0, longitude: 0 };
|
|
axios
|
|
.get(
|
|
`https://api.open-meteo.com/v1/forecast?latitude=${latitude}&longitude=${longitude}&daily=weathercode,temperature_2m_max,temperature_2m_min¤t_weather=true&timezone=Europe%2FLondon`
|
|
)
|
|
.then((res) => {
|
|
setWeather(res.data);
|
|
});
|
|
});
|
|
}, [cityInput]);
|
|
if (!weather.current_weather) {
|
|
return (
|
|
<>
|
|
<Skeleton height={40} width={100} mb="xl" />
|
|
<Group noWrap direction="row">
|
|
<Skeleton height={50} circle />
|
|
<Group>
|
|
<Skeleton height={25} width={70} mr="lg" />
|
|
<Skeleton height={25} width={70} />
|
|
</Group>
|
|
</Group>
|
|
</>
|
|
);
|
|
}
|
|
function usePerferedUnit(value: number): string {
|
|
return isFahrenheit ? `${(value * (9 / 5) + 32).toFixed(1)}°F` : `${value.toFixed(1)}°C`;
|
|
}
|
|
return (
|
|
<Group p="sm" spacing="xs" direction="column">
|
|
<Title>{usePerferedUnit(weather.current_weather.temperature)}</Title>
|
|
<Group spacing={0}>
|
|
<WeatherIcon code={weather.current_weather.weathercode} />
|
|
<Space mx="sm" />
|
|
<span>{usePerferedUnit(weather.daily.temperature_2m_max[0])}</span>
|
|
<ArrowUpRight size={16} style={{ right: 15 }} />
|
|
<Space mx="sm" />
|
|
<span>{usePerferedUnit(weather.daily.temperature_2m_min[0])}</span>
|
|
<ArrowDownRight size={16} />
|
|
</Group>
|
|
</Group>
|
|
);
|
|
}
|