[E/Ex-Hentai] AutoLogin

E/Ex - 共享帳號登入、自動獲取 Cookies、手動輸入 Cookies、本地備份以及查看備份,自動檢測登入

2024-07-15 या दिनांकाला. सर्वात नवीन आवृत्ती पाहा.

// ==UserScript==
// @name         [E/Ex-Hentai] AutoLogin
// @name:zh-TW   [E/Ex-Hentai] 自動登入
// @name:zh-CN   [E/Ex-Hentai] 自动登入
// @name:ja      [E/Ex-Hentai] 自動ログイン
// @name:ko      [E/Ex-Hentai] 자동 로그인
// @name:en      [E/Ex-Hentai] AutoLogin
// @version      0.0.29
// @author       Canaan HS
// @description         E/Ex - 共享帳號登入、自動獲取 Cookies、手動輸入 Cookies、本地備份以及查看備份,自動檢測登入
// @description:zh-TW   E/Ex - 共享帳號登入、自動獲取 Cookies、手動輸入 Cookies、本地備份以及查看備份,自動檢測登入
// @description:zh-CN   E/Ex - 共享帐号登录、自动获取 Cookies、手动输入 Cookies、本地备份以及查看备份,自动检测登录
// @description:ja      E/Ex - 共有アカウントでのログイン、クッキーの自动取得、クッキーの手动入力、ローカルバックアップおよびバックアップの表示、自动ログインの検出
// @description:ko      E/Ex - 공유 계정 로그인, 자동으로 쿠키 가져오기, 쿠키 수동 입력, 로컬 백업 및 백업 보기, 자동 로그인 감지
// @description:en      E/Ex - Shared account login, automatic cookie retrieval, manual cookie input, local backup, and backup viewing, automatic login detection

// @connect      *
// @match        *://e-hentai.org/*
// @match        *://exhentai.org/*
// @icon         https://e-hentai.org/favicon.ico

// @license      MIT
// @namespace    https://greasyfork.org/users/989635

// @run-at       document-start
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_notification
// @grant        GM_xmlhttpRequest
// @grant        GM_getResourceText
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        GM_addValueChangeListener

// @require      https://cdnjs.cloudflare.com/ajax/libs/jquery/3.7.1/jquery.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/js-cookie/3.0.5/js.cookie.min.js
// @require      https://update.greasyfork.org/scripts/495339/1404326/ObjectSyntax_min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/jquery-jgrowl/1.4.9/jquery.jgrowl.min.js
// @resource     jgrowl-css https://cdnjs.cloudflare.com/ajax/libs/jquery-jgrowl/1.4.9/jquery.jgrowl.min.css
// ==/UserScript==

