Greasy Fork is available in English.

剪贴板 FillText

快速填写预设文本到输入框,适用于所有https网站。

// ==UserScript==
// @name         剪贴板 FillText
// @namespace    http://tampermonkey.net/
// @version      1.13
// @description  快速填写预设文本到输入框,适用于所有https网站。
// @author       Unitiny
// @match        https://*/*
// @icon         
// @grant        unsafeWindow
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_addValueChangeListener
// @license MIT
// ==/UserScript==

let Global = {};
let GlobalDoms = {};
const GlobalKey = "fillTextGlobal";
const CSS = {
    flex: {
        bs: "space-between", start: "flex-start", end: "flex-end", center: "center", column: "column", row: "row"
    },
};
(function () {
    Global = getGlobal()

    Global.renderStatus = 1
    loadResource()
    initStyle()
    setInputDom()
    operationBoard()
    Global.renderStatus = 0

    getGlobalHook()
})();

/**
 * 初始化
 */
//设置input节点
function setInputDom() {
    let t = setInterval(() => {
        let downingKeys = [];
        let editKey = "";
        let inputDoms = document.querySelectorAll('input');
        inputDoms.forEach(function (dom, i) {
            dom.setAttribute(Global.attributeName, i)
            dom.addEventListener("click", function (event) {
                let d = event.target
                Global.curInputDom = d.getAttribute(Global.attributeName)
                if (Global.order === 1) {
                    d.value = getSpanText(Global.curSpan)
                }
            })
            dom.addEventListener("keydown", function (event) {
                //快捷键模式,按下Alt开始
                if (event.key === "Alt") {
                    downingKeys = []
                    downingKeys.push(event.key)
                    editKey = event.key
                    return
                }
                if (downingKeys.length > 0) {
                    if (downingKeys.indexOf(event.key) !== -1) return;
                    downingKeys.push(event.key)
                    editKey += "+" + event.key
                    getCurCateVal().spans.map(v => {
                        if (editKey === v.key) {
                            dom.value = getSpanText(v.index) || dom.value
                            event.preventDefault()
                            downingKeys = []
                            editKey = ""
                        }
                    })
                    return;
                }

                //编辑模式
                let str = dom.value + event.key
                getCurCateVal().spans.map(v => {
                    if (str === v.key.split("+").join("")) {
                        dom.value = getSpanText(v.index) || dom.value
                        event.preventDefault() //阻止写入该字符
                    }
                })
            })
            dom.addEventListener("keyup", function (event) {
                let i = downingKeys.indexOf(event.key)
                i !== -1 ? downingKeys.splice(i, 1) : null
                if (event.key === "Alt") {
                    downingKeys = []
                    editKey = ""
                }
            })
        })
        GlobalDoms.inputDoms = inputDoms
        //有拿到inputDoms
        if (inputDoms.length > 0) {
            clearInterval(t)
        }
    }, 1000)
}

/**
 * 工具函数
 */

function isOnline() {
    return !window.location.host.includes("localhost") && !window.location.host.includes("127.0.0.1")
}

//添加前缀来防止与原页面id重复
function prefixStr(str) {
    return `${Global.prefix}-${str}`
}

function getLink() {
    let link = document.createElement("link");
    link.rel = "stylesheet";
    link.href = "https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@24,400,0,0";
    return link
}

function loadResource() {
    document.head.appendChild(getLink());
}

function containKeys(a, b) {
    const aKeys = Object.keys(a);
    const bKeys = Object.keys(b);

    return aKeys.length >= bKeys.length && bKeys.every(key => aKeys.includes(key));
}

function domCtl(showDoms, hiddenDoms) {
    showDoms?.forEach(v => {
        showDom(v)
    })
    hiddenDoms?.forEach(v => {
        hiddenDom(v)
    })
}

function emptyFunc() {

}

function getCloseSpan() {
    let close = document.createElement("span")
    close.style = CSS.iconStyle
    close.style.cssText += `
    position: absolute;
    right: -9px;
    top: -14px;`
    close.innerText = "×"
    return close
}

/**
 * 样式区
 */

