flash-game-downloader

一键下载 flash 游戏(swf),有限地支持(1)4399(2)7k7k(3)nitrome

// ==UserScript==
// @name            flash-game-downloader
// @namespace       http://tampermonkey.net/
// @version         0.0.6
// @description     一键下载 flash 游戏(swf),有限地支持(1)4399(2)7k7k(3)nitrome
// @author          [email protected]
// @match           https://www.4399.com/flash/*
// @match           https://s2.4399.com
// @match           http://www.7k7k.com/swf/*.htm*
// @match           *://www.nitrome.com/*
// @require         https://cdn.staticfile.org/jszip/3.7.1/jszip.min.js
// @require         https://cdn.staticfile.org/sweetalert2/11.7.5/sweetalert2.all.min.js
// @icon            
// @grant           none
// @run-at          document-idle
// @license         GPL-3.0-only
// ==/UserScript==


(function() {
    /**
     * 脚本级全局常量
     */

    FLASH_ICON = ``;


    /**
     * 脚本级公用函数和对象
     */

    /**
     * 元素选择器
     * @param {string} selector 选择器
     * @returns {Array<HTMLElement>} 元素列表
     */
    function $(selector) {
        const self = this?.querySelectorAll ? this : document;
        return [...self.querySelectorAll(selector)];
    }


    /**
     * 安全元素选择器,直到元素存在时才返回元素列表,最多等待5秒
     * @param {string} selector 选择器
     * @returns {Promise<Array<HTMLElement>>} 元素列表
     */
    async function $$(selector) {
        const self = this?.querySelectorAll ? this : document;

        for (let i = 0; i < 10; i++) {
            let elems = [...self.querySelectorAll(selector)];
            if (elems.length > 0) {
                return elems;
            }
            await new Promise(r => setTimeout(r, 500));
        }
        throw Error(`"${selector}" not found`);
    }


    const util = {
        /**
         * 查找数组中某元素的全部位置,找不到返回空列表
         * @param {Array} arr 
         * @param {Array} elem 
         * @returns {Array<number>}
         */
        get_indexes: function(arr, elem) {
            const indexes = [];
            let from = 0;
            let i = arr.indexOf(elem, from);

            while (i !== -1) {
                indexes.push(i);
                from = i + 1;
                i = arr.indexOf(elem, from);
            }
            return indexes;
        },

        /**
         * 返回子数组位置,找不到返回-1
         * @param {Array<number>} arr 父数组
         * @param {Array<number>} sub_arr 子数组
         * @param {number} from 开始位置
         * @returns {number} index
         */
        index_of_sub_arr: function(arr, sub_arr, from) {
            // 如果子数组为空,则返回-1
            if (sub_arr.length === 0) return -1;
            // 初始化当前位置为from
            let position = from;
            // 算出最大循环次数
            const length = arr.length - sub_arr.length + 1;

            // 循环查找子数组直到没有更多
            while (position < length) {
                // 如果当前位置的元素与子数组的第一个元素相等,则开始比较后续元素
                if (arr[position] === sub_arr[0]) {
                    // 初始化匹配标志为真
                    let match = true;
                    // 循环比较后续元素,如果有不相等的,则将匹配标志设为假,并跳出循环
                    for (let i = 1; i < sub_arr.length; i++) {
                        if (arr[position + i] !== sub_arr[i]) {
                            match = false;
                            break;
                        }
                    }
                    // 如果匹配标志为真,则说明找到了子数组,返回当前位置
                    if (match) return position;
                }
                // 更新当前位置为下一个位置
                position++;
            }
            // 如果循环结束还没有找到子数组,则返回-1
            return -1;
        },

        Socket: class Socket {
            /**
            * 创建套接字对象
            * @param {Window} target 目标窗口
            */
            constructor(target) {
                if (!(target.window && (target === target.window))) {
                    console.log(target);
                    throw new Error(`target is not a [Window Object]`); 
                }
                this.target = target;
                this.connected = false;
                this.listeners = new Set();
            }
        
            get [Symbol.toStringTag]() { return "Socket"; }
        
            /**
            * 向目标窗口发消息
            * @param {*} message 
            */
            talk(message) {
                if (!this.target) {
                    throw new TypeError(
                        `socket.target is not a window: ${this.target}`
                    );
                }
                this.target.postMessage(message, "*");
            }
        
            /**
            * 添加捕获型监听器,返回实际添加的监听器
            * @param {Function} listener (e: MessageEvent) => {...}
            * @param {boolean} once 是否在执行后自动销毁,默认 false;如为 true 则使用自动包装过的监听器
            * @returns {Function} listener
            */
            listen(listener, once=false) {
                if (this.listeners.has(listener)) {
                    return;
                }
        
                let real_listener = listener;
                // 包装监听器
                if (once) {
                    const self = this;
                    function wrapped(e) {
                        listener(e);
                        self.not_listen(wrapped);
                    }
                    real_listener = wrapped;
                }
                
                // 添加监听器
                this.listeners.add(real_listener);
                window.addEventListener(
                    "message", real_listener, true
                );
                return real_listener;
            }
        
            /**
            * 移除socket上的捕获型监听器
            * @param {Function} listener (e: MessageEvent) => {...}
            */
            not_listen(listener) {
                console.log(listener);
                console.log(
                    "listener delete operation:",
                    this.listeners.delete(listener)
                );
                window.removeEventListener("message", listener, true);
            }
        
            /**
            * 检查对方来信是否为pong消息
            * @param {MessageEvent} e 
            * @param {Function} resolve 
            */
            _on_pong(e, resolve) {
                // 收到pong消息
                if (e.data.pong) {
                    this.connected = true;
                    this.listeners.forEach(
                        listener => listener.ping ? this.not_listen(listener) : 0
                    );
                    console.log("Client: Connected!\n" + new Date());
                    resolve(this);
                }
            }
        
            /**
            * 向对方发送ping消息
            * @returns {Promise<Socket>}
            */
            _ping() {
                return new Promise((resolve, reject) => {
                    // 绑定pong检查监听器
                    const listener = this.listen(
                        e => this._on_pong(e, resolve)
                    );
                    listener.ping = true;
        
                    // 5分钟后超时
                    setTimeout(
                        () => reject(new Error(`Timeout Error during receiving pong (>5min)`)),
                        5 * 60 * 1000
                    );
                    // 发送ping消息
                    this.talk({ ping: true });
                });
            }
        
            /**
            * 检查对方来信是否为ping消息
            * @param {MessageEvent} e 
            * @param {Function} resolve 
            */
            _on_ping(e, resolve) {
                // 收到ping消息
                if (e.data.ping) {
                    this.target = e.source;
                    this.connected = true;
                    this.listeners.forEach(
                        listener => listener.pong ? this.not_listen(listener) : 0
                    );
                    console.log("Server: Connected!\n" + new Date());
                    
                    // resolve 后期约状态无法回退
                    // 但后续代码仍可执行
                    resolve(this);
                    // 回应pong消息
                    this.talk({ pong: true });
                }
            }
        
            /**
            * 当对方来信是为ping消息时回应pong消息
            * @returns {Promise<Socket>}
            */
            _pong() {
                return new Promise(resolve => {
                    // 绑定ping检查监听器
                    const listener = this.listen(
                        e => this._on_ping(e, resolve)
                    );
                    listener.pong = true;
                });
            }
        
            /**
            * 连接至目标窗口
            * @param {boolean} talk_first 是否先发送ping消息
            * @param {Window} target 目标窗口
            * @returns {Promise<Socket>}
            */
            connect(talk_first) {
                // 先发起握手
                if (talk_first) {
                    return this._ping();
                }
                // 后发起握手
                return this._pong();
            }
        },

        /**
         * 以指定原因弹窗提示并抛出错误
         * @param {string} reason 
         */
        raise: function(reason) {
            alert(reason);
            throw new Error(reason);
        },
    
        /**
         * 返回一个包含计数器的迭代器, 其每次迭代值为 [index, value]
         * @param {Iterable} iterable 
         * @returns 
         */
        enumerate: function* (iterable) {
            let i = 0;
            for (let value of iterable) {
                yield [i++, value];
            }
        },
    
        /**
         * 同步的迭代若干可迭代对象
         * @param  {...Iterable} iterables 
         * @returns 
         */
        zip: function* (...iterables) {
            // 强制转为迭代器
            const iterators = iterables.map(
                iterable => iterable[Symbol.iterator]()
            );
    
            // 逐次迭代
            while (true) {
                let [done, values] = base.getAllValus(iterators);
                if (done) {
                    return;
                }
                if (values.length === 1) {
                    yield values[0];
                } else {
                    yield values;
                }
            }
        },
    
        /**
         * 返回指定范围整数生成器
         * @param {number} end 如果只提供 end, 则返回 [0, end)
         * @param {number} end2 如果同时提供 end2, 则返回 [end, end2)
         * @param {number} step 步长, 可以为负数,不能为 0
         * @returns 
         */
        range: function*(end, end2=null, step=1) {
            // 参数合法性校验
            if (step === 0) {
                throw new RangeError("step can't be zero");
            }
            const len = end2 - end;
            if (end2 && len && step && (len * step < 0)) {
                throw new RangeError(`[${end}, ${end2}) with step ${step} is invalid`);
            }
    
            // 生成范围
            end2 = end2 === null ? 0 : end2;
            let [small, big] = [end, end2].sort((a, b) => a - b);
            // 开始迭代
            if (step > 0) {
                for (let i = small; i < big; i += step) {
                    yield i;
                }
            } else {
                for (let i = big; i > small; i += step) {
                    yield i;
                }
            };
        },
    
        /**
         * 复制text到剪贴板
         * @param {string} text 
         * @returns 
         */
        copy_text: function(text) {
            // 输出到控制台和剪贴板
            console.log(
                text.length > 20 ?
                    text.slice(0, 21) + "..." : text
            );
            
            if (!navigator.clipboard) {
                base.oldCopy(text);
                return;
            };
    
            navigator.clipboard
                .writeText(text)
                .catch(_ => base.oldCopy(text));
        },
    
        /**
         * 复制媒体到剪贴板
         * @param {Blob} blob
         */
        copy: async function(blob) {
            const data = [new ClipboardItem({ [blob.type]: blob })];
            try {
                await navigator.clipboard.write(data);
                console.log(`${blob.type} 成功复制到剪贴板`);
            } catch (err) {
                console.error(err.name, err.message);
            }
        },
    
        /**
         * 创建并下载文件
         * @param {string} file_name 文件名
         * @param {ArrayBuffer | ArrayBufferView | Blob | string} content 内容
         * @param {string} type 媒体类型,需要符合 MIME 标准 
         */
        save: function(file_name, content, type="") {
            const blob = new Blob(
                [content], { type }
            );
            const size = (blob.size / 1024).toFixed(1);
            console.log(`blob saved, size: ${size} kb, type: ${blob.type}`);
    
            const url = URL.createObjectURL(blob);
            const a = document.createElement("a");
            a.download = file_name || "未命名文件";
            a.href = url;
            a.click();
            URL.revokeObjectURL(url);
        },
    
        sleep: async function(delay_ms) {
            return new Promise(
                resolve => setTimeout(resolve, delay_ms)
            );
        },
    
        /**
         * 取得get参数key对应的value
         * @param {string} key
         * @returns {string} value
         */
        get_param: function(key) {
            return new URL(location.href).searchParams.get(key);
        },
    
        /**
         * 等待直到函数返回true
         * @param {Function} is_ok 判断条件达成与否的函数
         * @param {number} timeout 最大等待秒数, 默认5000毫秒
         */
        wait_until: async function(is_ok, timeout=5000) {
            const gap = 200;
            let chances = parseInt(timeout / gap);
            chances = chances < 1 ? 1 : chances;
            
            while (! await is_ok()) {
                await this.sleep(200);
                chances -= 1;
                if (!chances) {
                    break;
                }
            }
        },
    
        /**
         * 用try移除元素
         * @param {HTMLElement} element 要移除的元素
         */
        remove: function(element) {
            try {
                element.remove();
            } catch (e) {}
        },
    
        /**
         * 等待全部任务落定后返回值的列表
         * @param {Iterable<Promise>} tasks 
         * @returns {Promise<Array>} values
         */
        gather: async function(tasks) {
            const results = await Promise.allSettled(tasks);
            return results
                .filter(result => result.value)
                .map(result => result.value);
        },
    
        /**
         * 使用xhr异步GET请求目标url,返回响应体blob
         * @param {string} url 
         * @returns {Promise<Blob>} blob
         */
        xhr_get_blob: async function(url) {
            const xhr = new XMLHttpRequest();
            xhr.open("GET", url);
            xhr.responseType = "blob";
            
            return new Promise((resolve, reject) => {
                xhr.onload = () => {
                    const code = xhr.status;
                    if (code >= 200 && code <= 299) {
                        resolve(xhr.response);
                    }
                    else {
                        reject(new Error(`Network Error: ${code}`));
                    }
                }
                xhr.send();
            });
        },
    
        /**
         * 加载CDN脚本
         * @param {string} url 
         */
        load_web_script: async function(url) {
            try {
                // xhr+eval方式
                Function(
                    await (await this.xhr_get_blob(url)).text()
                )();
            } catch(e) {
                console.error(e);
                // 嵌入<script>方式
                const script = document.createElement("script");
                script.src = url;
                document.body.append(script);
            }
        },
    };
    

    /**
     * 域名级主函数
     */


    /**
     * 启动下载 4399 flash 游戏
     */
    function dl_flash_4399() {
        /**
         * 域名级全局常量、变量
         */

        BASE_URL = "https://s2.4399.com/4399swf";
        let sock;


        async function send_url() {
            const title = $(".name a")[0].textContent.trim() || "flash游戏";
            const path = window._strGamePath;

            if (!path) util.raise(
                "_strGamePath 不存在,找不到游戏文件路径"
            );
            if (!path.endsWith(".swf")) util.raise(
                `当前游戏不是 flash 游戏。\n游戏路径为:${path}`
            );

            const id = "flash-dl-src";
            let iframe = $(`#${id}`)[0];

            if (!iframe) {
                iframe = document.createElement("iframe");
                iframe.id = id;
                iframe.src = "https://s2.4399.com";
                document.body.append(iframe);
                sock = new util.Socket(iframe.contentWindow);
                await sock.connect(false);
            }
            
            sock.talk({
                flash_dl: true,
                url: BASE_URL + path,
                title,
            });
        }

        function add_style() {
            const style = `
            <style>
                #flash-dl-btn {
                    text-align: center;
                    background: url("${FLASH_ICON}");
                    background-repeat: no-repeat;
                    background-position: top;
                    width: 40px;
                    padding-top: 30px;
                    margin: 0 10px;
                    float: left;
                    display: inline;
                    cursor: pointer;
                }

                #flash-dl-src {
                    display: none;
                }
            <style>
            `;
            document.head.insertAdjacentHTML(
                "beforeend", style
            );
        }

        async function add_dl_btn() {
            const box = (await $$("#uplayer .fr"))[0];

            // 修改误导性的下载按钮文本(下载4399游戏盒子)
            $("#down_a")[0].textContent = "盒子";
            
            // 新按钮
            const btn = document.createElement("a");
            btn.id = "flash-dl-btn";
            btn.textContent = "下载";
            btn.onfocus = () => btn.blur();
            btn.onclick = send_url;
            box.insertAdjacentElement("afterbegin", btn);
        }

        (() => {
            console.log("enter: dl_flash");
            add_style();
            add_dl_btn();
        })();
    }

    /**
     * 执行下载 4399 flash 游戏
     */
    function dl_flash_4399_in_origin() {
        /**
         * @param {MessageEvent} e 
         */
        async function on_msg(e) {
            if (!e.data.flash_dl) return;

            const { url, title } = e.data;
            const resp = await fetch(url, {
                headers: {
                    "Host": "szhong.4399.com",
                    "X-Requested-With": "ShockwaveFlash/34.0.0.282",
                }
            });
            if (!resp.ok) util.raise(
                `游戏下载失败,错误代码:${resp.status},原因:${resp.statusText}`
            );

            const blob = await resp.blob();
            util.save(
                title.endsWith(".swf") ? title : title + ".swf",
                blob,
                "application/x-shockwave-flash"
            );
        }

        (() => {
            console.log("enter: dl_flash_in_origin")
            if (window.top === window) return;

            const sock = new util.Socket(window.top);
            sock.listen(on_msg);
            sock.connect(true);
        })();
    }

    /**
     * 下载 7k7k flash 游戏
     */
    function dl_flash_7k7k() {
        /**
         * 域名级全局常量变量
         */

        let swf_url;
        let dl_btn;
        const fnames = ["启动器.swf"];
        const HOW_TO_PLAY = `
            【如何游玩多 SWF 文件组成的 Flash 游戏?】
            1. 在你的电脑上下载并安装 python
            2. 将 python 解释器目录加入环境变量
            3. 在解压为文件夹的游戏目录下打开 cmd 或 powershell
            4. 输入命令:python -m http.server --bind 0.0.0.0 5678
            5. 回车执行上述命令
            6. 用支持 Flash 的浏览器(如 [cef flash browser](https://github.com/Mzying2001/CefFlashBrowser) 访问:http://127.0.0.1:5678/启动器.swf
        `.replace(/ {2,}/g, ""); 


        /**
         * @returns {number} 
         */
        function get_game_id() {
            return window?.gameInfo?.gameId ||
                parseInt(
                    // http://www.7k7k.com/swf/28079.htm?abc
                    location.pathname.match(/(?<=[/])[0-9]+?(?=[.]htm)/)[0]
                );
        }


        /**
         * @param {string | URL} url 
         * @returns {Promise<ArrayBuffer>}
         */
        async function fetch_as_buffer(url) {
            const resp = await fetch(url);
            console.log(resp);
            if (!resp.ok) util.raise(`资源获取失败:${resp.status}`);
            return await resp.arrayBuffer();
        }


        /**
         * @param {string} fname 
         */
        function update_url(fname) {
            const parts = swf_url.pathname.split("/");
            parts.splice(-1, 1, fname);
            swf_url.pathname = parts.join("/");
        }


        /**
         * @param {number} game_id 
         * @returns {Promise<ArrayBuffer>}
         */
        async function get_swf(game_id) {
            // 查询游戏信息
            const info_url = `http://www.7k7k.com/swf/game/${game_id}/?time`;
            const resp = await fetch(info_url);
            console.log(resp);
            if (!resp.ok) util.raise(`游戏信息查询失败:${resp.status}`);

            const info = await resp.json();
            console.log(info);

            // 查询游戏页面 url
            const iframe_url = info?.result?.url;
            console.log(iframe_url);
            if (!iframe_url) util.raise(
                `找不到游戏页面路径:<游戏信息>.result.url 不存在`
            );

            // 如果是游戏文件链接,直接下载,返回空结果用于终止后续函数
            if (iframe_url.endsWith(".swf")) {
                const swf = await fetch_as_buffer(iframe_url);
                const blob = new Blob(
                    [swf], { type: "application/x-shockwave-flash" }
                );
                util.save(get_title() + ".swf", blob);
                return;
            }

            // 从游戏页面 html 中提取游戏链接
            const resp2 = await fetch(iframe_url);
            console.log(resp2);
            if (!resp2.ok) util.raise(`游戏页面获取失败:${resp2.status}`);

            const html = await resp2.text();
            const matches = html.match(/_src_\s*?=\s*?(['"])(.+)?\1/)
                || html.match(/var\s+?p\s*?=\s*(['"])(.+)?\1/);
            console.log(matches);

            const swf_name = matches[2];
            console.log(swf_name);

            if (!swf_name) {
                console.log(html);
                util.raise(`游戏路径查询失败:游戏页面中找不到 _src_ = "..."`);
            }

            swf_url = new URL(iframe_url);
            update_url(swf_name);

            // 下载游戏文件
            return await fetch_as_buffer(swf_url);
        }


        function get_title() {
            return document.title.split(",")[0];
        }


        /**
         * @param {ArrayBuffer} data
         * @returns {string} 
         */
        function get_sub_fname(data) {
            const bytes = new Uint8Array(data);
            const end = util.index_of_sub_arr(
                //             .swf
                bytes, [0x2e, 0x73, 0x77, 0x66], 0
            );
            if (end === -1) {
                console.log(`找不到子文件路径:找不到 .swf 字符串`);
                return "";
            }

            const begin = bytes.lastIndexOf(0, end);
            if (begin === -1) {
                console.log(`找不到子文件路径:找不到 .swf 前的 \x00`);
                return "";
            }

            return new TextDecoder()
                .decode(bytes.subarray(begin + 1, end)) + ".swf";
        }


        /**
         * @param {ArrayBuffer} swf 
         * @param {Array<Blob>} files 
         * @returns {Promise<void>} 
         */
        async function collect_swfs(swf, files) {
            const fname = get_sub_fname(swf);
            if (!fname) return;

            fnames.push(fname);
            update_url(fname);

            const new_swf = await fetch_as_buffer(swf_url);
            files.push(new Blob(
                [new_swf], { type: "application/x-shockwave-flash" }
            ));
            collect_swfs(new_swf, files);
        }


        async function download_game() {
            const game_id = get_game_id();
            const swf = await get_swf(game_id);
            if (!swf) return;

            const files = [new Blob(
                [swf], { type: "application/x-shockwave-flash" }
            )];

            await collect_swfs(swf, files);
            const title = get_title();
            // 单文件游戏直接下载
            if (files.length === 1) {
                util.save(title + ".swf", files[0]);
                return;
            }

            // 多文件游戏下载压缩包
            const zip = new window.JSZip();
            files.forEach((blob, i) => zip.file(
                fnames[i], blob, { binary: true }
            ));
            const help = new Blob([HOW_TO_PLAY]);
            zip.file("使用说明.txt", help, { binary: true });

            // 导出
            const zip_blob = await zip.generateAsync({ type: "blob" });
            console.log(zip_blob);
            util.save(`${title}.zip`, zip_blob);
        }


        function add_style() {
            const style = `
            <style>
                #flash-dl-btn {
                    background: url("${FLASH_ICON}");
                    background-repeat: no-repeat;
                    background-position: center;
                    width: 40px;
                    height: 100%;
                    cursor: pointer;
                }

                .play_header {
                    display: flex !important;
                    flex-direction: row;
                    justify-content: space-between;
                }

                .disabled {
                    filter: grayscale(75%);
                    pointer-events: none;
                }
            <style>
            `;
            document.head.insertAdjacentHTML(
                "beforeend", style
            );
        }


        async function add_btn() {
            dl_btn = document.createElement("button");
            dl_btn.id = "flash-dl-btn";

            dl_btn.onclick = async () => {
                dl_btn.classList.add("disabled");
                try {
                    await download_game(); 
                } catch (err) {
                    console.error(err);
                    alert(`下载失败,请在脚本主页反馈并附上网址,谢谢`);
                    dl_btn.classList.remove("disabled");
                }
                dl_btn.classList.remove("disabled");
            };

            const targets = await $$(".play_header");
            const target = targets[0];
            target.insertAdjacentElement("beforeend", dl_btn);
        }


        (() => {
            add_style();
            add_btn();
        })();
    }


    /**
     * 下载 nitrome flash 游戏
     */
    function dl_flash_nitrome() {
        function on_game_page() {
            function add_style() {
                const style = `
                <style>
                    #flash-dl-btn {
                        background: url("${FLASH_ICON}");
                        background-repeat: no-repeat;
                        background-position: center;
                        width: 100%;
                        height: 70px;
                        cursor: pointer;
                        display: flex;
                        flex-direction: row;
                        justify-content: space-around;
                    }
    
                    .comment-info {
                        flex-direction: column !important;
                    }
                <style>
                `;
                document.head.insertAdjacentHTML(
                    "beforeend", style
                );
            }
    
    
            function add_btn() {
                const dl_btn = document.createElement("a");
                // http://www.nitrome.com/games/finalninja/
                const fname = location.pathname.split("/").at(-2) + ".swf";
                dl_btn.download = fname;
                dl_btn.href = fname;
                dl_btn.target = "_blank";
                dl_btn.id = "flash-dl-btn";
                dl_btn.textContent = "下载游戏文件";
    
                $(".comment-info")[0].insertAdjacentElement(
                    "beforeend", dl_btn
                );
            }
    
    
            function main() {
                add_style();
                add_btn();
            }
    
    
            setTimeout(main, 1000);
        }


        function on_list_page() {
            const DL_BTN = `
            <a id="flash-dl-btn" data-src="$1" onclick="copy_link(this)"></a>
            `;


            function add_style() {
                const style = `
                <style>
                    #flash-dl-btn {
                        background: url("${FLASH_ICON}");
                        background-repeat: no-repeat;
                        background-position: center;
                        width: 32px;
                        height: 33px;
                        cursor: cell;
                        position: absolute;
                        z-index: 200;
                        margin-left: -28px;
                        transform: scale(0.7);
                        filter: hue-rotate(185deg);
                    }

                    #flash-dl-btn:hover {
                        filter: none;
                    }

                    .copy-icon {
                        border: none !important;
                        margin: 0 1.25em !important;
                        margin: 0 0 0 10px !important;
                    }

                    .copy-container {
                        margin: 8px 16px !important;
                        padding: 0 !important;
                        font-size: 14px !important;
                        
                    }
                    
                    .copy-popup {
                        top: 60px;
                        padding: 4px 10px !important;
                        height: 44px !important;
                        font-size: 12px !important;
                        width: fit-content !important;
                        align-content: center;
                        box-shadow: rgba(0, 0, 0, 0.2) 0px 12px 28px 0px, rgba(0, 0, 0, 0.1) 0px 2px 4px 0px, rgba(255, 255, 255, 0.05) 0px 0px 0px 1px inset !important;
                    }

                    .swal2-popup {
                        border-radius:0 !important;    
                    }
                </style>
                `;
                document.head.insertAdjacentHTML(
                    "beforeend", style
                );
            }


            /**
             * @param {HTMLAnchorElement} elem 
             */
            window.copy_link = function(elem) {
                const link = elem.dataset.src;
                console.log(link);
                navigator.clipboard.writeText(link);
                Sweetalert2.fire({
                    text: "复制成功,可以粘贴咯~",
                    toast: true,
                    timer: 2000,
                    showConfirmButton: false,
                    icon: "success",
                    position: "top",
                    customClass: {
                        popup: "copy-popup",
                        htmlContainer: "copy-container",
                        icon: "copy-icon"
                    }
                });
            };


            function add_btn() {
                $(".box_wrap").forEach(box => {
                    if (box.querySelector("#flash-dl-btn")) return;

                    const game = box
                        .querySelector("[itemprop=link]")
                        .href
                        .split("/")
                        .at(-2);

                    console.log(`game name: ${game}`);

                    const href = `http://www.nitrome.com/games/${game}/${game}.swf`;
                    const btn = DL_BTN.replace("$1", href);
                    box.insertAdjacentHTML("beforeend", btn);
                });
            }


            (() => {
                add_style();
                add_btn();
            })();
        }


        (() => {
            console.log("enter: sub route");

            const path = location.pathname.toLowerCase();
            const game_types = [
                "/all-games/",
                "/multiplayer-games/",
                "/hearted-games/",
                "/demos/"
            ];

            const map = new Map([
                ...game_types.map(type => [type, on_list_page]),
                ["/games/.+", on_game_page]
            ]);
            
            for (const [pattern, handler] of map.entries()) {
                if (new RegExp(`^${pattern}$`).test(path)) {
                    return handler();
                }
            }
            console.log(`不受支持的路径:${path}`);
        })();
    }


    /**
     * 路由函数,脚本主函数入口
     */
    function route() {
        console.log("enter: main route");

        const host = location.hostname;
        const map = new Map([
            ["www.4399.com", dl_flash_4399],
            ["s2.4399.com", dl_flash_4399_in_origin],
            ["www.7k7k.com", dl_flash_7k7k],
            ["www.nitrome.com", dl_flash_nitrome],
        ]);

        if (!map.has(host)) {
            console.log(`不受支持的域名:${host}`);
            return;
        }
        map.get(host)();
    }


    setTimeout(route, 500);

    /**
     * 更新日志
     * ---
     * 更新日期:2023/4/28
     * 更新版本:0.0.1
     * - 完成第一版  4399 flash 文件下载脚本
     * ---
     * 更新日期:2023/5/18
     * 更新版本:0.0.2
     * - 脚本名称变更
     * - 新增支持 7k7k
     * ---
     * 更新日期:2023/5/19
     * 更新版本:0.0.3
     * - 7k7k 游戏文件地址搜索增强
     * ---
     * 更新日期:2023/5/19
     * 更新版本:0.0.4
     * - 新增支持 nitrome
     * ---
     * 更新日期:2023/5/19
     * 更新版本:0.0.5
     * - 修复 7k7k 部分游戏下载失败的 bug
     * ---
     * 更新日期:2023/5/22
     * 更新版本:0.0.6
     * - 在 nitrome 游戏列表页增加了复制下载链接按钮
     */
})();