(async () => {
    const lang = language(Syn.Device.Lang);
    const domain = Syn.Device.Host;
    const CKOP = CookieFactory();
    new class AutoLogin {
        constructor() {
            this.modal = null;
            this.Share = Syn.Store("g", "Share") ?? this.UpdateShared();
            this.on = async (element, type, listener) => {
                $(element).on(type, listener);
            };
            this.Growl = async (message, theme, life) => {
                $.jGrowl(`  ${message}  `, {
                    theme: theme,
                    life: life
                });
            };
            this.CreateDetection = () => {
                const detection = $(".modal-background");
                detection[0] && detection.remove();
            };
            this.CreateMenu = async () => {
                $(document.body).append(this.modal);
                requestAnimationFrame(() => {
                    $(".modal-background").css({
                        opacity: "1",
                        "background-color": "rgba(0,0,0,0.5)",
                        transform: "translate(-50%, -50%) scale(1)"
                    });
                });
            };
            this.DeleteMenu = async () => {
                const modal = $(".modal-background");
                modal.css({
                    opacity: "0",
                    "pointer-events": "none",
                    "background-color": "rgba(0,0,0,0)",
                    transform: "translate(-50%, -50%) scale(0)"
                });
                setTimeout(() => {
                    modal.remove();
                }, 1300);
            };
            this.GlobalMenuToggle = async () => {
                Syn.StoreListen(["Expand"], listen => {
                    listen.far && this.MenuToggle();
                });
            };
            this.MenuToggle = async () => {
                const state = Syn.Store("g", "Expand", false), disp = state ? lang.RM_C1 : lang.RM_C0;
                Syn.Menu({
                    [disp]: {
                        func: () => {
                            state ? Syn.Store("s", "Expand", false) : Syn.Store("s", "Expand", true);
                            this.MenuToggle();
                        },
                        hotkey: "c",
                        close: false
                    }
                }, "Switch");
                state ? this.Expand() : this.Collapse();
            };
            this.Expand = async () => {
                Syn.Menu({
                    [lang.RM_01]: {
                        func: () => this.GetCookieAutomatically()
                    },
                    [lang.RM_02]: {
                        func: () => this.ManualSetting()
                    },
                    [lang.RM_03]: {
                        func: () => this.ViewSaveCookie()
                    },
                    [lang.RM_04]: {
                        func: () => this.CookieInjection()
                    },
                    [lang.RM_05]: {
                        func: () => this.ClearLogin()
                    }
                }, "Expand");
            };
            this.Collapse = async () => {
                for (let i = 1; i <= 5; i++) {
                    GM_unregisterMenuCommand("Expand-" + i);
                }
            };
        }
        async Main() {
            let CurrentTime = new Date(), DetectionTime = Syn.Storage("DetectionTime", {
                type: localStorage
            });
            DetectionTime = DetectionTime ? new Date(DetectionTime) : new Date(CurrentTime.getTime() + 11 * 60 * 1e3);
            const Conversion = Math.abs(DetectionTime - CurrentTime) / (1e3 * 60);
            console.warn(Conversion);
            if (Conversion >= 10) {
                const cookie = Syn.Store("gj", "E/Ex_Cookies");
                cookie && CKOP.Verify(cookie);
            }
            Syn.Menu({
                [lang.RM_00]: {
                    func: () => this.SharedLogin()
                }
            });
            this.MenuToggle();
            this.GlobalMenuToggle();
        }
        async GetSharedDict() {
            return new Promise((resolve, reject) => {
                GM_xmlhttpRequest({
                    method: "GET",
                    responseType: "json",
                    url: "https://raw.githubusercontent.com/Canaan-HS/Script-DataBase/main/Share/ExShare.json",
                    onload: response => {
                        if (response.status === 200) {
                            const data = response.response;
                            if (typeof data === "object" && Object.keys(data).length > 0) {
                                resolve(data);
                            } else {
                                console.error(lang.SS_05);
                                resolve({});
                            }
                        } else {
                            console.error(lang.SS_06);
                            resolve({});
                        }
                    },
                    onerror: error => {
                        console.error(lang.SS_07, error);
                        resolve({});
                    }
                });
            });
        }
        async UpdateShared() {
            const Shared = await this.GetSharedDict();
            if (Object.keys(Shared).length > 0) {
                this.Share = Shared;
                Syn.Store("s", "Share", Shared);
                this.Growl(lang.SS_03, "jGrowl", 1500);
                const modal = Syn.$$(".modal-background");
                if (modal) {
                    setTimeout(() => {
                        modal.remove();
                        this.SharedLogin();
                    }, 800);
                }
            } else {
                this.Growl(lang.SS_04, "jGrowl", 1500);
            }
        }
        async SharedLogin() {
            this.CreateDetection();
            const Share = this.Share, AccountQuantity = Object.keys(Share).length, Igneous = CKOP.Get().igneous;
            let Select = $(`<select id="account-select" class="acc-select"></select>`), Value;
            for (let i = 1; i <= AccountQuantity; i++) {
                if (Share[i][0].value == Igneous) {
                    Value = i;
                }
                Select.append($("<option>").attr({
                    value: i
                }).text(`${lang.SM_16} ${i}`));
            }
            this.modal = $(`
                <div class="modal-background">
                    <div class="acc-modal">
                        <h1>${lang.SM_15}</h1>
                        <div class="acc-select-flex">${Select.prop("outerHTML")}</div>
                        <div class="acc-button-flex">
                            <button class="modal-button" id="update">${lang.SM_17}</button>
                            <button class="modal-button" id="login">${lang.SM_18}</button>
                        </div>
                    </div>
                </div>
            `);
            this.CreateMenu();
            Value && $("#account-select").val(Value);
            const self = this;
            self.on(".modal-background", "click", function (click) {
                click.stopImmediatePropagation();
                const target = click.target;
                if (target.id == "login") {
                    CKOP.ReAdd(Share[+$("#account-select").val()]);
                } else if (target.id == "update") {
                    self.UpdateShared();
                } else if (target.className == "modal-background") {
                    self.DeleteMenu();
                }
            });
        }
        async GetCookieAutomatically() {
            let cookie_box = [];
            for (const [name, value] of Object.entries(CKOP.Get())) {
                cookie_box.push({
                    name: name,
                    value: value
                });
            }
            cookie_box.length > 1 ? this.Cookie_Show(JSON.stringify(cookie_box, null, 4)) : alert(lang.SS_01);
        }
        async Cookie_Show(cookies) {
            this.CreateDetection();
            this.modal = `
                <div class="modal-background">
                    <div class="show-modal">
                    <h1 style="text-align: center;">${lang.SM_01}</h1>
                        <pre><b>${cookies}</b></pre>
                        <div style="text-align: right;">
                            <button class="modal-button" id="save">${lang.SM_02}</button>
                            <button class="modal-button" id="close">${lang.SM_03}</button>
                        </div>
                    </div>
                </div>
            `;
            this.CreateMenu();
            const self = this;
            self.on(".modal-background", "click", function (click) {
                click.stopImmediatePropagation();
                const target = click.target;
                if (target.id == "save") {
                    Syn.Store("s", "E/Ex_Cookies", cookies);
                    self.Growl(lang.SM_05, "jGrowl", 1500);
                    self.DeleteMenu();
                } else if (target.className == "modal-background" || target.id == "close") {
                    self.DeleteMenu();
                }
            });
        }
        async ManualSetting() {
            this.CreateDetection();
            this.modal = `
                <div class="modal-background">
                    <div class="set-modal">
                    <h1>${lang.SM_09}</h1>
                        <form id="set_cookies">
                            <div id="input_cookies" class="set-box">
                                <label>[igneous]:</label><input class="set-list" type="text" name="igneous" placeholder="${lang.SM_10}"><br>
                                <label>[ipb_member_id]:</label><input class="set-list" type="text" name="ipb_member_id" placeholder="${lang.SM_11}" required><br>
                                <label>[ipb_pass_hash]:</label><input class="set-list" type="text" name="ipb_pass_hash" placeholder="${lang.SM_11}" required><hr>
                                <h3>${lang.SM_12}</h3>
                                <label>[sl]:</label><input class="set-list" type="text" name="sl" value="dm_2"><br>
                                <label>[sk]:</label><input class="set-list" type="text" name="sk"><br>
                            </div>
                            <button type="submit" class="modal-button" id="save">${lang.SM_02}</button>
                            <button class="modal-button" id="close">${lang.SM_04}</button>
                        </form>
                    </div>
                </div>
            `;
            this.CreateMenu();
            let cookie;
            const textarea = $("<textarea>").attr({
                style: "margin: 1.15rem auto 0 auto",
                rows: 18,
                cols: 40,
                readonly: true
            }), self = this;
            self.on("#set_cookies", "submit", function (submit) {
                submit.preventDefault();
                submit.stopImmediatePropagation();
                const cookie_list = Array.from($("#set_cookies .set-list")).map(function (input) {
                    const value = $(input).val();
                    return value.trim() !== "" ? {
                        name: $(input).attr("name"),
                        value: value
                    } : null;
                }).filter(Boolean);
                cookie = JSON.stringify(cookie_list, null, 4);
                textarea.val(cookie);
                $("#set_cookies div").append(textarea);
                self.Growl(lang.SM_13, "jGrowl", 3e3);
            });
            self.on(".modal-background", "click", function (click) {
                click.stopImmediatePropagation();
                const target = click.target;
                if (target.className == "modal-background" || target.id == "close") {
                    click.preventDefault();
                    cookie && Syn.Store("s", "E/Ex_Cookies", cookie);
                    self.DeleteMenu();
                }
            });
        }
        async ViewSaveCookie() {
            this.CreateDetection();
            this.modal = `
                <div class="modal-background">
                    <div class="set-modal">
                    <h1>${lang.SM_14}</h1>
                        <div id="view_cookies" style="margin: 0.6rem"></div>
                        <button class="modal-button" id="save">${lang.SM_06}</button>
                        <button class="modal-button" id="close">${lang.SM_04}</button>
                    </div>
                </div>
            `;
            this.CreateMenu();
            const cookie = Syn.Store("gj", "E/Ex_Cookies");
            const textarea = $("<textarea>").attr({
                rows: 20,
                cols: 50,
                id: "view_SC",
                style: "margin-top: 1.25rem;"
            }), self = this;
            textarea.val(JSON.stringify(cookie, null, 4));
            $("#view_cookies").append(textarea);
            self.on(".modal-background", "click", function (click) {
                click.stopImmediatePropagation();
                const target = click.target;
                if (target.id == "save") {
                    GM_notification({
                        title: lang.SM_07,
                        text: lang.SM_08,
                        image: "https://cdn-icons-png.flaticon.com/512/5234/5234222.png",
                        timeout: 3e3
                    });
                    Syn.Store("sj", "E/Ex_Cookies", JSON.parse($("#view_SC").val()));
                    self.DeleteMenu();
                } else if (target.className == "modal-background" || target.id == "close") {
                    self.DeleteMenu();
                }
            });
        }
        async CookieInjection() {
            try {
                CKOP.ReAdd(Syn.Store("gj", "E/Ex_Cookies"));
            } catch (error) {
                alert(lang.SS_02);
            }
        }
        async ClearLogin() {
            CKOP.Delete();
            location.reload();
        }
    }().Main();
    new class Style {
        async Import() {
            let show_style, button_style, button_hover, jGrowl_style, acc_style;
            if (domain == "e-hentai.org") {
                button_hover = "color: #8f4701;";
                jGrowl_style = "background-color: #5C0D12; color: #fefefe;";
                show_style = "background-color: #fefefe; border: 3px ridge #34353b;";
                acc_style = "color: #5C0D12; background-color: #fefefe; border: 2px solid #B5A4A4;";
                button_style = "color: #5C0D12; border: 2px solid #B5A4A4; background-color: #fefefe;";
            } else if (domain == "exhentai.org") {
                button_hover = "color: #989898;";
                jGrowl_style = "background-color: #fefefe; color: #5C0D12;";
                show_style = "background-color: #34353b; border: 2px ridge #5C0D12;";
                acc_style = "color: #f1f1f1; background-color: #34353b; border: 2px solid #8d8d8d;";
                button_style = "color: #fefefe; border: 2px solid #8d8d8d; background-color: #34353b;";
                Syn.AddStyle(`
                    body {
                        padding: 2px;
                        color: #f1f1f1;
                        text-align: center;
                        background: #34353b;
                    }
                `);
            }
            Syn.AddStyle(`
                ${GM_getResourceText("jgrowl-css")}
                .jGrowl {
                    ${jGrowl_style}
                    top: 0;
                    left: 50%;
                    width: auto;
                    z-index: 9999;
                    font-size: 1.3rem;
                    border-radius: 2px;
                    text-align: center;
                    white-space: nowrap;
                    transform: translateX(-50%);
                }
                .modal-background {
                    top: 50%;
                    left: 50%;
                    opacity: 0;
                    width: 100%;
                    height: 100%;
                    z-index: 9999;
                    overflow: auto;
                    position: fixed;
                    transition: 0.6s ease;
                    background-color: rgba(0,0,0,0);
                    transform: translate(-50%, -50%) scale(0.3);
                }
                .acc-modal {
                    ${show_style}
                    width: 16%;
                    overflow: auto;
                    margin: 10rem auto;
                    border-radius: 10px;
                }
                .acc-select-flex {
                    display: flex;
                    align-items: center;
                    flex-direction: initial;
                    justify-content: space-around;
                }
                .acc-button-flex {
                    display: flex;
                    justify-content: center;
                }
                .acc-select {
                    ${acc_style}
                    width: 10rem;
                    padding: 4px;
                    margin: 1.1rem 1.4rem 1.5rem 1.4rem;
                    font-weight: bold;
                    cursor: pointer;
                    font-size: 1.2rem;
                    text-align: center;
                    border-radius: 5px;
                }
                .show-modal {
                    ${show_style}
                    width: 25%;
                    padding: 1.5rem;
                    overflow: auto;
                    margin: 5rem auto;
                    text-align: left;
                    border-radius: 10px;
                    border-collapse: collapse;
                }
                .modal-button {
                    ${button_style}
                    top: 0;
                    margin: 3% 2%;
                    font-size: 14px;
                    font-weight: bold;
                    border-radius: 3px;
                }
                .modal-button:hover, .modal-button:focus {
                    ${button_hover}
                    cursor: pointer;
                    text-decoration: none;
                }
                .set-modal {
                    ${show_style}
                    width: 35rem;
                    padding: 0.3rem;
                    overflow: auto;
                    border-radius: 10px;
                    text-align: center;
                    border-collapse: collapse;
                    margin: 2% auto 8px auto;
                }
                .set-box {
                    display: flex;
                    margin: 0.6rem;
                    font-weight: bold;
                    flex-direction: column;
                    align-items: flex-start;
                }
                .set-list {
                    width: 95%;
                    font-weight: 550;
                    font-size: 1.1rem;
                    text-align: center;
                }
                hr {
                    width: 98%;
                    opacity: 0.2;
                    border: 1px solid;
                    margin-top: 1.3rem;
                }
                label {
                    margin: 0.4rem;
                    font-size: 0.9rem;
                }
            `);
        }
    }().Import();
    function CookieFactory() {
        let Cookie = undefined;
        let Today = new Date();
        Today.setFullYear(Today.getFullYear() + 1);
        let RequiredCookie = ["ipb_member_id", "ipb_pass_hash"];
        if (domain == "exhentai.org") RequiredCookie.unshift("igneous");
        return {
            Get: () => Cookies.get(),
            Add: function (AddCookies) {
                for (Cookie of AddCookies) {
                    Cookies.set(Cookie.name, Cookie.value, {
                        expires: Today
                    });
                }
                Syn.Storage("DetectionTime", {
                    type: localStorage,
                    value: new Date().getTime()
                });
                location.reload();
            },
            Delete: function () {
                for (Cookie of Object.keys(this.Get())) {
                    Cookies.remove(Cookie, {
                        path: "/"
                    });
                    Cookies.remove(Cookie, {
                        path: "/",
                        domain: `.${domain}`
                    });
                }
            },
            ReAdd: function (Cookies) {
                this.Delete();
                this.Add(Cookies);
            },
            Verify: function (Cookies) {
                const VCookie = new Set(Object.keys(this.Get()));
                const Result = RequiredCookie.every(Name => VCookie.has(Name));
                if (!Result) {
                    this.Delete();
                    this.Add(Cookies);
                }
            }
        };
    }
    function language(lang) {
        const Display = {
            Traditional: {
                RM_00: "🍪 共享登入",
                RM_C0: "📂 展開菜單",
                RM_C1: "📁 摺疊菜單",
                RM_01: "📜 自動獲取",
                RM_02: "📝 手動輸入",
                RM_03: "🔍 查看保存",
                RM_04: "🔃 手動注入",
                RM_05: "🗑️ 清除登入",
                SM_01: "確認選擇的 Cookies",
                SM_02: "確認保存",
                SM_03: "取消退出",
                SM_04: "退出選單",
                SM_05: "保存成功!",
                SM_06: "更改保存",
                SM_07: "變更通知",
                SM_08: "已保存變更",
                SM_09: "設置 Cookies",
                SM_10: "要登入 Ex 才需要填寫",
                SM_11: "必填項目",
                SM_12: "下方選填 也可不修改",
                SM_13: "[確認輸入正確]按下退出選單保存",
                SM_14: "當前設置 Cookies",
                SM_15: "帳戶選擇",
                SM_16: "帳戶",
                SM_17: "更新",
                SM_18: "登入",
                SS_01: "未獲取到 Cookies !!\n\n請先登入帳戶",
                SS_02: "未檢測到可注入的 Cookies !!\n\n請從選單中進行設置",
                SS_03: "共享數據獲取成功",
                SS_04: "共享數據獲取失敗",
                SS_05: "請求為空數據",
                SS_06: "連線異常, 地址類型可能是錯的",
                SS_07: "請求錯誤: "
            },
            Simplified: {
                RM_00: "🍪 共享登录",
                RM_C0: "📂 展开菜单",
                RM_C1: "📁 折叠菜单",
                RM_01: "📜 自动获取",
                RM_02: "📝 手动输入",
                RM_03: "🔍 查看保存",
                RM_04: "🔃 手动注入",
                RM_05: "🗑️ 清除登录",
                SM_01: "确认选择的 Cookies",
                SM_02: "确认保存",
                SM_03: "取消退出",
                SM_04: "退出菜单",
                SM_05: "保存成功!",
                SM_06: "更改保存",
                SM_07: "变更通知",
                SM_08: "已保存变更",
                SM_09: "设置 Cookies",
                SM_10: "要登录 Ex 才需要填写",
                SM_11: "必填项目",
                SM_12: "下方选填 也可不修改",
                SM_13: "[确认输入正确]按下退出菜单保存",
                SM_14: "当前设置 Cookies",
                SM_15: "账户选择",
                SM_16: "账户",
                SM_17: "更新",
                SM_18: "登录",
                SS_01: "未获取到 Cookies !!\n\n请先登录账户",
                SS_02: "未检测到可注入的 Cookies !!\n\n请从菜单中进行设置",
                SS_03: "共享数据获取成功",
                SS_04: "共享数据获取失败",
                SS_05: "请求为空数据",
                SS_06: "连接异常, 地址类型可能是错的",
                SS_07: "请求错误: "
            },
            English: {
                RM_00: "🍪 Shared Login",
                RM_C0: "📂 Expand Menu",
                RM_C1: "📁 Collapse Menu",
                RM_01: "📜 Auto Retrieve",
                RM_02: "📝 Manual Input",
                RM_03: "🔍 View Saved",
                RM_04: "🔃 Manual Injection",
                RM_05: "🗑️ Clear Login",
                SM_01: "Confirm Selected Cookies",
                SM_02: "Confirm Save",
                SM_03: "Cancel Exit",
                SM_04: "Exit Menu",
                SM_05: "Save Successful!",
                SM_06: "Change Saved",
                SM_07: "Change Notification",
                SM_08: "Changes Saved",
                SM_09: "Set Cookies",
                SM_10: "Required for Ex Login",
                SM_11: "Mandatory Field",
                SM_12: "Optional Below, No Changes Needed",
                SM_13: "[Confirm Correct Input] Press Exit Menu to Save",
                SM_14: "Current Set Cookies",
                SM_15: "Account Selection",
                SM_16: "Account",
                SM_17: "Update",
                SM_18: "Login",
                SS_01: "No Cookies Retrieved !!\n\nPlease Login First",
                SS_02: "No Injectable Cookies Detected !!\n\nPlease Set in Menu",
                SS_03: "Shared Data Retrieval Successful",
                SS_04: "Shared Data Retrieval Failed",
                SS_05: "Request Contains No Data",
                SS_06: "Connection Error, Address Type May Be Wrong",
                SS_07: "Request Error: "
            },
            Korea: {
                RM_00: "🍪 공유 로그인",
                RM_C0: "📂 메뉴 확장",
                RM_C1: "📁 메뉴 축소",
                RM_01: "📜 자동 가져오기",
                RM_02: "📝 수동 입력",
                RM_03: "🔍 저장 보기",
                RM_04: "🔃 수동 주입",
                RM_05: "🗑️ 로그인 지우기",
                SM_01: "선택한 쿠키 확인",
                SM_02: "저장 확인",
                SM_03: "취소 종료",
                SM_04: "메뉴 종료",
                SM_05: "저장 성공!",
                SM_06: "변경 저장",
                SM_07: "변경 알림",
                SM_08: "변경 사항 저장됨",
                SM_09: "쿠키 설정",
                SM_10: "Ex 로그인에 필요",
                SM_11: "필수 항목",
                SM_12: "아래 선택 항목, 변경 필요 없음",
                SM_13: "[입력 정확성 확인] 메뉴 종료를 눌러 저장",
                SM_14: "현재 설정된 쿠키",
                SM_15: "계정 선택",
                SM_16: "계정",
                SM_17: "업데이트",
                SM_18: "로그인",
                SS_01: "쿠키를 가져오지 못했습니다 !!\n\n먼저 로그인 해주세요",
                SS_02: "주입 가능한 쿠키를 감지하지 못했습니다 !!\n\n메뉴에서 설정해 주세요",
                SS_03: "공유 데이터 가져오기 성공",
                SS_04: "공유 데이터 가져오기 실패",
                SS_05: "요청 데이터가 비어 있습니다",
                SS_06: "연결 오류, 주소 유형이 잘못되었을 수 있습니다",
                SS_07: "요청 오류: "
            },
            Japan: {
                RM_00: "🍪 共有ログイン",
                RM_C0: "📂 メニュー展開",
                RM_C1: "📁 メニュー折りたたみ",
                RM_01: "📜 自動取得",
                RM_02: "📝 手動入力",
                RM_03: "🔍 保存を表示",
                RM_04: "🔃 手動注入",
                RM_05: "🗑️ ログインクリア",
                SM_01: "選択したクッキーを確認",
                SM_02: "保存を確認",
                SM_03: "キャンセルして終了",
                SM_04: "メニューを終了",
                SM_05: "保存成功!",
                SM_06: "変更を保存",
                SM_07: "変更通知",
                SM_08: "変更が保存されました",
                SM_09: "クッキーを設定",
                SM_10: "Exログインに必要",
                SM_11: "必須項目",
                SM_12: "下の選択肢、変更の必要はありません",
                SM_13: "[入力が正しいことを確認] メニュー終了を押して保存",
                SM_14: "現在設定されているクッキー",
                SM_15: "アカウント選択",
                SM_16: "アカウント",
                SM_17: "更新",
                SM_18: "ログイン",
                SS_01: "クッキーを取得できませんでした!!\n\n先にログインしてください",
                SS_02: "注入可能なクッキーが検出されませんでした!!\n\nメニューから設定してください",
                SS_03: "共有データの取得に成功しました",
                SS_04: "共有データの取得に失敗しました",
                SS_05: "リクエストが空データです",
                SS_06: "接続エラー、アドレスタイプが間違っている可能性があります",
                SS_07: "リクエストエラー: "
            }
        }, Match = {
            ko: Display.Korea,
            ja: Display.Japan,
            "en-US": Display.English,
            "zh-CN": Display.Simplified,
            "zh-SG": Display.Simplified,
            "zh-TW": Display.Traditional,
            "zh-HK": Display.Traditional,
            "zh-MO": Display.Traditional
        };
        return Match[lang] ?? Match["en-US"];
    }
})();