mirror of
https://github.com/MichMich/MagicMirror.git
synced 2025-06-27 03:39:55 +00:00
## [2.29.0] - 2024-10-01 Thanks to: @bugsounet, @dkallen78, @jargordon, @khassel, @KristjanESPERANTO, @MarcLandis, @rejas, @ryan-d-williams, @sdetweil, @skpanagiotis. > ⚠️ This release needs nodejs version `v20` or `v22`, minimum version is `v20.9.0` ### Added - [compliments] Added support for cron type date/time format entries mm hh DD MM dow (minutes/hours/days/months and day of week) see https://crontab.cronhub.io for construction (#3481) - [core] Check config at every start of MagicMirror² (#3450) - [core] Add spelling check (cspell): `npm run test:spelling` and handle spelling issues (#3544) - [core] removed `config.paths.vendor` (could not work because `vendor` is hardcoded in `index.html`), renamed `config.paths.modules` to `config.foreignModulesDir`, added variable `MM_CUSTOMCSS_FILE` which - if set - overrides `config.customCss`, added variable `MM_MODULES_DIR` which - if set - overrides `config.foreignModulesDir`, added test for `MM_MODULES_DIR` (#3530) - [core] elements are now removed from `index.html` when loading script or stylesheet files fails - [core] Added `MODULE_DOM_UPDATED` notification each time the DOM is re-rendered via `updateDom` (#3534) - [tests] added minimal needed node version to tests (currently v20.9.0) to avoid releases with wrong node version info - [tests] Added `node-libgpiod` library to electron-rebuild tests (#3563) ### Removed - [core] removed installer only files (#3492) - [core] removed raspberry object from systeminformation (#3505) - [linter] removed `eslint-plugin-import`, because it doesn't support ESLint v9. We will reenter it later when it does. - [tests] removed `onoff` library from electron-rebuild tests (#3563) ### Updated - [weather] Updated `apiVersion` default from 2.5 to 3.0 (#3424) - [core] Updated dependencies including stylistic-eslint - [core] nail down `node-ical` version to `0.18.0` with exception `allow-ghsas: GHSA-8hc4-vh64-cxmj` in `dep-review.yaml` (which should removed after next `node-ical` update) - [core] Updated SocketIO catch all to new API - [core] Allow custom modules positions by scanning index.html for the defined regions, instead of hard coded (PR #3518 fixes issue #3504) - [core] Detail optimizations in `config_check.js` - [core] Updated minimal needed node version in `package.json` (currently v20.9.0) (#3559) and except for v21 (no security updates) (#3561) - [linter] Switch to ESLint v9 and flat config and replace `eslint-plugin-unicorn` by `@eslint/js` - [core] fix discovering module positions twice after #3450 ### Fixed - Fixed `checks` badge in README.md - [weather] Fixed issue with the UK Met Office provider following a change in their API paths and header info. - [core] add check for node_helper loading for multiple instances of same module (#3502) - [weather] Fixed issue for respecting unit config on broadcasted notifications - [tests] Fixes calendar test by moving it from e2e to electron with fixed date (#3532) - [calendar] fixed sliceMultiDayEvents getting wrong count and displaying incorrect entries, Europe/Berlin (#3542) - [tests] ignore `js/positions.js` when linting (this file is created at runtime) - [calendar] fixed sliceMultiDayEvents showing previous day without config enabled --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Michael Teeuw <michael@xonaymedia.nl> Co-authored-by: Kristjan ESPERANTO <35647502+KristjanESPERANTO@users.noreply.github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Ross Younger <crazyscot@gmail.com> Co-authored-by: Veeck <github@veeck.de> Co-authored-by: Bugsounet - Cédric <github@bugsounet.fr> Co-authored-by: jkriegshauser <joshuakr@nvidia.com> Co-authored-by: illimarkangur <116028111+illimarkangur@users.noreply.github.com> Co-authored-by: sam detweiler <sdetweil@gmail.com> Co-authored-by: vppencilsharpener <tim.pray@gmail.com> Co-authored-by: veeck <michael.veeck@nebenan.de> Co-authored-by: Paranoid93 <6515818+Paranoid93@users.noreply.github.com> Co-authored-by: Brian O'Connor <btoconnor@users.noreply.github.com> Co-authored-by: WallysWellies <59727507+WallysWellies@users.noreply.github.com> Co-authored-by: Jason Stieber <jrstieber@gmail.com> Co-authored-by: jargordon <50050429+jargordon@users.noreply.github.com> Co-authored-by: Daniel <32464403+dkallen78@users.noreply.github.com> Co-authored-by: Ryan Williams <65094007+ryan-d-williams@users.noreply.github.com> Co-authored-by: Panagiotis Skias <panagiotis.skias@gmail.com> Co-authored-by: Marc Landis <dirk.rettschlag@gmail.com>
206 lines
5.7 KiB
JavaScript
206 lines
5.7 KiB
JavaScript
/* global WeatherProvider, WeatherObject */
|
|
|
|
/*
|
|
* This class is a provider for Weatherbit,
|
|
* see https://www.weatherbit.io/
|
|
*/
|
|
WeatherProvider.register("weatherbit", {
|
|
|
|
/*
|
|
* Set the name of the provider.
|
|
* Not strictly required, but helps for debugging.
|
|
*/
|
|
providerName: "Weatherbit",
|
|
|
|
// Set the default config properties that is specific to this provider
|
|
defaults: {
|
|
apiBase: "https://api.weatherbit.io/v2.0",
|
|
apiKey: "",
|
|
lat: 0,
|
|
lon: 0
|
|
},
|
|
|
|
fetchedLocation () {
|
|
return this.fetchedLocationName || "";
|
|
},
|
|
|
|
fetchCurrentWeather () {
|
|
this.fetchData(this.getUrl())
|
|
.then((data) => {
|
|
if (!data || !data.data[0] || typeof data.data[0].temp === "undefined") {
|
|
// No usable data?
|
|
return;
|
|
}
|
|
|
|
const currentWeather = this.generateWeatherDayFromCurrentWeather(data);
|
|
this.setCurrentWeather(currentWeather);
|
|
})
|
|
.catch(function (request) {
|
|
Log.error("Could not load data ... ", request);
|
|
})
|
|
.finally(() => this.updateAvailable());
|
|
},
|
|
|
|
fetchWeatherForecast () {
|
|
this.fetchData(this.getUrl())
|
|
.then((data) => {
|
|
if (!data || !data.data) {
|
|
// No usable data?
|
|
return;
|
|
}
|
|
|
|
const forecast = this.generateWeatherObjectsFromForecast(data.data);
|
|
this.setWeatherForecast(forecast);
|
|
|
|
this.fetchedLocationName = `${data.city_name}, ${data.state_code}`;
|
|
})
|
|
.catch(function (request) {
|
|
Log.error("Could not load data ... ", request);
|
|
})
|
|
.finally(() => this.updateAvailable());
|
|
},
|
|
|
|
/**
|
|
* Overrides method for setting config to check if endpoint is correct for hourly
|
|
* @param {object} config The configuration object
|
|
*/
|
|
setConfig (config) {
|
|
this.config = config;
|
|
if (!this.config.weatherEndpoint) {
|
|
switch (this.config.type) {
|
|
case "hourly":
|
|
this.config.weatherEndpoint = "/forecast/hourly";
|
|
break;
|
|
case "daily":
|
|
case "forecast":
|
|
this.config.weatherEndpoint = "/forecast/daily";
|
|
break;
|
|
case "current":
|
|
this.config.weatherEndpoint = "/current";
|
|
break;
|
|
default:
|
|
Log.error("weatherEndpoint not configured and could not resolve it based on type");
|
|
}
|
|
}
|
|
},
|
|
|
|
// Create a URL from the config and base URL.
|
|
getUrl () {
|
|
return `${this.config.apiBase}${this.config.weatherEndpoint}?lat=${this.config.lat}&lon=${this.config.lon}&units=M&key=${this.config.apiKey}`;
|
|
},
|
|
|
|
// Implement WeatherDay generator.
|
|
generateWeatherDayFromCurrentWeather (currentWeatherData) {
|
|
//Calculate TZ Offset and invert to convert Sunrise/Sunset times to Local
|
|
const d = new Date();
|
|
let tzOffset = d.getTimezoneOffset();
|
|
tzOffset = tzOffset * -1;
|
|
|
|
const currentWeather = new WeatherObject();
|
|
|
|
currentWeather.date = moment.unix(currentWeatherData.data[0].ts);
|
|
currentWeather.humidity = parseFloat(currentWeatherData.data[0].rh);
|
|
currentWeather.temperature = parseFloat(currentWeatherData.data[0].temp);
|
|
currentWeather.windSpeed = parseFloat(currentWeatherData.data[0].wind_spd);
|
|
currentWeather.windFromDirection = currentWeatherData.data[0].wind_dir;
|
|
currentWeather.weatherType = this.convertWeatherType(currentWeatherData.data[0].weather.icon);
|
|
currentWeather.sunrise = moment(currentWeatherData.data[0].sunrise, "HH:mm").add(tzOffset, "m");
|
|
currentWeather.sunset = moment(currentWeatherData.data[0].sunset, "HH:mm").add(tzOffset, "m");
|
|
|
|
this.fetchedLocationName = `${currentWeatherData.data[0].city_name}, ${currentWeatherData.data[0].state_code}`;
|
|
|
|
return currentWeather;
|
|
},
|
|
|
|
generateWeatherObjectsFromForecast (forecasts) {
|
|
const days = [];
|
|
|
|
for (const forecast of forecasts) {
|
|
const weather = new WeatherObject();
|
|
|
|
weather.date = moment(forecast.datetime, "YYYY-MM-DD");
|
|
weather.minTemperature = forecast.min_temp;
|
|
weather.maxTemperature = forecast.max_temp;
|
|
weather.precipitationAmount = forecast.precip;
|
|
weather.precipitationProbability = forecast.pop;
|
|
weather.weatherType = this.convertWeatherType(forecast.weather.icon);
|
|
|
|
days.push(weather);
|
|
}
|
|
|
|
return days;
|
|
},
|
|
|
|
// Map icons from Dark Sky to our icons.
|
|
convertWeatherType (weatherType) {
|
|
const weatherTypes = {
|
|
t01d: "day-thunderstorm",
|
|
t01n: "night-alt-thunderstorm",
|
|
t02d: "day-thunderstorm",
|
|
t02n: "night-alt-thunderstorm",
|
|
t03d: "thunderstorm",
|
|
t03n: "thunderstorm",
|
|
t04d: "day-thunderstorm",
|
|
t04n: "night-alt-thunderstorm",
|
|
t05d: "day-sleet-storm",
|
|
t05n: "night-alt-sleet-storm",
|
|
d01d: "day-sprinkle",
|
|
d01n: "night-alt-sprinkle",
|
|
d02d: "day-sprinkle",
|
|
d02n: "night-alt-sprinkle",
|
|
d03d: "day-shower",
|
|
d03n: "night-alt-shower",
|
|
r01d: "day-shower",
|
|
r01n: "night-alt-shower",
|
|
r02d: "day-rain",
|
|
r02n: "night-alt-rain",
|
|
r03d: "day-rain",
|
|
r03n: "night-alt-rain",
|
|
r04d: "day-sprinkle",
|
|
r04n: "night-alt-sprinkle",
|
|
r05d: "day-shower",
|
|
r05n: "night-alt-shower",
|
|
r06d: "day-shower",
|
|
r06n: "night-alt-shower",
|
|
f01d: "day-sleet",
|
|
f01n: "night-alt-sleet",
|
|
s01d: "day-snow",
|
|
s01n: "night-alt-snow",
|
|
s02d: "day-snow-wind",
|
|
s02n: "night-alt-snow-wind",
|
|
s03d: "snowflake-cold",
|
|
s03n: "snowflake-cold",
|
|
s04d: "day-rain-mix",
|
|
s04n: "night-alt-rain-mix",
|
|
s05d: "day-sleet",
|
|
s05n: "night-alt-sleet",
|
|
s06d: "day-snow",
|
|
s06n: "night-alt-snow",
|
|
a01d: "day-haze",
|
|
a01n: "dust",
|
|
a02d: "smoke",
|
|
a02n: "smoke",
|
|
a03d: "day-haze",
|
|
a03n: "dust",
|
|
a04d: "dust",
|
|
a04n: "dust",
|
|
a05d: "day-fog",
|
|
a05n: "night-fog",
|
|
a06d: "fog",
|
|
a06n: "fog",
|
|
c01d: "day-sunny",
|
|
c01n: "night-clear",
|
|
c02d: "day-sunny-overcast",
|
|
c02n: "night-alt-partly-cloudy",
|
|
c03d: "day-cloudy",
|
|
c03n: "night-alt-cloudy",
|
|
c04d: "cloudy",
|
|
c04n: "cloudy",
|
|
u00d: "rain-mix",
|
|
u00n: "rain-mix"
|
|
};
|
|
|
|
return weatherTypes.hasOwnProperty(weatherType) ? weatherTypes[weatherType] : null;
|
|
}
|
|
});
|