mirror of
https://github.com/MichMich/MagicMirror.git
synced 2025-06-27 03:39:55 +00:00
## [2.26.0] - 01-01-2024 Thanks to: @bnitkin, @bugsounet, @dependabot, @jkriegshauser, @kaennchenstruggle, @KristjanESPERANTO and @Ybbet. Special thanks to @khassel, @rejas and @sdetweil for taking over most (if not all) of the work on this release as project collaborators. This version would not be there without their effort. Thank you guys! You are awesome! This release also marks the latest release by Michael Teeuw. For more info, please read the following post: [A New Chapter for MagicMirror: The Community Takes the Lead](https://forum.magicmirror.builders/topic/18329/a-new-chapter-for-magicmirror-the-community-takes-the-lead). ### Added - Added update notification updater (for 3rd party modules) - Added node 21 to the test matrix - Added transform object to calendar:customEvents - Added ESLint rules for jest (including jest/expect-expect and jest/no-done-callback) ### Removed - Removed Codecov workflow (not working anymore, other workflow required) (#3107) - Removed titleReplace from calendar, replaced + extended by customEvents (backward compatibility included) (#3249) - Removed failing unit test (#3254) - Removed some unused variables ### Updated - Update electron to v27 and update other dependencies as well as github actions - Update newsfeed: Use `html-to-text` instead of regex for transform description - Review ESLint config (#3269) - Updated dependencies - Clock module: optionally display current moon phase in addition to rise/set times - electron is now per default started without gpu, if needed it must be enabled with new env var `ELECTRON_ENABLE_GPU=1` on startup (#3226) - Replace prettier by stylistic in ESLint config to lint JavaScript (and disable some rules for `config/config.js*` files) - Update node-ical to v0.17.1 and fix tests ### Fixed - Avoid fade out/in on updateDom when many calendars are used - Fix the option eventClass on customEvents. - Fix yr API version in locationforecast and sunrise call (#3227) - Fix cloneObject() function to respect RegExp (#3237) - Fix newsfeed module for feeds using "a10:updated" tag (#3238) - Fix issue template (#3167) - Fix #3256 filter out bad results from rrule.between - Fix calendar events sometimes not respecting deleted events (#3250) - Fix electron loadurl locally on Windows when address "0.0.0.0" (#2550) - Fix updatanotification (update_helper.js): catch error if reponse is not an JSON format (check PM2) - Fix missing typeof in calendar module - Fix style issues after prettier update - Fix calendar test (#3291) by moving "Exdate check" from e2e to electron to run on a Thursday - Fix calendar config params `fetchInterval` and `excludedEvents` were never used from single calendar config (#3297) - Fix MM_PORT variable not used in electron and allow full path for MM_CONFIG_FILE variable (#3302) --------- Signed-off-by: naveen <172697+naveensrinivasan@users.noreply.github.com> Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Karsten Hassel <hassel@gmx.de> Co-authored-by: Malte Hallström <46646495+SkySails@users.noreply.github.com> Co-authored-by: Veeck <github@veeck.de> Co-authored-by: veeck <michael@veeck.de> Co-authored-by: dWoolridge <dwoolridge@charter.net> Co-authored-by: Johan <jojjepersson@yahoo.se> Co-authored-by: Dario Mratovich <dario_mratovich@hotmail.com> Co-authored-by: Dario Mratovich <dario.mratovich@outlook.com> Co-authored-by: Magnus <34011212+MagMar94@users.noreply.github.com> Co-authored-by: Naveen <172697+naveensrinivasan@users.noreply.github.com> Co-authored-by: buxxi <buxxi@omfilm.net> Co-authored-by: Thomas Hirschberger <47733292+Tom-Hirschberger@users.noreply.github.com> Co-authored-by: Kristjan ESPERANTO <35647502+KristjanESPERANTO@users.noreply.github.com> Co-authored-by: Andrés Vanegas Jiménez <142350+angeldeejay@users.noreply.github.com> Co-authored-by: Dave Child <dave@addedbytes.com> Co-authored-by: grenagit <46225780+grenagit@users.noreply.github.com> Co-authored-by: Grena <grena@grenabox.fr> Co-authored-by: Magnus Marthinsen <magmar@online.no> Co-authored-by: Patrick <psieg@users.noreply.github.com> Co-authored-by: Piotr Rajnisz <56397164+rajniszp@users.noreply.github.com> Co-authored-by: Suthep Yonphimai <tomzt@users.noreply.github.com> Co-authored-by: CarJem Generations (Carter Wallace) <cwallacecs@gmail.com> Co-authored-by: Nicholas Fogal <nfogal.misc@gmail.com> Co-authored-by: JakeBinney <126349119+JakeBinney@users.noreply.github.com> Co-authored-by: OWL4C <124401812+OWL4C@users.noreply.github.com> Co-authored-by: Oscar Björkman <17575446+oscarb@users.noreply.github.com> Co-authored-by: Ismar Slomic <ismar@slomic.no> Co-authored-by: Jørgen Veum-Wahlberg <jorgen.wahlberg@amedia.no> Co-authored-by: Eddie Hung <6740044+eddiehung@users.noreply.github.com> Co-authored-by: Bugsounet - Cédric <github@bugsounet.fr> Co-authored-by: bugsounet <bugsounet@bugsounet.fr> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Knapoc <Knapoc@users.noreply.github.com> Co-authored-by: sam detweiler <sdetweil@gmail.com> Co-authored-by: veeck <michael.veeck@nebenan.de> Co-authored-by: Paranoid93 <6515818+Paranoid93@users.noreply.github.com> Co-authored-by: NolanKingdon <27908974+NolanKingdon@users.noreply.github.com> Co-authored-by: J. Kenzal Hunter <kenzal.hunter@gmail.com> Co-authored-by: Teddy <teddy.payet@gmail.com> Co-authored-by: TeddyStarinvest <teddy.payet@starinvest.com> Co-authored-by: martingron <61826403+martingron@users.noreply.github.com> Co-authored-by: dgoth <132394363+dgoth@users.noreply.github.com> Co-authored-by: kaennchenstruggle <54073894+kaennchenstruggle@users.noreply.github.com> Co-authored-by: jkriegshauser <jkriegshauser@gmail.com> Co-authored-by: Ben Nitkin <ben@nitkin.net>
511 lines
14 KiB
JavaScript
511 lines
14 KiB
JavaScript
/* global Class, cloneObject, Loader, MMSocket, nunjucks, Translator */
|
|
|
|
/* MagicMirror²
|
|
* Module Blueprint.
|
|
* @typedef {Object} Module
|
|
*
|
|
* By Michael Teeuw https://michaelteeuw.nl
|
|
* MIT Licensed.
|
|
*/
|
|
const Module = Class.extend({
|
|
|
|
/*********************************************************
|
|
* All methods (and properties) below can be subclassed. *
|
|
*********************************************************/
|
|
|
|
// Set the minimum MagicMirror² module version for this module.
|
|
requiresVersion: "2.0.0",
|
|
|
|
// Module config defaults.
|
|
defaults: {},
|
|
|
|
// Timer reference used for showHide animation callbacks.
|
|
showHideTimer: null,
|
|
|
|
// Array to store lockStrings. These strings are used to lock
|
|
// visibility when hiding and showing module.
|
|
lockStrings: [],
|
|
|
|
// Storage of the nunjucks Environment,
|
|
// This should not be referenced directly.
|
|
// Use the nunjucksEnvironment() to get it.
|
|
_nunjucksEnvironment: null,
|
|
|
|
/**
|
|
* Called when the module is instantiated.
|
|
*/
|
|
init () {
|
|
//Log.log(this.defaults);
|
|
},
|
|
|
|
/**
|
|
* Called when the module is started.
|
|
*/
|
|
async start () {
|
|
Log.info(`Starting module: ${this.name}`);
|
|
},
|
|
|
|
/**
|
|
* Returns a list of scripts the module requires to be loaded.
|
|
* @returns {string[]} An array with filenames.
|
|
*/
|
|
getScripts () {
|
|
return [];
|
|
},
|
|
|
|
/**
|
|
* Returns a list of stylesheets the module requires to be loaded.
|
|
* @returns {string[]} An array with filenames.
|
|
*/
|
|
getStyles () {
|
|
return [];
|
|
},
|
|
|
|
/**
|
|
* Returns a map of translation files the module requires to be loaded.
|
|
*
|
|
* return Map<String, String> -
|
|
* @returns {*} A map with langKeys and filenames.
|
|
*/
|
|
getTranslations () {
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* Generates the dom which needs to be displayed. This method is called by the MagicMirror² core.
|
|
* This method can to be subclassed if the module wants to display info on the mirror.
|
|
* Alternatively, the getTemplate method could be subclassed.
|
|
* @returns {HTMLElement|Promise} The dom or a promise with the dom to display.
|
|
*/
|
|
getDom () {
|
|
return new Promise((resolve) => {
|
|
const div = document.createElement("div");
|
|
const template = this.getTemplate();
|
|
const templateData = this.getTemplateData();
|
|
|
|
// Check to see if we need to render a template string or a file.
|
|
if ((/^.*((\.html)|(\.njk))$/).test(template)) {
|
|
// the template is a filename
|
|
this.nunjucksEnvironment().render(template, templateData, function (err, res) {
|
|
if (err) {
|
|
Log.error(err);
|
|
}
|
|
|
|
div.innerHTML = res;
|
|
|
|
resolve(div);
|
|
});
|
|
} else {
|
|
// the template is a template string.
|
|
div.innerHTML = this.nunjucksEnvironment().renderString(template, templateData);
|
|
|
|
resolve(div);
|
|
}
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Generates the header string which needs to be displayed if a user has a header configured for this module.
|
|
* This method is called by the MagicMirror² core, but only if the user has configured a default header for the module.
|
|
* This method needs to be subclassed if the module wants to display modified headers on the mirror.
|
|
* @returns {string} The header to display above the header.
|
|
*/
|
|
getHeader () {
|
|
return this.data.header;
|
|
},
|
|
|
|
/**
|
|
* Returns the template for the module which is used by the default getDom implementation.
|
|
* This method needs to be subclassed if the module wants to use a template.
|
|
* It can either return a template sting, or a template filename.
|
|
* If the string ends with '.html' it's considered a file from within the module's folder.
|
|
* @returns {string} The template string of filename.
|
|
*/
|
|
getTemplate () {
|
|
return `<div class="normal">${this.name}</div><div class="small dimmed">${this.identifier}</div>`;
|
|
},
|
|
|
|
/**
|
|
* Returns the data to be used in the template.
|
|
* This method needs to be subclassed if the module wants to use a custom data.
|
|
* @returns {object} The data for the template
|
|
*/
|
|
getTemplateData () {
|
|
return {};
|
|
},
|
|
|
|
/**
|
|
* Called by the MagicMirror² core when a notification arrives.
|
|
* @param {string} notification The identifier of the notification.
|
|
* @param {*} payload The payload of the notification.
|
|
* @param {Module} sender The module that sent the notification.
|
|
*/
|
|
notificationReceived (notification, payload, sender) {
|
|
if (sender) {
|
|
// Log.log(this.name + " received a module notification: " + notification + " from sender: " + sender.name);
|
|
} else {
|
|
// Log.log(this.name + " received a system notification: " + notification);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Returns the nunjucks environment for the current module.
|
|
* The environment is checked in the _nunjucksEnvironment instance variable.
|
|
* @returns {object} The Nunjucks Environment
|
|
*/
|
|
nunjucksEnvironment () {
|
|
if (this._nunjucksEnvironment !== null) {
|
|
return this._nunjucksEnvironment;
|
|
}
|
|
|
|
this._nunjucksEnvironment = new nunjucks.Environment(new nunjucks.WebLoader(this.file(""), { async: true }), {
|
|
trimBlocks: true,
|
|
lstripBlocks: true
|
|
});
|
|
|
|
this._nunjucksEnvironment.addFilter("translate", (str, variables) => {
|
|
return nunjucks.runtime.markSafe(this.translate(str, variables));
|
|
});
|
|
|
|
return this._nunjucksEnvironment;
|
|
},
|
|
|
|
/**
|
|
* Called when a socket notification arrives.
|
|
* @param {string} notification The identifier of the notification.
|
|
* @param {*} payload The payload of the notification.
|
|
*/
|
|
socketNotificationReceived (notification, payload) {
|
|
Log.log(`${this.name} received a socket notification: ${notification} - Payload: ${payload}`);
|
|
},
|
|
|
|
/**
|
|
* Called when the module is hidden.
|
|
*/
|
|
suspend () {
|
|
Log.log(`${this.name} is suspended.`);
|
|
},
|
|
|
|
/**
|
|
* Called when the module is shown.
|
|
*/
|
|
resume () {
|
|
Log.log(`${this.name} is resumed.`);
|
|
},
|
|
|
|
/*********************************************
|
|
* The methods below don't need subclassing. *
|
|
*********************************************/
|
|
|
|
/**
|
|
* Set the module data.
|
|
* @param {object} data The module data
|
|
*/
|
|
setData (data) {
|
|
this.data = data;
|
|
this.name = data.name;
|
|
this.identifier = data.identifier;
|
|
this.hidden = false;
|
|
this.hasAnimateIn = false;
|
|
this.hasAnimateOut = false;
|
|
|
|
this.setConfig(data.config, data.configDeepMerge);
|
|
},
|
|
|
|
/**
|
|
* Set the module config and combine it with the module defaults.
|
|
* @param {object} config The combined module config.
|
|
* @param {boolean} deep Merge module config in deep.
|
|
*/
|
|
setConfig (config, deep) {
|
|
this.config = deep ? configMerge({}, this.defaults, config) : Object.assign({}, this.defaults, config);
|
|
},
|
|
|
|
/**
|
|
* Returns a socket object. If it doesn't exist, it's created.
|
|
* It also registers the notification callback.
|
|
* @returns {MMSocket} a socket object
|
|
*/
|
|
socket () {
|
|
if (typeof this._socket === "undefined") {
|
|
this._socket = new MMSocket(this.name);
|
|
}
|
|
|
|
this._socket.setNotificationCallback((notification, payload) => {
|
|
this.socketNotificationReceived(notification, payload);
|
|
});
|
|
|
|
return this._socket;
|
|
},
|
|
|
|
/**
|
|
* Retrieve the path to a module file.
|
|
* @param {string} file Filename
|
|
* @returns {string} the file path
|
|
*/
|
|
file (file) {
|
|
return `${this.data.path}/${file}`.replace("//", "/");
|
|
},
|
|
|
|
/**
|
|
* Load all required stylesheets by requesting the MM object to load the files.
|
|
* @returns {Promise<void>}
|
|
*/
|
|
loadStyles () {
|
|
return this.loadDependencies("getStyles");
|
|
},
|
|
|
|
/**
|
|
* Load all required scripts by requesting the MM object to load the files.
|
|
* @returns {Promise<void>}
|
|
*/
|
|
loadScripts () {
|
|
return this.loadDependencies("getScripts");
|
|
},
|
|
|
|
/**
|
|
* Helper method to load all dependencies.
|
|
* @param {string} funcName Function name to call to get scripts or styles.
|
|
* @returns {Promise<void>}
|
|
*/
|
|
async loadDependencies (funcName) {
|
|
let dependencies = this[funcName]();
|
|
|
|
const loadNextDependency = async () => {
|
|
if (dependencies.length > 0) {
|
|
const nextDependency = dependencies[0];
|
|
await Loader.loadFileForModule(nextDependency, this);
|
|
dependencies = dependencies.slice(1);
|
|
await loadNextDependency();
|
|
} else {
|
|
return Promise.resolve();
|
|
}
|
|
};
|
|
|
|
await loadNextDependency();
|
|
},
|
|
|
|
/**
|
|
* Load all translations.
|
|
* @returns {Promise<void>}
|
|
*/
|
|
async loadTranslations () {
|
|
const translations = this.getTranslations() || {};
|
|
const language = config.language.toLowerCase();
|
|
|
|
const languages = Object.keys(translations);
|
|
const fallbackLanguage = languages[0];
|
|
|
|
if (languages.length === 0) {
|
|
return;
|
|
}
|
|
|
|
const translationFile = translations[language];
|
|
const translationsFallbackFile = translations[fallbackLanguage];
|
|
|
|
if (!translationFile) {
|
|
return Translator.load(this, translationsFallbackFile, true);
|
|
}
|
|
|
|
await Translator.load(this, translationFile, false);
|
|
|
|
if (translationFile !== translationsFallbackFile) {
|
|
return Translator.load(this, translationsFallbackFile, true);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Request the translation for a given key with optional variables and default value.
|
|
* @param {string} key The key of the string to translate
|
|
* @param {string|object} [defaultValueOrVariables] The default value or variables for translating.
|
|
* @param {string} [defaultValue] The default value with variables.
|
|
* @returns {string} the translated key
|
|
*/
|
|
translate (key, defaultValueOrVariables, defaultValue) {
|
|
if (typeof defaultValueOrVariables === "object") {
|
|
return Translator.translate(this, key, defaultValueOrVariables) || defaultValue || "";
|
|
}
|
|
return Translator.translate(this, key) || defaultValueOrVariables || "";
|
|
},
|
|
|
|
/**
|
|
* Request an (animated) update of the module.
|
|
* @param {number|object} [updateOptions] The speed of the animation or object with for updateOptions (speed/animates)
|
|
*/
|
|
updateDom (updateOptions) {
|
|
MM.updateDom(this, updateOptions);
|
|
},
|
|
|
|
/**
|
|
* Send a notification to all modules.
|
|
* @param {string} notification The identifier of the notification.
|
|
* @param {*} payload The payload of the notification.
|
|
*/
|
|
sendNotification (notification, payload) {
|
|
MM.sendNotification(notification, payload, this);
|
|
},
|
|
|
|
/**
|
|
* Send a socket notification to the node helper.
|
|
* @param {string} notification The identifier of the notification.
|
|
* @param {*} payload The payload of the notification.
|
|
*/
|
|
sendSocketNotification (notification, payload) {
|
|
this.socket().sendNotification(notification, payload);
|
|
},
|
|
|
|
/**
|
|
* Hide this module.
|
|
* @param {number} speed The speed of the hide animation.
|
|
* @param {Function} callback Called when the animation is done.
|
|
* @param {object} [options] Optional settings for the hide method.
|
|
*/
|
|
hide (speed, callback, options = {}) {
|
|
let usedCallback = callback || function () {};
|
|
let usedOptions = options;
|
|
|
|
if (typeof callback === "object") {
|
|
Log.error("Parameter mismatch in module.hide: callback is not an optional parameter!");
|
|
usedOptions = callback;
|
|
usedCallback = function () {};
|
|
}
|
|
|
|
MM.hideModule(
|
|
this,
|
|
speed,
|
|
() => {
|
|
this.suspend();
|
|
usedCallback();
|
|
},
|
|
usedOptions
|
|
);
|
|
},
|
|
|
|
/**
|
|
* Show this module.
|
|
* @param {number} speed The speed of the show animation.
|
|
* @param {Function} callback Called when the animation is done.
|
|
* @param {object} [options] Optional settings for the show method.
|
|
*/
|
|
show (speed, callback, options) {
|
|
let usedCallback = callback || function () {};
|
|
let usedOptions = options;
|
|
|
|
if (typeof callback === "object") {
|
|
Log.error("Parameter mismatch in module.show: callback is not an optional parameter!");
|
|
usedOptions = callback;
|
|
usedCallback = function () {};
|
|
}
|
|
|
|
MM.showModule(
|
|
this,
|
|
speed,
|
|
() => {
|
|
this.resume();
|
|
usedCallback();
|
|
},
|
|
usedOptions
|
|
);
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Merging MagicMirror² (or other) default/config script by @bugsounet
|
|
* Merge 2 objects or/with array
|
|
*
|
|
* Usage:
|
|
* -------
|
|
* this.config = configMerge({}, this.defaults, this.config)
|
|
* -------
|
|
* arg1: initial object
|
|
* arg2: config model
|
|
* arg3: config to merge
|
|
* -------
|
|
* why using it ?
|
|
* Object.assign() function don't to all job
|
|
* it don't merge all thing in deep
|
|
* -> object in object and array is not merging
|
|
* -------
|
|
*
|
|
* Todo: idea of Mich determinate what do you want to merge or not
|
|
* @param {object} result the initial object
|
|
* @returns {object} the merged config
|
|
*/
|
|
function configMerge (result) {
|
|
const stack = Array.prototype.slice.call(arguments, 1);
|
|
let item, key;
|
|
|
|
while (stack.length) {
|
|
item = stack.shift();
|
|
for (key in item) {
|
|
if (item.hasOwnProperty(key)) {
|
|
if (typeof result[key] === "object" && result[key] && Object.prototype.toString.call(result[key]) !== "[object Array]") {
|
|
if (typeof item[key] === "object" && item[key] !== null) {
|
|
result[key] = configMerge({}, result[key], item[key]);
|
|
} else {
|
|
result[key] = item[key];
|
|
}
|
|
} else {
|
|
result[key] = item[key];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
Module.definitions = {};
|
|
|
|
Module.create = function (name) {
|
|
// Make sure module definition is available.
|
|
if (!Module.definitions[name]) {
|
|
return;
|
|
}
|
|
|
|
const moduleDefinition = Module.definitions[name];
|
|
const clonedDefinition = cloneObject(moduleDefinition);
|
|
|
|
// Note that we clone the definition. Otherwise the objects are shared, which gives problems.
|
|
const ModuleClass = Module.extend(clonedDefinition);
|
|
|
|
return new ModuleClass();
|
|
};
|
|
|
|
Module.register = function (name, moduleDefinition) {
|
|
if (moduleDefinition.requiresVersion) {
|
|
Log.log(`Check MagicMirror² version for module '${name}' - Minimum version: ${moduleDefinition.requiresVersion} - Current version: ${window.mmVersion}`);
|
|
if (cmpVersions(window.mmVersion, moduleDefinition.requiresVersion) >= 0) {
|
|
Log.log("Version is ok!");
|
|
} else {
|
|
Log.warn(`Version is incorrect. Skip module: '${name}'`);
|
|
return;
|
|
}
|
|
}
|
|
Log.log(`Module registered: ${name}`);
|
|
Module.definitions[name] = moduleDefinition;
|
|
};
|
|
|
|
window.Module = Module;
|
|
|
|
/**
|
|
* Compare two semantic version numbers and return the difference.
|
|
* @param {string} a Version number a.
|
|
* @param {string} b Version number b.
|
|
* @returns {number} A positive number if a is larger than b, a negative
|
|
* number if a is smaller and 0 if they are the same
|
|
*/
|
|
function cmpVersions (a, b) {
|
|
const regExStrip0 = /(\.0+)+$/;
|
|
const segmentsA = a.replace(regExStrip0, "").split(".");
|
|
const segmentsB = b.replace(regExStrip0, "").split(".");
|
|
const l = Math.min(segmentsA.length, segmentsB.length);
|
|
|
|
for (let i = 0; i < l; i++) {
|
|
let diff = parseInt(segmentsA[i], 10) - parseInt(segmentsB[i], 10);
|
|
if (diff) {
|
|
return diff;
|
|
}
|
|
}
|
|
return segmentsA.length - segmentsB.length;
|
|
}
|