function initStyle() {
    CSS.parentBoardStyle = `
    ${flexStyle({jc: CSS.flex.start})}
    position: absolute;
    left: 0px;
    top: 0px;
    width: auto;
    overflow: auto;
    z-index:999;`
    CSS.boardStyle = `
    ${flexStyle({jc: CSS.flex.bs, fd: CSS.flex.column})}
    position: relative;
    left: 0px;
    top: 0px;
    width: 330px;
    height: 400px;
    margin: 10px;
    background: white;
    border: 5px solid ${themeColor()};
    border-radius: 5px;
    overflow: auto;
    z-index:999;`
    CSS.rightBoardStyle = `
    ${flexStyle({jc: CSS.flex.end, fd: CSS.flex.column})}
    position: relative;
    left: 0px;
    top: 0px;
    width: 330px;
    height: 400px;
    margin: 10px;
    background: white;
    border: 5px solid ${themeColor()};
    border-radius: 5px;
    overflow: auto;
    z-index:999;`
    CSS.hiddenBoardStyle = `
    ${flexStyle({})}
    position: absolute;
    left: 0px;
    top: 0px;
    width: 40px;
    height: 40px;
    margin: 10px;
    border: 5px solid skyblue;
    border-radius: 50%;
    overflow: auto;
    color: white;
    background: skyblue;
    z-index:999;
    display: none;`
    CSS.divStyle = `
                ${flexStyle({fd: CSS.flex.column})}
                width: 300px;
                height: 400px;
                margin: 10px;
                border: 1px solid skyblue;
                border-radius: 5px;
                overflow: auto`
    CSS.cateSpanStyle = `
                margin: 3px;
                padding: 5px 12px;
                width: auto;
                height: 35px;
                border: 1px solid ${themeColor()};
                border-radius: 5px;
                background: white;
                font-size: 16px;
                text-align: center;
                line-height: 33px;
                margin-right: 10px;`
    CSS.spanStyle = `
                ${flexStyle({jc: CSS.flex.center, ai: CSS.flex.start, fd: CSS.flex.column})}
                flex-grow: 0;
                flex-shrink: 0;
                min-width: 250px;
                max-width: 250px;
                min-height: 100px;
                max-height: 100px;
                margin: 10px 20px;
                border: 1px solid ${themeColor()};
                border-radius: 7px;
                height: 70px;
                padding: 0 10px 0 10px;`
    CSS.pStyle = `
    word-break: break-all;
    overflow: hidden;
    text-overflow: ellipsis;
    display: -webkit-box;
    -webkit-box-orient: vertical;
    -webkit-line-clamp: 2;`
    CSS.operationStyle = `
    padding: 0 12px 0 12px;
    margin: 5px;
    width: auto;
    height: 35px;
    border-radius: 5px;
    background: #4bb84b;
    color: white;
    font-size: 16px;
    text-align: center;
    line-height: 33px;
    margin-right: 10px;`
    CSS.textArea = `
    border: 1px solid ${themeColor()};
    border-radius: 5px;
    flex-grow: 0;
    flex-shrink: 0;
    min-width: 90%;
    max-width: 90%;
    min-height: 100px;
    max-height: 100px;
    margin: 10px 5px 10px 5px;`
    CSS.iconStyle = `
    font-family: 'Material Symbols Outlined';
    font-weight: normal;
    font-style: normal;
    font-size: 24px;
    line-height: 1;
    letter-spacing: normal;
    text-transform: none;
    display: inline-block;
    white-space: nowrap;
    word-wrap: normal;
    direction: ltr;
    -webkit-font-feature-settings: 'liga';
    -webkit-font-smoothing: antialiased;`
    CSS.spanSettingStyle = `
    width: 100%;
    ${flexStyle({jc: CSS.flex.end})}
    `
    CSS.displayNone = "display: none;"
    CSS.displayFlex = "display: flex;"
    CSS.borderRed = "border: 1px solid red;"
    CSS.borderSkyblue = "border: 1px solid skyblue;"
}

function themeColor() {
    return Global.theme ? "skyblue" : "black"
}

function isShow(show) {
    return show ? "" : CSS.displayNone
}

function showDom(dom, add = CSS.displayFlex) {
    dom.style.cssText = dom.style.cssText.replace(isShow(false), "")
    dom.style.cssText += add
}

function hiddenDom(dom) {
    dom.style.cssText += isShow(false)
}

function replaceStyle(dom, key, value) {
    let list = dom.style.cssText.split(";");
    for (let i = 0; i < list.length; i++) {
        if (list[i].includes(key + ":")) {
            list[i] = `${key}:${value}`
            break
        }
    }
    dom.style.cssText = list.join(";")
}

function flexStyle({jc = CSS.flex.center, ai = CSS.flex.center, fd = CSS.flex.row}) {
    return `display: flex;
            justify-content: ${jc};
            align-items: ${ai};
            flex-direction: ${fd};`
}


/**
 * Global
 */
function defaultGlobal() {
    return {
        id: 0,
        version: Date.now(),
        renderStatus: 0, //1渲染中
        order: 0,
        style: {},
        parentShow: 0,
        show: 0,
        theme: true,
        curSpan: 0,
        curInputDom: 0,
        prefix: "fillText",
        attributeName: "data-fillText",
        hiddenKey: "Alt+q",
        inputDoms: [],
        boardChildNodes: 3,
        curCate: "default",
        category: [
            {
                name: "default",
                setting: {},
                spans: randSpans()
            },
            {
                name: "常用",
                setting: {},
                spans: []
            }
        ]
    }
}

function checkGlobal(data) {
    return data && JSON.stringify(data) !== "{}" && containKeys(data, defaultGlobal())
}

function saveGlobalInterval() {
    setInterval(function () {
        saveGlobal()
    }, 1000 * 30)
}

function getGlobalHook() {
    if (!isOnline()) {
        return
    }
    GM_addValueChangeListener(GlobalKey, function(name, old_value, new_value, remote) {
        console.log(name, old_value, new_value, remote)

        if(remote) {
            Global = getGlobal()
            Global.renderStatus = 1

            //重新渲染
            document.body.removeChild(GlobalDoms.parentBoard)
            document.body.removeChild(GlobalDoms.hiddenBoard)
            loadResource()
            initStyle()
            setInputDom()
            operationBoard()
            Global.renderStatus = 0
        }
    })
}

