MediaWiki:Gadget-MarkBlocked.js

MediaWiki系统消息页面
/**
 * SPDX-License-Identifier: CC-BY-SA-4.0
 * _addText: '{{Gadget Header|license=CC-BY-SA-4.0}}'
 *
 * @base {@link https://en.wikipedia.org/wiki/MediaWiki:Gadget-markblocked.js}
 * @source {@link https://git.qiuwen.net.cn/InterfaceAdmin/QiuwenGadgets/src/branch/master/src/MarkBlocked}
 * @license CC-BY-SA-4.0 {@link https://www.qiuwenbaike.cn/wiki/H:CC-BY-SA-4.0}
 */
/**
 * +------------------------------------------------------------+
 * |            === WARNING: GLOBAL GADGET FILE ===             |
 * +------------------------------------------------------------+
 * |       All changes should be made in the repository,        |
 * |                otherwise they will be lost.                |
 * +------------------------------------------------------------+
 * |        Changes to this page may affect many users.         |
 * | Please discuss changes by opening an issue before editing. |
 * +------------------------------------------------------------+
 */
/* <nowiki> */

(() => {

"use strict";

// dist/MarkBlocked/MarkBlocked.js
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) {
  try {
    var info = gen[key](arg);
    var value = info.value;
  } catch (error) {
    reject(error);
    return;
  }
  if (info.done) {
    resolve(value);
  } else {
    Promise.resolve(value).then(_next, _throw);
  }
}
function _asyncToGenerator(fn) {
  return function() {
    var self = this, args = arguments;
    return new Promise(function(resolve, reject) {
      var gen = fn.apply(self, args);
      function _next(value) {
        asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value);
      }
      function _throw(err) {
        asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err);
      }
      _next(void 0);
    });
  };
}
function _createForOfIteratorHelper(o, allowArrayLike) {
  var it = typeof Symbol !== "undefined" && o[Symbol.iterator] || o["@@iterator"];
  if (!it) {
    if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") {
      if (it)
        o = it;
      var i = 0;
      var F = function() {
      };
      return { s: F, n: function() {
        if (i >= o.length)
          return { done: true };
        return { done: false, value: o[i++] };
      }, e: function(e) {
        throw e;
      }, f: F };
    }
    throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
  }
  var normalCompletion = true, didErr = false, err;
  return { s: function() {
    it = it.call(o);
  }, n: function() {
    var step = it.next();
    normalCompletion = step.done;
    return step;
  }, e: function(e) {
    didErr = true;
    err = e;
  }, f: function() {
    try {
      if (!normalCompletion && it.return != null)
        it.return();
    } finally {
      if (didErr)
        throw err;
    }
  } };
}
function _unsupportedIterableToArray(o, minLen) {
  if (!o)
    return;
  if (typeof o === "string")
    return _arrayLikeToArray(o, minLen);
  var n = Object.prototype.toString.call(o).slice(8, -1);
  if (n === "Object" && o.constructor)
    n = o.constructor.name;
  if (n === "Map" || n === "Set")
    return Array.from(o);
  if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))
    return _arrayLikeToArray(o, minLen);
}
function _arrayLikeToArray(arr, len) {
  if (len == null || len > arr.length)
    len = arr.length;
  for (var i = 0, arr2 = new Array(len); i < len; i++)
    arr2[i] = arr[i];
  return arr2;
}
//! src/MarkBlocked/modules/i18n.ts
var import_ext_gadget = require("ext.gadget.i18n");
var getI18nMessages = () => {
  return {
    infinity: (0, import_ext_gadget.localize)({
      en: "infinity",
      "zh-hans": "无限期",
      "zh-hant": "無限期"
    }),
    Blocked: (0, import_ext_gadget.localize)({
      en: "; $5blocked ($1) by $2: $3 ($4 ago)",
      "zh-hans": ";由$2$5封禁$1:$3($4前)",
      "zh-hant": ";由$2$5封鎖$1:$3($4前)"
    }),
    partial: (0, import_ext_gadget.localize)({
      en: "partial ",
      zh: "部分"
    }),
    d: (0, import_ext_gadget.localize)({
      en: "day",
      zh: "天"
    }),
    s: (0, import_ext_gadget.localize)({
      en: "second",
      zh: "秒"
    })
  };
};
var i18nMessages = getI18nMessages();
var getMessage = (key) => {
  return i18nMessages[key] || key;
};
//! src/MarkBlocked/modules/util/parseTime.ts
var addZero = (v) => {
  return v <= 9 ? "0".concat(v) : "".concat(v);
};
var inHours = (ms) => {
  let mm = Math.floor(ms / 6e4);
  if (!mm) {
    return "".concat(Math.floor(ms / 1e3)).concat(getMessage("s"));
  }
  let hh = Math.floor(mm / 60);
  mm %= 60;
  const dd = Math.floor(hh / 24);
  hh %= 24;
  if (dd) {
    return "".concat(dd + (dd < 10 ? ".".concat(addZero(hh)) : "")).concat(getMessage("d"));
  }
  return "".concat(hh, ":").concat(addZero(mm));
};
var parseTS = (string) => {
  const m = string.replace(/\D/g, "").match(/(\d\d\d\d)(\d\d)(\d\d)(\d\d)(\d\d)(\d\d)/);
  return new Date(Date.UTC(+m[1], +m[2] - 1, +m[3], +m[4], +m[5], +m[6])).getTime();
};
//! src/MarkBlocked/modules/MarkBlocked.module.less
var loading = "MarkBlocked-module__loading_xtAfOG";
var userlink = "MarkBlocked-module__userlink_xtAfOG";
var userlinkIndef = "MarkBlocked-module__userlinkIndef_xtAfOG";
var userlinkPartial = "MarkBlocked-module__userlinkPartial_xtAfOG";
var userlinkTemp = "MarkBlocked-module__userlinkTemp_xtAfOG";
//! src/MarkBlocked/options.json
var contributionsPageAlias = "Special:Contributions";
var userNamespaceNumbers = [2, 3];
var version = "2.0";
//! src/MarkBlocked/modules/api.ts
var import_ext_gadget2 = require("ext.gadget.Util");
var api = (0, import_ext_gadget2.initMwApi)("MarkBlocked/".concat(version));
//! src/MarkBlocked/modules/util/generateUserNamespaceTitles.ts
var generateUserNamespaceTitles = () => {
  const {
    wgNamespaceIds
  } = mw.config.get();
  const userNamespaceTitles2 = [];
  for (var _i = 0, _Object$entries = Object.entries(wgNamespaceIds); _i < _Object$entries.length; _i++) {
    const [namespace, id] = _Object$entries[_i];
    if (!userNamespaceNumbers.includes(id)) {
      continue;
    }
    userNamespaceTitles2[userNamespaceTitles2.length] = "".concat(mw.util.escapeRegExp(namespace.replace(/_/g, " ")), ":");
  }
  return userNamespaceTitles2;
};
//! src/MarkBlocked/modules/util/generateUserLinks.ts
var userNamespaceTitles = generateUserNamespaceTitles();
var userTitleRegex = new RegExp("^(".concat(userNamespaceTitles.join("|"), "|").concat(contributionsPageAlias, "\\/)+([^\\/#]+)$"), "i");
var {
  wgArticlePath,
  wgScript
} = mw.config.get();
var articleRegex = new RegExp("".concat(wgArticlePath.replace("$1", ""), "([^#]+)"));
var scriptRegex = new RegExp("^".concat(wgScript, "\\?title=([^#&]+)"));
var generateUserLinks = ($content) => {
  const userLinks = {};
  var _iterator = _createForOfIteratorHelper($content.find("a")), _step;
  try {
    for (_iterator.s(); !(_step = _iterator.n()).done; ) {
      var _user, _userLinks$_user;
      const element = _step.value;
      const $element = $(element);
      if ($element.hasClass("mw-changeslist-date") || $element.parent("span").hasClass("mw-history-undo") || $element.parent("span").hasClass("mw-rollback-link")) {
        continue;
      }
      const href = $element.attr("href");
      if (!href) {
        continue;
      }
      if (mw.util.isIPv6Address(href.replace(/^(?:https?:\/\/)/i, ""))) {
        continue;
      }
      try {
        if (new mw.Uri(href).host !== location.host) {
          continue;
        }
      } catch {
        try {
          if (new mw.Uri(location.href + href).host !== location.host) {
            continue;
          }
        } catch {
          continue;
        }
      }
      let pageTitle;
      if (articleRegex.test(href)) {
        const match = articleRegex.exec(href);
        pageTitle = match[1];
      } else if (scriptRegex.test(href)) {
        const match = scriptRegex.exec(href);
        pageTitle = match[1];
      } else {
        continue;
      }
      pageTitle = decodeURIComponent(pageTitle).replace(/_/g, " ");
      const userExecArray = userTitleRegex.exec(pageTitle);
      if (!userExecArray || typeof userExecArray[2] !== "string") {
        continue;
      }
      let user = userExecArray[2].slice(0, 1).toUpperCase() + userExecArray[2].slice(1);
      if (mw.util.isIPv6Address(user)) {
        user = user.toUpperCase();
      }
      $element.addClass("".concat(userlink));
      (_userLinks$_user = userLinks[_user = user]) !== null && _userLinks$_user !== void 0 ? _userLinks$_user : userLinks[_user] = [];
      userLinks[user][userLinks[user].length] = $element;
    }
  } catch (err) {
    _iterator.e(err);
  } finally {
    _iterator.f();
  }
  return userLinks;
};
//! src/MarkBlocked/modules/markBlockedUser.ts
var markBlockedUser = ($content) => {
  const userLinks = generateUserLinks($content);
  const users = Object.keys(userLinks);
  if (!users.length) {
    return;
  }
  $content.addClass("".concat(loading));
  const markLinks = (response) => {
    var _response$query;
    if (!((_response$query = response["query"]) !== null && _response$query !== void 0 && _response$query.blocks)) {
      return;
    }
    var _iterator2 = _createForOfIteratorHelper(response["query"].blocks), _step2;
    try {
      for (_iterator2.s(); !(_step2 = _iterator2.n()).done; ) {
        const block = _step2.value;
        const isPartialBlcok = typeof block.restrictions === "string" && block.restrictions !== "";
        let className = "".concat(userlinkPartial);
        let blockTime = "";
        if (block.expiry.startsWith("in")) {
          if (!isPartialBlcok) {
            className = "".concat(userlinkIndef);
          }
          blockTime = getMessage("infinity");
        } else {
          if (!isPartialBlcok) {
            className = "".concat(userlinkTemp);
          }
          blockTime = inHours(parseTS(block.expiry) - parseTS(block.timestamp));
        }
        let tip = getMessage("Blocked").replace("$1", blockTime).replace("$2", block.by).replace("$3", block.reason).replace("$4", inHours(Date.now() - parseTS(block.timestamp)));
        tip = isPartialBlcok ? tip.replace("$5", getMessage("partial")) : tip.replace("$5", "");
        const $links = userLinks[block.user];
        if (!$links) {
          continue;
        }
        var _iterator3 = _createForOfIteratorHelper($links), _step3;
        try {
          for (_iterator3.s(); !(_step3 = _iterator3.n()).done; ) {
            const $link = _step3.value;
            $link.addClass(className).attr("title", $link.attr("title") + tip);
          }
        } catch (err) {
          _iterator3.e(err);
        } finally {
          _iterator3.f();
        }
      }
    } catch (err) {
      _iterator2.e(err);
    } finally {
      _iterator2.f();
    }
  };
  const promises = [];
  for (let i = 0; i < users.length; i++) {
    const bkusers = users.splice(0, 50);
    if (!bkusers.length) {
      continue;
    }
    promises[promises.length] = /* @__PURE__ */ _asyncToGenerator(function* () {
      const params = {
        bkusers,
        action: "query",
        format: "json",
        formatversion: "2",
        list: "blocks",
        bklimit: 100,
        bkprop: ["by", "expiry", "reason", "restrictions", "timestamp", "user"]
      };
      try {
        const response = yield api.post(params);
        markLinks(response);
      } catch (error) {
        console.error("[MarkBlocked] Ajax error:", error);
      }
    });
  }
  void _asyncToGenerator(function* () {
    for (var _i2 = 0, _promises = promises; _i2 < _promises.length; _i2++) {
      const promise = _promises[_i2];
      try {
        yield promise();
      } catch {
      }
    }
  })().then(() => {
    $content.removeClass("".concat(loading));
  });
};
//! src/MarkBlocked/modules/addHook.ts
var {
  wgAction,
  wgNamespaceNumber
} = mw.config.get();
var addHook = () => {
  let isInit = false;
  mw.hook("wikipage.content").add(function markBlocked($content) {
    if (isInit) {
      return;
    }
    isInit = true;
    if (wgAction === "view" && wgNamespaceNumber === 0) {
      $content = $content.find(".diff-title");
    }
    $content = $content.add("#ca-nstab-user");
    markBlockedUser($content);
  });
};
//! src/MarkBlocked/MarkBlocked.ts
if (!["edit", "submit"].includes(mw.config.get("wgAction"))) {
  addHook();
}

})();

