swagger查看Vo

实现swagger查看Vo

// ==UserScript==
// @name         swagger查看Vo
// @namespace    yoke
// @version      0.1.0
// @description  实现swagger查看Vo
// @author       yoke
// @match        *://*/*/swagger-ui/index.html*
// @license      MIT
// @require      https://cdnjs.cloudflare.com/ajax/libs/jquery/3.7.1/jquery.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/clipboard.js/2.0.11/clipboard.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/axios/1.7.3/axios.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.21/lodash.min.js
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/jquery.jsonview.min.js
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/clipboard.js/2.0.11/clipboard.min.js
// @resource viewCss  https://cdn.jsdelivr.net/npm/[email protected]/dist/jquery.jsonview.min.css
// @resource botCss https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css
// @icon         
// @grant        GM_addStyle
// @grant        GM_getResourceText
// ==/UserScript==
(function () {
    "use strict";
    $(function () {
        setTimeout(() => {
            function message(data) {
                // 创建消息提示框
                let messageBox = $(`
                <div id="message-box" style="opacity:0;position:fixed;top:0;left:50%;transform:translateX(-20%);background-color:#14CE66;color:#fff;padding:10px 20px;border-radius:5px;z-index:9999;display: flex;
    justify-content: center;
    align-items: center;">
                <svg t="1723080213021" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1842" width="16" height="16"><path d="M865.834667 225.834667a42.666667 42.666667 0 0 1 60.330666 60.330666l-476.586666 476.586667a42.666667 42.666667 0 0 1-60.373334 0l-291.413333-291.370667a42.666667 42.666667 0 1 1 60.373333-60.330666l261.205334 261.205333L865.834667 225.834667z" fill="#ffffff" p-id="1843"></path></svg>

<span style="margin-left:10px">${data}</span></div>`);
                $("body").append(messageBox);

                // 显示消息提示框,从上到下动画
                messageBox
                    .css("top", "-50px")
                    .animate({ top: "100px", opacity: 1 }, 200, function () {
                        // 2秒钟后隐藏消息提示框,从下到上动画
                        setTimeout(function () {
                            messageBox.animate(
                                { top: "-50px", opacity: 0 },
                                200,
                                function () {
                                    messageBox.remove();
                                }
                            );
                        }, 2000);
                    });
            }

            let doms = [];
            $(".opblock-summary").each(function (index, el) {
                let that = this;
                let dom = $(this).children(".view-line-link");
                let newCopy = dom
                    .clone()
                    .attr("title", "复制name")
                    .attr(
                        "data-clipboard-text",
                        $(this).find(".opblock-summary-description").text()
                    )
                    .attr("data-clipboard-action", "copy")
                    .addClass("copy_custom");
                dom.before(newCopy);
                doms.push(newCopy[0]); // 收集新创建的元素
            });

            // 创建选择器字符串
            let selector = doms
                .map((dom) =>
                    dom.className
                        .split(" ")
                        .map((cls) => `.${cls}`)
                        .join("")
                )
                .join(", ");
            var clipboard = new ClipboardJS(selector, {
                text: function (e) {
                    console.log(e);
                    return $(e).attr("data-clipboard-text");
                },
            });

            clipboard.on("success", function (e) {
                console.info("Action:", e.action);
                console.info("Text:", e.text);
                console.info("Trigger:", e.trigger);
                // 注销对象
                e.clearSelection();
                message("复制成功");
            });

            clipboard.on("error", function (e) {
                console.error("Action:", e.action);
                console.error("Trigger:", e.trigger);
            });
            function createModal(data) {
                $("body").append(`
<div class="modal fade" id="myModal" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true" data-bs-backdrop="static">
  <div class="modal-dialog modal-dialog-centered modal-xl">
    <div class="modal-content">
      <div class="modal-header">
        <h5 class="modal-title">当前Vo数据</h5>
        <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
      </div>
      <div class="modal-body">
        <div id="json" style="min-height:500px"></div>
      </div>
      <div class="modal-footer">
        <button type="button" class="btn btn-primary" data-bs-dismiss="modal">关闭</button>
      </div>
    </div>
  </div>
</div>
                 `);
            }
            createModal();

            const url = $(".info .url").text();
            if (!_.isEmpty(url)) {
                // 代码内部    引入bootstrap的css文件并加入html中
                const css = GM_getResourceText("viewCss");
                GM_addStyle(css);
                const botCss = GM_getResourceText("botCss");
                GM_addStyle(botCss);
                axios.interceptors.response.use((res) => {
                    return res.data;
                });

                axios(url).then((res) => {
                    console.log(res);

                    const data = res;

                    function getSchemas(
                        key,
                        value,
                        schema,
                        rootData,
                        visitedRefs = new Set()
                    ) {
                        // 处理 $ref 的情况
                        if (value.$ref) {
                            const refPath = value.$ref.split("/").at(-1);
                            console.log(refPath, visitedRefs);

                            // 如果已经访问过这个 $ref,说明存在循环引用
                            if (visitedRefs.has(refPath)) {
                                return `循环引用:${refPath}`;
                            }

                            // 将当前 $ref 标记为已访问
                            visitedRefs.add(refPath);

                            return parseSchema(refPath, rootData, visitedRefs);
                        } else {
                            // 判断是否必填
                            const isRequired = schema?.required?.includes(key)
                                ? "必填--"
                                : "";
                            return `${isRequired}${value.type}--${value.description ?? "没有字段描述?"}`;
                        }
                    }

                    // 递归解析并生成目标对象
                    function parseSchema(name, rootData, visitedRefs = new Set()) {
                        const result = {};
                        const schema = _.get(rootData, `components.schemas.${name}`);
                        console.log(`Parsing schema: ${name}`, schema, rootData);

                        if (schema?.properties) {
                            for (let [key, value] of Object.entries(schema.properties)) {
                                console.log(`Processing property: ${key}`, value);

                                if (value.type === "object") {
                                    // 递归处理嵌套对象
                                    result[key] = getSchemas(
                                        key,
                                        value,
                                        schema,
                                        rootData,
                                        visitedRefs
                                    );
                                } else if (value.type === "array") {
                                    if (!_.has(value.items, "type")) {
                                        // 处理数组类型,递归解析 items
                                        result[key] = [
                                            getSchemas(
                                                key,
                                                value.items,
                                                schema,
                                                rootData,
                                                visitedRefs
                                            ),
                                        ];
                                    }
                                    if (value.items?.type === "string" || value.items?.type === "integer") {
                                        result[key] = [value.type === "string" ? '1' : 1, value.type];
                                    }

                                } else if (
                                    value.type === "string" ||
                                    value.type === "integer"
                                ) {
                                    result[key] = `${value.type}--${value.description ?? "没有字段描述?"}`;
                                } else {
                                    // 处理基本类型
                                    result[key] = getSchemas(
                                        key,
                                        value,
                                        schema,
                                        rootData,
                                        visitedRefs
                                    );
                                }
                            }
                        }
                        return result;
                    }

                    function createDoms({ modelBox, x, y, name }) {
                        if (name) {
                            modelBox.css({ position: "relative" });
                            const doms =
                                $(`<span class="see" style="position:absolute;bottom:${y};right:${x};font-size:22px;cursor:pointer" data-vo-name=${name} data-bs-toggle="tooltip" data-bs-placement="top" data-bs-title="查看完整json"><svg xmlns="http://www.w3.org/2000/svg" width="22" height="22" fill="currentColor" class="bi bi-eye" viewBox="0 0 16 16">
  <path d="M16 8s-3-5.5-8-5.5S0 8 0 8s3 5.5 8 5.5S16 8 16 8zM1.173 8a13.133 13.133 0 0 1 1.66-2.043C4.12 4.668 5.88 3.5 8 3.5c2.12 0 3.879 1.168 5.168 2.457A13.133 13.133 0 0 1 14.828 8c-.058.087-.122.183-.195.288-.335.48-.83 1.12-1.465 1.755C11.879 11.332 10.119 12.5 8 12.5c-2.12 0-3.879-1.168-5.168-2.457A13.134 13.134 0 0 1 1.172 8z"/>
  <path d="M8 5.5a2.5 2.5 0 1 0 0 5 2.5 2.5 0 0 0 0-5zM4.5 8a3.5 3.5 0 1 1 7 0 3.5 3.5 0 0 1-7 0z"/>
</svg></span>`);
                            modelBox.append(doms);
                            new bootstrap.Tooltip(doms.get(0));
                            doms.click(function (e) {
                                $(".modal-title").text(name);
                                $("#json").JSONView(parseSchema(name, data), {
                                    collapsed: false,
                                });
                                $("#myModal").modal("show");
                            });
                        }
                    }

                    // 显示接口详情json
                    $(document).click(function (e) {
                        if (
                            e.target.nodeName == "BUTTON" &&
                            e.target.className == "tablinks" &&
                            e.target.getAttribute("data-name") == "model"
                        ) {
                            console.log(e.target);
                            const modelBox = $(e.target)
                                .parent() // 获取当前元素的上一级
                                .parent() // 获取上一级的上一级
                                .next() // 获取上一级的下一个兄弟元素
                                .find(".model-box");
                            const name = modelBox.find(".model-title__text").eq(0).text();
                            createDoms({
                                x: "8px",
                                y: "8px",
                                modelBox,
                                name,
                            });
                            console.log(modelBox, name);
                        }
                    });

                    $(".no-margin").delegate(
                        ".model-container",
                        "mouseenter",
                        function (e) {
                            console.log(e.currentTarget);
                            const modelBox = $(e.currentTarget);
                            const name = $(e.currentTarget).attr("data-name");
                            createDoms({
                                x: "33px",
                                y: "calc(50% - 19px)",
                                modelBox,
                                name,
                            });
                        }
                    );

                    $(".no-margin").delegate(
                        ".model-container",
                        "mouseleave",
                        function (e) {
                            const modelBox = $(e.currentTarget);
                            console.log(modelBox.find(".see").eq(0));
                            modelBox.find(".see").eq(0).remove();
                        }
                    );
                });
            }
        }, 5000);
    });
})();