// ==UserScript== // @name 本地表格数据筛选 // @name:zh-CN 本地表格数据筛选 // @name:zh-TW 本地表格數據篩選 // @name:en Filter tabular data // @namespace http://tampermonkey.net/ // @version 1.0.2 // @license GPL-3.0 // @author ShineByPupil // @description 获取标签的表格元素,根据表头形成筛选列表,本地对数据进行筛选 // @description:zh-CN 获取
标签的表格元素,根据表头形成筛选列表,本地对数据进行筛选 // @description:zh-TW 獲取
標簽的表格元素,根據表頭形成篩選列表,本地對數據進行篩選 // @description:en Obtain the table element of the
tag, form a filtering list based on the table header, and locally filter the data // @match *://*/* // @icon  // @noframes // @grant none // @downloadURL none // ==/UserScript== (function () { "use strict"; let searchDialogDOM = null; const utils = { /** * 在数组中查找第一个空单元的索引。 * * @param {Array} array - 要查找empty的数组。 * @return {number} 第一个empty的索引,如果没有找到空单元则返回数组的长度。 */ findEmptyIndex: function (array) { // 寻找第一个空单元的索引 const index = array.findIndex((_, i) => !(i in array)); // 如果找到空单元,则将新元素插入 if (index !== -1) { return index; } else { // 如果数组中没有空单元,则将新元素追加到数组末尾 return array.length; } }, messageBox: null, /** * 在屏幕上显示指定时间长度的消息。 * * @param {string} message - 要显示的消息。 * @param {number} [duration=2500] - 消息应显示的毫秒数。默认为2500毫秒。 * @return {void} 此函数不返回值。 */ showMessage: function (message, duration = 2500) { if (!this.messageBox) { this.messageBox = this.createNode( `
` ); document.body.appendChild(this.messageBox); } this.messageBox.textContent = message; this.messageBox.style.display = "block"; // 显示消息 // 设置一定时间后自动隐藏消息 setTimeout(() => { this.messageBox.style.display = "none"; }, duration); }, /** * 从提供的模板字符串创建一个新的 DOM 节点。 * * @param {string} template - 要创建节点的 HTML 模板字符串。 * @return {Node} 新创建的 DOM 节点。 */ createNode: function (template) { const div = document.createElement("div"); div.innerHTML = template.trim(); return div.firstChild; }, }; /** * 初始化函数,用于加载表格筛选。 * */ function init() { window.addEventListener("load", function (event) { console.log("加载表格筛选"); renderCSS(); findTable(); }); } /** * 在页面上查找所有的表格,并为每个表格添加一个按钮,当点击该按钮时,会打开搜索对话框。 * 该按钮位于表格的左上角。 * * @return {void} 该函数没有返回值。 */ function findTable() { const tableList = document.querySelectorAll("table"); if (tableList.length) { document.querySelectorAll("table").forEach((tableDOM) => { if ( tableDOM.querySelector("thead") && tableDOM.querySelector("tbody") ) { const btn = document.createElement("button"); btn.innerHTML = "F"; btn.title = "打开筛选弹窗"; btn.onclick = () => showSearchDialog(tableDOM); tableDOM.appendChild(btn); tableDOM.style.position = "relative"; const btn_style = { position: "absolute", top: "0", left: "0", with: "20px", height: "20px", lineHeight: "20px", padding: "0 4px", backgroundColor: "#fff", border: "1px solid #409eff", }; Object.keys(btn_style).forEach((key) => { btn.style[key] = btn_style[key]; }); } else { console.log("没有找到表格:", tableDOM); } }); } } const showSearchDialog = (function () { const weakMap = new WeakMap(); /** * 从给定的表格 DOM 元素中解析表格数据。 * * @param {Element} tableDOM - 要解析的表格 DOM 元素。 * @return {Object} 包含解析后的数据、表头、过滤器映射和表单 DOM 的对象。 */ function parse(tableDOM) { let data = []; // 表格数据 let header = []; // 表头数据 tableDOM.querySelectorAll("thead tr").forEach((trDOM) => { header.push( Array.from(trDOM.querySelectorAll("th")).map((n) => { return { label: n.innerText.replaceAll("\n", "
"), rowspan: n.rowSpan ?? 1, // 高度 colspan: n.colSpan ?? 1, // 宽度 }; }) ); }); tableDOM.querySelectorAll("tbody tr").forEach((trDOM) => { data.push( Array.from(trDOM.querySelectorAll("td")).map((n) => { return n.innerText.replaceAll("\n", "
"); }) ); }); let dp = new Array(header.length).fill(0).map((n) => new Array()); // 多级表头结构 header.forEach((tr, i) => { tr.forEach((td, j) => { const index = utils.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.replaceAll("
", ""); } } }); }); 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); } } } const formDOM = renderForm({ filterMap, data, tableDOM }); return weakMap .set(tableDOM, { data, header, filterMap, formDOM }) .get(tableDOM); } return function (tableDOM) { if (!searchDialogDOM) { searchDialogDOM = renderDialog(); document.body.appendChild(searchDialogDOM); } const { formDOM } = weakMap.has(tableDOM) ? weakMap.get(tableDOM) : parse(tableDOM); const content = searchDialogDOM.querySelector(".content"); content.childNodes.forEach((node) => { content.removeChild(node); }); content.appendChild(formDOM); searchDialogDOM._show(); }; })(); /** * 渲染一个带有搜索功能的对话框。 * * @return {HTMLElement} 渲染的对话框。 */ function renderDialog() { // 主体 const dialog = utils.createNode(` `); const header = dialog.querySelector(".searchDialog__header"); // 方法 dialog._show = () => { dialog.style.display = "block"; dialog.style.left = "calc(50% - 15vw)"; dialog.style.top = "10vh"; dialog.classList.remove("fade-out"); dialog.classList.add("fade-in"); }; dialog._hidden = () => { dialog.classList.add("fade-out"); dialog.classList.remove("fade-in"); dialog.onanimationend = () => { dialog.style.display = "none"; dialog.onanimationend = null; }; }; // 事件 dialog.addEventListener("click", (event) => { const { target, target: { className, tagName }, } = event; if (className.includes("closeBtn")) { // 关闭 dialog._hidden(); } else if (className.includes("resetBtn")) { // 重置 document.dispatchEvent( new CustomEvent("btnEvent", { detail: { type: "reset" } }) ); } else if (className.includes("confirmBtn")) { // 确定 const notClose = dialog.querySelector("input[type=checkbox]").checked; document.dispatchEvent( new CustomEvent("btnEvent", { detail: { type: "confirm", notClose } }) ); } }); // 监听键盘按下事件 document.addEventListener("keydown", (event) => { if (dialog.style.display === "block" && event.key === "Escape") dialog._hidden(); }); let offsetX, offsetY; header.addEventListener("dragstart", (event) => { event.dataTransfer.effectAllowed = "move"; // 获取拖动开始时鼠标相对于拖动元素的偏移 offsetX = event.clientX - header.getBoundingClientRect().left; offsetY = event.clientY - header.getBoundingClientRect().top; }); header.addEventListener("drag", function (event) { if (event.clientX && event.clientY) { // 计算拖动后的位置 const x = event.clientX - offsetX; const y = event.clientY - offsetY; // 设置拖动元素的新位置 dialog.style.left = x + "px"; dialog.style.top = y + "px"; } else { dialog.style.left = "calc(50% - 15vw)"; dialog.style.top = "10vh"; } }); header.addEventListener("dragover", function (event) { event.preventDefault(); }); return dialog; } /** * 根据提供的数据和表格 DOM 渲染一个带有筛选选项的表单。 * * @param {Object} filterMap - 筛选选项的映射。 * @param {Array} data - 用于筛选的数据。 * @param {HTMLElement} tableDOM - 表格的 DOM 元素。 * @return {HTMLElement} 带有筛选选项的表单的 DOM 元素。 */ function renderForm({ filterMap, data, tableDOM }) { const formDOM = utils.createNode(`
添加
`); const inputDOM = utils.createNode(`
删除
`); const form = formDOM.querySelector("form"); /** * 验证表单,检查所有输入字段是否有值。 * * @return {Promise} 如果所有输入字段都有值,则解析;否则拒绝 */ function validate() { let flag = true; form.childNodes.forEach((node) => { const [, , input] = node.children; if (!input.value) { flag = false; input.classList.add("error", "shake"); } else { input.classList.remove("error"); } }); return new Promise((resolve, reject) => { if (flag) { resolve(); } else { utils.showMessage("表单验证未通过"); reject(new Error("表单验证未通过")); } }); } /** * 从表单中获取过滤规则。 * * @return {Object} 包含过滤规则的对象。该对象有三个属性: * - rulse_AND:表示并且过滤规则的数组。每个对象有两个属性: * - keyword:表示要过滤的关键字的字符串。 * - colIndexs:表示要过滤的列索引的数组。 * - rulse_OR:表示或者过滤规则的数组。结构与rulse_AND相同。 * - rulse_NOT:表示非过滤规则的数组。结构与rulse_AND相同。 */ function getRules() { const rulse_AND = []; const rulse_OR = []; const rulse_NOT = []; form.childNodes.forEach((node) => { const [select1, select2, input] = node.children; switch (select1.value) { case "AND": rulse_AND.push({ keyword: input.value, colIndexs: Array.from(filterMap.get(select2.value)), }); break; case "OR": rulse_OR.push({ keyword: input.value, colIndexs: Array.from(filterMap.get(select2.value)), }); break; case "NOT": rulse_NOT.push({ keyword: input.value, colIndexs: Array.from(filterMap.get(select2.value)), }); break; } }); return { rulse_AND, rulse_OR, rulse_NOT, }; } /** * 根据给定的规则确定表格行是否可见。 * * @param {Array} trData - 表格行数据。 * @param {Object} rules - 过滤规则。 * @param {Array} rules.rulse_AND - AND 过滤规则。 * @param {Array} rules.rulse_OR - OR 过滤规则。 * @param {Array} rules.rulse_NOT - NOT 过滤规则。 * @param {Object} rules.rulse_AND[].rule - AND 过滤规则。 * @param {string} rules.rulse_AND[].rule.keyword - 过滤关键字。 * @param {Array} rules.rulse_AND[].rule.colIndexs - 过滤列索引。 * @param {Object} rules.rulse_OR[].rule - OR 过滤规则。 * @param {string} rules.rulse_OR[].rule.keyword - 过滤关键字。 * @param {Array} rules.rulse_OR[].rule.colIndexs - 过滤列索引。 * @param {Object} rules.rulse_NOT[].rule - NOT 过滤规则。 * @param {string} rules.rulse_NOT[].rule.keyword - 过滤关键字。 * @param {Array} rules.rulse_NOT[].rule.colIndexs - 过滤列索引。 * @return {boolean} 如果表格行可见,则为 true,否则为 false。 */ const isVisible = function (trData, rules) { const { rulse_AND, rulse_OR, rulse_NOT } = rules; const isVisible_AND = rulse_AND.length && rulse_AND.every((rule) => { const { keyword, colIndexs } = rule; return colIndexs.some((index) => trData?.[index]?.includes(keyword)); }); const isVisible_OR = rulse_OR.length && rulse_OR.some((rule) => { const { keyword, colIndexs } = rule; return colIndexs.some((index) => trData?.[index]?.includes(keyword)); }); const isVisible_NOT = rulse_NOT.length && rulse_NOT.every((rule) => { const { keyword, colIndexs } = rule; return !colIndexs.some((index) => trData?.[index]?.includes(keyword)); }); return isVisible_AND || isVisible_OR || isVisible_NOT; }; /** * 处理根据给定规则筛选表格行。如果提供了规则, * 则根据规则筛选表格行,并显示成功消息以及筛选行数。 * 如果没有提供规则,则重置所有表格行的可见性,并显示成功消息以及全部行数。 * * @return {void} 此函数不返回任何内容。 */ function handleFilter() { const rules = getRules(); const trList = Array.from(tableDOM.querySelector("tbody").children); let count = 0; if (Object.values(rules).flat().length) { // 筛选 data.forEach((trData, i) => { if (isVisible(trData, rules)) { trList[i].style.visibility = "visible"; count++; } else { trList[i].style.visibility = "collapse"; } }); utils.showMessage(`搜索成功,一共查询出 ${count} 数据`); } else { // 重置 trList.forEach((tr) => (tr.style.visibility = "visible")); utils.showMessage(`重置成功,一共查询出 ${trList.length} 条数据`); } } form.addEventListener("submit", (event) => { event.preventDefault(); validate() .then(() => { handleFilter(); searchDialogDOM._hidden(); }) .catch((e) => { console.error(e.message); }); }); formDOM.addEventListener("wheel", (event) => { if (event.target.tagName === "SELECT") { event.preventDefault(); const length = event.target.options.length; const index = event.target.selectedIndex; const direction = event.wheelDeltaY > 0 ? "up" : "down"; event.target.selectedIndex = direction === "up" ? index === 0 ? length - 1 : index - 1 : index === length - 1 ? 0 : index + 1; } }); formDOM.addEventListener("click", (event) => { if (event.target.className.includes("add")) { form.appendChild(inputDOM.cloneNode(true)); } else if (event.target.className.includes("del")) { // 删除规则 event.target.parentNode.remove(); } }); formDOM.addEventListener("animationend", (event) => { if (event.target.className.includes("shake")) { event.target.classList.remove("shake"); } }); formDOM.addEventListener("input", (event) => { if (event.target.tagName === "INPUT") { event.target.value ? event.target.classList.remove("error") : event.target.classList.add("error"); } }); document.addEventListener("btnEvent", (event) => { if (formDOM.parentElement) { switch (event?.detail?.type) { case "confirm": validate() .then(() => { handleFilter(); if (!event?.detail?.notClose) { searchDialogDOM._hidden(); } }) .catch((e) => { console.error(e.message); }); break; case "reset": form.innerHTML = ""; break; } } }); return formDOM; } // 放弃。todo (不定高)虚拟滚动 function renderTable(header, dataSource) { const container = utils.createNode(`
`); const tableDOM = container.querySelector("table"); const colgroupDOM = container.querySelector("colgroup"); const thDOM = container.querySelector("thead"); const tbDOM = container.querySelector("tbody"); // 根据表头计算colgroup let i = 23; while (i--) { let colDOM = document.createElement("col"); colDOM.setAttribute("width", "100px"); colgroupDOM.appendChild(colDOM); } window.colgroupDOM = colgroupDOM; tableDOM.style.width = 23 * 100 + "px"; // 渲染表头 header.forEach((tr) => { let trDOM = document.createElement("tr"); thDOM.appendChild(trDOM); tr.forEach((td) => { let thDOM = document.createElement("th"); trDOM.appendChild(thDOM); thDOM.innerHTML = td.label; thDOM.rowspan = td.rowspan; thDOM.colspan = td.colspan; thDOM.setAttribute("rowspan", td.rowspan); thDOM.setAttribute("colspan", td.colspan); }); }); // 渲染表体 dataSource.forEach((td) => { let trDOM = document.createElement("tr"); trDOM.innerHTML = td.map((n) => `

${n}

`).join(""); tbDOM.appendChild(trDOM); }); return container; } /** * 渲染搜索对话框和页面上其他元素的CSS样式。 * * @return {void} 此函数不返回任何内容。 */ function renderCSS() { let style = utils.createNode(` `); document.head.appendChild(style); } init(); })();