MediaWiki:Gadget-OnlineAdmins.js

MediaWiki系统消息页面
/**
 * SPDX-License-Identifier: CC-BY-SA-4.0
 * _addText: '{{Gadget Header|license=CC-BY-SA-4.0}}'
 *
 * @base {@link https://zh.wikipedia.org/wiki/User:Vanished_user_1929210/js/OnlineAdmins.js}
 * @source {@link https://git.qiuwen.net.cn/InterfaceAdmin/QiuwenGadgets/src/branch/master/src/OnlineAdmins}
 * @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/OnlineAdmins/OnlineAdmins.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;
}
var __create = Object.create;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __getProtoOf = Object.getPrototypeOf;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __copyProps = (to, from, except, desc) => {
  if (from && typeof from === "object" || typeof from === "function") {
    var _iterator = _createForOfIteratorHelper(__getOwnPropNames(from)), _step;
    try {
      for (_iterator.s(); !(_step = _iterator.n()).done; ) {
        let key = _step.value;
        if (!__hasOwnProp.call(to, key) && key !== except)
          __defProp(to, key, {
            get: () => from[key],
            enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
          });
      }
    } catch (err) {
      _iterator.e(err);
    } finally {
      _iterator.f();
    }
  }
  return to;
};
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
  // If the importer is in node compatibility mode or this is not an ESM
  // file that has been converted to a CommonJS file using a Babel-
  // compatible transform (i.e. "__esModule" has not been set), then set
  // "default" to the CommonJS "module.exports" for node compatibility.
  isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", {
    value: mod,
    enumerable: true
  }) : target,
  mod
));
//! src/OnlineAdmins/modules/components/OnlineAdmins.module.less
var section = "OnlineAdmins-module__section_WS9IGa";
var sectionList = "OnlineAdmins-module__sectionList_WS9IGa";
var talkPageLink = "OnlineAdmins-module__talkPageLink_WS9IGa";
//! src/OnlineAdmins/modules/components/groupList.tsx
var import_ext_gadget2 = __toESM(require("ext.gadget.React"), 1);
//! src/OnlineAdmins/modules/i18n.ts
var import_ext_gadget = require("ext.gadget.i18n");
var getI18nMessages = () => {
  return {
    " ($1 online):": (0, import_ext_gadget.localize)({
      en: " ($1 online):",
      ja: "($1人オンライン中):",
      "zh-hans": "($1个在线):",
      "zh-hant": "($1個在線):"
    }),
    Admins: (0, import_ext_gadget.localize)({
      en: "Admins",
      ja: "管理者",
      "zh-hans": "管理员",
      "zh-hant": "管理員"
    }),
    Patrollers: (0, import_ext_gadget.localize)({
      en: "Patrollers",
      ja: "巡回者",
      "zh-hans": "巡查员",
      "zh-hant": "巡查員"
    }),
    Stewards: (0, import_ext_gadget.localize)({
      en: "Stewards",
      ja: "スチュワード",
      "zh-hans": "裁决委员",
      "zh-hant": "裁決委員"
    }),
    "Network error": (0, import_ext_gadget.localize)({
      en: "Network error",
      ja: "ネットワークエラー",
      "zh-hans": "网络异常",
      "zh-hant": "網路異常"
    }),
    NoOnline: (0, import_ext_gadget.localize)({
      en: "Currently there are no high privilege users online",
      ja: "現在、高権限利用者はオンラインにいません",
      "zh-hans": "目前没有站务人员在线",
      "zh-hant": "目前沒有站務人員在線"
    }),
    Online: (0, import_ext_gadget.localize)({
      en: "Online admins",
      ja: "オンラインの高権限利用者",
      "zh-hans": "在线站务人员",
      "zh-hant": "線上站務人員"
    }),
    OnlineWithin30Minutes: (0, import_ext_gadget.localize)({
      en: "High privilege users online within 30 minutes:",
      ja: "30分以内にオンラインで高権限利用者:",
      "zh-hans": "下面是最近30分钟内在线的站务人员:",
      "zh-hant": "下面是最近30分鐘內的線上站務人員:"
    })
  };
};
var i18nMessages = getI18nMessages();
var getMessage = (key) => {
  return i18nMessages[key] || key;
};
//! src/OnlineAdmins/modules/components/groupList.tsx
var sanitize = (string) => string.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&lt;").replace(/"/g, "&quot;");
var UserLink = ({
  userName
}) => /* @__PURE__ */ import_ext_gadget2.default.createElement("li", null, /* @__PURE__ */ import_ext_gadget2.default.createElement("a", {
  href: mw.util.getUrl("User:".concat(sanitize(userName))),
  rel: "noopener noreferrer",
  target: "_blank"
}, sanitize(userName)), " ", /* @__PURE__ */ import_ext_gadget2.default.createElement("span", {
  className: talkPageLink
}, "(", /* @__PURE__ */ import_ext_gadget2.default.createElement("a", {
  href: mw.util.getUrl("User_talk:".concat(sanitize(userName))),
  rel: "noopener noreferrer",
  target: "_blank"
}, "留言"), ")"));
var onlineCountText = getMessage(" ($1 online):");
var GroupList = ({
  groupName,
  userNames
}) => /* @__PURE__ */ import_ext_gadget2.default.createElement("div", {
  className: section
}, /* @__PURE__ */ import_ext_gadget2.default.createElement("span", null, groupName), /* @__PURE__ */ import_ext_gadget2.default.createElement("span", null, onlineCountText.replace("$1", String(userNames.length))), /* @__PURE__ */ import_ext_gadget2.default.createElement("ul", {
  className: sectionList
}, userNames.map((user) => /* @__PURE__ */ import_ext_gadget2.default.createElement(UserLink, {
  key: user,
  userName: user
}))));
var groupListElement = (groupName, userNames) => /* @__PURE__ */ import_ext_gadget2.default.createElement(GroupList, {
  groupName,
  userNames
});
var listTitle = () => /* @__PURE__ */ import_ext_gadget2.default.createElement("p", null, getMessage("OnlineWithin30Minutes"));
//! src/OnlineAdmins/modules/constant.ts
var BLACK_LIST = ["滥用过滤器"];
//! src/OnlineAdmins/options.json
var version = "2.0";
//! src/OnlineAdmins/modules/api.ts
var import_ext_gadget3 = require("ext.gadget.Util");
var api = (0, import_ext_gadget3.initMwApi)("OnlineAdmins/".concat(version));
//! src/OnlineAdmins/modules/doClick.ts
var import_ext_gadget4 = require("ext.gadget.Util");
var queryRecentChanges = /* @__PURE__ */ function() {
  var _ref = _asyncToGenerator(function* (rcstart, rcend) {
    const params = {
      rcstart,
      rcend,
      action: "query",
      format: "json",
      formatversion: "2",
      list: "recentchanges",
      rcprop: "user",
      rcshow: ["!bot", "!anon"],
      rclimit: 500
    };
    const response = yield api.post(params);
    return response;
  });
  return function queryRecentChanges2(_x, _x2) {
    return _ref.apply(this, arguments);
  };
}();
var queryLogEvents = /* @__PURE__ */ function() {
  var _ref2 = _asyncToGenerator(function* (lestart, leend) {
    const params = {
      lestart,
      leend,
      action: "query",
      format: "json",
      formatversion: "2",
      list: "logevents",
      leprop: "user",
      lelimit: 500
    };
    const response = yield api.post(params);
    return response;
  });
  return function queryLogEvents2(_x3, _x4) {
    return _ref2.apply(this, arguments);
  };
}();
var queryUserProps = /* @__PURE__ */ function() {
  var _ref3 = _asyncToGenerator(function* (ususers) {
    const params = {
      ususers,
      action: "query",
      format: "json",
      formatversion: "2",
      list: "users",
      usprop: "groups"
    };
    const response = yield api.post(params);
    return response;
  });
  return function queryUserProps2(_x5) {
    return _ref3.apply(this, arguments);
  };
}();
var doClick = /* @__PURE__ */ function() {
  var _ref4 = _asyncToGenerator(function* (event) {
    event.preventDefault();
    let users = [];
    const usersExt = [];
    const stewards = [];
    const admins = [];
    const patrollers = [];
    const time = /* @__PURE__ */ new Date();
    const rcstart = time.toISOString();
    time.setMinutes(time.getMinutes() - 30);
    const rcend = time.toISOString();
    try {
      const recentchanges = yield queryRecentChanges(rcstart, rcend);
      var _iterator2 = _createForOfIteratorHelper(recentchanges["query"].recentchanges), _step2;
      try {
        for (_iterator2.s(); !(_step2 = _iterator2.n()).done; ) {
          const {
            user
          } = _step2.value;
          users[users.length] = user;
        }
      } catch (err) {
        _iterator2.e(err);
      } finally {
        _iterator2.f();
      }
      const logevents = yield queryLogEvents(rcstart, rcend);
      var _iterator3 = _createForOfIteratorHelper(logevents["query"].logevents), _step3;
      try {
        for (_iterator3.s(); !(_step3 = _iterator3.n()).done; ) {
          const {
            user
          } = _step3.value;
          usersExt[usersExt.length] = user;
        }
      } catch (err) {
        _iterator3.e(err);
      } finally {
        _iterator3.f();
      }
      users = (0, import_ext_gadget4.uniqueArray)([...users, ...usersExt]);
      const promises = [];
      for (let i = 0; i < users.length; i++) {
        const ususers = users.splice(0, 50);
        if (!ususers.length) {
          continue;
        }
        promises[promises.length] = /* @__PURE__ */ _asyncToGenerator(function* () {
          const response = yield queryUserProps(ususers);
          var _iterator4 = _createForOfIteratorHelper(response["query"].users), _step4;
          try {
            for (_iterator4.s(); !(_step4 = _iterator4.n()).done; ) {
              const {
                groups,
                name
              } = _step4.value;
              if (groups.includes("bot") || BLACK_LIST.includes(name) || !name) {
                continue;
              }
              if (groups.includes("steward")) {
                stewards[stewards.length] = name;
              }
              if (groups.includes("sysop")) {
                admins[admins.length] = name;
              }
              if (groups.includes("patroller")) {
                patrollers[patrollers.length] = name;
              }
            }
          } catch (err) {
            _iterator4.e(err);
          } finally {
            _iterator4.f();
          }
        });
      }
      void _asyncToGenerator(function* () {
        for (var _i = 0, _promises = promises; _i < _promises.length; _i++) {
          const promise = _promises[_i];
          try {
            yield promise();
          } catch {
          }
        }
      })().then(() => {
        if (stewards.length + admins.length + patrollers.length) {
          const elements = [listTitle()];
          if (stewards.length) {
            elements[elements.length] = groupListElement(getMessage("Stewards"), stewards);
          }
          if (admins.length) {
            elements[elements.length] = groupListElement(getMessage("Admins"), admins);
          }
          if (patrollers.length) {
            elements[elements.length] = groupListElement(getMessage("Patrollers"), patrollers);
          }
          void mw.notify($("<div>").append(elements), {
            tag: "onlineAdmins"
          });
        } else {
          void mw.notify(getMessage("NoOnline"), {
            tag: "onlineAdmins",
            type: "warn"
          });
        }
      });
    } catch {
      void mw.notify(getMessage("Network error"), {
        tag: "onlineAdmins",
        type: "error"
      });
    }
  });
  return function doClick2(_x6) {
    return _ref4.apply(this, arguments);
  };
}();
//! src/OnlineAdmins/OnlineAdmins.ts
var addPortletLink = () => {
  const portletId = document.querySelector("#p-cactions") ? "p-cactions" : "p-tb";
  const element = mw.util.addPortletLink(portletId, "#", getMessage("Online"), "t-onlineadmin");
  if (!element) {
    return;
  }
  $(element).find("a").on("click", (event) => {
    void doClick(event);
  });
};
$(addPortletLink);

})();