/* </nowiki> */

//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/MarkBlocked/modules/i18n.ts", "src/MarkBlocked/modules/util/parseTime.ts", "src/MarkBlocked/modules/MarkBlocked.module.less", "src/MarkBlocked/options.json", "src/MarkBlocked/modules/api.ts", "src/MarkBlocked/modules/util/generateUserNamespaceTitles.ts", "src/MarkBlocked/modules/util/generateUserLinks.ts", "src/MarkBlocked/modules/markBlockedUser.ts", "src/MarkBlocked/modules/addHook.ts", "src/MarkBlocked/MarkBlocked.ts"],
  "sourcesContent": ["import {localize} from 'ext.gadget.i18n';\n\nconst getI18nMessages = () => {\n\treturn {\n\t\tinfinity: localize({\n\t\t\ten: 'infinity',\n\t\t\t'zh-hans': '无限期',\n\t\t\t'zh-hant': '無限期',\n\t\t}),\n\t\tBlocked: localize({\n\t\t\ten: '; $5blocked ($1) by $2: $3 ($4 ago)',\n\t\t\t'zh-hans': '；由$2$5封禁$1：$3（$4前）',\n\t\t\t'zh-hant': '；由$2$5封鎖$1：$3（$4前）',\n\t\t}),\n\t\tpartial: localize({\n\t\t\ten: 'partial ',\n\t\t\tzh: '部分',\n\t\t}),\n\t\td: localize({\n\t\t\ten: 'day',\n\t\t\tzh: '天',\n\t\t}),\n\t\ts: localize({\n\t\t\ten: 'second',\n\t\t\tzh: '秒',\n\t\t}),\n\t};\n};\n\nconst i18nMessages = getI18nMessages();\n\nconst getMessage: GetMessages<typeof i18nMessages> = (key) => {\n\treturn i18nMessages[key] || key;\n};\n\nexport {getMessage};\n", "import {getMessage} from '../i18n';\n\nconst addZero = (v: number): string => {\n\treturn v <= 9 ? `0${v}` : `${v}`;\n};\n\nconst inHours = (ms: number): string => {\n\tlet mm: number = Math.floor(ms / 6e4);\n\tif (!mm) {\n\t\treturn `${Math.floor(ms / 1e3)}${getMessage('s')}`;\n\t}\n\n\tlet hh: number = Math.floor(mm / 60);\n\tmm %= 60;\n\n\tconst dd: number = Math.floor(hh / 24);\n\thh %= 24;\n\n\tif (dd) {\n\t\treturn `${dd + (dd < 10 ? `.${addZero(hh)}` : '')}${getMessage('d')}`;\n\t}\n\n\treturn `${hh}:${addZero(mm)}`;\n};\n\n// 20081226220605 or 2008-01-26T06:34:19Z -> number\nconst parseTS = (string: string): number => {\n\tconst m: RegExpMatchArray = string\n\t\t.replace(/\\D/g, '')\n\t\t.match(/(\\d\\d\\d\\d)(\\d\\d)(\\d\\d)(\\d\\d)(\\d\\d)(\\d\\d)/) as RegExpMatchArray;\n\n\treturn new Date(\n\t\tDate.UTC(\n\t\t\t+(m[1] as string),\n\t\t\t+(m[2] as string) - 1,\n\t\t\t+(m[3] as string),\n\t\t\t+(m[4] as string),\n\t\t\t+(m[5] as string),\n\t\t\t+(m[6] as string)\n\t\t)\n\t).getTime();\n};\n\nexport {inHours, parseTS};\n", "import \"esbuild-css-modules-plugin-ns-css:src/MarkBlocked/modules/MarkBlocked.module.less\";\nexport const loading = \"MarkBlocked-module__loading_xtAfOG\";\nexport const userlink = \"MarkBlocked-module__userlink_xtAfOG\";\nexport const userlinkIndef = \"MarkBlocked-module__userlinkIndef_xtAfOG\";\nexport const userlinkPartial = \"MarkBlocked-module__userlinkPartial_xtAfOG\";\nexport const userlinkTemp = \"MarkBlocked-module__userlinkTemp_xtAfOG\";\n\nexport default {\n  \"loading\": loading,\n  \"userlink\": userlink,\n  \"userlinkIndef\": userlinkIndef,\n  \"userlinkPartial\": userlinkPartial,\n  \"userlinkTemp\": userlinkTemp\n};\n      ", "{\n\t\"contributionsPageAlias\": \"Special:Contributions\",\n\t\"userNamespaceNumbers\": [2, 3],\n\t\"version\": \"2.0\"\n}\n", "import * as OPTIONS from '../options.json';\nimport {initMwApi} from 'ext.gadget.Util';\n\nconst api: mw.Api = initMwApi(`MarkBlocked/${OPTIONS.version}`);\n\nexport {api};\n", "import * as OPTIONS from '../../options.json';\n\nconst generateUserNamespaceTitles = (): string[] => {\n\tconst {wgNamespaceIds} = mw.config.get();\n\n\tconst userNamespaceTitles: string[] = [];\n\n\tfor (const [namespace, id] of Object.entries(wgNamespaceIds)) {\n\t\tif (!OPTIONS.userNamespaceNumbers.includes(id)) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tuserNamespaceTitles[userNamespaceTitles.length] = `${mw.util.escapeRegExp(namespace.replace(/_/g, ' '))}:`; // Replace `[].push()` to avoid polyfilling core-js\n\t}\n\n\treturn userNamespaceTitles;\n};\n\nexport {generateUserNamespaceTitles};\n", "import * as OPTIONS from '../../options.json';\nimport {generateUserNamespaceTitles} from './generateUserNamespaceTitles';\nimport {userlink} from '../MarkBlocked.module.less';\n\n// Get all aliases for user: & user_talk:\nconst userNamespaceTitles: string[] = generateUserNamespaceTitles();\n// RegExp for all titles that are  User:| User_talk: | Special:Contributions/ (for userscripts)\nconst userTitleRegex: RegExp = new RegExp(\n\t`^(${userNamespaceTitles.join('|')}|${OPTIONS.contributionsPageAlias}\\\\/)+([^\\\\/#]+)$`,\n\t'i'\n);\n\n// RegExp for links\n// articleRegex also matches external links in order to support the noping template\nconst {wgArticlePath, wgScript} = mw.config.get();\nconst articleRegex: RegExp = new RegExp(`${wgArticlePath.replace('$1', '')}([^#]+)`);\nconst scriptRegex: RegExp = new RegExp(`^${wgScript}\\\\?title=([^#&]+)`);\n\nconst generateUserLinks = ($content: JQuery): Record<string, JQuery[]> => {\n\t// Find all \"user\" links and save them in userLinks : { 'users': [<link1>, <link2>, ...], 'user2': [<link3>, <link3>, ...], ... }\n\tconst userLinks: Record<string, JQuery[]> = {};\n\n\tfor (const element of $content.find('a')) {\n\t\tconst $element: JQuery<HTMLAnchorElement> = $(element);\n\t\tif (\n\t\t\t$element.hasClass('mw-changeslist-date') ||\n\t\t\t$element.parent('span').hasClass('mw-history-undo') ||\n\t\t\t$element.parent('span').hasClass('mw-rollback-link')\n\t\t) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst href: string | undefined = $element.attr('href');\n\t\tif (!href) {\n\t\t\tcontinue;\n\t\t}\n\t\tif (mw.util.isIPv6Address(href.replace(/^(?:https?:\\/\\/)/i, ''))) {\n\t\t\tcontinue;\n\t\t}\n\t\ttry {\n\t\t\tif (new mw.Uri(href).host !== location.host) {\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t} catch {\n\t\t\ttry {\n\t\t\t\tif (new mw.Uri(location.href + href).host !== location.host) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t} catch {\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t}\n\n\t\tlet pageTitle: string | undefined;\n\t\tif (articleRegex.test(href)) {\n\t\t\tconst match: RegExpExecArray = articleRegex.exec(href) as RegExpExecArray;\n\t\t\tpageTitle = match[1] as string;\n\t\t} else if (scriptRegex.test(href)) {\n\t\t\tconst match: RegExpExecArray = scriptRegex.exec(href) as RegExpExecArray;\n\t\t\tpageTitle = match[1] as string;\n\t\t} else {\n\t\t\tcontinue;\n\t\t}\n\n\t\tpageTitle = decodeURIComponent(pageTitle).replace(/_/g, ' ');\n\n\t\tconst userExecArray: RegExpExecArray | null = userTitleRegex.exec(pageTitle);\n\t\tif (!userExecArray || typeof userExecArray[2] !== 'string') {\n\t\t\tcontinue;\n\t\t}\n\n\t\tlet user: string = userExecArray[2].slice(0, 1).toUpperCase() + userExecArray[2].slice(1);\n\t\tif (mw.util.isIPv6Address(user)) {\n\t\t\tuser = user.toUpperCase();\n\t\t}\n\n\t\t// The following classes are used here:\n\t\t// * see ../Markblocked.module.less\n\t\t// * for more information\n\t\t$element.addClass(`${userlink}`);\n\n\t\tuserLinks[user] ??= [];\n\n\t\t(userLinks[user] as (typeof userLinks)[keyof typeof userLinks])[\n\t\t\t(userLinks[user] as (typeof userLinks)[keyof typeof userLinks]).length\n\t\t] = $element; // Replace `[].push()` to avoid polyfilling core-js\n\t}\n\n\treturn userLinks;\n};\n\nexport {generateUserLinks};\n", "import {inHours, parseTS} from './util/parseTime';\nimport {loading, userlinkIndef, userlinkPartial, userlinkTemp} from './MarkBlocked.module.less';\nimport {api} from './api';\nimport {generateUserLinks} from './util/generateUserLinks';\nimport {getMessage} from './i18n';\n\nconst markBlockedUser = ($content: JQuery): void => {\n\t// Find all \"user\" links and save them in userLinks : { 'users': [<link1>, <link2>, ...], 'user2': [<link3>, <link3>, ...], ... }\n\tconst userLinks: Record<string, JQuery[]> = generateUserLinks($content);\n\n\t// Convert users into array\n\tconst users: string[] = Object.keys(userLinks);\n\tif (!users.length) {\n\t\treturn;\n\t}\n\n\t// The following classes are used here:\n\t// * see ./Markblocked.module.less\n\t// * for more information\n\t$content.addClass(`${loading}`);\n\n\t// API request\n\ttype Response = {\n\t\tquery: {\n\t\t\tblocks?: Array<{\n\t\t\t\tby: string;\n\t\t\t\texpiry: string;\n\t\t\t\treason: string;\n\t\t\t\trestrictions: string | string[];\n\t\t\t\ttimestamp: string;\n\t\t\t\tuser: string;\n\t\t\t}>;\n\t\t};\n\t};\n\n\t// Callback: receive data and mark links\n\tconst markLinks = (response: Response): void => {\n\t\tif (!response['query']?.blocks) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor (const block of response['query'].blocks) {\n\t\t\tconst isPartialBlcok: boolean = typeof block.restrictions === 'string' && block.restrictions !== ''; // Partial block\n\n\t\t\tlet className: string = `${userlinkPartial}`;\n\t\t\tlet blockTime: string = '';\n\t\t\tif (block.expiry.startsWith('in')) {\n\t\t\t\tif (!isPartialBlcok) {\n\t\t\t\t\tclassName = `${userlinkIndef}`;\n\t\t\t\t}\n\t\t\t\tblockTime = getMessage('infinity');\n\t\t\t} else {\n\t\t\t\tif (!isPartialBlcok) {\n\t\t\t\t\tclassName = `${userlinkTemp}`;\n\t\t\t\t}\n\t\t\t\tblockTime = inHours(parseTS(block.expiry) - parseTS(block.timestamp));\n\t\t\t}\n\n\t\t\tlet tip: string = getMessage('Blocked')\n\t\t\t\t.replace('$1', blockTime)\n\t\t\t\t.replace('$2', block.by)\n\t\t\t\t.replace('$3', block.reason)\n\t\t\t\t.replace('$4', inHours(Date.now() - parseTS(block.timestamp)));\n\t\t\ttip = isPartialBlcok ? tip.replace('$5', getMessage('partial')) : tip.replace('$5', '');\n\n\t\t\tconst $links: JQuery[] | undefined = userLinks[block.user];\n\t\t\tif (!$links) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tfor (const $link of $links) {\n\t\t\t\t// The following classes are used here:\n\t\t\t\t// * see ./Markblocked.module.less\n\t\t\t\t// * for more information\n\t\t\t\t$link.addClass(className).attr('title', $link.attr('title') + tip);\n\t\t\t}\n\t\t}\n\t};\n\n\tconst promises: (() => Promise<void>)[] = [];\n\n\tfor (let i = 0; i < users.length; i++) {\n\t\tconst bkusers = users.splice(0, 50);\n\t\tif (!bkusers.length) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tpromises[promises.length] = async (): Promise<void> => {\n\t\t\tconst params: ApiQueryBlocksParams = {\n\t\t\t\tbkusers,\n\t\t\t\taction: 'query',\n\t\t\t\tformat: 'json',\n\t\t\t\tformatversion: '2',\n\t\t\t\tlist: 'blocks',\n\t\t\t\tbklimit: 100,\n\t\t\t\tbkprop: ['by', 'expiry', 'reason', 'restrictions', 'timestamp', 'user'],\n\t\t\t};\n\n\t\t\ttry {\n\t\t\t\tconst response = await api.post(params);\n\t\t\t\tmarkLinks(response as Response);\n\t\t\t} catch (error: unknown) {\n\t\t\t\tconsole.error('[MarkBlocked] Ajax error:', error);\n\t\t\t}\n\t\t};\n\t}\n\n\tvoid (async () => {\n\t\tfor (const promise of promises) {\n\t\t\ttry {\n\t\t\t\tawait promise();\n\t\t\t} catch {}\n\t\t}\n\t})().then(() => {\n\t\t// The following classes are used here:\n\t\t// * see ./Markblocked.module.less\n\t\t// * for more information\n\t\t$content.removeClass(`${loading}`);\n\t});\n};\n\nexport {markBlockedUser};\n", "import {markBlockedUser} from './markBlockedUser';\n\nconst {wgAction, wgNamespaceNumber} = mw.config.get();\n\nconst addHook = (): void => {\n\tlet isInit: boolean = false;\n\n\tmw.hook('wikipage.content').add(function markBlocked($content): void {\n\t\tif (isInit) {\n\t\t\treturn;\n\t\t}\n\t\tisInit = true;\n\n\t\t// On the first call after initial page load, container is mw.util.$content\n\t\t// Used to limit mainspace activity to just the diff definitions\n\t\tif (wgAction === 'view' && wgNamespaceNumber === 0) {\n\t\t\t$content = $content.find('.diff-title');\n\t\t}\n\t\t// On page load, also update the namespace tab\n\t\t$content = $content.add('#ca-nstab-user');\n\n\t\tmarkBlockedUser($content);\n\t});\n};\n\nexport {addHook};\n", "import {addHook} from './modules/addHook';\n\nif (!['edit', 'submit'].includes(mw.config.get('wgAction'))) {\n\taddHook();\n}\n"],
  "mappings": "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",
  "names": ["import_ext_gadget", "require", "getI18nMessages", "infinity", "localize", "en", "Blocked", "partial", "zh", "d", "s", "i18nMessages", "getMessage", "key", "addZero", "v", "concat", "inHours", "ms", "mm", "Math", "floor", "hh", "dd", "parseTS", "string", "m", "replace", "match", "Date", "UTC", "getTime", "loading", "userlink", "userlinkIndef", "userlinkPartial", "userlinkTemp", "contributionsPageAlias", "userNamespaceNumbers", "version", "import_ext_gadget2", "api", "initMwApi", "generateUserNamespaceTitles", "wgNamespaceIds", "mw", "config", "get", "userNamespaceTitles2", "_i", "_Object$entries", "Object", "entries", "length", "namespace", "id", "includes", "util", "escapeRegExp", "userNamespaceTitles", "userTitleRegex", "RegExp", "join", "wgArticlePath", "wgScript", "articleRegex", "scriptRegex", "generateUserLinks", "$content", "userLinks", "_iterator", "_createForOfIteratorHelper", "find", "_step", "n", "done", "_user", "_userLinks$_user", "element", "value", "$element", "$", "hasClass", "parent", "href", "attr", "isIPv6Address", "Uri", "host", "location", "pageTitle", "test", "exec", "decodeURIComponent", "userExecArray", "user", "slice", "toUpperCase", "addClass", "err", "e", "f", "markBlockedUser", "users", "keys", "markLinks", "response", "_response$query", "blocks", "_iterator2", "_step2", "block", "isPartialBlcok", "restrictions", "className", "blockTime", "expiry", "startsWith", "timestamp", "tip", "by", "reason", "now", "$links", "_iterator3", "_step3", "$link", "promises", "i", "bkusers", "splice", "_asyncToGenerator", "params", "action", "format", "formatversion", "list", "bklimit", "bkprop", "post", "error", "console", "_i2", "_promises", "promise", "then", "removeClass", "wgAction", "wgNamespaceNumber", "addHook", "isInit", "hook", "add", "markBlocked"]
}
