User:Evad37/livenotifications/sandbox.js

(function{function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})({1:[function(require,module,exports){ "use strict";

var _App = _interopRequireDefault(require("./modules/App"));

function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }

Promise.all([// Resource loader modules mw.loader.using(['mediawiki.util', 'mediawiki.api', 'mediawiki.Title']), // Page ready $.ready]).then(function {  (0, _App["default"]); });

},{"./modules/App":2}],2:[function(require,module,exports){ "use strict";

Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0;

var _util = require("./util");

var _spinner = _interopRequireDefault(require("./spinner"));

function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }

var App = function App { var api = new mw.Api({   ajax: {      headers: {        "Api-User-Agent": "livenotifications/1.0.0" + " ( https://en.wikipedia.org/wiki/User:Evad37/livenotifications )"      }    }  }); var config = mw.config.get(["wgServer"]); var waitTimeMilliseconds = 1000 * 60; var lastCheckedTimestamp = new Date; // Main function for retreiving and displaying notifications

var mainloop = function mainloop { // Only make the request if the window/tab is active (focused) if (document.hasFocus) { return api.get({       "action": "query",        "format": "json",        "formatversion": "2",        "curtimestamp": 1,        "meta": "notifications",        "notfilter": "!read",        "notformat": "model",        "notlimit": "20"      }).then(function (response) {        // Find notifications which are new since the last request        var notifications = (response && response.query && response.query.notifications && response.query.notifications.list || []).filter(function (notification) { return lastCheckedTimestamp < new Date(notification.timestamp.utciso8601); }).forEach(function (notification) { var iconFulllUrl = "https:" + config.wgServer + notification["*"].iconUrl; var readableDate = notification.timestamp.utciso8601.replace("T", " ").replace(/:\d\dZ/, " (UTC)"); var messageDiv = (0, _util.makeElement)("div", null, [(0, _util.makeElement)("span", { height: "30px", style: "float:right; margin:-0.2em -0.2em 0.2em 0.5em" }, "X"), (0, _util.makeElement)("a", { href: notification["*"].links.primary.url, title: notification["*"].links.primary.label, target: "_blank", style: "display:block" }, [(0, _util.makeElement)("img", {           src: iconFulllUrl,            height: "30px",            style: "display:block; float:left; margin:0 0.5em 22em -0.5em; color:#666; font-size:88%"          }), (0, _util.parseHtml)(notification["*"].header), (0, _util.makeElement)("span", {            style: "display: block; color: #666; font-size:88%"          }, readableDate)].flatMap(function (x) {            return x;          }) // Flattening in case parseHtml returns an array )]);         mw.notify(messageDiv, {            autoHide: false          }); }); // Update the last checked timestamp (for the next request)

lastCheckedTimestamp = new Date(response.curtimestamp); })["catch"](function (error, details) { return console.log("[livenotifications] error", {         error: error,          details: details        }); });   }  };

window.setInterval(function {    var loop = mainloop;

if (loop) { _spinner["default"].start;

loop.then(function {        return _spinner["default"].stop;      }); } }, waitTimeMilliseconds); };

var _default = App; exports["default"] = _default;

},{"./spinner":3,"./util":4}],3:[function(require,module,exports){ "use strict";

Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0;

function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }

function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }

function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; }

