本地表格數據篩選

獲取<table>標簽的表格元素,根據表頭形成篩選列表,本地對數據進行篩選

// ==UserScript==
// @name         本地表格数据筛选
// @name:zh-CN   本地表格数据筛选
// @name:zh-TW   本地表格數據篩選
// @name:en      Filter tabular data
// @namespace    http://tampermonkey.net/
// @version      2.0.0
// @license      GPL-3.0
// @author       ShineByPupil
// @description  获取<table>标签的表格元素,根据表头形成筛选列表,本地对数据进行筛选
// @description:zh-CN  获取<table>标签的表格元素,根据表头形成筛选列表,本地对数据进行筛选
// @description:zh-TW  獲取<table>標簽的表格元素,根據表頭形成篩選列表,本地對數據進行篩選
// @description:en  Obtain the table element of the <table> tag, form a filtering list based on the table header, and locally filter the data
// @match        *://*/*
// @icon 
// @noframes
// @grant        none
// ==/UserScript==

(function () {
  "use strict";

  function findEmptyIndex(array) {
    // 寻找第一个空单元的索引
    const index = array.findIndex((_, i) => !(i in array));
    // 如果找到空单元,则将新元素插入
    if (index !== -1) {
      return index;
    } else {
      // 如果数组中没有空单元,则将新元素追加到数组末尾
      return array.length;
    }
  }
  // 渲染帧优化
  const rafDebounce = function (fn) {
    let rafId = null;

    return function (...args) {
      rafId && cancelAnimationFrame(rafId);
      rafId = requestAnimationFrame(() => {
        fn.apply(this, args);
        rafId = null;
      });
    };
  };

  class MessageBox extends HTMLElement {
    constructor() {
      super();

      this.attachShadow({ mode: "open" });
      this.shadowRoot.innerHTML = `
        <div class="message"></div>
        
        <style>
          .message {
            position: fixed;
            z-index: 100;
            top: 20px;
            left: 50%;
            transform: translateX(-50%);
            padding: 10px 20px;
            background-color: #333;
            color: #fff;
            border-radius: 5px;
            box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2);
            display: none; /* 默认隐藏 */
          }
        </style>
      `;

      this.message = this.shadowRoot.querySelector(".message");
    }

    show(message, duration = 2500) {
      this.message.textContent = message;
      this.message.style.display = "block"; // 显示消息

      // 设置一定时间后自动隐藏消息
      setTimeout(() => {
        this.message.style.display = "none";
      }, duration);
    }
  }

  class SearchDialog extends HTMLElement {
    /**
     * WeakMap 存储结构说明:
     *
     * 键:table 表格元素
     *
     * 值:SearchTable 表格状态容器
     */
    weakMap = new Map();
    form = null;
    startX = 0;
    startY = 0;
    initialX = 0;
    initialY = 0;
    x = 0;
    y = 0;

    constructor() {
      super();

      this.attachShadow({ mode: "open" });
      this.shadowRoot.innerHTML = `
        <div class="searchDialog">
          <div class="searchDialog__header">
            <label class="searchDialog__title">搜索</label>
            <span class="closeBtn">✕</span>
          </div>
    
          <div class="searchDialog__content scroll-bar"></div>
    
          <div class="searchDialog__footer">
            <label>
              <input class="searchDialog__notClose" type="checkbox" checked/>
              <span style="margin-left: 4px">查询后不关闭</span>
            </label>
            <input type="button" class="resetBtn" value="重置"/>
            <input type="button" class="closeBtn" value="取消"/>
            <input type="button" class="confirmBtn primary" value="确定"/>
          </div>
        </div>
        
        <style>
          .searchDialog {
            font-size: 12px;
            font-family: "Microsoft YaHei", sans-serif;
            display: none;
            z-index: 9999;
            flex-direction: column;
            width: 30vw;
            min-width: 600px;
            box-sizing: border-box;
            position: fixed;
            background-color: #fff;
            border: 1px solid #ddd;
            box-shadow: 0 0 10px rgba(0, 0, 0, 0.2);
            border-radius: 5px;
            user-select: none;
          }
          
          .searchDialog__header {
            display: flex;
            justify-content: space-between;
            align-items: center;
            padding: 10px 20px;
            font-size: 18px;
            cursor: move;
          }
          .searchDialog__header span {
            transition: color 0.3s;
            cursor: pointer;
            font-size: 20px;
          }
          .searchDialog__header span:hover {
            color: red;
          }
          
          .searchDialog__content {
            margin: 20px;
            padding: 0 10px;
            flex: 1;
            max-height: 400px;
            overflow-y: auto;
            overflow-x: hidden;
          }
          
          .searchDialog__footer {
            padding: 10px;
            display: flex;
            justify-content: flex-end;
            align-items: center;
            user-select: none;
          }
          .searchDialog__footer > label {
            display: flex;
          }
          .searchDialog__footer > label > * {
            cursor: pointer;
          }
          
          .fade-in {
            animation: fadeIn 0.3s;
          }
          .fade-out {
            animation: fadeOut 0.3s;
          }
          @keyframes fadeIn {
            from {
              opacity: 0;
            }
            to {
              opacity: 1;
            }
          }
          @keyframes fadeOut {
            from {
              opacity: 1;
            }
            to {
              opacity: 0;
            }
          }
          
          input[type='button'] {
            display: inline-block;
            padding: 4px 10px;
            border-radius: 4px;
            background-color: #fff;
            border: 1px solid #dcdfe6;
            color: #606266;
            text-decoration: none;
            cursor: pointer;
            margin-left: 10px;
          }
          input[type='button']:hover {
            background-color: #f5f7fa;
            border-color: #409eff;
            color: #409eff;
          }
          input[type='button'].primary {
            background-color: #409eff;
            border-color: #409eff;
            color: #fff;
          }
          input[type='button'].primary:hover {
            background-color: #66b1ff;
            border-color: #66b1ff;
          }
          
          .scroll-bar {
            overflow-y: scroll;
          }
          .scroll-bar::-webkit-scrollbar {
            margin: 10px;
            height: 10px;
            width: 10px;
            border: 2px solid #333; /* 设置滚动条的边框颜色 */
          }
          .scroll-bar::-webkit-scrollbar-track {
            background-color: #f1f1f1;
          }
          .scroll-bar::-webkit-scrollbar-thumb {
            background-color: #888;
            border-radius: 5px;
          }
          .scroll-bar::-webkit-scrollbar-thumb:hover {
            background-color: #555;
          }
        </style>
      `;

      this.dialog = this.shadowRoot.querySelector(".searchDialog");
      this.content = this.shadowRoot.querySelector(".searchDialog__content");
      this.notClose = this.dialog.querySelector(".searchDialog__notClose");

      this.init();
    }

    show(table) {
      if (!this.weakMap.has(table)) {
        this.weakMap.set(table, new SearchTable(table));
      }

      const searchFrom = this.weakMap.get(table).searchFrom;

      if (this.form !== searchFrom) {
        this.form?.remove();
        this.form = searchFrom;
        this.content.appendChild(searchFrom);
      }

      requestAnimationFrame(() => {
        this.dialog.style.display = "flex";
        this.dialog.style.left = "calc(50% - 15vw)";
        this.dialog.style.top = "10vh";
        this.dialog.classList.remove("fade-out");
        this.dialog.classList.add("fade-in");
      });
    }

    close() {
      this.dialog.classList.add("fade-out");
      this.dialog.classList.remove("fade-in");
      this.dialog.onanimationend = function () {
        this.style.display = "none";
        this.onanimationend = null;
      };
    }

    init() {
      this.initEvent();
      this.initTable();
    }

    initEvent() {
      // 点击事件 - footer 按钮组
      this.dialog.addEventListener("click", async (event) => {
        const {
          target,
          target: { className, tagName },
        } = event;

        if (className.includes("closeBtn")) {
          this.close(); // 关闭
        } else if (className.includes("resetBtn")) {
          this.form.reset(); // 重置
        } else if (className.includes("confirmBtn")) {
          await this.form.confirm(); // 确定

          !this.notClose.checked && this.close();
        } else if (tagName === "INPUT" && target.type === "checkbox") {
          target.parentElement.classList.toggle("active");
        }
      });

      // 键盘事件 - esc 关闭弹窗
      document.addEventListener("keydown", (event) => {
        if (this.dialog.style.display === "flex" && event.key === "Escape")
          this.close();
      });

      const startDrag = (event) => {
        this.startX = event.clientX;
        this.startY = event.clientY;
        this.initialX = this.x;
        this.initialY = this.y;

        document.addEventListener("mousemove", onDrag);
        document.addEventListener("mouseup", endDrag);
      };
      const onDrag = rafDebounce((event) => {
        event.preventDefault();

        const dx = event.clientX - this.startX;
        const dy = event.clientY - this.startY;
        this.x = this.initialX + dx;
        this.y = this.initialY + dy;

        this.dialog.style.transform = `translate(${this.x}px, ${this.y}px)`;
      });
      const endDrag = () => {
        document.removeEventListener("mousemove", onDrag);
        document.removeEventListener("mouseup", endDrag);
      };

      // 拖动事件
      this.dialog
        .querySelector(".searchDialog__header")
        .addEventListener("mousedown", startDrag);
    }

    initTable() {
      document.querySelectorAll("table").forEach((table) => {
        const thead = table.querySelector("thead");
        const tbody = table.querySelector("tbody");

        if (thead && tbody) {
          const parent = table.parentElement;
          parent.classList.add("filter-table");
          const template_style = document.createElement("template");

          table.classList.add("scroll-bar");
          template_style.innerHTML = `
            <style>
              table {
                position: relative;
                max-height: 50vh;
                overflow-y: auto !important;
              }
              .scroll-bar {
                overflow-y: scroll;
              }
              .scroll-bar::-webkit-scrollbar {
                margin: 10px;
                height: 10px;
                width: 10px;
                border: 2px solid #333; /* 设置滚动条的边框颜色 */
              }
              .scroll-bar::-webkit-scrollbar-track {
                background-color: #f1f1f1;
              }
              .scroll-bar::-webkit-scrollbar-thumb {
                background-color: #888;
                border-radius: 5px;
              }
              .scroll-bar::-webkit-scrollbar-thumb:hover {
                background-color: #555;
              }
            </style>
          `;
          document.head.appendChild(template_style.content);

          const wrapper = document.createElement("div");
          const shadow = wrapper.attachShadow({ mode: "open" });

          shadow.innerHTML = `
            <button class="open-filter-Dialog-btn" title="打开筛选弹窗">F</button>
            <style>
              button {
                position: absolute;
                top: 0;
                left: 0;
                width: 20px;
                height: 20px;
                line-height: 20px;
                padding: 0 4px;
                background-color: #fff;
                border: 1px solid #409eff;
                cursor: pointer;
              }
            </style>
          `;
          shadow.querySelector("button").onclick = () => this.show(table);

          thead.appendChild(wrapper);
        }
      });
    }
  }

  class SearchFrom extends HTMLElement {
    table = null;
    filterMap = null;

    constructor() {
      super();
      this.attachShadow({ mode: "open" });
      this.shadowRoot.innerHTML = `
        <article class="filter_form">
          <span class="add">添加</span>
    
          <form></form>
        </article>
        
        <style>
          .filter_form {
            font-size: 12px;
          }
          .filter_form form {
            position: relative;
            margin-top: 4px;
          }
          .filter_form .add,
          .filter_form .del {
            user-select: none;
            cursor: pointer;
          }
        </style>
      `;

      this.article = this.shadowRoot.querySelector("article");
      this.form = this.shadowRoot.querySelector("form");
      this.init();
    }

    init() {
      this.initEvent();
    }

    initEvent() {
      this.article.addEventListener("wheel", (event) => {
        const innerElement = event.composedPath()[0];

        if (innerElement.tagName === "SELECT") {
          event.preventDefault();

          const length = innerElement.options.length;
          const index = innerElement.selectedIndex;
          const direction = event.wheelDeltaY > 0 ? "up" : "down";

          innerElement.selectedIndex =
            direction === "up"
              ? index === 0
                ? length - 1
                : index - 1
              : index === length - 1
                ? 0
                : index + 1;
        }
      });
      this.article.addEventListener("click", (event) => {
        const innerElement = event.composedPath()[0];

        if (innerElement.className.includes("add")) {
          this.addSearchInput();
        } else if (innerElement.className.includes("del")) {
          // 删除规则
          innerElement.parentNode.remove();
        }
      });

      this.article.addEventListener("input", (event) => {
        const innerElement = event.composedPath()[0];

        if (innerElement.tagName === "INPUT") {
          innerElement.value
            ? innerElement.classList.remove("error")
            : innerElement.classList.add("error");
        }
      });
    }

    reset() {
      this.form.innerHTML = "";
    }

    async confirm() {
      try {
        this.formTrim();
        await this.validate();
        this.filter();
      } catch (e) {
        console.error(e);
      }
    }

    addSearchInput() {
      const searchInput = document.createElement("search-input");
      searchInput.setAttribute(
        "options",
        JSON.stringify([...this.filterMap.keys()]),
      );
      searchInput.filterMap = this.filterMap;
      this.form.appendChild(searchInput);
    }

    formTrim() {
      this.form.querySelectorAll("input[type=text]").forEach((input) => {
        input.value = input.value.includes(",")
          ? input.value
              .split(",")
              .map((n) => n.trim())
              .filter((n) => n)
              .join(", ")
          : input.value.trim();
      });
    }

    validate() {
      let flag = true;

      this.form.childNodes.forEach((node) => {
        const { checkbox, input } = node;
        if (checkbox.checked && !input.value) {
          flag = false;
          input.classList.add("error", "shake");
        } else {
          input.classList.remove("error");
        }
      });

      return new Promise((resolve, reject) => {
        if (flag) {
          resolve();
        } else {
          messageBox.show("表单验证未通过");
          reject(new Error("表单验证未通过"));
        }
      });
    }

    getRules() {
      const rules_AND = [];
      const rules_OR = [];
      const rules_NOT = [];

      this.form.childNodes.forEach((node) => {
        const { checkbox, select1, select2, input } = node;

        if (checkbox.checked) {
          const rules = input.value.split(",").map((keyword) => ({
            keyword: keyword.trim(),
            colIndexs: Array.from(this.filterMap.get(select2.value)),
          }));

          switch (select1.value) {
            case "AND":
              rules_AND.push(...rules);
              break;
            case "OR":
              rules_OR.push(...rules);
              break;
            case "NOT":
              rules_NOT.push(...rules);
              break;
          }
        }
      });

      return { rules_AND, rules_OR, rules_NOT };
    }

    filter() {
      const data = this.table.searchTable.data;
      const rules = this.getRules();
      const trList = Array.from(this.table.querySelector("tbody").children);
      let count = 0;

      // 筛选
      data.forEach((trData, i) => {
        if (this.isVisible(trData, rules)) {
          trList[i].style.display = "";
          count++;
        } else {
          trList[i].style.display = "none";
        }
      });

      messageBox.show(`搜索成功,一共查询出 ${count} 数据`);
    }
    // 根据给定的规则确定表格行是否可见
    isVisible(trData, rules) {
      const { rules_AND, rules_OR, rules_NOT } = rules;

      const isVisible_AND =
        rules_AND.length &&
        rules_AND.every((rule) => {
          const { keyword, colIndexs } = rule;
          return colIndexs.some((index) => trData?.[index]?.includes(keyword));
        });
      const isVisible_OR =
        rules_OR.length &&
        rules_OR.some((rule) => {
          const { keyword, colIndexs } = rule;
          return colIndexs.some((index) => trData?.[index]?.includes(keyword));
        });
      const isVisible_NOT = rules_NOT.length
        ? rules_NOT.every((rule) => {
            const { keyword, colIndexs } = rule;
            return !colIndexs.some((index) =>
              trData?.[index]?.includes(keyword),
            );
          })
        : true;

      return (
        isVisible_NOT &&
        (rules_AND.length || rules_OR.length
          ? isVisible_AND || isVisible_OR
          : true)
      );
    }
  }

  class SearchInput extends HTMLElement {
    static get observedAttributes() {
      return ["options"];
    }

    options = [];
    filterMap = null;

    constructor() {
      super();
      this.attachShadow({ mode: "open" });
      this.shadowRoot.innerHTML = `
        <div class="form-example active">
          <input type="checkbox" checked>
  
          <select>
            <option label="AND" value="AND"></option>
            <option label="OR" value="OR"></option>
            <option label="NOT" value="NOT"></option>
          </select>
    
          <select></select>
    
          <input type="text"/>
    
          <span class="del">删除</span>
        </div>
        
        <style>
          .form-example {
            display: flex;
            align-items: center;
            margin: 0 -4px 6px;
          }
          .form-example > * {
            margin: 0 4px;
          }
          .form-example:not(.active) * {
            border-color: #d9d9d9;
            color: #d9d9d9;
          }
          input[type='text'] {
            flex: 1;
          }
          input[type='text'].error {
            border-color: red;
          }
          input[type='text'].shake {
            animation: shake 0.6s ease-in-out 1;
          }
          
          select, input[type='text'] {
            box-sizing: border-box;
            height: 32px;
            padding: 4px 11px;
            border-radius: 4px;
            border: 1px solid #d9d9d9;
            transition: all 0.3s;
          }
          select:focus, input:focus {
            outline: none;
            border-color: #4096ff;
            border-inline-end-width: 1px;
          }
          select:hover, input:hover {
            border-color: #4096ff;
            border-inline-end-width: 1px;
          }
          
          @keyframes shake {
            0% { transform: translateX(0); }
            25% { transform: translateX(-6px); }
            50% { transform: translateX(4px); }
            75% { transform: translateX(-2px); }
            100% { transform: translateX(0); }
          }
        </style>
      `;

      this.example = this.shadowRoot.querySelector(".form-example");
      this.checkbox = this.shadowRoot.querySelector("input[type='checkbox']");
      this.select1 = this.shadowRoot.querySelector("select:nth-child(2)");
      this.select2 = this.shadowRoot.querySelector("select:nth-child(3)");
      this.input = this.shadowRoot.querySelector("input[type='text']");
      this.del = this.shadowRoot.querySelector(".del");

      this.init();
    }

    attributeChangedCallback(attrName, oldVal, newVal) {
      if (attrName === "options") {
        this.options = JSON.parse(newVal);
        this.select2.innerHTML = this.options
          .map((n) => `<option label="${n}" value="${n}"></option>`)
          .join("");
      }
    }

    init() {
      this.initEvent();
    }

    initEvent() {
      this.checkbox.onclick = () => {
        this.example.classList.toggle("active");
      };
      this.example.onanimationend = function (event) {
        const innerElement = event.composedPath()[0];

        if (innerElement.className.includes("shake")) {
          innerElement.classList.remove("shake");
        }
      };
    }
  }

  class SearchTable {
    constructor(table) {
      this.table = table;
      table.searchTable = this;

      const { header, filterMap } = this.parseTable(table);
      this.header = header;
      this.filterMap = filterMap;
      this.searchFrom = document.createElement("search-from");
      this.searchFrom.table = table;
      this.searchFrom.filterMap = filterMap;
    }

    parseTable(table) {
      const thead = table.querySelector("thead");
      if (!thead) {
        throw new Error("缺少表头");
      }

      let header = []; // 表头数据
      thead.querySelectorAll("tr").forEach((tr) => {
        header.push(
          Array.from(tr.querySelectorAll("th")).map((n) => {
            return {
              label: n.textContent,
              rowspan: n.rowSpan ?? 1, // 高度
              colspan: n.colSpan ?? 1, // 宽度
            };
          }),
        );
      });

      let dp = new Array(header.length).fill(0).map(() => []);
      header.forEach((tr, i) => {
        tr.forEach((td) => {
          const index = findEmptyIndex(dp[i]);
          const { colspan, rowspan } = td;

          for (let k = i; k < i + rowspan; k++) {
            for (let l = index; l < index + colspan; l++) {
              dp[k][l] ??= td.label;
            }
          }
        });
      });

      let filterMap = new Map(); // 过滤器映射
      for (let i = 0; i < dp.length; i++) {
        for (let j = 0; j < dp[i].length; j++) {
          if (dp[i][j]) {
            filterMap.has(dp[i][j]) || filterMap.set(dp[i][j], new Set());
            filterMap.get(dp[i][j]).add(j);
          }
        }
      }

      return { header, filterMap };
    }

    get data() {
      return Array.from(this.table.querySelectorAll("tbody > tr")).map((tr) => {
        return Array.from(tr.querySelectorAll("td")).map((n) => {
          return n.textContent;
        });
      });
    }
  }

  customElements.define("message-box", MessageBox);
  customElements.define("search-dialog", SearchDialog);
  customElements.define("search-from", SearchFrom);
  customElements.define("search-input", SearchInput);

  const messageBox = document.createElement("message-box");
  document.body.appendChild(messageBox);

  window.addEventListener("load", function (event) {
    const searchDialog = document.createElement("search-dialog");
    document.body.appendChild(searchDialog);
  });
})();