308 lines
8.7 KiB
JavaScript
Raw Normal View History

/* global WeatherProvider, WeatherUtils */
/* MagicMirror²
* Module: Weather
*
2020-04-28 23:05:28 +02:00
* By Michael Teeuw https://michaelteeuw.nl
* MIT Licensed.
*/
Module.register("weather", {
// Default module config.
defaults: {
weatherProvider: "openweathermap",
2017-09-22 13:26:44 +02:00
roundTemp: false,
type: "current", // current, forecast, daily (equivalent to forecast), hourly (only with OpenWeatherMap /onecall endpoint)
2017-10-18 11:58:45 +02:00
units: config.units,
tempUnits: config.units,
windUnits: config.units,
2017-10-18 11:58:45 +02:00
updateInterval: 10 * 60 * 1000, // every 10 minutes
animationSpeed: 1000,
timeFormat: config.timeFormat,
showPeriod: true,
showPeriodUpper: false,
showWindDirection: true,
showWindDirectionAsArrow: false,
lang: config.language,
showHumidity: false,
2020-05-28 10:10:00 +02:00
showSun: true,
2017-10-18 11:58:45 +02:00
degreeLabel: false,
decimalSymbol: ".",
2017-10-18 11:58:45 +02:00
showIndoorTemperature: false,
showIndoorHumidity: false,
2019-01-06 12:34:44 +01:00
maxNumberOfDays: 5,
maxEntries: 5,
2021-05-29 16:11:39 +02:00
ignoreToday: false,
fade: true,
fadePoint: 0.25, // Start on 1/4th of the list.
2017-10-18 11:58:45 +02:00
initialLoadDelay: 0, // 0 seconds delay
appendLocationNameToHeader: true,
calendarClass: "calendar",
2018-12-27 18:52:35 +01:00
tableClass: "small",
2017-10-18 11:58:45 +02:00
onlyTemp: false,
showPrecipitationAmount: false,
colored: false,
showFeelsLike: true,
absoluteDates: false
},
// Module properties.
weatherProvider: null,
// Can be used by the provider to display location of event if nothing else is specified
firstEvent: null,
// Define required scripts.
getStyles: function () {
2017-10-18 13:38:56 +02:00
return ["font-awesome.css", "weather-icons.css", "weather.css"];
},
// Return the scripts that are necessary for the weather module.
getScripts: function () {
New weather provider: Yr.no (#2948) # Added Yr.no as a weather provider Yr.no is a free Norwegian weather service. The configuration is quite simple: ```js { weatherProvider: "yr", lat: 59.9171, lon: 10.7276, altitude: 30 } ``` The latitude and longitude cannot have more than 4 decimals, but that should be plenty. To quote yr: "There is no need to ask for weather forecasts with nanometer precision!". The altitude should be meters above sea level and defaults to 0. If `type` is set to `current` the symbol can display the next 1, 6 or 12 hours by setting `currentForecastHours` (default is 1). It states in [Getting started-guide](https://developer.yr.no/doc/GettingStarted/) that users of the API should cache the results and use the `Expires`-header to know when to ask for new data. By using the `If-Modified-Since`-header we can avoid downloading the same data over and over again. I chose not to override the `User-Agent`-header set in [`server.js`](https://github.com/MichMich/MagicMirror/blob/a328ce5/js/server.js#L97) even though it does not comply with [the terms of service](https://developer.yr.no/doc/TermsOfService/). It currently works with the default header, and by searching the web for MagicMirror the GitHub-repo should be easy to find without an explicit link. I also had to make some minor changes to `server.js` and `weatherprovider.js` to be able to send and return HTTP headers. To handle the HTTP 304 response without body I chose to return `undefined` so we easily can use the response as a condition: `if (response) ...`. The documentation for the API is available here: - [API Reference overview](https://api.met.no/weatherapi/) - [Locationforecast](https://api.met.no/weatherapi/locationforecast/2.0/) - Used to get the weather forecast - [Sunrise](https://api.met.no/weatherapi/sunrise/2.0/documentation) - used to find sunrise and sunset times Co-authored-by: Veeck <github@veeck.de>
2022-11-08 06:18:47 +01:00
return ["moment.js", this.file("../utils.js"), "weatherutils.js", "weatherprovider.js", "weatherobject.js", "suncalc.js", this.file("providers/" + this.config.weatherProvider.toLowerCase() + ".js")];
},
2018-12-27 17:13:49 +01:00
// Override getHeader method.
getHeader: function () {
if (this.config.appendLocationNameToHeader && this.weatherProvider) {
if (this.data.header) return this.data.header + " " + this.weatherProvider.fetchedLocation();
else return this.weatherProvider.fetchedLocation();
2018-12-27 17:13:49 +01:00
}
return this.data.header ? this.data.header : "";
2018-12-27 17:13:49 +01:00
},
// Start the weather module.
start: function () {
2018-12-27 17:13:49 +01:00
moment.locale(this.config.lang);
if (this.config.useKmh) {
Log.warn("Your are using the deprecated config values 'useKmh'. Please switch to windUnits!");
this.windUnits = "kmh";
} else if (this.config.useBeaufort) {
Log.warn("Your are using the deprecated config values 'useBeaufort'. Please switch to windUnits!");
this.windUnits = "beaufort";
}
// Initialize the weather provider.
2018-05-21 10:56:46 +02:00
this.weatherProvider = WeatherProvider.initialize(this.config.weatherProvider, this);
// Let the weather provider know we are starting.
2018-05-21 10:56:46 +02:00
this.weatherProvider.start();
2017-10-18 13:38:56 +02:00
// Add custom filters
2018-05-21 10:56:46 +02:00
this.addFilters();
2017-10-18 13:38:56 +02:00
// Schedule the first update.
2018-12-27 19:37:02 +01:00
this.scheduleUpdate(this.config.initialLoadDelay);
},
2018-07-02 15:43:24 +02:00
// Override notification handler.
notificationReceived: function (notification, payload, sender) {
2018-07-02 15:43:24 +02:00
if (notification === "CALENDAR_EVENTS") {
const senderClasses = sender.data.classes.toLowerCase().split(" ");
2018-07-02 15:43:24 +02:00
if (senderClasses.indexOf(this.config.calendarClass.toLowerCase()) !== -1) {
this.firstEvent = null;
for (let event of payload) {
2018-07-02 15:43:24 +02:00
if (event.location || event.geo) {
this.firstEvent = event;
Log.debug("First upcoming event with location: ", event);
2018-07-02 15:43:24 +02:00
break;
}
}
}
2018-12-27 19:37:02 +01:00
} else if (notification === "INDOOR_TEMPERATURE") {
2018-07-02 15:43:24 +02:00
this.indoorTemperature = this.roundValue(payload);
this.updateDom(300);
2018-12-27 19:37:02 +01:00
} else if (notification === "INDOOR_HUMIDITY") {
2018-07-02 15:43:24 +02:00
this.indoorHumidity = this.roundValue(payload);
this.updateDom(300);
}
},
2017-10-01 13:50:15 +02:00
// Select the template depending on the display type.
getTemplate: function () {
switch (this.config.type.toLowerCase()) {
case "current":
return "current.njk";
case "hourly":
return "hourly.njk";
case "daily":
case "forecast":
return "forecast.njk";
//Make the invalid values use the "Loading..." from forecast
default:
return "forecast.njk";
}
2017-10-01 13:50:15 +02:00
},
// Add all the data to the template.
getTemplateData: function () {
2021-05-29 16:11:39 +02:00
const forecast = this.weatherProvider.weatherForecast();
2017-10-01 13:50:15 +02:00
return {
config: this.config,
current: this.weatherProvider.currentWeather(),
forecast: forecast,
hourly: this.weatherProvider.weatherHourly(),
2018-07-02 15:43:24 +02:00
indoor: {
humidity: this.indoorHumidity,
temperature: this.indoorTemperature
}
2019-06-05 10:23:58 +02:00
};
},
// What to do when the weather provider has new information available?
updateAvailable: function () {
2018-05-21 10:56:46 +02:00
Log.log("New weather information available.");
2018-12-27 19:37:02 +01:00
this.updateDom(0);
this.scheduleUpdate();
2021-01-17 15:00:34 +01:00
if (this.weatherProvider.currentWeather()) {
this.sendNotification("CURRENTWEATHER_TYPE", { type: this.weatherProvider.currentWeather().weatherType.replace("-", "_") });
}
2022-04-26 13:55:39 +02:00
2022-04-26 17:37:23 +02:00
const notificationPayload = {
2022-04-26 14:52:05 +02:00
currentWeather: this.weatherProvider?.currentWeatherObject?.simpleClone() ?? null,
forecastArray: this.weatherProvider?.weatherForecastArray?.map((ar) => ar.simpleClone()) ?? [],
hourlyArray: this.weatherProvider?.weatherHourlyArray?.map((ar) => ar.simpleClone()) ?? [],
locationName: this.weatherProvider?.fetchedLocationName,
providerName: this.weatherProvider.providerName
2022-04-26 13:55:39 +02:00
};
this.sendNotification("WEATHER_UPDATED", notificationPayload);
},
scheduleUpdate: function (delay = null) {
let nextLoad = this.config.updateInterval;
if (delay !== null && delay >= 0) {
nextLoad = delay;
}
setTimeout(() => {
switch (this.config.type.toLowerCase()) {
case "current":
this.weatherProvider.fetchCurrentWeather();
break;
case "hourly":
this.weatherProvider.fetchWeatherHourly();
break;
case "daily":
case "forecast":
this.weatherProvider.fetchWeatherForecast();
break;
default:
Log.error(`Invalid type ${this.config.type} configured (must be one of 'current', 'hourly', 'daily' or 'forecast')`);
2017-09-22 13:26:44 +02:00
}
}, nextLoad);
2017-10-18 13:38:56 +02:00
},
roundValue: function (temperature) {
const decimals = this.config.roundTemp ? 0 : 1;
const roundValue = parseFloat(temperature).toFixed(decimals);
2021-01-07 11:51:10 +01:00
return roundValue === "-0" ? 0 : roundValue;
2018-12-27 17:13:49 +01:00
},
2018-07-02 15:43:24 +02:00
2017-10-18 13:38:56 +02:00
addFilters() {
this.nunjucksEnvironment().addFilter(
"formatTime",
function (date) {
date = moment(date);
if (this.config.timeFormat !== 24) {
if (this.config.showPeriod) {
if (this.config.showPeriodUpper) {
return date.format("h:mm A");
} else {
return date.format("h:mm a");
}
2017-10-18 13:38:56 +02:00
} else {
return date.format("h:mm");
2017-10-18 13:38:56 +02:00
}
2018-05-21 10:56:46 +02:00
}
return date.format("HH:mm");
}.bind(this)
);
2018-05-21 10:56:46 +02:00
this.nunjucksEnvironment().addFilter(
"unit",
function (value, type) {
if (type === "temperature") {
value = this.roundValue(WeatherUtils.convertTemp(value, this.config.tempUnits)) + "°";
if (this.config.degreeLabel) {
if (this.config.tempUnits === "metric") {
value += "C";
} else if (this.config.tempUnits === "imperial") {
value += "F";
} else {
value += "K";
}
}
} else if (type === "precip") {
if (value === null || isNaN(value) || value === 0 || value.toFixed(2) === "0.00") {
value = "";
2018-05-21 10:56:46 +02:00
} else {
if (this.config.weatherProvider === "ukmetoffice" || this.config.weatherProvider === "ukmetofficedatahub") {
value += "%";
} else {
value = `${value.toFixed(2)} ${this.config.units === "imperial" ? "in" : "mm"}`;
}
2018-05-21 10:56:46 +02:00
}
} else if (type === "humidity") {
value += "%";
} else if (type === "wind") {
value = WeatherUtils.convertWind(value, this.config.windUnits);
2017-10-18 13:38:56 +02:00
}
return value;
}.bind(this)
);
this.nunjucksEnvironment().addFilter(
"roundValue",
function (value) {
return this.roundValue(value);
}.bind(this)
);
this.nunjucksEnvironment().addFilter(
"decimalSymbol",
function (value) {
return value.toString().replace(/\./g, this.config.decimalSymbol);
}.bind(this)
);
this.nunjucksEnvironment().addFilter(
"calcNumSteps",
function (forecast) {
return Math.min(forecast.length, this.config.maxNumberOfDays);
}.bind(this)
);
this.nunjucksEnvironment().addFilter(
"calcNumEntries",
2021-05-29 16:11:39 +02:00
function (dataArray) {
2020-06-30 04:18:03 -04:00
return Math.min(dataArray.length, this.config.maxEntries);
}.bind(this)
);
this.nunjucksEnvironment().addFilter(
"opacity",
function (currentStep, numSteps) {
if (this.config.fade && this.config.fadePoint < 1) {
if (this.config.fadePoint < 0) {
this.config.fadePoint = 0;
}
const startingPoint = numSteps * this.config.fadePoint;
const numFadesteps = numSteps - startingPoint;
if (currentStep >= startingPoint) {
return 1 - (currentStep - startingPoint) / numFadesteps;
} else {
return 1;
}
} else {
return 1;
}
}.bind(this)
);
2017-10-01 16:19:14 +02:00
}
});