function saveGlobal(g = Global) {
    if (!isOnline() || Global.renderStatus === 1) {
        return false
    }

    console.log("saveGlobal", g)
    let lastG = GM_getValue(GlobalKey)
    if (!checkGlobal(lastG)) {
        g.version = Date.now()
        GM_setValue(GlobalKey, g)
        return true
    }

    if (checkGlobal(g)) {
        g.version = Date.now()
        GM_setValue(GlobalKey, g)
        return true
    }
    return false
}

function getGlobal() {
    if (!isOnline()) {
        return defaultGlobal();
    }

    let data = GM_getValue(GlobalKey)
    if (!checkGlobal(data)) {
        saveGlobal(defaultGlobal())
        return defaultGlobal()
    }

    return data
}

/**
 * 分类
 */

function getCurCateVal() {
    for (const v of Global.category) {
        if (v.name === Global.curCate) {
            return v
        }
    }
    return Global.category[0]
}

function showCateInput() {
    let divDom = GlobalDoms.categoryAreaDom.getElementsByTagName("div")[0];
    if (divDom.getElementsByClassName("cateInput").length > 0) {
        return
    }

    let el = document.createElement("input");
    el.className = "cateInput"
    el.placeholder = "请输入内容,按回车添加"
    el.style.cssText = CSS.textArea
    replaceStyle(el, "border", `1px solid ${themeColor()}`)
    el.addEventListener("keydown", function (event) {
        if (event.key === "Enter") {
            if (addCategory(event.target.value)) {
                renderCateSpan()
                renderBoard(renderSpan())
                // alert("添加分类成功!")
                event.stopImmediatePropagation()
            }
        }
    })

    divDom.insertBefore(el, divDom.childNodes[0])
}

function addCategory(name) {
    if (name.length > 10) {
        alert("分类名过长")
        return false
    }
    let has = false
    Global.category.forEach((v, i) => {
        if (v.name === name) {
            alert("已存在该分类")
            has = true
            return
        }
    })

    if (has) return
    Global.category.push({
        name: name,
        spansText: [],
        setting: {},
        spans: []
    })
    Global.curCate = name
    saveGlobal()

    changeCurCateText()
    return true
}

function delCategory(name) {
    if (Global.category.length <= 1) {
        alert("必须有一个分类存在")
        return false
    }
    let index = 0
    Global.category.map((v, i) => {
        if (v.name === name) {
            index = i
        }
    })
    Global.category.splice(index, 1)
    if (name === Global.curCate) {
        Global.curCate = Global.category[0].name
    }
    saveGlobal()

    changeCurCateText()
    return true
}

function categoryList() {
    let divDom = GlobalDoms.categoryAreaDom.getElementsByTagName("div")[0];
    if (divDom.getElementsByClassName(prefixStr("categoryList")).length > 0) {
        return
    }

    let list = []
    for (let i = 0; i < Global.category.length; i++) {
        list.push(getCategorySpan(i, Global.category[i]))
    }
    let div = document.createElement("div")
    div.className = prefixStr("categoryList")
    div.setAttribute("id", prefixStr("categoryList"))
    div.style.cssText = `${flexStyle({jc: CSS.flex.start})}flex-flow:row wrap;width:80%; margin: 0 20px 10px 0px;`
    div.append(...list)

    divDom.insertBefore(div, divDom.childNodes[0])
    toBottom()
}

function getCategorySpan(i, cate) {
    let close = getCloseSpan()
    close.addEventListener("click", function (event) {
        let curCate = Global.curCate
        if (delCategory(cate.name)) {
            if (cate.name === curCate) {
                renderBoard(renderSpan())
            }
            renderCateSpan()
            event.stopImmediatePropagation()
        }
    })

    let span = document.createElement("span")
    span.innerText = cate.name
    span.style.cssText = CSS.cateSpanStyle
    span.style.cssText += `position: relative;`
    span.addEventListener("click", function (event) {
        Global.curSpan = 0
        Global.curCate = cate.name
        renderBoard(renderSpan())
        changeCurCateText()
    })

    span.appendChild(close)
    return span
}

function changeCurCateText() {
    document.getElementById(prefixStr("curCate")).innerText = `当前分类:${Global.curCate}`
}

function renderCateSpan() {
    let list = []
    for (let i = 0; i < Global.category.length; i++) {
        list.push(getCategorySpan(i, Global.category[i]))
    }

    let div = document.getElementById(prefixStr("categoryList"))
    div.innerHTML = ""
    div.append(...list)
    return div
}

/**
 * 文本栏
 */

