ES自定义解析

使用自定义的结构解析ES查询结果 使查询结果易于解读

// ==UserScript==
// @name         ES自定义解析
// @version      20250417001
// @description  使用自定义的结构解析ES查询结果 使查询结果易于解读
// @author       HP
// @match        *://*/app/discover*
// @namespace    http://tampermonkey.net/
// @license      WTFPL
// @icon         
// ==/UserScript==

let targetNode = document.body;
let observerOptions = {
    childList: true, // 观察目标子节点的变化,添加或删除
    subtree: true, //默认是false,设置为true后可观察后代节点
}

function waitPageLoad(mutationsList, observer) {
    let shutdown = false;
    mutationsList.forEach(mutation => {
        if (shutdown) {
            return;
        }
        mutation.addedNodes.forEach(node => {
            if (node.nodeType === Node.ELEMENT_NODE && node.matches('.euiDataGridRow')) {
                // 找到目标元素后,停止观察器
                observer.disconnect();
                shutdown = true;
                // 在此处插入你需要在找到该元素后执行的脚本
                console.log('目标已加载 添加按钮');
                addButton();
            }
        });
    });
}

function addButton() {
    // 获取包含动态生成div的父容器(确保这个容器不是动态生成的)
    const parentContainer = document.querySelector('.euiDataGrid__focusWrap'); // 假设整个数据网格是一个固定的父容器
    let logId = 0;

    // 创建一个 MutationObserver 实例
    const observer = new MutationObserver(mutationsList => {
        mutationsList.forEach(mutation => {
            mutation.addedNodes.forEach(node => {
                if (node.nodeType === Node.ELEMENT_NODE && node.matches('.euiDataGridRowCell__expandActions')) {
                    const newButton = document.createElement('button');
                    newButton.className = 'my_analyze euiButtonIcon euiButtonIcon--primary euiButtonIcon--empty euiButtonIcon--xSmall euiDataGridRowCell__actionButtonIcon';
                    newButton.style.backgroundColor = 'lightgreen';
                    newButton.addEventListener('click', function () {
                        const parentNode = this.parentNode.parentNode;
                        const firstDiv = parentNode.querySelector('div');
                        const firstDl = firstDiv.querySelector('dl');

                        for (let dtElement of firstDl.querySelectorAll('dt')) {
                            if (dtElement.textContent.trim() === 'message') {
                                const ddElement = dtElement.nextElementSibling;
                                const messageJson = JSON.parse(ddElement.innerHTML);
                                console.log(messageJson.data); // 打印dd元素的内容
                                navigator.clipboard.writeText(messageJson.data) // 写入剪切板

                                // 创建新的子div并设置内容
                                const thisLogDivId = `successTipDiv_${++logId}`;
                                const thisLogHeaderId = `successTipHeader_${++logId}`;
                                const thisLogSpanId = `successTipSpan_${++logId}`;
                                const logDiv = document.querySelector('.euiGlobalToastList.css-q9rn2p-euiGlobalToastList-right');
                                const successTipHtml = `<div  id=${thisLogDivId}><div aria-label="通知" data-test-subj="euiToastHeader"  id=${thisLogHeaderId}><span data-test-subj="euiToastHeader__title" id=${thisLogSpanId}>已解析并复制</span></div></div>`
                                logDiv.insertAdjacentHTML('beforeend', successTipHtml);
                                const successTipDiv = document.getElementById(thisLogDivId);
                                const successTipHeader = document.getElementById(thisLogHeaderId);
                                const successTipSpan = document.getElementById(thisLogSpanId);

                                // 5秒后关闭提示框
                                setTimeout(function () {
                                    logDiv.removeChild(successTipDiv)
                                }, 5000);

                                successTipDiv.style.borderRadius = '6px';
                                successTipDiv.style.boxShadow = 'rgba(0, 0, 0, 0.1) 0px 1px 5px, rgba(0, 0, 0, 0.07) 0px 3.6px 13px, rgba(0, 0, 0, 0.06) 0px 8.4px 23px, rgba(0, 0, 0, 0.05) 0px 23px 35px';
                                successTipDiv.style.position = 'relative';
                                successTipDiv.style.paddingInline = '16px';
                                successTipDiv.style.backgroundColor = 'rgb(255, 255, 255)';
                                successTipDiv.style.inlineSize = '100%';
                                successTipDiv.style.borderTop = '2px solid rgb(0, 119, 204)';
                                successTipDiv.style.marginBlockEnd = '16px';
                                successTipDiv.style.animation = '250ms cubic-bezier(0.694, 0.0482, 0.335, 1) 0s 1 normal none running animation-1idg3ww';
                                successTipDiv.style.opacity = '1';

                                successTipHeader.style.paddingInlineEnd = '24px';
                                successTipHeader.style.paddingTop = '16px';
                                successTipHeader.style.paddingBottom = '16px';
                                successTipHeader.style.paddingLeft = '25px';
                                successTipHeader.style.display = 'flex';
                                successTipHeader.style.alignItems = 'baseline';

                                successTipSpan.style.fontSize = '1.1429rem';
                                successTipSpan.style.lineHeight = '1.7143rem';
                                successTipSpan.style.color = 'rgb(26, 28, 33)';
                                successTipSpan.style.fontWeight = '700';
                            }
                        }
                    });

                    node.appendChild(newButton);
                }
            });
        });
    });

    // 开始观察指定的DOM节点及其子树的变化
    observer.observe(parentContainer, observerOptions);
}

// 开始观察指定的DOM节点及其子树的变化
let observer = new MutationObserver(waitPageLoad);
observer.observe(targetNode, observerOptions);