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://github.com/TopRealm/YsArchives-Gadgets/tree/master/src/MarkBlocked}
 * @license CC-BY-SA-4.0 {@link https://youshou.wiki/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 _createForOfIteratorHelper(r, e) {
  var t = "undefined" != typeof Symbol && r[Symbol.iterator] || r["@@iterator"];
  if (!t) {
    if (Array.isArray(r) || (t = _unsupportedIterableToArray(r)) || e && r && "number" == typeof r.length) {
      t && (r = t);
      var n = 0, F = function() {
      };
      return { s: F, n: function() {
        return n >= r.length ? { done: true } : { done: false, value: r[n++] };
      }, e: function(r2) {
        throw r2;
      }, 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 o, a = true, u = false;
  return { s: function() {
    t = t.call(r);
  }, n: function() {
    var r2 = t.next();
    return a = r2.done, r2;
  }, e: function(r2) {
    u = true, o = r2;
  }, f: function() {
    try {
      a || null == t.return || t.return();
    } finally {
      if (u) throw o;
    }
  } };
}
function _unsupportedIterableToArray(r, a) {
  if (r) {
    if ("string" == typeof r) return _arrayLikeToArray(r, a);
    var t = {}.toString.call(r).slice(8, -1);
    return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0;
  }
}
function _arrayLikeToArray(r, a) {
  (null == a || a > r.length) && (a = r.length);
  for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e];
  return n;
}
//! src/MarkBlocked/modules/constant.ts
var CLASS_NAME = "gadget-mark_blocked";
var CLASS_NAME_LOADING = "".concat(CLASS_NAME, "--loading");
var CLASS_NAME_USERLINK = "".concat(CLASS_NAME, "-userlink");
var CLASS_NAME_USERLINK_INDEF = "".concat(CLASS_NAME_USERLINK, "--indef");
var CLASS_NAME_USERLINK_PARTIAL = "".concat(CLASS_NAME_USERLINK, "--partial");
var CLASS_NAME_USERLINK_TEMP = "".concat(CLASS_NAME_USERLINK, "--temp");
var WG_ACTION = mw.config.get("wgAction");
var WG_ARTICLE_PATH = mw.config.get("wgArticlePath");
var WG_NAMESPACE_IDS = mw.config.get("wgNamespaceIds");
var WG_NAMESPACE_NUMBER = mw.config.get("wgNamespaceNumber");
var WG_SCRIPT = mw.config.get("wgScript");
var IS_WG_EDIT_OR_SUBMIT_ACTION = ["edit", "submit"].includes(WG_ACTION);
//! 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 zz = (v) => {
  let _v = String(v);
  if (v <= 9) {
    _v = "0".concat(v);
  }
  return _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(zz(hh)) : "")).concat(getMessage("d"));
  }
  return "".concat(hh, ":").concat(zz(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/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 userNamespaceTitles2 = [];
  for (var _i = 0, _Object$entries = Object.entries(WG_NAMESPACE_IDS); _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 articleRegex = new RegExp("".concat(WG_ARTICLE_PATH.replace("$1", ""), "([^#]+)"));
var scriptRegex = new RegExp("^".concat(WG_SCRIPT, "\\?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(CLASS_NAME_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(CLASS_NAME_LOADING);
  let apiRequestCount = 0;
  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 = CLASS_NAME_USERLINK_PARTIAL;
        let blockTime = "";
        if (block.expiry.startsWith("in")) {
          if (!isPartialBlcok) {
            className = CLASS_NAME_USERLINK_INDEF;
          }
          blockTime = getMessage("infinity");
        } else {
          if (!isPartialBlcok) {
            className = CLASS_NAME_USERLINK_TEMP;
          }
          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();
    }
    if (--apiRequestCount === 0) {
      $content.removeClass(CLASS_NAME_LOADING);
    }
  };
  while (users.length > 0) {
    apiRequestCount++;
    const params = {
      action: "query",
      format: "json",
      formatversion: "2",
      list: "blocks",
      bklimit: 100,
      bkprop: ["by", "expiry", "reason", "restrictions", "timestamp", "user"],
      bkusers: users.splice(0, 50).join("|")
    };
    api.post(params).then((response) => {
      markLinks(response);
    }).catch((error) => {
      console.error("[MarkBlocked] Ajax error:", error);
    });
  }
};
//! src/MarkBlocked/modules/addHook.ts
var addHook = () => {
  let isInit = false;
  mw.hook("wikipage.content").add(function markBlocked($content) {
    if (isInit) {
      return;
    }
    isInit = true;
    if (WG_ACTION === "view" && WG_NAMESPACE_NUMBER === 0) {
      $content = $content.find(".diff-title");
    }
    $content = $content.add("#ca-nstab-user");
    markBlockedUser($content);
  });
};
//! src/MarkBlocked/MarkBlocked.ts
if (!IS_WG_EDIT_OR_SUBMIT_ACTION) {
  addHook();
}

})();

/* </nowiki> */

//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/MarkBlocked/modules/constant.ts", "src/MarkBlocked/modules/i18n.ts", "src/MarkBlocked/modules/util/parseTime.ts", "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": ["const CLASS_NAME: string = 'gadget-mark_blocked';\nconst CLASS_NAME_LOADING: string = `${CLASS_NAME}--loading`;\nconst CLASS_NAME_USERLINK: string = `${CLASS_NAME}-userlink`;\nconst CLASS_NAME_USERLINK_INDEF: string = `${CLASS_NAME_USERLINK}--indef`;\nconst CLASS_NAME_USERLINK_PARTIAL: string = `${CLASS_NAME_USERLINK}--partial`;\nconst CLASS_NAME_USERLINK_TEMP: string = `${CLASS_NAME_USERLINK}--temp`;\n\nconst WG_ACTION: MediaWikiConfigMapWgAction = mw.config.get('wgAction');\nconst WG_ARTICLE_PATH: string = mw.config.get('wgArticlePath');\nconst WG_NAMESPACE_IDS: Record<string, number> = mw.config.get('wgNamespaceIds');\nconst WG_NAMESPACE_NUMBER: number = mw.config.get('wgNamespaceNumber');\nconst WG_SCRIPT: string = mw.config.get('wgScript');\n\nconst IS_WG_EDIT_OR_SUBMIT_ACTION: boolean = ['edit', 'submit'].includes(WG_ACTION);\n\nexport {\n\tCLASS_NAME_LOADING,\n\tCLASS_NAME_USERLINK,\n\tCLASS_NAME_USERLINK_INDEF,\n\tCLASS_NAME_USERLINK_PARTIAL,\n\tCLASS_NAME_USERLINK_TEMP,\n\tWG_ACTION,\n\tWG_ARTICLE_PATH,\n\tWG_NAMESPACE_IDS,\n\tWG_NAMESPACE_NUMBER,\n\tWG_SCRIPT,\n\tIS_WG_EDIT_OR_SUBMIT_ACTION,\n};\n", "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 zz = (v: number): string => {\n\tlet _v: string = String(v);\n\tif (v <= 9) {\n\t\t_v = `0${v}`;\n\t}\n\n\treturn _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 ? `.${zz(hh)}` : '')}${getMessage('d')}`;\n\t}\n\n\treturn `${hh}:${zz(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", "{\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';\nimport {WG_NAMESPACE_IDS} from '../constant';\n\nconst generateUserNamespaceTitles = (): string[] => {\n\tconst userNamespaceTitles: string[] = [];\n\n\tfor (const [namespace, id] of Object.entries(WG_NAMESPACE_IDS)) {\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 {CLASS_NAME_USERLINK, WG_ARTICLE_PATH, WG_SCRIPT} from '../constant';\nimport {generateUserNamespaceTitles} from './generateUserNamespaceTitles';\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 articleRegex: RegExp = new RegExp(`${WG_ARTICLE_PATH.replace('$1', '')}([^#]+)`);\nconst scriptRegex: RegExp = new RegExp(`^${WG_SCRIPT}\\\\?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 constant.ts\n\t\t// * for more information\n\t\t$element.addClass(CLASS_NAME_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 {\n\tCLASS_NAME_LOADING,\n\tCLASS_NAME_USERLINK_INDEF,\n\tCLASS_NAME_USERLINK_PARTIAL,\n\tCLASS_NAME_USERLINK_TEMP,\n} from './constant';\nimport {inHours, parseTS} from './util/parseTime';\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 constant.ts\n\t// * for more information\n\t$content.addClass(CLASS_NAME_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\tlet apiRequestCount: number = 0;\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 = CLASS_NAME_USERLINK_PARTIAL;\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 = CLASS_NAME_USERLINK_INDEF;\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 = CLASS_NAME_USERLINK_TEMP;\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 constant.ts for more information\n\t\t\t\t// * gadget-mark_blocked-userlink--indef\n\t\t\t\t// * gadget-mark_blocked-userlink--temp\n\t\t\t\t// * gadget-mark_blocked-userlink--partial\n\t\t\t\t$link.addClass(className).attr('title', $link.attr('title') + tip);\n\t\t\t}\n\t\t}\n\n\t\tif (--apiRequestCount === 0) {\n\t\t\t// The following classes are used here:\n\t\t\t// * see constant.ts\n\t\t\t// * for more information\n\t\t\t$content.removeClass(CLASS_NAME_LOADING);\n\t\t}\n\t};\n\n\twhile (users.length > 0) {\n\t\tapiRequestCount++;\n\t\tconst params: ApiQueryBlocksParams = {\n\t\t\taction: 'query',\n\t\t\tformat: 'json',\n\t\t\tformatversion: '2',\n\t\t\tlist: 'blocks',\n\t\t\tbklimit: 100,\n\t\t\tbkprop: ['by', 'expiry', 'reason', 'restrictions', 'timestamp', 'user'],\n\t\t\tbkusers: users.splice(0, 50).join('|'),\n\t\t};\n\t\tapi.post(params)\n\t\t\t.then((response): void => {\n\t\t\t\tmarkLinks(response as Response);\n\t\t\t})\n\t\t\t.catch((error: never): void => {\n\t\t\t\tconsole.error('[MarkBlocked] Ajax error:', error);\n\t\t\t});\n\t}\n};\n\nexport {markBlockedUser};\n", "import {WG_ACTION, WG_NAMESPACE_NUMBER} from './constant';\nimport {markBlockedUser} from './markBlockedUser';\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 (WG_ACTION === 'view' && WG_NAMESPACE_NUMBER === 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 {IS_WG_EDIT_OR_SUBMIT_ACTION} from './modules/constant';\nimport {addHook} from './modules/addHook';\n\nif (!IS_WG_EDIT_OR_SUBMIT_ACTION) {\n\taddHook();\n}\n"],
  "mappings": "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",
  "names": ["CLASS_NAME", "CLASS_NAME_LOADING", "concat", "CLASS_NAME_USERLINK", "CLASS_NAME_USERLINK_INDEF", "CLASS_NAME_USERLINK_PARTIAL", "CLASS_NAME_USERLINK_TEMP", "WG_ACTION", "mw", "config", "get", "WG_ARTICLE_PATH", "WG_NAMESPACE_IDS", "WG_NAMESPACE_NUMBER", "WG_SCRIPT", "IS_WG_EDIT_OR_SUBMIT_ACTION", "includes", "import_ext_gadget", "require", "getI18nMessages", "infinity", "localize", "en", "Blocked", "partial", "zh", "d", "s", "i18nMessages", "getMessage", "key", "zz", "v", "_v", "String", "inHours", "ms", "mm", "Math", "floor", "hh", "dd", "parseTS", "string", "m", "replace", "match", "Date", "UTC", "getTime", "contributionsPageAlias", "userNamespaceNumbers", "version", "import_ext_gadget2", "api", "initMwApi", "generateUserNamespaceTitles", "userNamespaceTitles2", "_i", "_Object$entries", "Object", "entries", "length", "namespace", "id", "util", "escapeRegExp", "userNamespaceTitles", "userTitleRegex", "RegExp", "join", "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", "apiRequestCount", "markLinks", "response", "_response$query", "blocks", "_iterator2", "_step2", "block", "isPartialBlcok", "restrictions", "className", "blockTime", "expiry", "startsWith", "timestamp", "tip", "by", "reason", "now", "$links", "_iterator3", "_step3", "$link", "removeClass", "params", "action", "format", "formatversion", "list", "bklimit", "bkprop", "bkusers", "splice", "post", "then", "catch", "error", "console", "addHook", "isInit", "hook", "add", "markBlocked"]
}