//输入栏添加span
function showInput() {
    // if (GlobalDoms.board.childNodes.length > GlobalDoms.boardChildNodes + 1) {
    //     return
    // }
    let el = document.createElement("input");
    el.placeholder = "请输入内容,按回车添加"
    el.style.cssText = CSS.textArea
    replaceStyle(el, "border", `1px solid ${themeColor()}`)
    el.addEventListener("keydown", function (event) {
        if (event.key === "Enter") {
            addSpan(event.target.value, {index: getCurCateVal().spans.length})
            renderBoard(renderSpan())
            toBottom()
            saveGlobal()
        }
    })

    let divDom = GlobalDoms.textAreaDom.getElementsByTagName("div")[0];
    divDom.insertBefore(el, divDom.childNodes[0])

    toBottom()
}

function addSpan(text, {
    cancel = false,
    editKey = "",
    finish = false,
    index = 0,
    input = false,
    key = "",
    set = false,
    show = false
}) {
    let s = createSpanData(text, getCurCateVal().spans.length)
    s.show = show
    s.input = input
    s.set = set
    s.finish = finish
    s.cancel = cancel
    s.index = index
    s.key = key
    s.editKey = editKey
    getCurCateVal().spans.push(s)
}

function createSpanData(text, i = 0) {
    return {
        text: text,
        cancel: false,
        editKey: "",
        finish: false,
        index: i,
        input: false,
        key: "",
        set: false,
        show: false,
    }
}

function randSpans() {
    if (isOnline()) {
        return [
            createSpanData("账号: 123456"),
            createSpanData("密码: 123456"),
            createSpanData("点击输入框,再点击任意文本即可自动填写。也可设置快捷键填写。")
        ]
    }
    let list = []
    for (let i = 0; i < Math.round(Math.random() * 10 + 3); i++) {
        let s = ""
        for (let j = 0; j < Math.round(Math.random() * 50 + 10); j++) {
            s += Math.round(Math.random() * 10).toString()
        }
        list.push(createSpanData(s, i))
    }
    return list
}

function getSpanText(index) {
    return getCurCateVal().spans[index].text
}

function getSpans() {
    let list = []
    for (let i = 0; i < getCurCateVal().spans.length; i++) {
        list.push(getSpan(i, getSpanText(i)))
    }
    return list
}

function getSpan(i, text) {
    let span = document.createElement("span")
    span.tabIndex = 0
    span.innerHTML = `
<p style="${CSS.pStyle}">${text}</p>
<div style="${CSS.spanSettingStyle}${isShow(getSpanData(i).show)}">

<input id="settingInput${i}" style="width: 80%; height: 30px;${isShow(getSpanData(i).input)}" type="text" placeholder="请按键设置" readonly>

<span style="${CSS.iconStyle}${isShow(getSpanData(i).set)}">settings</span>

<span style="${CSS.iconStyle}${isShow(getSpanData(i).finish)}">done</span>

<span style="${CSS.iconStyle}${isShow(getSpanData(i).cancel)}">close</span>
</div>
`
    span.style.cssText = CSS.spanStyle
    replaceStyle(span, "border", `1px solid ${themeColor()}`)
    span.addEventListener("click", function (event) {
        selectSpan(i)
        span.removeEventListener("keydown", moveSpan)
        span.addEventListener("keydown", moveSpan)
        if (Global.order === 0) {
            let target = `${Global.attributeName}="${Global.curInputDom}"`
            let element = document.querySelector(`input[${target}]`);
            element.value = text
        }
    })

    let childSpan = span.getElementsByTagName("span")
    childSpan[0].addEventListener("click", function () {
        settingCtl({input: true, set: false, finish: true, cancel: true, index: i})
    })
    childSpan[1].addEventListener("click", function () {
        saveKey(i)
    })
    childSpan[2].addEventListener("click", function () {
        cancel(i)
    })
    return span
}

function selectSpan(i = Global.curSpan) {
    let spanList = getCurCateVal().spanList
    if (spanList.length === 0 || spanList.childNodes.length === 0) {
        return
    }

    let spanNodes = spanList.childNodes
    replaceStyle(spanNodes[Global.curSpan], "border", `1px solid ${themeColor()}`)

    Global.curSpan = i
    replaceStyle(spanNodes[Global.curSpan], "border", "1px solid red")
}

function moveSpan(event) {
    if (event.altKey) {
        let arrow = 0
        if (event.key === 'ArrowDown') {
            if (Global.curSpan >= getCurCateVal().spans.length - 1) {
                return
            }
            arrow = 1
        } else if (event.key === 'ArrowUp') {
            if (Global.curSpan <= 0) {
                return
            }
            arrow = -1
        }
        if (arrow !== 0) {
            let target = Global.curSpan + arrow
            let s = getCurCateVal().spans[Global.curSpan]
            getCurCateVal().spans.splice(Global.curSpan, 1)
            getCurCateVal().spans.splice(target, 0, s)
            exchangeSpanText(Global.curSpan, target)
            selectSpan(target)
            Global.curSpan = target
            // moveSpan无法多次触发问题,好bug,找了好久
            // 由于之前selectSpan()写法会先移除moveSpan事件,导致当前执行的moveSpan直接return了,
            // 无法触发后面添加moveSpan逻辑。
        }
    }
}

function exchangeSpanText(i, j) {
    let spanNodes = getCurCateVal().spanList.childNodes
    let a = spanNodes[i].getElementsByTagName("p")[0]
    let b = spanNodes[j].getElementsByTagName("p")[0]

    let t = a.innerText
    a.innerText = b.innerText
    b.innerText = t
}

