// ==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);
});
})();