var Spinner = /*#__PURE__*/function { function Spinner { _classCallCheck(this, Spinner);

this.badges = document.querySelectorAll(".mw-echo-notifications-badge"); this.minDuration = 3000; // milliseconds }

_createClass(Spinner, [{   key: "start",    value: function start {      this.startTime = new Date;      this.badges.forEach(function (el) { return el.setAttribute("style", "background-image:url(https://upload.wikimedia.org/wikipedia/commons/thumb/7/7a/Ajax_loader_metal_512.gif/20px-Ajax_loader_metal_512.gif)"); });   }  }, {    key: "stop",    value: function stop {      var _this = this;

var timeUsed = new Date - this.startTime; var remainingTime = Math.max(0, this.minDuration - timeUsed); window.setTimeout(function {        return _this.badges.forEach(function (el) { return el.removeAttribute("style"); });     }, remainingTime); } }]);

return Spinner; };

var spinner = new Spinner; var _default = spinner; exports["default"] = _default;

},{}],4:[function(require,module,exports){ "use strict";

Object.defineProperty(exports, "__esModule", { value: true }); exports.parseHtml = exports.makeElement = void 0;

/** * * @param {string} tagName * @param {object} [attributes] * @param {Node|string|Node[]|string[]} [contents] * @returns {HTMLElement} */ var makeElement = function makeElement(tagName, attributes, contents) { var el = document.createElement(tagName);

if (attributes) { for (var prop in attributes) { el.setAttribute(prop, attributes[prop]); } }

if (contents) { (Array.isArray(contents) ? contents : [contents]).forEach(function (content) {     var node = content && content.nodeType ? content : document.createTextNode(content);      el.appendChild(node);    }); }

return el; }; /** * @param {string} htmlString * @return {Node|Node[]} Node or Nodes parsed from HTML */

exports.makeElement = makeElement;

var parseHtml = function parseHtml(htmlString) { var div = document.createElement("div"); div.innerHTML = htmlString; var children = Array.from(div.childNodes); return children.length === 1 ? children[0] : children; };

exports.parseHtml = parseHtml;

},{}]},{},[1]) //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","livenotifications-src/index.js","livenotifications-src/modules/App.js","livenotifications-src/modules/spinner.js","livenotifications-src/modules/util.js"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()","import App from './modules/App';\n\nPromise.all([\n\t// Resource loader modules\n\tmw.loader.using(['mediawiki.util', 'mediawiki.api', 'mediawiki.Title']),\n\t// Page ready\n\t$.ready\n]).then(function() {\n\tApp();\n});\n","\nimport {makeElement, parseHtml} from \"./util\";\nimport spinner from \"./spinner\";\n\nconst App = () => {\n\tconst api = new mw.Api({\n\t\tajax: {\n\t\t\theaders: { \n\t\t\t\t\"Api-User-Agent\": \"livenotifications/1.0.0\" + \n\t\t\t\t\t\" ( https://en.wikipedia.org/wiki/User:Evad37/livenotifications )\"\n\t\t\t}\n\t\t}\n\t});\n\tconst config = mw.config.get([\n\t\t\"wgServer\"\n\t]);\n\tconst waitTimeMilliseconds = 1000 * 60;\n\tlet lastCheckedTimestamp = new Date();\n\n\t// Main function for retreiving and displaying notifications\n\tconst mainloop = function mainloop() {\n\t\t// Only make the request if the window/tab is active (focused)\n\t\tif ( document.hasFocus() ) {\n\t\t\treturn api\n\t\t\t\t.get({\t\t\t\t\n\t\t\t\t\t\"action\": \"query\",\n\t\t\t\t\t\"format\": \"json\",\n\t\t\t\t\t\"formatversion\": \"2\",\n\t\t\t\t\t\"curtimestamp\": 1,\n\t\t\t\t\t\"meta\": \"notifications\",\n\t\t\t\t\t\"notfilter\": \"!read\",\n\t\t\t\t\t\"notformat\": \"model\",\n\t\t\t\t\t\"notlimit\": \"20\"\n\t\t\t\t})\n\t\t\t\t.then(response => {\n\t\t\t\t\t// Find notifications which are new since the last request\n\t\t\t\t\tconst notifications = (response && response.query && response.query.notifications && response.query.notifications.list || [])\n\t\t\t\t\t\t.filter(notification => lastCheckedTimestamp < new Date(notification.timestamp.utciso8601))\n\t\t\t\t\t\t.forEach(notification => {\n\t\t\t\t\t\t\tconst iconFulllUrl = \"https:\" + config.wgServer + notification[\"*\"].iconUrl;\n\t\t\t\t\t\t\tconst readableDate = notification.timestamp.utciso8601.replace(\"T\", \" \").replace(/:\\d\\dZ/, \" (UTC)\");\n\t\t\t\t\t\t\tconst messageDiv = makeElement(\"div\", null, [\n\t\t\t\t\t\t\t\tmakeElement(\"span\", {\n\t\t\t\t\t\t\t\t\theight: \"30px\",\n\t\t\t\t\t\t\t\t\tstyle: \"float:right; margin:-0.2em -0.2em 0.2em 0.5em\"\n\t\t\t\t\t\t\t\t}, \"X\"),\n\t\t\t\t\t\t\t\tmakeElement(\"a\", {\n\t\t\t\t\t\t\t\t\thref: notification[\"*\"].links.primary.url,\n\t\t\t\t\t\t\t\t\ttitle: notification[\"*\"].links.primary.label,\n\t\t\t\t\t\t\t\t\ttarget: \"_blank\",\n\t\t\t\t\t\t\t\t\tstyle: \"display:block\"\n\t\t\t\t\t\t\t\t}, [\n\t\t\t\t\t\t\t\t\tmakeElement(\"img\", {\n\t\t\t\t\t\t\t\t\t\tsrc: iconFulllUrl,\n\t\t\t\t\t\t\t\t\t\theight: \"30px\",\n\t\t\t\t\t\t\t\t\t\tstyle: \"display:block; float:left; margin:0 0.5em 22em -0.5em; color:#666; font-size:88%\"\n\t\t\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t\t\t\tparseHtml(notification[\"*\"].header),\n\t\t\t\t\t\t\t\t\tmakeElement(\"span\", {\n\t\t\t\t\t\t\t\t\t\tstyle: \"display: block; color: #666; font-size:88%\"\n\t\t\t\t\t\t\t\t\t}, readableDate)\n\t\t\t\t\t\t\t\t].flatMap(x=>x) // Flattening in case parseHtml() returns an array\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t]);\n\t\t\t\t\t\t\tmw.notify(messageDiv, {autoHide:false});\n\t\t\t\t\t\t});\n\t\t\t\t\t// Update the last checked timestamp (for the next request)\n\t\t\t\t\tlastCheckedTimestamp = new Date(response.curtimestamp)\n\t\t\t\t})\n\t\t\t\t.catch((error, details) => console.log(\"[livenotifications] error\", {error, details}));\n\t\t}\n\t}\n\n\twindow.setInterval(() => {\n\t\tconst loop = mainloop();\n\t\tif (loop) {\n\t\t\tspinner.start();\n\t\t\tloop.then(() => spinner.stop());\n\t\t}\n\t}, waitTimeMilliseconds);\n}\n\nexport default App;","class Spinner {\n\tconstructor() {\n\t\tthis.badges = document.querySelectorAll(\".mw-echo-notifications-badge\");\n\t\tthis.minDuration = 3000; // milliseconds\n\t}\n\n\tstart() {\n\t\tthis.startTime = new Date();\n\t\tthis.badges.forEach(el => el.setAttribute(\"style\", \"background-image:url(https://upload.wikimedia.org/wikipedia/commons/thumb/7/7a/Ajax_loader_metal_512.gif/20px-Ajax_loader_metal_512.gif)\"))\n\t}\n\n\tstop() {\n\t\tconst timeUsed = new Date() - this.startTime;\n\t\tconst remainingTime = Math.max(0, this.minDuration - timeUsed);\n\t\twindow.setTimeout(\n\t\t\t() => this.badges.forEach(el => el.removeAttribute(\"style\")),\n\t\t\tremainingTime\n\t\t);\n\t}\n}\n\nconst spinner = new Spinner();\nexport default spinner;","/**\n * \n * @param {string} tagName \n * @param {object<string,string|number>} [attributes]\n * @param {Node|string|Node[]|string[]} [contents]\n * @returns {HTMLElement}\n */\nconst makeElement = function(tagName, attributes, contents) {\n\tconst el = document.createElement(tagName);\n\tif (attributes) {\n\t\tfor (const prop in attributes) {\n\t\t\tel.setAttribute(prop, attributes[prop]);\n\t\t}\n\t}\n\tif (contents) {\n\t\t(Array.isArray(contents) ? contents : [contents])\n\t\t.forEach(content => {\n\t\t\tconst node = content && content.nodeType ? content : document.createTextNode(content);\n\t\t\tel.appendChild(node);\n\t\t});\n\t}\n\treturn el;\n}\n\n/**\n * @param {string} htmlString\n * @return {Node|Node[]} Node or Nodes parsed from HTML\n */\nconst parseHtml = function(htmlString) {\n\tconst div = document.createElement(\"div\");\n\tdiv.innerHTML = htmlString;\n\tconst children = Array.from(div.childNodes);\n\treturn children.length === 1 ? children[0] : children;\n}\n\nexport {makeElement, parseHtml}"]}