function delSpan() {
    getCurCateVal().spans.splice(Global.curSpan, 1)
    Global.curSpan = 0
    renderBoard(renderSpan())
    saveGlobal()
}

function delAllSpan() {
    getCurCateVal().spans = []
    renderBoard(renderSpan())
    saveGlobal()
}

function renderSpan() {
    let div = document.createElement("div");
    div.setAttribute("id", "textSpanList")
    div.style.cssText = flexStyle({fd: CSS.flex.column})
    div.append(...getSpans())
    getCurCateVal().spanList = div
    return div
}

function getSpanData(index) {
    return index >= 0 ? getCurCateVal().spans[index] : getCurCateVal().setting
}

/**
 * 快捷键
 */

function changeOrder() {
    let d = document.getElementById(prefixStr("orderOperation"))
    d.innerText = d.innerText === "先填后选" ? "先选后填" : "先填后选"
    Global.order = Global.order ? 0 : 1
    saveGlobal()
}

function editKey() {
    settingCtl({show: !getCurCateVal().setting.show, set: !getCurCateVal().setting.set})
}

function settingCtl({show = true, input = false, set = true, finish = false, cancel = false, index = -1}) {
    if (index !== -1) {
        //单span控制
        let span = getCurCateVal().spans[index]
        getCurCateVal().spans[index].show = show
        getCurCateVal().spans[index].input = input
        getCurCateVal().spans[index].set = set
        getCurCateVal().spans[index].finish = finish
        getCurCateVal().spans[index].cancel = cancel
    } else {
        //全局控制
        getCurCateVal().setting.show = show
        getCurCateVal().setting.input = input
        getCurCateVal().setting.set = set
        getCurCateVal().setting.finish = finish
        getCurCateVal().setting.cancel = cancel
        getCurCateVal().spans.map(v => {
            v.show = show
            v.input = input
            v.set = set
            v.finish = finish
            v.cancel = cancel
        })
    }
    renderBoard(renderSpan())
    if (input) {
        listenKey(index)
    }

    //阻止冒泡
    let event = window.event || arguments.callee.caller.arguments[0];
    event.stopPropagation()
}

function listenKey(index) {
    let ss = getSpanData(index)
    let input = document.getElementById("settingInput" + index);
    input.focus()
    input.value = ss.key
    ss.editKey = ss.key
    let downingKeys = [] //按下未松开的keys
    input.addEventListener("keydown", function (event) {
        if (event.key === "Backspace") {
            //如果存在,则移除按下中的key
            let i = downingKeys.indexOf(event.key)
            i !== -1 ? downingKeys.splice(i, 1) : null

            let arr = ss.editKey.split("+")
            arr.pop()
            ss.editKey = arr.join("+")
        } else if (!downingKeys.includes(event.key)) {
            if (ss.editKey === "") {
                ss.editKey += event.key
            } else {
                ss.editKey += "+" + event.key
            }
            downingKeys.push(event.key)
        }
        input.value = ss.editKey
    });
    input.addEventListener("keyup", function (event) {
        if (event.key !== "Backspace") {
            let i = downingKeys.indexOf(event.key)
            i !== -1 ? downingKeys.splice(i, 1) : null
        }
    });
    let event = window.event || arguments.callee.caller.arguments[0];
    event.stopPropagation()
}

function saveKey(index) {
    let ss = getSpanData(index)
    let has = false
    getCurCateVal().spans.map(v => {
        if (v.key === ss.editKey && v.index !== ss.index) {
            alert(`文本 ${getSpanText(v.index)} 已设置该快捷键`)
            has = true
        }
    })
    if (!has) {
        ss.key = ss.editKey
        ss.editKey = ""
        settingCtl({index: index})
    }
    saveGlobal()
}

function cancel(index) {
    let ss = getSpanData(index)
    ss.editKey = ""
    settingCtl({index: index})
}

/**
 * 右侧操作面板
 */
function rightOperateBoard() {
    let b = document.createElement("div");
    b.style.cssText = CSS.rightBoardStyle
    b.setAttribute("id", "fillTextOperateBoard")

    GlobalDoms.textAreaDom = textArea()
    GlobalDoms.categoryAreaDom = categoryArea()
    GlobalDoms.settingAreaDom = settingArea()
    domCtl([], [GlobalDoms.textAreaDom, GlobalDoms.categoryAreaDom, GlobalDoms.settingAreaDom])
    b.appendChild(GlobalDoms.textAreaDom)
    b.appendChild(GlobalDoms.categoryAreaDom)
    b.appendChild(GlobalDoms.settingAreaDom)

    GlobalDoms.rightBoard = b
    return b
}

