QQ群管理者

QQ群管理者,一键导出QQ群成员信息,需要先进入QQ群官网!

目前为 2023-11-30 提交的版本。查看 最新版本

// ==UserScript==
// @name         QQ群管理者
// @namespace    http://www.bmqy.net/
// @version      3.0.1
// @author       bmqy
// @description  QQ群管理者,一键导出QQ群成员信息,需要先进入QQ群官网!
// @license      ISC
// @icon         
// @homepage     https://github.com/bmqy/qq-group-manager#readme
// @homepageURL  https://github.com/bmqy/qq-group-manager#readme
// @source       https://github.com/bmqy/qq-group-manager.git
// @supportURL   https://github.com/bmqy/qq-group-manager/issues
// @match        https://qun.qq.com/*
// @match        https://qun.qq.com/#/*
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/vue.global.prod.js
// @require      data:application/javascript,%3Bwindow.Vue%3DVue%3B
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/index.full.min.js
// @require      https://cdn.jsdelivr.net/npm/@element-plus/[email protected]/dist/index.iife.min.js
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/js.cookie.min.js
// @resource     animate.css                  https://cdn.jsdelivr.net/npm/[email protected]/animate.css
// @resource     element-plus/dist/index.css  https://cdn.jsdelivr.net/npm/[email protected]/dist/index.css
// @grant        GM_addStyle
// @grant        GM_getResourceText
// @grant        GM_info
// @grant        GM_setClipboard
// @grant        GM_xmlhttpRequest
// ==/UserScript==

(e=>{if(typeof GM_addStyle=="function"){GM_addStyle(e);return}const a=document.createElement("style");a.textContent=e,document.head.append(a)})(" .bmqyQQGroupManagerOpen[data-v-0ec73a95]{position:fixed;top:110px;right:100px}.bmqyQQGroupManagerBox[data-v-0ec73a95]{width:300px;position:fixed;top:110px;right:100px;border-radius:10px;z-index:999}.bmqyQQGroupManagerBox .margin-top[data-v-0ec73a95]{margin-top:15px}.bmqyQQGroupManagerBox[data-v-0ec73a95] .el-card__header .title{font-size:14px;font-weight:600}.bmqyQQGroupManagerBox[data-v-0ec73a95] .el-card__header .el-button--text{padding:0}.bmqyQQGroupManagerBox[data-v-0ec73a95] fieldset{margin-top:10px;padding:10px;border:1px solid #ddd}.bmqyQQGroupManagerBox[data-v-0ec73a95] fieldset legend b{font-size:14px}.bmqyQQGroupManagerBox p[data-v-0ec73a95],.bmqyQQGroupManagerBox[data-v-0ec73a95] .el-checkbox__label{font-size:12px;line-height:28px}.bmqyQQGroupManagerBox .el-progress[data-v-0ec73a95]{position:absolute;line-height:1;bottom:0;z-index:9;left:0;right:1px}.bmqyQQGroupManagerBox .flex[data-v-0ec73a95]{display:flex;justify-content:space-between;align-items:center}.bmqyQQGroupManagerBox .bmqyQQGroupBaseInfo[data-v-0ec73a95]{font-size:12px}.bmqyQQGroupManagerBox .qrcodeCont[data-v-0ec73a95]{background-color:#fff;position:absolute;right:85px;top:67px;width:100px;border:1px solid #e6e6e6;border-radius:5px;box-shadow:0 0 1px #ccc}.bmqyQQGroupManagerBox .qrcodeCont img[data-v-0ec73a95]{width:100px;height:100px}.bmqyQQGroupManagerBox .qrcodeCont span[data-v-0ec73a95]{display:inline-block;text-align:center;color:#999;font-weight:400} ");