/* </nowiki> */

//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/OnlineAdmins/modules/components/OnlineAdmins.module.less", "src/OnlineAdmins/modules/components/groupList.tsx", "src/OnlineAdmins/modules/i18n.ts", "src/OnlineAdmins/modules/constant.ts", "src/OnlineAdmins/options.json", "src/OnlineAdmins/modules/api.ts", "src/OnlineAdmins/modules/doClick.ts", "src/OnlineAdmins/OnlineAdmins.ts"],
  "sourcesContent": ["import \"esbuild-css-modules-plugin-ns-css:src/OnlineAdmins/modules/components/OnlineAdmins.module.less\";\nexport const section = \"OnlineAdmins-module__section_WS9IGa\";\nexport const sectionList = \"OnlineAdmins-module__sectionList_WS9IGa\";\nexport const talkPageLink = \"OnlineAdmins-module__talkPageLink_WS9IGa\";\n\nexport default {\n  \"section\": section,\n  \"sectionList\": sectionList,\n  \"talkPageLink\": talkPageLink\n};\n      ", "import {section, sectionList, talkPageLink} from './OnlineAdmins.module.less';\nimport React from 'ext.gadget.React';\nimport {getMessage} from '../i18n';\n\nconst sanitize = (string: string) =>\n\tstring.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&lt;').replace(/\"/g, '&quot;');\n\ninterface UserLinkProps {\n\tuserName: string;\n}\n\ninterface GroupListProps {\n\tgroupName: string;\n\tuserNames: string[];\n}\n\nconst UserLink = ({userName}: UserLinkProps) => (\n\t<li>\n\t\t<a href={mw.util.getUrl(`User:${sanitize(userName)}`)} rel=\"noopener noreferrer\" target=\"_blank\">\n\t\t\t{sanitize(userName)}\n\t\t</a>\n\t\t&nbsp;\n\t\t<span className={talkPageLink}>\n\t\t\t（\n\t\t\t<a href={mw.util.getUrl(`User_talk:${sanitize(userName)}`)} rel=\"noopener noreferrer\" target=\"_blank\">\n\t\t\t\t留言\n\t\t\t</a>\n\t\t\t）\n\t\t</span>\n\t</li>\n);\n\nconst onlineCountText: string = getMessage(' ($1 online):');\n\nconst GroupList = ({groupName, userNames}: GroupListProps) => (\n\t<div className={section}>\n\t\t<span>{groupName}</span>\n\t\t<span>{onlineCountText.replace('$1', String(userNames.length))}</span>\n\t\t<ul className={sectionList}>\n\t\t\t{userNames.map((user) => (\n\t\t\t\t<UserLink key={user} userName={user} />\n\t\t\t))}\n\t\t</ul>\n\t</div>\n);\n\nconst groupListElement = (groupName: string, userNames: string[]) => (\n\t<GroupList groupName={groupName} userNames={userNames} />\n);\n\nconst listTitle = () => <p>{getMessage('OnlineWithin30Minutes')}</p>;\n\nexport {groupListElement, listTitle};\n", "import {localize} from 'ext.gadget.i18n';\n\nconst getI18nMessages = () => {\n\treturn {\n\t\t' ($1 online):': localize({\n\t\t\ten: ' ($1 online):',\n\t\t\tja: '（$1人オンライン中）：',\n\t\t\t'zh-hans': '（$1个在线）：',\n\t\t\t'zh-hant': '（$1個在線）：',\n\t\t}),\n\t\tAdmins: localize({\n\t\t\ten: 'Admins',\n\t\t\tja: '管理者',\n\t\t\t'zh-hans': '管理员',\n\t\t\t'zh-hant': '管理員',\n\t\t}),\n\t\tPatrollers: localize({\n\t\t\ten: 'Patrollers',\n\t\t\tja: '巡回者',\n\t\t\t'zh-hans': '巡查员',\n\t\t\t'zh-hant': '巡查員',\n\t\t}),\n\t\tStewards: localize({\n\t\t\ten: 'Stewards',\n\t\t\tja: 'スチュワード',\n\t\t\t'zh-hans': '裁决委员',\n\t\t\t'zh-hant': '裁決委員',\n\t\t}),\n\t\t'Network error': localize({\n\t\t\ten: 'Network error',\n\t\t\tja: 'ネットワークエラー',\n\t\t\t'zh-hans': '网络异常',\n\t\t\t'zh-hant': '網路異常',\n\t\t}),\n\t\tNoOnline: localize({\n\t\t\ten: 'Currently there are no high privilege users online',\n\t\t\tja: '現在、高権限利用者はオンラインにいません',\n\t\t\t'zh-hans': '目前没有站务人员在线',\n\t\t\t'zh-hant': '目前沒有站務人員在線',\n\t\t}),\n\t\tOnline: localize({\n\t\t\ten: 'Online admins',\n\t\t\tja: 'オンラインの高権限利用者',\n\t\t\t'zh-hans': '在线站务人员',\n\t\t\t'zh-hant': '線上站務人員',\n\t\t}),\n\t\tOnlineWithin30Minutes: localize({\n\t\t\ten: 'High privilege users online within 30 minutes:',\n\t\t\tja: '30分以内にオンラインで高権限利用者：',\n\t\t\t'zh-hans': '下面是最近30分钟内在线的站务人员：',\n\t\t\t'zh-hant': '下面是最近30分鐘內的線上站務人員：',\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", "const BLACK_LIST: string[] = ['滥用过滤器'];\n\nexport {BLACK_LIST};\n", "{\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(`OnlineAdmins/${OPTIONS.version}`);\n\nexport {api};\n", "import {groupListElement, listTitle} from './components/groupList';\nimport {BLACK_LIST} from './constant';\nimport {api} from './api';\nimport {getMessage} from './i18n';\nimport {uniqueArray} from 'ext.gadget.Util';\n\nconst queryRecentChanges = async (rcstart: string, rcend: string) => {\n\tconst params: ApiQueryRecentChangesParams = {\n\t\trcstart,\n\t\trcend,\n\t\taction: 'query',\n\t\tformat: 'json',\n\t\tformatversion: '2',\n\t\tlist: 'recentchanges',\n\t\trcprop: 'user',\n\t\trcshow: ['!bot', '!anon'],\n\t\trclimit: 500,\n\t};\n\tconst response = await api.post(params);\n\n\treturn response;\n};\n\nconst queryLogEvents = async (lestart: string, leend: string) => {\n\tconst params: ApiQueryLogEventsParams = {\n\t\tlestart,\n\t\tleend,\n\t\taction: 'query',\n\t\tformat: 'json',\n\t\tformatversion: '2',\n\t\tlist: 'logevents',\n\t\tleprop: 'user',\n\t\tlelimit: 500,\n\t};\n\tconst response = await api.post(params);\n\n\treturn response;\n};\n\nconst queryUserProps = async (ususers: string | string[]) => {\n\tconst params: ApiQueryUsersParams = {\n\t\tususers,\n\t\taction: 'query',\n\t\tformat: 'json',\n\t\tformatversion: '2',\n\t\tlist: 'users',\n\t\tusprop: 'groups',\n\t};\n\tconst response = await api.post(params);\n\n\treturn response;\n};\n\nconst doClick = async (event: JQuery.ClickEvent<HTMLAnchorElement>): Promise<void> => {\n\tevent.preventDefault();\n\n\tlet users: string[] = [];\n\tconst usersExt: string[] = [];\n\tconst stewards: string[] = [];\n\tconst admins: string[] = [];\n\tconst patrollers: string[] = [];\n\n\tconst time: Date = new Date();\n\tconst rcstart: string = time.toISOString();\n\ttime.setMinutes(time.getMinutes() - 30); // 最近更改30分钟内的编辑用户\n\tconst rcend: string = time.toISOString();\n\n\ttry {\n\t\tconst recentchanges = await queryRecentChanges(rcstart, rcend);\n\n\t\tfor (const {user} of recentchanges['query'].recentchanges as {user: string}[]) {\n\t\t\tusers[users.length] = user; // Replace `[].push()` to avoid polyfilling core-js\n\t\t}\n\n\t\tconst logevents = await queryLogEvents(rcstart, rcend);\n\n\t\tfor (const {user} of logevents['query'].logevents as {user: string}[]) {\n\t\t\tusersExt[usersExt.length] = user;\n\t\t}\n\n\t\t// 用户名列表合并、去重、分割\n\t\tusers = uniqueArray([...users, ...usersExt]); // Replace `[...new Set()]` to avoid polyfilling core-js\n\n\t\tconst promises: (() => Promise<void>)[] = [];\n\n\t\tfor (let i = 0; i < users.length; i++) {\n\t\t\tconst ususers = users.splice(0, 50);\n\t\t\tif (!ususers.length) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tpromises[promises.length] = async (): Promise<void> => {\n\t\t\t\tconst response = await queryUserProps(ususers);\n\n\t\t\t\tfor (const {groups, name} of response['query'].users as {groups: string[]; name: string}[]) {\n\t\t\t\t\t// 找到管理人员，去除机器人，消除name的空值\n\t\t\t\t\tif (groups.includes('bot') || BLACK_LIST.includes(name) || !name) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\t\t\t\t\tif (groups.includes('steward')) {\n\t\t\t\t\t\tstewards[stewards.length] = name;\n\t\t\t\t\t}\n\t\t\t\t\tif (groups.includes('sysop')) {\n\t\t\t\t\t\tadmins[admins.length] = name;\n\t\t\t\t\t}\n\t\t\t\t\tif (groups.includes('patroller')) {\n\t\t\t\t\t\tpatrollers[patrollers.length] = name;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t};\n\t\t}\n\n\t\tvoid (async () => {\n\t\t\t// 查询用户权限\n\t\t\tfor (const promise of promises) {\n\t\t\t\ttry {\n\t\t\t\t\tawait promise();\n\t\t\t\t} catch {}\n\t\t\t}\n\t\t})().then(() => {\n\t\t\tif (stewards.length + admins.length + patrollers.length) {\n\t\t\t\tconst elements: Element[] = [listTitle()];\n\n\t\t\t\tif (stewards.length) {\n\t\t\t\t\telements[elements.length] = groupListElement(getMessage('Stewards'), stewards);\n\t\t\t\t}\n\t\t\t\tif (admins.length) {\n\t\t\t\t\telements[elements.length] = groupListElement(getMessage('Admins'), admins);\n\t\t\t\t}\n\t\t\t\tif (patrollers.length) {\n\t\t\t\t\telements[elements.length] = groupListElement(getMessage('Patrollers'), patrollers);\n\t\t\t\t}\n\t\t\t\tvoid mw.notify($('<div>').append(elements), {tag: 'onlineAdmins'});\n\t\t\t} else {\n\t\t\t\tvoid mw.notify(getMessage('NoOnline'), {\n\t\t\t\t\ttag: 'onlineAdmins',\n\t\t\t\t\ttype: 'warn',\n\t\t\t\t});\n\t\t\t}\n\t\t});\n\t} catch {\n\t\tvoid mw.notify(getMessage('Network error'), {tag: 'onlineAdmins', type: 'error'});\n\t}\n};\n\nexport {doClick};\n", "import {doClick} from './modules/doClick';\nimport {getMessage} from './modules/i18n';\n\nconst addPortletLink = (): void => {\n\t// Create portlet link\n\tconst portletId: 'p-cactions' | 'p-tb' = document.querySelector('#p-cactions') ? 'p-cactions' : 'p-tb';\n\tconst element: HTMLLIElement | null = mw.util.addPortletLink(portletId, '#', getMessage('Online'), 't-onlineadmin');\n\tif (!element) {\n\t\treturn;\n\t}\n\n\t// Bind click listener\n\t$(element)\n\t\t.find('a')\n\t\t.on('click', (event) => {\n\t\t\tvoid doClick(event);\n\t\t});\n};\n\n$(addPortletLink);\n"],
  "mappings": "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",
  "names": ["section", "sectionList", "talkPageLink", "import_ext_gadget2", "__toESM", "require", "import_ext_gadget", "getI18nMessages", "localize", "en", "ja", "Admins", "Patrollers", "Stewards", "NoOnline", "Online", "OnlineWithin30Minutes", "i18nMessages", "getMessage", "key", "sanitize", "string", "replace", "UserLink", "userName", "default", "createElement", "href", "mw", "util", "getUrl", "concat", "rel", "target", "className", "onlineCountText", "GroupList", "groupName", "userNames", "String", "length", "map", "user", "groupListElement", "listTitle", "BLACK_LIST", "version", "import_ext_gadget3", "api", "initMwApi", "import_ext_gadget4", "queryRecentChanges", "_ref", "_asyncToGenerator", "rcstart", "rcend", "params", "action", "format", "formatversion", "list", "rcprop", "rcshow", "rclimit", "response", "post", "_x", "_x2", "apply", "arguments", "queryLogEvents", "_ref2", "lestart", "leend", "leprop", "lelimit", "_x3", "_x4", "queryUserProps", "_ref3", "ususers", "usprop", "_x5", "doClick", "_ref4", "event", "preventDefault", "users", "usersExt", "stewards", "admins", "patrollers", "time", "Date", "toISOString", "setMinutes", "getMinutes", "recentchanges", "_iterator2", "_createForOfIteratorHelper", "_step2", "s", "n", "done", "value", "err", "e", "f", "logevents", "_iterator3", "_step3", "uniqueArray", "promises", "i", "splice", "_iterator4", "_step4", "groups", "name", "includes", "_i", "_promises", "promise", "then", "elements", "notify", "$", "append", "tag", "type", "_x6", "addPortletLink", "portletId", "document", "querySelector", "element", "find", "on"]
}