function textArea() {
    let operation = document.createElement("div")
    operation.style.width = "100%"
    operation.style.cssText += flexStyle({})

    operation.innerHTML = `
<div id="${prefixStr("bottomOperation")}">
    <div style="${flexStyle({jc: CSS.flex.start})}margin: 0 20px 10px 0px;flex-flow:row wrap;">
        <span style="${CSS.operationStyle}" >快捷键</span>
        <span style="${CSS.operationStyle}">添加</span>
        <span style="${CSS.operationStyle}background: #ea5c5c;">删除</span>
        <span style="${CSS.operationStyle}margin-right: 0px;background: #ea5c5c;">清空</span>
    </div>
    <div style="${flexStyle({jc: CSS.flex.start})}flex-flow:row wrap;">
        <span style="${CSS.operationStyle}">隐藏</span>
    </div>
</div>`

    let list = [editKey, showInput, delSpan, delAllSpan, domCtl]
    list.forEach((v, i) => {
        if (v.name === "domCtl") {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                v([], [GlobalDoms.rightBoard])
            })
            return
        }
        operation.getElementsByTagName("span")[i].addEventListener("click", v)
    })
    return operation
}

function categoryArea() {
    let operation = document.createElement("div")
    operation.style.width = "100%"
    operation.style.cssText += flexStyle({})
    operation.style.cssText += "flex-flow:row wrap;"

    operation.innerHTML = `
<div id="${prefixStr("bottomOperation")}">
    <div style="${flexStyle({jc: CSS.flex.start})}margin: 0 20px 10px 0px;flex-flow:row wrap;">
        <span style="${CSS.operationStyle}">添加分类</span>
        <span style="${CSS.operationStyle}margin-right: 0px;">分类列表</span>
        <span id="${prefixStr("curCate")}" style="${CSS.operationStyle}">当前分类:${Global.curCate}</span>
    </div>
    <div style="${flexStyle({jc: CSS.flex.start})}flex-flow:row wrap;">
        <span style="${CSS.operationStyle}">隐藏</span>
    </div>
</div>`

    let list = [showCateInput, categoryList, emptyFunc, domCtl]
    list.forEach((v, i) => {
        if (v.name === "domCtl") {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                v([], [GlobalDoms.rightBoard])
            })
            return
        }
        operation.getElementsByTagName("span")[i].addEventListener("click", v)
    })
    return operation
}

function settingArea() {
    let operation = document.createElement("div")
    operation.style.width = "100%"
    operation.style.cssText += flexStyle({})
    operation.style.cssText += "flex-flow:row wrap;"
    operation.innerHTML = `
<div id="${prefixStr("bottomOperation")}">
    <div style="${flexStyle({jc: CSS.flex.start})}margin: 0 20px 10px 0px;flex-flow:row wrap;">
        <span style="${CSS.operationStyle}">⬆</span>
        <span style="${CSS.operationStyle}">⬇</span>
        <span style="${CSS.operationStyle}">说明</span>
        <span id="${prefixStr("orderOperation")}" style="${CSS.operationStyle}">先填后选</span>
        <span id="${prefixStr("orderOperation")}" style="${CSS.operationStyle}margin-right: 0px;">编辑隐藏快捷键</span>
    </div>
    <div style="${flexStyle({jc: CSS.flex.start})}flex-flow:row wrap;">
        <span style="${CSS.operationStyle}">隐藏</span>
    </div>
</div>`

    let list = [changeBoardHeight, changeBoardHeight, doc, changeOrder, showEditKeyInput, domCtl]
    list.forEach((v, i) => {
        if (v.name === "changeBoardHeight") {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                i % 2 === 0 ? v(40) : v(-40)
            })
        } else if (v.name === "domCtl") {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                v([], [GlobalDoms.rightBoard])
            })
        } else if (v.name === "showEditKeyInput") {
            operation.getElementsByTagName("span")[i].addEventListener("click", v)
        } else {
            operation.getElementsByTagName("span")[i].addEventListener("click", v)
        }
    })
    return operation
}

function showEditKeyInput() {
    let divDom = GlobalDoms.settingAreaDom.getElementsByTagName("div")[0];
    if (divDom.getElementsByClassName("editKeyInput").length > 0) {
        return
    }

    let inputDom = document.createElement("div");
    inputDom.className = "editKeyInput"
    inputDom.innerHTML = `
    <input id="editKeyInput" style="width: 80%; height: 30px;" type="text" value="${Global.hiddenKey}"
           placeholder="请按键设置" readOnly>
    `
    divDom.insertBefore(inputDom, divDom.childNodes[0])
    listenHiddenKey()
}

function listenHiddenKey() {
    let input = GlobalDoms.settingAreaDom.getElementsByTagName("input")[0]
    input.focus()
    input.value = Global.hiddenKey

    let editKey = ""
    let downingKeys = [] //按下未松开的keys
    input.addEventListener("keydown", function (event) {
        if (event.key === "Backspace") {
            //如果存在,则移除按下中的key
            let i = downingKeys.indexOf(event.key)
            i !== -1 ? downingKeys.splice(i, 1) : null

            let arr = editKey.split("+")
            arr.pop()
            editKey = arr.join("+")
        } else if(event.key === "Enter") {
            let arr = editKey.split("+")
            if(arr.length === 2 && arr[0] === "Alt" && arr[1] !== "Alt") {
                Global.hiddenKey = editKey
                let divDom = GlobalDoms.settingAreaDom.getElementsByTagName("div")[0]
                divDom.removeChild(divDom.getElementsByTagName("div")[0])
                saveGlobal()
            } else {
                alert("只能设置为Alt+任意键,长度为2")
            }
        } else if (!downingKeys.includes(event.key)) {
            if (editKey === "") {
                editKey += event.key
            } else {
                editKey += "+" + event.key
            }
            downingKeys.push(event.key)
        }
        input.value = editKey
    });
    input.addEventListener("keyup", function (event) {
        if (event.key !== "Backspace") {
            let i = downingKeys.indexOf(event.key)
            i !== -1 ? downingKeys.splice(i, 1) : null
        }
    });
    let event = window.event || arguments.callee.caller.arguments[0];
    event.stopPropagation()
}