(function (vue, ElementPlus, ElementPlusIconsVue, Cookies) {
  'use strict';

  function _interopNamespaceDefault(e) {
    const n = Object.create(null, { [Symbol.toStringTag]: { value: 'Module' } });
    if (e) {
      for (const k in e) {
        if (k !== 'default') {
          const d = Object.getOwnPropertyDescriptor(e, k);
          Object.defineProperty(n, k, d.get ? d : {
            enumerable: true,
            get: () => e[k]
          });
        }
      }
    }
    n.default = e;
    return Object.freeze(n);
  }

  const ElementPlusIconsVue__namespace = /*#__PURE__*/_interopNamespaceDefault(ElementPlusIconsVue);

  const cssLoader = (e) => {
    const t = GM_getResourceText(e);
    return GM_addStyle(t), t;
  };
  cssLoader("element-plus/dist/index.css");
  cssLoader("animate.css");
  var _GM_info = /* @__PURE__ */ (() => typeof GM_info != "undefined" ? GM_info : void 0)();
  var _GM_setClipboard = /* @__PURE__ */ (() => typeof GM_setClipboard != "undefined" ? GM_setClipboard : void 0)();
  var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)();
  const _export_sfc = (sfc, props) => {
    const target = sfc.__vccOpts || sfc;
    for (const [key, val] of props) {
      target[key] = val;
    }
    return target;
  };
  const _withScopeId = (n) => (vue.pushScopeId("data-v-0ec73a95"), n = n(), vue.popScopeId(), n);
  const _hoisted_1 = {
    slot: "header",
    class: "clearfix"
  };
  const _hoisted_2 = { class: "title" };
  const _hoisted_3 = { style: { "color": "#909399", "font-size": "12px", "font-weight": "normal" } };
  const _hoisted_4 = {
    key: 0,
    style: { "color": "#f56c6c" }
  };
  const _hoisted_5 = { key: 1 };
  const _hoisted_6 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("legend", null, [
    /* @__PURE__ */ vue.createElementVNode("b", null, "1、选择一个QQ群:")
  ], -1));
  const _hoisted_7 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("legend", null, [
    /* @__PURE__ */ vue.createElementVNode("b", null, "群信息:")
  ], -1));
  const _hoisted_8 = {
    key: 0,
    style: { "color": "#f56c6c", "font-size": "12px" }
  };
  const _hoisted_9 = {
    key: 1,
    class: "bmqyQQGroupBaseInfo"
  };
  const _hoisted_10 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("legend", null, [
    /* @__PURE__ */ vue.createElementVNode("b", null, "2、选择导出项目:")
  ], -1));
  const _hoisted_11 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("legend", null, [
    /* @__PURE__ */ vue.createElementVNode("b", null, "3、选择导出方式:")
  ], -1));
  const _hoisted_12 = { class: "margin-top flex" };
  const _hoisted_13 = ["innerHTML"];
  const _hoisted_14 = { class: "qrcodeCont" };
  const _hoisted_15 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("p", null, [
    /* @__PURE__ */ vue.createElementVNode("span", null, [
      /* @__PURE__ */ vue.createElementVNode("img", {
        src: "",
        alt: "支付宝收款码"
      }),
      /* @__PURE__ */ vue.createElementVNode("br"),
      /* @__PURE__ */ vue.createTextVNode("支付宝打赏")
    ]),
    /* @__PURE__ */ vue.createElementVNode("span", null, [
      /* @__PURE__ */ vue.createElementVNode("img", {
        src: "",
        alt: "微信收款码"
      }),
      /* @__PURE__ */ vue.createElementVNode("br"),
      /* @__PURE__ */ vue.createTextVNode("微信打赏")
    ])
  ], -1));
  const _hoisted_16 = [
    _hoisted_15
  ];
  const _sfc_main = {
    __name: "app",
    setup(__props) {
      const { proxy } = vue.getCurrentInstance();
      const name = vue.ref("");
      const moreTips = vue.ref("如果对您有帮助,\n感谢您打赏^_^");
      vue.ref(false);
      const exportField = vue.ref([
        {
          name: "uin",
          title: "QQ号",
          checked: true,
          disabled: true
        },
        {
          name: "role",
          title: "群职务",
          checked: false
        },
        {
          name: "nick",
          title: "昵称",
          checked: false
        },
        {
          name: "card",
          title: "群名片",
          checked: false
        },
        {
          name: "g",
          title: "性别",
          checked: false
        },
        {
          name: "qage",
          title: "Q龄",
          checked: false
        },
        {
          name: "join_time",
          title: "入群时间",
          checked: false
        },
        {
          name: "lv",
          title: "等级(积分)",
          checked: false,
          disabled: true
        },
        {
          name: "last_speak_time",
          title: "最后发言",
          checked: false
        }
      ]);
      const exportMode = vue.ref([
        {
          key: "plain",
          val: "纯文本"
        },
        {
          key: "xlsx",
          val: "电子表格"
        }
      ]);
      const currentMode = vue.ref("plain");
      const btnText = vue.ref("开始");
      const btnLoading = vue.ref(false);
      const api = vue.ref({
        queryGroupList: "https://qun.qq.com/cgi-bin/qun_mgr/get_group_list",
        queryGrouopInfo: "https://qun.qq.com/cgi-bin/qun_mgr/search_group_members",
        queryGrouopMemberList: "https://qun.qq.com/cgi-bin/qun_mgr/search_group_members"
        //queryGrouopMemberList: 'https://apinew.bmqy.net/api/qq/',
        //queryGrouopPublish: 'https://qun.qq.com/cgi-bin/qiandao/sign/publish',
        //queryGrouopGalleryTemplate: 'https://qun.qq.com/cgi-bin/qiandao/gallery_template?gc=authorQGroup&time=替换文本',
        //queryFriendList: 'https://qun.qq.com/cgi-bin/qun_mgr/get_friend_list'
      });
      const qqbkn = vue.ref("");
      const groupList = vue.ref({});
      const currentGc = vue.ref("");
      const groupInfo = vue.ref({
        gc: 0,
        gn: "",
        owner: 0,
        max_count: 0,
        count: 0,
        adm_num: 0,
        levelname: null,
        mems: []
      });
      const percentage = vue.ref(0);
      const result = vue.ref("");
      const appShow = vue.ref(true);
      const showQrcode = vue.ref(false);
      const getqqbkn = () => {
        let skey = proxy.$cookie.get("skey");
        if (!skey) {
          qqbkn.value = "";
          return false;
        }
        let bkn = null;
        for (var e = skey, t = 5381, n = 0, o = e.length; o > n; ++n) {
          t += (t << 5) + e.charAt(n).charCodeAt();
        }
        bkn = 2147483647 & t;
        qqbkn.value = bkn;
      };
      const queryGroupList = () => {
        _GM_xmlhttpRequest({
          method: "POST",
          url: api.value.queryGroupList,
          data: `bkn=${qqbkn.value}`,
          responseType: "json",
          onload: function(xhr) {
            if (xhr.status == 200) {
              let res = xhr.response;
              if (res.errcode == 0 && res.ec == 0) {
                groupList.value = res;
              } else {
                qqbkn.value = "";
                proxy.$message.error("登录失效,请重新登录!");
              }
            } else {
              console.log("BmqyQQGroup: error!");
            }
          },
          onerror: function(xhr) {
            console.log("BmqyQQGroup: error!");
          }
        });
      };
      const onChangeGc = () => {
        setGroupBaseInfo();
        queryGroupInfo();
        percentage.value = 0;
      };
      const setGroupBaseInfo = () => {
        if (currentGc.value == "") {
          return false;
        }
        for (let k in groupList.value) {
          if (k == "create" || k == "join" || k == "manage") {
            let gl = groupList.value[k];
            gl.forEach((e) => {
              if (e.gc == currentGc.value) {
                groupInfo.value.owner = e.owner;
                groupInfo.value.gn = e.gn;
                return false;
              }
            });
          }
        }
      };
      const queryGroupInfo = () => {
        _GM_xmlhttpRequest({
          method: "POST",
          url: api.value.queryGrouopInfo,
          data: `bkn=${qqbkn.value}&gc=${currentGc.value}&st=0&end=0&sort=0&bkn=${qqbkn.value}`,
          responseType: "json",
          onload: function(xhr) {
            if (xhr.status == 200) {
              let res = xhr.response;
              if (res.ec === 0) {
                groupInfo.value.max_count = res.max_count;
                groupInfo.value.count = res.count;
                groupInfo.value.adm_num = res.adm_num;
                if (res.levelname) {
                  groupInfo.value.levelname = res.levelname;
                  exportField.value[7].disabled = false;
                } else {
                  exportField.value[7].checked = false;
                  exportField.value[7].disabled = true;
                }
              }
            } else {
              console.log("BmqyQQGroup: error!");
            }
          },
          onerror: function(xhr) {
            console.log("BmqyQQGroup: error!");
          }
        });
      };
      const start = async () => {
        doQueryGroupMemberList();
      };
      const doQueryGroupMemberList = async () => {
        let arr = [];
        let maxCount = groupInfo.value.count;
        let ps = 40;
        btnText.value = "加载中...";
        btnLoading.value = true;
        for (let i = 0; i * ps < maxCount; i++) {
          let b = i * ps + (i == 0 ? i : 1), e = i * ps + ps > maxCount ? maxCount : i * ps + ps;
          await proxy.$app.delay();
          console.log("🚀 ~ file: App.vue:213 ~ doQueryGroupMemberList ~ b, e:", b, e);
          let r = await queryGroupMemberList(b, e);
          arr.push(...r);
          percentage.value = parseInt(e / maxCount * 100);
        }
        btnText.value = "开始";
        btnLoading.value = false;
        groupInfo.value.mems = arr;
      };
      const queryGroupMemberList = async (b, e) => {
        let begin = b;
        let end = e;
        return new Promise((resolve, reject) => {
          _GM_xmlhttpRequest({
            method: "POST",
            url: api.value.queryGrouopMemberList,
            data: `bkn=${qqbkn.value}&gc=${currentGc.value}&st=${begin}&end=${end}&sort=0`,
            headers: {
              "Content-Type": "application/x-www-form-urlencoded"
            },
            responseType: "json",
            onload: function(xhr) {
              if (xhr.status == 200) {
                let res = xhr.response;
                if (res.ec == 0) {
                  resolve(res.mems);
                } else {
                  console.log("BmqyQQGroupError:", res);
                  reject("BmqyQQGroupError!");
                }
              } else {
                console.log("BmqyQQGroupError:", xhr.response);
                reject("BmqyQQGroupError!");
              }
            },
            onerror: function(xhr) {
              console.log("BmqyQQGroupError:", err);
              reject("BmqyQQGroupError!");
            }
          });
        });
      };
      const formatGroupMember = () => {
        groupInfo.value.mems.forEach((item, i) => {
          for (let i2 = 0; i2 < exportField.value.length; i2++) {
            let field = exportField.value[i2];
            let key = field.name;
            if (field.checked) {
              switch (key) {
                case "g":
                  item[key] = getGenderText(item[key]);
                  break;
                case "role":
                  item[key] = getRoleText(item[key]);
                  break;
                case "lv":
                  item[key] = getLvText(item[key]);
                  break;
                case "join_time":
                  item[key] = proxy.$app.formatDateTime(item[key], "YYYY-mm-dd HH:MM");
                  break;
                case "last_speak_time":
                  item[key] = proxy.$app.formatDateTime(item[key], "YYYY-mm-dd HH:MM");
                  break;
                default:
                  item[key] = item[key];
                  break;
              }
            }
          }
        });
        switch (currentMode.value) {
          case "plain":
            exportGroupMemberListToPlain();
            break;
          case "xlsx":
            exportGroupMemberListToXlsx();
            break;
        }
      };
      const exportGroupMemberListToPlain = () => {
        let memberList = groupInfo.value.mems;
        let k = enabledExportFieldCount.value;
        let sResult = "";
        for (let i = 0; i < memberList.length; i++) {
          for (let j = 0; j < exportField.value.length; j++) {
            let field = exportField.value[j];
            let l = 1;
            let key = field.name;
            if (field.checked) {
              if (l < k) {
                sResult += memberList[i][key] + "	";
              } else {
                sResult += memberList[i][key];
              }
              l++;
            }
          }
          if (i < memberList.length - 1) {
            sResult += "\r\n";
          }
        }
        try {
          _GM_setClipboard(sResult, {
            type: "text",
            mimetype: "text/plain"
          });
          result.value = "已复制到剪贴板!";
          proxy.$message.success("已复制到剪贴板!");
        } catch (e) {
          navigator.clipboard.writeText(sResult).then(() => {
            result.value = "已复制到剪贴板!";
            proxy.$message.success("已复制到剪贴板!");
          }).catch((err2) => {
            console.error("无法复制此文本:", err2);
          });
        }
      };
      const exportGroupMemberListToXlsx = () => {
        let memberList = groupInfo.value.mems;
        let k = enabledExportFieldCount.value;
        let str = ``;
        for (let i = 0; i < memberList.length; i++) {
          for (let j = 0; j < exportField.value.length; j++) {
            let field = exportField.value[j];
            let key = field.name;
            let l = 1;
            if (field.checked) {
              if (l < k) {
                str += `"${memberList[i][key]}	",`;
              } else {
                str += `"${memberList[i][key]}	"`;
              }
              l++;
            }
          }
          str += "\n";
        }
        let uri = "data:text/xlsx;charset=utf-8,\uFEFF" + encodeURIComponent(str);
        result.value = '<a href="' + uri + `" download="QQ群成员列表-${groupInfo.value.gn}.xlsx">已导出:点此下载</a>`;
        proxy.$message.success("已导出请下载!");
      };
      const getGenderText = (val) => {
        if (val == 0) {
          return "男";
        } else if (val == 1) {
          return "女";
        } else {
          return "未知";
        }
      };
      const getRoleText = (val) => {
        if (val == 0) {
          return "群主";
        } else if (val == 1) {
          return "管理员";
        } else {
          return "群员";
        }
      };
      const getLvText = (json) => {
        return groupInfo.value.levelname[json.level] + "(" + json.point + ")";
      };
      const btnDisabled = vue.computed(() => {
        return currentGc.value == "" || percentage.value == 100;
      });
      const enabledExportFieldCount = vue.computed(() => {
        let i = 0;
        for (let key in exportField.value) {
          if (exportField.value[key].checked) {
            i++;
          }
        }
        return i;
      });
      vue.watch(percentage, async (n, o) => {
        if (n == 100) {
          await formatGroupMember();
        }
      });
      vue.watch(currentMode, async (n, o) => {
        if (percentage.value == 100) {
          switch (n) {
            case "plain":
              exportGroupMemberListToPlain();
              break;
            case "xlsx":
              exportGroupMemberListToXlsx();
              break;
          }
        }
      });
      vue.onBeforeMount(() => {
        getqqbkn();
        queryGroupList();
      });
      return (_ctx, _cache) => {
        const _component_el_button = vue.resolveComponent("el-button");
        const _component_Tools = vue.resolveComponent("Tools");
        const _component_el_icon = vue.resolveComponent("el-icon");
        const _component_el_dropdown_item = vue.resolveComponent("el-dropdown-item");
        const _component_el_dropdown_menu = vue.resolveComponent("el-dropdown-menu");
        const _component_el_dropdown = vue.resolveComponent("el-dropdown");
        const _component_el_option = vue.resolveComponent("el-option");
        const _component_el_option_group = vue.resolveComponent("el-option-group");
        const _component_el_select = vue.resolveComponent("el-select");
        const _component_el_checkbox = vue.resolveComponent("el-checkbox");
        const _component_el_col = vue.resolveComponent("el-col");
        const _component_el_row = vue.resolveComponent("el-row");
        const _component_el_progress = vue.resolveComponent("el-progress");
        const _component_el_card = vue.resolveComponent("el-card");
        return vue.openBlock(), vue.createElementBlock("div", null, [
          vue.withDirectives(vue.createVNode(_component_el_button, {
            class: "bmqyQQGroupManagerOpen animate__animated animate__fadeInRight",
            type: "primary",
            icon: "Promotion",
            circle: "",
            onClick: _cache[0] || (_cache[0] = ($event) => appShow.value = true),
            title: "打开面板"
          }, null, 512), [
            [vue.vShow, !vue.unref(appShow)]
          ]),
          vue.withDirectives(vue.createVNode(_component_el_card, { class: "box-card bmqyQQGroupManagerBox animate__animated animate__bounce" }, {
            default: vue.withCtx(() => [
              vue.createElementVNode("div", _hoisted_1, [
                vue.createElementVNode("span", _hoisted_2, [
                  vue.createTextVNode(vue.toDisplayString(vue.unref(name)) + " ", 1),
                  vue.createElementVNode("span", _hoisted_3, vue.toDisplayString(`ver${_ctx.$app.getVersion()}`), 1)
                ]),
                vue.createVNode(_component_el_dropdown, {
                  style: { "float": "right", "padding": "3px 0" },
                  trigger: "click"
                }, {
                  dropdown: vue.withCtx(() => [
                    vue.createVNode(_component_el_dropdown_menu, null, {
                      default: vue.withCtx(() => [
                        vue.createVNode(_component_el_dropdown_item, {
                          icon: "CoffeeCup",
                          onMousemove: _cache[1] || (_cache[1] = ($event) => showQrcode.value = true),
                          onMouseout: _cache[2] || (_cache[2] = ($event) => showQrcode.value = _ctx.fasle),
                          title: vue.unref(moreTips)
                        }, {
                          default: vue.withCtx(() => [
                            vue.createTextVNode("请喝咖啡")
                          ]),
                          _: 1
                        }, 8, ["title"]),
                        vue.createVNode(_component_el_dropdown_item, {
                          icon: "CircleClose",
                          onClick: _cache[3] || (_cache[3] = ($event) => appShow.value = false)
                        }, {
                          default: vue.withCtx(() => [
                            vue.createTextVNode("关闭面板")
                          ]),
                          _: 1
                        })
                      ]),
                      _: 1
                    })
                  ]),
                  default: vue.withCtx(() => [
                    vue.createVNode(_component_el_button, { type: "text" }, {
                      default: vue.withCtx(() => [
                        vue.createVNode(_component_el_icon, null, {
                          default: vue.withCtx(() => [
                            vue.createVNode(_component_Tools)
                          ]),
                          _: 1
                        })
                      ]),
                      _: 1
                    })
                  ]),
                  _: 1
                })
              ]),
              vue.unref(qqbkn) == "" ? (vue.openBlock(), vue.createElementBlock("div", _hoisted_4, "请先登录你的QQ账号!")) : vue.createCommentVNode("", true),
              vue.unref(qqbkn) != "" ? (vue.openBlock(), vue.createElementBlock("div", _hoisted_5, [
                vue.createElementVNode("fieldset", null, [
                  _hoisted_6,
                  vue.createVNode(_component_el_select, {
                    modelValue: vue.unref(currentGc),
                    "onUpdate:modelValue": _cache[4] || (_cache[4] = ($event) => vue.isRef(currentGc) ? currentGc.value = $event : null),
                    filterable: "",
                    size: "mini",
                    placeholder: "请选择",
                    onChange: onChangeGc
                  }, {
                    default: vue.withCtx(() => [
                      vue.unref(groupList)["create"] ? (vue.openBlock(), vue.createBlock(_component_el_option_group, {
                        key: "create",
                        label: "创建的群"
                      }, {
                        default: vue.withCtx(() => [
                          (vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(vue.unref(groupList)["create"], (item) => {
                            return vue.openBlock(), vue.createBlock(_component_el_option, {
                              key: item.gc,
                              label: _ctx.$app.htmlDecode(item.gn),
                              "data-owner": item.owner,
                              value: item.gc
                            }, null, 8, ["label", "data-owner", "value"]);
                          }), 128))
                        ]),
                        _: 1
                      })) : vue.createCommentVNode("", true),
                      vue.unref(groupList)["join"] ? (vue.openBlock(), vue.createBlock(_component_el_option_group, {
                        key: "join",
                        label: "加入的群"
                      }, {
                        default: vue.withCtx(() => [
                          (vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(vue.unref(groupList)["join"], (item) => {
                            return vue.openBlock(), vue.createBlock(_component_el_option, {
                              key: item.gc,
                              label: _ctx.$app.htmlDecode(item.gn),
                              "data-owner": item.owner,
                              value: item.gc
                            }, null, 8, ["label", "data-owner", "value"]);
                          }), 128))
                        ]),
                        _: 1
                      })) : vue.createCommentVNode("", true),
                      vue.unref(groupList)["manage"] ? (vue.openBlock(), vue.createBlock(_component_el_option_group, {
                        key: "manage",
                        label: "管理的群"
                      }, {
                        default: vue.withCtx(() => [
                          (vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(vue.unref(groupList)["manage"], (item) => {
                            return vue.openBlock(), vue.createBlock(_component_el_option, {
                              key: item.gc,
                              label: _ctx.$app.htmlDecode(item.gn),
                              "data-owner": item.owner,
                              value: item.gc
                            }, null, 8, ["label", "data-owner", "value"]);
                          }), 128))
                        ]),
                        _: 1
                      })) : vue.createCommentVNode("", true)
                    ]),
                    _: 1
                  }, 8, ["modelValue"])
                ]),
                vue.createElementVNode("fieldset", null, [
                  _hoisted_7,
                  vue.unref(currentGc) == "" ? (vue.openBlock(), vue.createElementBlock("p", _hoisted_8, "请先选择一个群!")) : vue.createCommentVNode("", true),
                  vue.unref(currentGc) != "" ? (vue.openBlock(), vue.createElementBlock("div", _hoisted_9, [
                    vue.createElementVNode("p", null, "群名称:" + vue.toDisplayString(_ctx.$app.htmlDecode(vue.unref(groupInfo).gn)), 1),
                    vue.createElementVNode("p", null, "群号码:" + vue.toDisplayString(vue.unref(currentGc)), 1),
                    vue.createElementVNode("p", null, "群主QQ:" + vue.toDisplayString(vue.unref(groupInfo).owner), 1),
                    vue.createElementVNode("p", null, "群成员数:" + vue.toDisplayString(`${vue.unref(groupInfo).count}/${vue.unref(groupInfo).max_count}`), 1)
                  ])) : vue.createCommentVNode("", true)
                ]),
                vue.createElementVNode("fieldset", null, [
                  _hoisted_10,
                  vue.createVNode(_component_el_row, { gutter: 20 }, {
                    default: vue.withCtx(() => [
                      (vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(vue.unref(exportField), (item, index) => {
                        return vue.openBlock(), vue.createBlock(_component_el_col, {
                          span: 12,
                          offset: 0,
                          key: index
                        }, {
                          default: vue.withCtx(() => [
                            vue.createVNode(_component_el_checkbox, {
                              modelValue: item.checked,
                              "onUpdate:modelValue": ($event) => item.checked = $event,
                              value: item.name,
                              disabled: item.disabled
                            }, {
                              default: vue.withCtx(() => [
                                vue.createTextVNode(vue.toDisplayString(item.title), 1)
                              ]),
                              _: 2
                            }, 1032, ["modelValue", "onUpdate:modelValue", "value", "disabled"])
                          ]),
                          _: 2
                        }, 1024);
                      }), 128))
                    ]),
                    _: 1
                  })
                ]),
                vue.createElementVNode("fieldset", null, [
                  _hoisted_11,
                  vue.createVNode(_component_el_select, {
                    modelValue: vue.unref(currentMode),
                    "onUpdate:modelValue": _cache[5] || (_cache[5] = ($event) => vue.isRef(currentMode) ? currentMode.value = $event : null),
                    size: "mini",
                    placeholder: "请选择"
                  }, {
                    default: vue.withCtx(() => [
                      (vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(vue.unref(exportMode), (item, index) => {
                        return vue.openBlock(), vue.createBlock(_component_el_option, {
                          key: index,
                          label: item.val,
                          value: item.key,
                          "v-model": vue.unref(currentMode)
                        }, null, 8, ["label", "value", "v-model"]);
                      }), 128))
                    ]),
                    _: 1
                  }, 8, ["modelValue"])
                ]),
                vue.createVNode(_component_el_progress, {
                  percentage: vue.unref(percentage),
                  "show-text": false,
                  status: "success"
                }, null, 8, ["percentage"]),
                vue.createElementVNode("div", _hoisted_12, [
                  vue.createVNode(_component_el_button, {
                    type: "primary",
                    round: "",
                    disabled: vue.unref(btnDisabled),
                    loading: vue.unref(btnLoading),
                    onClick: start
                  }, {
                    default: vue.withCtx(() => [
                      vue.createTextVNode(vue.toDisplayString(vue.unref(btnText)), 1)
                    ]),
                    _: 1
                  }, 8, ["disabled", "loading"]),
                  vue.unref(percentage) == 100 ? (vue.openBlock(), vue.createElementBlock("div", {
                    key: 0,
                    innerHTML: vue.unref(result)
                  }, null, 8, _hoisted_13)) : vue.createCommentVNode("", true)
                ])
              ])) : vue.createCommentVNode("", true),
              vue.withDirectives(vue.createElementVNode("div", _hoisted_14, _hoisted_16, 512), [
                [vue.vShow, vue.unref(showQrcode)]
              ])
            ]),
            _: 1
          }, 512), [
            [vue.vShow, vue.unref(appShow)]
          ])
        ]);
      };
    }
  };
  const App = /* @__PURE__ */ _export_sfc(_sfc_main, [["__scopeId", "data-v-0ec73a95"]]);
  const app = vue.createApp(App);
  for (const [key, component] of Object.entries(ElementPlusIconsVue__namespace)) {
    app.component(key, component);
  }
  app.use(ElementPlus);
  app.config.globalProperties.$cookie = Cookies;
  app.config.globalProperties.$app = {
    getName() {
      return _GM_info["script"]["name"];
    },
    getNameSpace() {
      return _GM_info["script"]["namespace"];
    },
    getVersion() {
      return _GM_info["script"]["version"];
    },
    delay() {
      let ms = Math.floor(Math.random() * 3 + 1) * 1e3;
      return new Promise((resolve, reject) => setTimeout(resolve, ms));
    },
    htmlDecode(html) {
      if (html.length == 0)
        return "";
      html = html.replace(/&amp;/g, "&");
      html = html.replace(/&lt;/g, "<");
      html = html.replace(/&gt;/g, ">");
      html = html.replace(/&nbsp;/g, " ");
      html = html.replace(/&quot/g, "'");
      return html;
    },
    formatDateTime(timestamp, format) {
      let dt = new Date(timestamp * 1e3);
      let ret;
      const opt = {
        "Y+": dt.getFullYear().toString(),
        // 年
        "m+": (dt.getMonth() + 1).toString(),
        // 月
        "d+": dt.getDate().toString(),
        // 日
        "H+": dt.getHours().toString(),
        // 时
        "M+": dt.getMinutes().toString(),
        // 分
        "S+": dt.getSeconds().toString()
        // 秒
        // 有其他格式化字符需求可以继续添加,必须转化成字符串
      };
      for (let k in opt) {
        ret = new RegExp("(" + k + ")").exec(format);
        if (ret) {
          format = format.replace(ret[1], ret[1].length == 1 ? opt[k] : opt[k].padStart(ret[1].length, "0"));
        }
      }
      return format;
    }
  };
  app.mount(
    (() => {
      const $appRoot = document.createElement("div");
      $appRoot.id = "qqGroupManagerApp";
      document.body.appendChild($appRoot);
      return $appRoot;
    })()
  );

})(Vue, ElementPlus, ElementPlusIconsVue, Cookies);