// ==UserScript==
// @name 本地表格数据筛选
// @name:zh-CN 本地表格数据筛选
// @name:zh-TW 本地表格數據篩選
// @name:en Filter tabular data
// @namespace http://tampermonkey.net/
// @version 1.0.4
// @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';
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(
`<div id="messageBox" class="c-message"></div>`
);
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 thead = tableDOM.querySelector('thead');
const scrollDOM = utils.createNode(
`<div class="filter-table"></div>`
);
tableDOM.parentElement.insertBefore(scrollDOM, tableDOM);
tableDOM.remove();
scrollDOM.appendChild(tableDOM);
const btn = utils.createNode(
`<button class="open-filter-Dialog-btn" title="打开筛选弹窗">F</button>`
);
thead.appendChild(btn);
btn.onclick = () => showSearchDialog(tableDOM);
}
});
}
}
const showSearchDialog = (function () {
const weakMap = new WeakMap();
/**
* 从给定的表格 DOM 元素中解析表格数据。
*
* @param {Element} tableDOM - 要解析的表格 DOM 元素。
* @return {Object} 包含解析后的数据、表头、过滤器映射和表单 DOM 的对象。
*/
function parse(tableDOM) {
let header = []; // 表头数据
// 解析表头
tableDOM.querySelectorAll('thead tr').forEach((trDOM) => {
header.push(
Array.from(trDOM.querySelectorAll('th')).map((n) => {
return {
label: n.textContent,
rowspan: n.rowSpan ?? 1, // 高度
colspan: n.colSpan ?? 1, // 宽度
};
})
);
});
// 每次筛选时,动态解析表格数据,防止表格排序导致顺序变化
const getData = () => {
return Array.from(tableDOM.querySelectorAll('tbody tr')).map(
(trDOM) => {
return Array.from(trDOM.querySelectorAll('td')).map((n) => {
return n.textContent;
});
}
);
};
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;
}
}
});
});
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, getData, tableDOM });
return weakMap
.set(tableDOM, { 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(`
<div id="searchDialog" style="display: none">
<div class="searchDialog__header" draggable="true">
<label class="searchDialog__title">搜索</label>
<span class="closeBtn">✕</span>
</div>
<div class="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="setMaxHeight" value="表格最大高度开关"></input>
<input type="button" class="resetBtn" value="重置"></input>
<input type="button" class="closeBtn" value="取消"></input>
<input type="button" class="confirmBtn primary" value="确定"></input>
</div>
</div>
`);
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].searchDialog__notClose'
).checked;
document.dispatchEvent(
new CustomEvent('btnEvent', { detail: { type: 'confirm', notClose } })
);
} else if (tagName === 'INPUT' && target.type === 'checkbox') {
target.parentElement.classList.toggle('active');
} else if (className.includes('setMaxHeight')) {
document.dispatchEvent(
new CustomEvent('btnEvent', { detail: { type: 'setMaxHeight' } })
);
}
});
// 监听键盘按下事件
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, getData, tableDOM }) {
const formDOM = utils.createNode(`
<article class="filter_form">
<span class="add">添加</span>
<form></form>
</article>
`);
const inputDOM = utils.createNode(`
<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>
${[...filterMap.keys()]
.map((n) => `<option label="${n}" value="${n}"></option>`)
.join('')}
</select>
<input type="text"/>
<span class="del">删除</span>
</div>
`);
const form = formDOM.querySelector('form');
function formTrim() {
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();
});
}
/**
* 验证表单,检查所有输入字段是否有值。
*
* @return {Promise} 如果所有输入字段都有值,则解析;否则拒绝
*/
function validate() {
let flag = true;
form.childNodes.forEach((node) => {
const [checkbox, , , input] = node.children;
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 {
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 [checkbox, select1, select2, input] = node.children;
if (checkbox.checked) {
const rules = input.value.split(',').map((keyword) => ({
keyword: keyword.trim(),
colIndexs: Array.from(filterMap.get(select2.value)),
}));
switch (select1.value) {
case 'AND':
rulse_AND.push(...rules);
break;
case 'OR':
rulse_OR.push(...rules);
break;
case 'NOT':
rulse_NOT.push(...rules);
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)
);
})
: true;
return (
isVisible_NOT &&
(rulse_AND.length || rulse_OR.length
? isVisible_AND || isVisible_OR
: true)
);
};
/**
* 处理根据给定规则筛选表格行。如果提供了规则,
* 则根据规则筛选表格行,并显示成功消息以及筛选行数。
* 如果没有提供规则,则重置所有表格行的可见性,并显示成功消息以及全部行数。
*
* @return {void} 此函数不返回任何内容。
*/
function handleFilter() {
const data = getData();
const rules = getRules();
const trList = Array.from(tableDOM.querySelector('tbody').children);
let count = 0;
// 筛选
data.forEach((trData, i) => {
if (isVisible(trData, rules)) {
trList[i].style.visibility = 'visible';
count++;
} else {
trList[i].style.visibility = 'collapse';
}
});
utils.showMessage(`搜索成功,一共查询出 ${count} 数据`);
}
form.addEventListener('submit', async (event) => {
try {
event.preventDefault();
formTrim();
await validate();
handleFilter();
searchDialogDOM._hidden();
} catch (e) {
console.error(e);
}
});
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', async (event) => {
if (formDOM.parentElement) {
switch (event?.detail?.type) {
case 'confirm':
try {
formTrim();
await validate();
handleFilter();
if (!event?.detail?.notClose) {
searchDialogDOM._hidden();
}
} catch (e) {
console.error(e);
}
break;
case 'reset':
form.innerHTML = '';
break;
case 'setMaxHeight':
tableDOM.parentElement.classList.toggle('scroll-bar')
? utils.showMessage('设置滚动条')
: utils.showMessage('恢复原状');
break;
}
}
});
return formDOM;
}
// 放弃。todo (不定高)虚拟滚动
function renderTable(header, dataSource) {
const container = utils.createNode(`
<div class="c-table">
<div class="c-table__content scroll-bar">
<table>
<colgroup></colgroup>
<thead></thead>
<tbody></tbody>
</table>
</div>
</div>
`);
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) => `<td><p>${n}</p></td>`).join('');
tbDOM.appendChild(trDOM);
});
return container;
}
/**
* 渲染搜索对话框和页面上其他元素的CSS样式。
*
* @return {void} 此函数不返回任何内容。
*/
function renderCSS() {
let style = utils.createNode(`
<style>
/* 滚动条样式 */
.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;
}
/* 表格样式 */
.filter-table.scroll-bar {
max-height: 80vh;
}
.filter-table table {
}
.filter-table table thead {
position: sticky;
top: 0;
}
.filter-table button.open-filter-Dialog-btn {
position: absolute;
top: 0;
left: 0;
width: 20px;
height: 20px;
line-height: 20px;
padding: 0 4px;
background-color: #fff;
border: 1px solid #409eff;
}
/* 弹窗样式 */
#searchDialog {
font-size: 12px;
font-family: "Microsoft YaHei", sans-serif;
display: flex;
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;
}
#searchDialog .searchDialog__header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 10px 20px;
font-size: 18px;
}
#searchDialog .searchDialog__header span {
transition: color 0.3s;
cursor: pointer;
font-size: 20px;
}
#searchDialog .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 .searchDialog__footer {
padding: 10px;
display: flex;
justify-content: flex-end;
align-items: center;
user-select: none;
}
#searchDialog .searchDialog__footer > label {
display: flex;
}
#searchDialog .searchDialog__footer > label > * {
cursor: pointer;
}
/* CSS 过渡效果 */
#searchDialog.fade-in {
animation: fadeIn 0.3s;
}
#searchDialog.fade-out {
animation: fadeOut 0.3s;
}
@keyframes fadeIn {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@keyframes fadeOut {
from {
opacity: 1;
}
to {
opacity: 0;
}
}
.c-message {
position: fixed;
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; /* 默认隐藏 */
}
/* 表单样式 */
#searchDialog article.filter_form {
font-size: 12px;
}
#searchDialog article.filter_form form {
position: relative;
margin-top: 4px;
}
#searchDialog article.filter_form .add,
#searchDialog article.filter_form .del {
user-select: none;
cursor: pointer;
}
#searchDialog article.filter_form form .form-example {
display: flex;
align-items: center;
margin: 0 -4px;
margin-bottom: 6px;
}
#searchDialog article.filter_form form .form-example > * {
margin: 0 4px;
}
#searchDialog article.filter_form form .form-example input[type='text'] {
flex: 1;
}
#searchDialog article.filter_form form .form-example input[type='text'].error {
border-color: red;
}
#searchDialog article.filter_form form .form-example input[type='text'].shake {
animation: shake 0.6s ease-in-out 1;
}
#searchDialog article.filter_form form .form-example:not(.active) * {
border-color: #d9d9d9;
color: #d9d9d9;
}
@keyframes shake {
0% { transform: translateX(0); }
25% { transform: translateX(-6px); }
50% { transform: translateX(4px); }
75% { transform: translateX(-2px); }
100% { transform: translateX(0); }
}
/* 组件样式 */
#searchDialog select,
#searchDialog input[type='text'] {
box-sizing: border-box;
height: 32px;
padding: 4px 11px;
border-radius: 4px;
border: 1px solid #d9d9d9;
transition: all 0.3s;
}
#searchDialog select:focus,
#searchDialog input:focus {
outline: none;
border-color: #4096ff;
border-inline-end-width: 1px;
}
#searchDialog select:hover,
#searchDialog input:hover {
border-color: #4096ff;
border-inline-end-width: 1px;
}
#searchDialog 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;
}
#searchDialog input[type='button']:hover {
background-color: #f5f7fa;
border-color: #409eff;
color: #409eff;
}
#searchDialog input[type='button'].primary {
background-color: #409eff;
border-color: #409eff;
color: #fff;
}
#searchDialog input[type='button'].primary:hover {
background-color: #66b1ff;
border-color: #66b1ff;
}
</style>
`);
document.head.appendChild(style);
}
init();
})();