/**
 * 控制面板
 */

function changeBoardHeight(h) {
    GlobalDoms.board.style.height = parseInt(GlobalDoms.board.style.height) + h + "px"
}

function renderBoard(spanListDom) {
    GlobalDoms.board.replaceChild(spanListDom, GlobalDoms.board.childNodes[1])
    let l = GlobalDoms.board.childNodes.length
    if (l > GlobalDoms.boardChildNodes) {
        for (let i = GlobalDoms.boardChildNodes; i < l; i++) {
            GlobalDoms.board.removeChild(GlobalDoms.board.lastChild)
        }
    }
    selectSpan()
}

function modeCtl() {
    if (Global.theme) {
        Global.theme = false
        replaceStyle(GlobalDoms.board, "border", `5px solid ${themeColor()}`)
        getCurCateVal().spanList.childNodes.forEach(v => {
            replaceStyle(v, "border", `1px solid ${themeColor()}`)
        })
        return
    }
    Global.theme = true
    replaceStyle(GlobalDoms.board, "border", `5px solid ${themeColor()}`)
    getCurCateVal().spanList.childNodes.forEach(v => {
        replaceStyle(v, "border", `1px solid ${themeColor()}`)
    })
    saveGlobal()
}

function topOperation() {
    let operation = document.createElement("div")
    operation.style.width = "100%"
    operation.innerHTML = `
<div id="${prefixStr("topOperation")}" style="${flexStyle({jc: CSS.flex.end})}margin: 0 20px 10px 0px;">
    <span style="${CSS.iconStyle}font-size: 20px;">light_mode</span>
    <span style="${CSS.iconStyle}">expand_more</span>
    <span style="${CSS.iconStyle}">remove</span>
</div>`

    let childSpan = operation.getElementsByTagName("span")
    childSpan[0].addEventListener("click", modeCtl)
    childSpan[1].addEventListener("click", toBottom)
    childSpan[2].addEventListener("click", function () {
        parentBoardCtl(1)
    })
    return operation
}

function bottomOperation() {
    let operation = document.createElement("div")
    operation.style.width = "100%"
    operation.innerHTML = `
<div id="${prefixStr("bottomOperation")}">
    <div style="${flexStyle({jc: CSS.flex.end})}margin: 0 20px 10px 0px;">
        <span style="${CSS.operationStyle}" >快捷键</span>
        <span style="${CSS.operationStyle}">添加</span>
        <span style="${CSS.operationStyle}background: #ea5c5c;">删除</span>
        <span style="${CSS.operationStyle}margin-right: 0px;background: #ea5c5c;">清空</span>
    </div>
    <div style="${flexStyle({jc: CSS.flex.end})}margin: 0 20px 10px 0px;">
        <span style="${CSS.operationStyle}">说明</span>
        <span style="${CSS.operationStyle}">保存配置</span>
        <span id="${prefixStr("orderOperation")}" style="${CSS.operationStyle}margin-right: 0px;">先填后选</span>
    </div>
    <div style="${flexStyle({jc: CSS.flex.end})}margin: 0 20px 10px 0px;">
        <span style="${CSS.operationStyle}">⬆</span>
        <span style="${CSS.operationStyle}">⬇</span>
        <span style="${CSS.operationStyle}">添加分类</span>
        <span style="${CSS.operationStyle}margin-right: 0px;">分类列表</span>
    </div>
    <div style="${flexStyle({jc: CSS.flex.end})}margin: 0 20px 10px 0px;">
        <span id="${prefixStr("curCate")}" style="${CSS.operationStyle}margin-right: 0px;">当前分类:${Global.curCate}</span>
    </div>
</div>`

    let list = [editKey, showInput, delSpan, delAllSpan, doc, saveGlobal, changeOrder, changeBoardHeight, changeBoardHeight, showCateInput, categoryList]
    list.forEach((v, i) => {
        if (v.name === "changeBoardHeight") {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                i % 2 === 0 ? v(40) : v(-40)
            })
        } else if (v.name === "saveGlobal") {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                if (saveGlobal()) {
                    alert(`保存成功!`)
                } else {
                    alert(`配置数据太旧,无法保存。请刷新页面后再编辑保存`)
                }
            })
        } else {
            operation.getElementsByTagName("span")[i].addEventListener("click", v)
        }
    })
    return operation
}

function bottomOperationV2() {
    let operation = document.createElement("div")
    operation.style.width = "100%"
    operation.innerHTML = `
<div id="${prefixStr("bottomOperationV2")}">
    <div style="${flexStyle({jc: CSS.flex.end})}margin: 0 20px 10px 0px;">
        <span style="${CSS.operationStyle}" >文本</span>
        <span style="${CSS.operationStyle}">分类</span>
        <span style="${CSS.operationStyle}margin-right: 0px;">设置</span>
    </div>
</div>`

    let list = [domCtl, domCtl, domCtl]
    list.forEach((v, i) => {
        if (i === 0) {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                v([GlobalDoms.rightBoard], [])
                v([GlobalDoms.textAreaDom], [GlobalDoms.categoryAreaDom, GlobalDoms.settingAreaDom])
            })
        } else if (i === 1) {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                v([GlobalDoms.rightBoard], [])
                v([GlobalDoms.categoryAreaDom], [GlobalDoms.textAreaDom, GlobalDoms.settingAreaDom])
            })
        } else if (i === 2) {
            operation.getElementsByTagName("span")[i].addEventListener("click", function () {
                v([GlobalDoms.rightBoard], [])
                v([GlobalDoms.settingAreaDom], [GlobalDoms.textAreaDom, GlobalDoms.categoryAreaDom])
            })
        }
    })
    return operation
}

function board() {
    let b = document.createElement("div");
    b.style.cssText = CSS.boardStyle
    b.setAttribute("id", "fillTextBoard")
    b.appendChild(topOperation())
    b.appendChild(renderSpan())
    b.appendChild(bottomOperationV2())
    GlobalDoms.board = b
    return b
}

function hiddenBoard() {
    let hb = document.createElement("div")
    hb.style.cssText = CSS.hiddenBoardStyle
    hb.setAttribute("id", "fillTextHiddenBoard")
    hb.innerHTML = `<span style="${CSS.iconStyle}font-size:30px;pointer-events: none;">add</span>`
    hb.addEventListener("click", function () {
        parentBoardCtl(0)
    })
    GlobalDoms.hiddenBoard = hb
    return hb
}

// 两个boardCtl优化为domCtl方法
function boardCtl(show) {
    if (show === 0) {
        hiddenDom(GlobalDoms.hiddenBoard)
        showDom(GlobalDoms.board)
    } else if (show === 1) {
        hiddenDom(GlobalDoms.board)
        showDom(GlobalDoms.hiddenBoard)
    } else if (show === -1) {
        hiddenDom(GlobalDoms.board)
        hiddenDom(GlobalDoms.hiddenBoard)
    }

    Global.show = show
    saveGlobal()
}

function parentBoardCtl(show) {
    if (show === 0) {
        hiddenDom(GlobalDoms.hiddenBoard)
        showDom(GlobalDoms.parentBoard)
    } else if (show === 1) {
        hiddenDom(GlobalDoms.parentBoard)
        showDom(GlobalDoms.hiddenBoard)
    } else if (show === -1) {
        hiddenDom(GlobalDoms.parentBoard)
        hiddenDom(GlobalDoms.hiddenBoard)
    }

    Global.parentShow = show
    saveGlobal()
}

function parentBoard() {
    let b = document.createElement("div");
    b.style.cssText = CSS.parentBoardStyle
    b.setAttribute("id", "fillTextParentBoard")
    b.appendChild(board())
    b.appendChild(rightOperateBoard())
    GlobalDoms.parentBoard = b
    return b
}

//操作面板
function operationBoard() {
    document.body.appendChild(parentBoard())
    document.body.appendChild(hiddenBoard())
    document.body.removeEventListener("keydown", listenBodyKey)
    document.body.addEventListener("keydown", listenBodyKey)

    parentBoardCtl(Global.parentShow)
    moveDom(GlobalDoms.parentBoard)
    domCtl([GlobalDoms.board], [GlobalDoms.rightBoard])
}

/**
 * 响应事件区
 */

function listenBodyKey (event) {
    if (event.altKey) {
        if ("Alt+" + event.key === Global.hiddenKey) {
            Global.parentShow === -1 ? parentBoardCtl(0) : parentBoardCtl(-1)
        }
    }
}

function toBottom() {
    GlobalDoms.board.scrollTop = GlobalDoms.board.scrollHeight
}

function moveDom(dom) {
    let isDragging = false;
    let x, y;

    dom.addEventListener("mousedown", function (event) {
        isDragging = true;
        x = event.x;
        y = event.y;
    });

    dom.addEventListener("mousemove", function (event) {
        if (isDragging) {
            let offsetX = event.x - x;
            let offsetY = event.y - y;

            dom.style.left = parseInt(dom.style.left) + offsetX + "px";
            dom.style.top = parseInt(dom.style.top) + offsetY + "px";
            x = event.x
            y = event.y
        }
    });

    dom.addEventListener("mouseup", function () {
        isDragging = false;
    });
}

function doc() {
    alert("面板隐藏快捷键默认为Alt+q,可自行更改(只能设置为Alt+任意键,修改后按下Enter保存)\n快捷键建议Alt开头或纯字母数字,如:Alt+1、qq、11\n")
}