Fast_Add_Cart

Add to cart without redirect to cart page, also provide import/export cart feature.

스크립트 설치?
개발자의 추천 스크립트

Hidden_DLC_Helper는 어떤가요?

스크립트 설치
// ==UserScript==
// @name:zh-CN      Steam快速添加购物车
// @name            Fast_Add_Cart
// @namespace       https://blog.chrxw.com
// @supportURL      https://blog.chrxw.com/scripts.html
// @contributionURL https://afdian.com/@chr233
// @version         4.6
// @description:zh-CN  超级方便的添加购物车体验, 不用跳转商店页, 附带导入导出购物车功能.
// @description     Add to cart without redirect to cart page, also provide import/export cart feature.
// @author          Chr_
// @match           https://store.steampowered.com/*
// @license         AGPL-3.0
// @icon            https://blog.chrxw.com/favicon.ico
// @grant           GM_addStyle
// @grant           GM_setClipboard
// @grant           GM_setValue
// @grant           GM_getValue
// @grant           GM_registerMenuCommand
// ==/UserScript==

(async () => {
    "use strict";

    // 多语言
    const LANG = {
        ZH: {
            langName: "中文",
            changeLang: "修改插件语言",
            facInputBoxPlaceHolder:
                "一行一条, 自动忽略【#】后面的内容, 支持的格式如下: (自动保存)",
            storeLink: "商店链接",
            steamDBLink: "DB链接",
            import: "导入",
            importDesc: "从文本框批量添加购物车(从上到下导入)",
            importDesc2: "当前页面无法导入购物车",
            export: "导出",
            exportDesc: "将购物车内容导出至文本框",
            exportConfirm: "输入框中含有内容, 请选择操作?",
            exportConfirmReplace: "覆盖原有内容",
            exportConfirmAppend: "添加到最后",
            copy: "复制",
            copyDesc: "复制文本框中的内容",
            copyDone: "复制到剪贴板成功",
            reset: "清除",
            resetDesc: "清除文本框和已保存的数据",
            resetConfirm: "您确定要清除文本框和已保存的数据吗?",
            history: "历史",
            historyDesc: "查看购物车历史记录",
            reload: "刷新",
            reloadDesc: "重新读取保存的购物车内容",
            reloadConfirm: "您确定要重新读取保存的购物车数据吗?",
            goBack: "返回",
            goBackDesc: "返回你当前的购物车",
            clear: "清空购物车",
            clearDesc: "清空购物车",
            clearConfirm: "您确定要移除所有您购物车中的物品吗?",
            help: "帮助",
            helpDesc: "显示帮助",
            helpTitle: "插件版本",
            formatError: "格式有误",
            chooseSub: "请选择SUB",
            operation: "操作中……",
            operationDone: "操作完成",
            addCart: "添加购物车",
            addCartTips: "添加到购物车……",
            addCartErrorSubNotFount: "未识别到SubID",
            noSubDesc: "可能尚未发行或者是免费游戏",
            inCart: "在购物车中",
            importingTitle: "正在导入购物车……",
            add: "添加",
            toCart: "到购物车",
            tips: "提示",
            ok: "是",
            no: "否",
            fetchingSubs: "读取可用SUB",
            noSubFound: "未找到可用SUB",
            networkError: "网络错误",
            addCartSuccess: "添加购物车成功",
            addCartError: "添加购物车失败",
            networkRequestError: "网络请求失败",
            unknownError: "未知错误",
            unrecognizedResult: "返回了未知结果",
            batchExtract: "批量提取",
            batchExtractDone: "批量提取完成",
            batchDesc: "AppID已提取, 可以在购物车页批量导入",
            onlyOnsale: " 仅打折",
            onlyOnsaleDesc: "勾选后批量导入时仅导入正在打折的游戏.",
            onlyOnsaleDesc2: "勾选后批量导出时仅导出正在打折的游戏.",
            notOnSale: "尚未打折, 跳过",
        },
        EN: {
            langName: "English",
            changeLang: "Change plugin language",
            facInputBoxPlaceHolder:
                "One line one item, ignore the content after #, support format: (auto save)",
            storeLink: "Store link",
            steamDBLink: "DB link",
            import: "Import",
            importDesc: "Batch add cart from textbox (from top to bottom)",
            importDesc2: "Current page can't import cart",
            export: "Export",
            exportDesc: "Export cart content to textbox",
            exportConfirm: "Textbox contains content, please choose operation?",
            exportConfirmReplace: "Replace original content",
            exportConfirmAppend: "Append to the end",
            copy: "Copy",
            copyDesc: "Copy textbox content",
            copyDone: "Copy to clipboard success",
            reset: "Reset",
            resetDesc: "Clear textbox and saved data",
            resetConfirm: "Are you sure to clear textbox and saved cart data?",
            history: "History",
            historyDesc: "View cart history",
            reload: "Reload",
            reloadDesc: "Reload saved cart date",
            reloadConfirm: "Are you sure to reload saved cart data?",
            goBack: "Back",
            goBackDesc: "Back to your cart",
            clear: "Clear",
            clearDesc: "Clear cart",
            clearConfirm: "Are you sure to remove all items in your cart?",
            help: "Help",
            helpDesc: "Show help",
            helpTitle: "Plugin Version",
            formatError: "Format error",
            chooseSub: "Please choose SUB",
            operation: "Operation in progress……",
            operationDone: "Operation done",
            addCart: "Add cart",
            addCartTips: "Adding to cart……",
            addCartErrorSubNotFount: "Unrecognized SubID",
            noSubDesc: "Maybe not released or free game",
            inCart: "In cart",
            importingTitle: "Importing cart……",
            add: "Add",
            toCart: "To cart",
            tips: "Tips",
            ok: "OK",
            no: "No",
            fetchingSubs: "Fetching available SUB",
            noSubFound: "No available SUB",
            networkError: "Network error",
            addCartSuccess: "Add cart success",
            addCartError: "Add cart failed",
            networkRequestError: "Network request failed",
            unknownError: "Unknown error",
            unrecognizedResult: "Returned unrecognized result",
            batchExtract: "Extract Items",
            batchExtractDone: "Batch Extract Done",
            batchDesc: "AppID list now saved, goto cart page to use batch import.",
            onlyOnsale: " Only on sale",
            onlyOnsaleDesc:
                "If checked, script will ignore games that is not on sale when import cart.",
            onlyOnsaleDesc2:
                "If checked, script will ignore games that is not on sale when export cart.",
            notOnSale: "Not on sale, skip",
        },
    };

    // 判断语言
    let language = GM_getValue("lang", "ZH");
    if (!language in LANG) {
        language = "ZH";
        GM_setValue("lang", language);
    }
    // 获取翻译文本
    const t = (key) => LANG[language][key] || key;

    {
        // 自动弹出提示
        const languageTips = GM_getValue("languageTips", true);
        if (languageTips && language === "ZH") {
            if (!document.querySelector("html").lang.startsWith("zh")) {
                ShowConfirmDialog(
                    "tips",
                    "Fast add cart now support English, switch?",
                    "Using English",
                    "Don't show again"
                )
                    .done(() => {
                        GM_setValue("lang", "EN");
                        GM_setValue("languageTips", false);
                        window.location.reload();
                    })
                    .fail((bool) => {
                        if (bool) {
                            showAlert(
                                "",
                                "You can switch the plugin's language using TamperMonkey's menu."
                            );
                            GM_setValue("languageTips", false);
                        }
                    });
            }
        }
        //注册菜单
        GM_registerMenuCommand(`${t("changeLang")} (${t("langName")})`, () => {
            switch (language) {
                case "EN":
                    language = "ZH";
                    break;
                case "ZH":
                    language = "EN";
                    break;
            }
            GM_setValue("lang", language);
            window.location.reload();
        });
    }

    //获取商店语言和区域
    const { LANGUAGE: storeLanguage, COUNTRY: userCountry } = JSON.parse(document.querySelector("#application_config")?.getAttribute("data-config") ?? "{}");
    const { webapi_token: accessToken } = JSON.parse(document.querySelector("#application_config")?.getAttribute("data-store_user_config") ?? "{}");

    const G_Objs = {};

    //初始化
    const pathname = window.location.pathname;
    if (
        pathname === "/search/" ||
        pathname === "/" ||
        pathname.startsWith("/tags/")
    ) {
        //搜索页,主页,标签页
        return;

    } else if (
        pathname.startsWith("/publisher/") ||
        pathname.startsWith("/franchise/") ||
        pathname.startsWith("/developer/")
    ) {
        //发行商主页
        return;

    } else if (
        pathname.startsWith("/app/") ||
        pathname.startsWith("/sub/") ||
        pathname.startsWith("/bundle/")
    ) {
        //商店详情页
        return;

    } else if (pathname.startsWith("/wishlist/")) {
        //愿望单页
        return;

    } else if (pathname.startsWith("/cart")) {
        //购物车页

        function genBr() {
            return document.createElement("br");
        }
        function genBtn(text, title, onclick) {
            let btn = document.createElement("button");
            btn.textContent = text;
            btn.title = title;
            btn.className = "btn_medium btnv6_blue_hoverfade fac_cartbtns";
            btn.addEventListener("click", onclick);
            return btn;
        }
        function genSpan(text) {
            let span = document.createElement("span");
            span.textContent = text;
            return span;
        }
        function genTxt(value, placeholder) {
            const t = document.createElement("textarea");
            t.className = "fac_inputbox";
            t.placeholder = placeholder;
            t.value = value;
            return t;
        }
        function genChk(name, title, checked = false) {
            const l = document.createElement("label");
            const i = document.createElement("input");
            const s = genSpan(name);
            i.textContent = name;
            i.title = title;
            i.type = "checkbox";
            i.className = "fac_checkbox";
            i.checked = checked;
            l.title = title;
            l.appendChild(i);
            l.appendChild(s);
            return [l, i];
        }

        const savedCart = GM_getValue("fac_cart") ?? "";
        const placeHolder = [
            t("facInputBoxPlaceHolder"),
            `1. ${t("storeLink")}: https://store.steampowered.com/app/xxx`,
            `2. ${t("steamDBLink")}:  https://steamdb.info/app/xxx`,
            "3. appID:       xxx a/xxx app/xxx",
            "4. subID:       s/xxx sub/xxx",
            "5. bundleID:    b/xxx bundle/xxx",
        ].join("\n");

        const inputBox = genTxt(savedCart, placeHolder);

        function fitInputBox() {
            inputBox.style.height =
                Math.min(inputBox.value.split("\n").length * 20 + 20, 900).toString() +
                "px";
        }

        inputBox.addEventListener("input", fitInputBox);
        G_Objs.inputBox = inputBox;
        fitInputBox();

        const originResetBtn = document.querySelector("div.remove_ctn");
        if (originResetBtn != null) {
            originResetBtn.style.display = "none";
        }

        const [lblDiscount, chkDiscount] = genChk(
            t("onlyOnsale"),
            t("onlyOnsaleDesc"),
            GM_getValue("fac_discount") ?? false
        );
        G_Objs.chkDiscount = chkDiscount;

        const btnImport = genBtn(`🔼${t("import")}`, t("importDesc"), async () => {
            inputBox.value = await importCart(
                inputBox.value,
                chkDiscount.checked
            );
        });

        const histryPage = pathname.search("history") !== -1;
        if (histryPage) {
            btnImport.disabled = true;
            btnImport.title = t("importDesc2");
        }

        const [lblDiscount2, chkDiscount2] = genChk(
            t("onlyOnsale"),
            t("onlyOnsaleDesc2"),
            GM_getValue("fac_discount2") ?? false
        );
        G_Objs.chkDiscount2 = chkDiscount2;

        const btnExport = genBtn(`🔽${t("export")}`, t("exportDesc"), async () => {
            let currentValue = inputBox.value.trim();
            const now = new Date().toLocaleString();

            var data = await exportCart(chkDiscount2.checked);

            if (currentValue !== "") {
                ShowConfirmDialog(
                    "",
                    t("exportConfirm"),
                    t("exportConfirmReplace"),
                    t("exportConfirmAppend")
                )
                    .done(() => {
                        inputBox.value = `========【${now}】=========\n` + data;
                        fitInputBox();
                    })
                    .fail((bool) => {
                        if (bool) {
                            inputBox.value = currentValue + `\n========【${now}】=========\n` + data;
                            fitInputBox();
                        }
                    });
            } else {
                inputBox.value =
                    `========【${now}】=========\n` + exportCart(chkDiscount2.checked);
                fitInputBox();
            }
        });

        const btnConvertToGift = genBtn("转送礼", "将购物车项目转换为送礼", () => {
            editCart(true, false);
        });

        const btnConvertToSelf = genBtn("转自用", "将购物车项目转换为为自己购买", () => {
            editCart(false, false);
        });

        const btnCopy = genBtn(`📋${t("copy")}`, t("copyDesc"), () => {
            GM_setClipboard(inputBox.value, "text");
            showAlert(t("tips"), t("copyDone"), true);
        });
        const btnClear = genBtn(`🗑️${t("reset")}`, t("resetDesc"), () => {
            ShowConfirmDialog("", t("resetConfirm"), t("ok"), t("no")).done(() => {
                inputBox.value = "";
                GM_setValue("fac_cart", "");
                fitInputBox();
            });
        });
        const btnReload = genBtn(`🔃${t("reload")}`, t("reloadDesc"), () => {
            ShowConfirmDialog("", t("reloadConfirm"), t("ok"), t("no")).done(() => {
                const s = GM_getValue("fac_cart") ?? "";
                inputBox.value = s;
                fitInputBox();
            });
        });
        const btnHistory = genBtn(`📜${t("history")}`, t("historyDesc"), () => {
            window.location.href =
                "https://help.steampowered.com/zh-cn/accountdata/ShoppingCartHistory";
        });
        const btnBack = genBtn(`↩️${t("goBack")}`, t("goBackDesc"), () => {
            window.location.href = "https://store.steampowered.com/cart/";
        });
        const btnForget = genBtn(`⚠️${t("clear")}`, t("clearDesc"), () => {
            ShowConfirmDialog("", t("clearConfirm"), t("ok"), t("no")).done(() => {
                deleteAccountCart()
                    .then(() => {
                        location.reload();
                    })
                    .catch((err) => {
                        console.error(err);
                        showAlert("出错", err, false);
                    });
            });
        });
        const btnHelp = genBtn(`🔣${t("help")}`, t("helpDesc"), () => {
            const {
                script: { version },
            } = GM_info;
            showAlert(
                `${t("helpTitle")} ${version}`,
                [
                    `<p>【🔼${t("import")}】${t("importDesc")}</p>`,
                    `<p>【✅${t("onlyOnsale")}】${t("onlyOnsaleDesc")}</p>`,
                    `<p>【🔽${t("export")}】${t("exportDesc")}</p>`,
                    `<p>【✅${t("onlyOnsale")}】${t("onlyOnsaleDesc2")}</p>`,
                    `<p>【📋${t("copy")}】${t("copyDesc")}</p>`,
                    `<p>【🗑️${t("reset")}】${t("resetDesc")}。</p>`,
                    `<p>【📜${t("history")}】${t("historyDesc")}</p>`,
                    `<p>【↩️${t("goBack")}】${t("goBackDesc")}</p>`,
                    `<p>【⚠️${t("clear")}】${t("clearDesc")}</p>`,
                    `<p>【🔣${t("help")}】${t("helpDesc")}</p>`,
                    `<p>【<a href="https://keylol.com/t747892-1-1" target="_blank">发布帖</a>】 【<a href="https://blog.chrxw.com/scripts.html" target="_blank">脚本反馈</a>】 【Developed by <a href="https://steamcommunity.com/id/Chr_" target="_blank">Chr_</a>】</p>`,
                ].join("<br>"),
                true
            );
        });

        const btnArea = document.createElement("div");
        btnArea.appendChild(btnImport);
        // btnArea.appendChild(btnImport2);
        btnArea.appendChild(lblDiscount);
        btnArea.appendChild(genSpan(" | "));
        btnArea.appendChild(btnExport);
        btnArea.appendChild(lblDiscount2);
        btnArea.appendChild(genSpan(" | "));
        btnArea.appendChild(btnConvertToGift);
        btnArea.appendChild(btnConvertToSelf);
        btnArea.appendChild(genSpan(" | "));
        btnArea.appendChild(btnHelp);

        const btnArea2 = document.createElement("div");
        btnArea2.appendChild(btnCopy);
        btnArea2.appendChild(btnClear);
        btnArea2.appendChild(btnReload);
        btnArea2.appendChild(genSpan(" | "));
        btnArea2.appendChild(histryPage ? btnBack : btnHistory);
        btnArea2.appendChild(genSpan(" | "));
        btnArea2.appendChild(btnForget);

        window.addEventListener("beforeunload", () => {
            GM_setValue("fac_cart", inputBox.value);
            GM_setValue("fac_discount", chkDiscount.checked);
            GM_setValue("fac_discount2", chkDiscount2.checked);
        });

        //等待购物车加载完毕, 显示额外面板
        const timer = setInterval(() => {
            const continer = document.querySelector("div[data-featuretarget='react-root']>div>div:last-child>div:last-child>div:first-child>div:last-child");
            if (continer) {
                clearInterval(timer);
                continer.appendChild(btnArea);
                continer.appendChild(genBr());
                continer.appendChild(inputBox);
                continer.appendChild(genBr());
                continer.appendChild(btnArea2);
            }
        }, 500);
    }

    // getStoreItem([730], null, null).then((data) => console.log(data)).catch(err => console.error(err))
    // getAccountCart().then((data) => console.log(data)).catch(err => console.error(err))
    // addItemsToAccountCart(null, [28627]).then((data) => console.log(data)).catch(err => console.error(err))

    //始终在右上角显示购物车按钮
    const cart_btn = document.getElementById("store_header_cart_btn");
    if (cart_btn !== null) {
        cart_btn.style.display = "";
    }

    //导入购物车
    function importCart(text, onlyOnSale = false) {
        const regFull = new RegExp(/(app|a|bundle|b|sub|s)\/(\d+)/);
        const regShort = new RegExp(/^([\s]*|)(\d+)/);

        return new Promise(async (resolve, reject) => {
            const dialog = showAlert(
                "导入购物车",
                `<h2 id="fac_diag" class="fac_diag">${t("operation")}</h2>`,
                true
            );

            const timer = setInterval(async () => {
                let txt = document.getElementById("fac_diag");
                if (txt) {
                    clearInterval(timer);

                    const txts = text.split("\n");

                    const result = [];

                    const appIds = [];
                    const subIds = [];
                    const bundleIds = [];

                    const targetSubIds = [];
                    const targetBundleIds = [];

                    try {
                        txt.textContent = "0/4 开始读取输入信息";

                        for (let line of txts) {
                            if (line.trim() === "") {
                                continue;
                            }
                            const tmp = line.split("#")[0];

                            const match = line.match(regFull) ?? line.match(regShort);
                            if (!match) {
                                if (line.search("=====") === -1) {
                                    result.push(`${tmp} #${t("formatError")}`);
                                } else {
                                    result.push(line);
                                }
                                continue;
                            }

                            let [_, type, subID] = match;
                            subID = parseInt(subID);
                            if (subID !== subID) {
                                result.push(`${tmp} #${t("formatError")}`);
                                continue;
                            }

                            switch (type.toLowerCase()) {
                                case "":
                                case "a":
                                case "app":
                                    type = "app";
                                    appIds.push(subID);
                                    break;
                                case "s":
                                case "sub":
                                    type = "sub";
                                    subIds.push(subID);
                                    break;
                                case "b":
                                case "bundle":
                                    type = "bundle";
                                    bundleIds.push(subID);
                                    break;
                                default:
                                    result.push(`${tmp} #${t("formatError")}`);
                                    continue;
                            }
                        }

                        const count = appIds.length + subIds.length + bundleIds.length;
                        txt.textContent = `1/4 成功读取 ${count} 个输入内容`;

                        if (count > 0) {
                            txt.textContent = "1/4 开始读取游戏信息";
                            const store_items = await getStoreItem(appIds, subIds, bundleIds);

                            console.log(store_items);

                            txt.textContent = "2/4 读取游戏信息成功";

                            for (let { appid, purchase_options } of store_items) {
                                if (!purchase_options) { continue; }

                                //输入值包含AppId, 解析可购买项
                                if (appIds.includes(appid)) {
                                    if (purchase_options.length >= 1) {
                                        const { packageid, bundleid, purchase_option_name: name, discount_pct: discount, formatted_final_price: price } = purchase_options[0];
                                        if (discount) {
                                            if (packageid) {
                                                result.push(`sub/${packageid} #app/${appid} #${name} 💳 ${price} 🔖 ${discount}`);
                                                targetSubIds.push(packageid);
                                            } else if (bundleid) {
                                                result.push(`bundle/${bundleid} #app/${appid} #${name} 💳 ${price} 🔖 ${discount}`);
                                                targetBundleIds.push(bundleid);
                                            }
                                        } else {
                                            if (packageid) {
                                                if (!onlyOnSale) {
                                                    result.push(`sub/${packageid} #app/${appid} #${name} 💳 ${price}`);
                                                    targetSubIds.push(packageid);
                                                } else {
                                                    result.push(`sub/${packageid} #app/${appid} #排除 #${name} 💳 ${price}`);
                                                }
                                            } else if (bundleid) {
                                                if (!onlyOnSale) {
                                                    result.push(`bundle/${bundleid} #app/${appid} #${name} 💳 ${price}`);
                                                    targetBundleIds.push(bundleid);
                                                } else {
                                                    result.push(`bundle/${bundleid} #app/${appid} #排除 #${name} 💳 ${price}`);
                                                }
                                            }
                                        }
                                    } else {
                                        result.push(`${tmp} #无可购买项目`);
                                    }
                                }

                                for (let { packageid, bundleid, purchase_option_name: name, discount_pct: discount, formatted_final_price: price } of purchase_options) {
                                    if (discount) {
                                        if (packageid && subIds.includes(packageid)) {
                                            result.push(`sub/${packageid} #${name} 💳 ${price} 🔖 ${discount}`);
                                            targetSubIds.push(packageid);
                                        } else if (bundleid && bundleIds.includes(bundleid)) {
                                            result.push(`bundle/${bundleid} #${name} 💳 ${price} 🔖 ${discount}`);
                                            targetBundleIds.push(bundleid);
                                        }
                                    } else {
                                        if (packageid && subIds.includes(packageid)) {
                                            if (!onlyOnSale) {
                                                result.push(`sub/${packageid} #${name} 💳 ${price}`);
                                                targetSubIds.push(packageid);
                                            } else {
                                                result.push(`sub/${packageid} #排除 #${name} 💳 ${price}`);

                                            }
                                        } else if (bundleid && bundleIds.includes(bundleid)) {
                                            if (!onlyOnSale) {
                                                result.push(`bundle/${bundleid} #${name} 💳 ${price}`);
                                                targetBundleIds.push(bundleid);
                                            } else {
                                                result.push(`bundle/${bundleid} #排除 #${name} 💳 ${price}`);
                                            }
                                        }
                                    }
                                }
                            }

                            txt.textContent = "3/4 解析游戏信息成功";

                            const data = await addItemsToAccountCart(targetSubIds, targetBundleIds, false);
                            console.log(data);

                            txt.textContent = "4/4 导入购物车成功";

                            dialog.Dismiss();

                            resolve(result.join("\n"));

                            setTimeout(() => {
                                window.location.reload();
                            }, 1000);

                        } else {
                            txt.textContent = "4/4 尚未输入有效内容";
                            resolve(result.join("\n"));
                        }

                    } catch (err) {
                        txt.textContent = "导出购物车失败";
                        console.error(err);
                        resolve(result.join("\n"));
                    }
                }
            }, 200);
        });
    }
    //导出购物车
    function exportCart(onlyOnsale = false) {
        return new Promise(async (resolve, reject) => {
            const dialog = showAlert(
                "导出购物车",
                `<h2 id="fac_diag" class="fac_diag">${t("operation")}</h2>`,
                true
            );

            const timer = setInterval(async () => {
                let txt = document.getElementById("fac_diag");
                if (txt) {
                    clearInterval(timer);

                    const result = [];

                    const subIds = [];
                    const bundleIds = [];
                    const gameNames = {};

                    try {
                        txt.textContent = "0/4 开始读取账号购物车";

                        const { line_items } = await getAccountCart();

                        if (line_items) {
                            for (let { packageid, bundleid } of line_items) {
                                if (packageid) {
                                    subIds.push(packageid);
                                }
                                else if (bundleid) {
                                    bundleIds.push(bundleid);
                                }
                            }
                        }

                        const count = subIds.length + bundleIds.length;
                        txt.textContent = `1/4 成功读取 ${count} 个购物车内容`;

                        if (count > 0) {
                            txt.textContent = "1/4 开始读取游戏信息";
                            const store_items = await getStoreItem(null, subIds, bundleIds);
                            txt.textContent = "2/4 读取游戏信息成功";

                            for (let { purchase_options } of store_items) {
                                if (!purchase_options) { continue; }

                                for (let { packageid, bundleid, purchase_option_name, discount_pct } of purchase_options) {
                                    let key;
                                    if (packageid) {
                                        key = `sub/${packageid}`;
                                    } else if (bundleid) {
                                        key = `bundle/${bundleid}`;
                                    }
                                    else {
                                        continue;
                                    }
                                    gameNames[key] = [`${purchase_option_name}`, discount_pct];
                                }
                            }

                            txt.textContent = "3/4 解析游戏信息成功";
                            txt.textContent = "3/4 开始导出购物车信息";
                            if (line_items) {
                                for (let { packageid, bundleid, price_when_added: { formatted_amount } } of line_items) {
                                    let key;
                                    if (packageid) {
                                        key = `sub/${packageid}`;
                                    } else if (bundleid) {
                                        key = `bundle/${bundleid}`;
                                    }
                                    const [name, discount] = gameNames[key] ?? "_";
                                    if (discount) {
                                        result.push(`${key} #${name} 💳 ${formatted_amount} 🔖 ${discount}`);
                                    } else if (!onlyOnsale) {
                                        result.push(`${key} #${name} 💳 ${formatted_amount}`);
                                    }
                                }
                            }

                            txt.textContent = "3/4 导出购物车信息成功";
                            dialog.Dismiss();

                            resolve(result.join("\n"));

                        } else {
                            txt.textContent = "4/4 购物车内容为空";
                            resolve(result.join("\n"));
                        }

                    } catch (err) {
                        txt.textContent = "读取账号购物车失败";
                        console.error(err);
                        resolve(result.join("\n"));
                    }
                }
            }, 200);
        });
    }

    //编辑购物车
    async function editCart(setToGift = false, setToPrivate = false) {
        const setGiftInfo = await inputGiftee(setToGift);

        return new Promise(async (resolve, reject) => {
            const dialog = showAlert(
                "编辑购物车",
                `<h2 id="fac_diag" class="fac_diag">${t("operation")}</h2>`,
                true
            );

            const timer = setInterval(async () => {
                let txt = document.getElementById("fac_diag");
                if (txt) {
                    clearInterval(timer);

                    const lineItemIds = [];

                    try {
                        txt.textContent = "0/3 开始读取账号购物车";

                        const { line_items } = await getAccountCart();

                        if (line_items) {
                            for (const { line_item_id, flags: { is_gift, is_private }, gift_info } of line_items) {
                                const accountid_giftee = gift_info?.accountid_giftee?.toString() ?? "";

                                console.log(line_item_id, is_gift, is_private, accountid_giftee);

                                //跳过无需处理的id
                                if (setToGift) {
                                    if (is_gift && setGiftInfo?.accountid_giftee == accountid_giftee) {
                                        continue;
                                    }
                                } else if (setToPrivate) {
                                    if (is_private) {
                                        continue;
                                    }
                                }
                                else {
                                    if (!is_gift && !is_private) {
                                        continue;
                                    }
                                }

                                lineItemIds.push(line_item_id);
                            }
                        }

                        console.log(lineItemIds);

                        const count = lineItemIds.length;
                        txt.textContent = `1/3 共计 ${count} 个待修改购物车内容`;

                        if (count > 0) {
                            for (let i = 0; i < count; i++) {
                                const itemId = lineItemIds[i];
                                await editAccountCart(itemId, setToGift, setToPrivate, setGiftInfo);
                                txt.textContent = `2/3 当前进度 ${i + 1} / ${count}`;
                            }

                            txt.textContent = "3/3 批量修改购物车成功";

                            setTimeout(() => dialog.Dismiss(), 1000);
                            resolve();
                        } else {
                            txt.textContent = "3/3 购物车无需修改";

                            setTimeout(() => dialog.Dismiss(), 1000);
                            resolve();
                        }

                    } catch (err) {
                        txt.textContent = "读取账号购物车失败";
                        console.error(err);
                        resolve();
                    }
                }
            }, 200);
        });
    }

    const steamIdConvert = BigInt("0x110000100000000");

    function inputGiftee(isGift = false) {
        const nickname = document.querySelector("#account_pulldown")?.textContent?.trim() ?? "unknown";

        return new Promise((resolve, reject) => {
            if (!isGift) {
                resolve(null);
            } else {
                ShowPromptDialog("提示", "请输入礼物接收人的 SteamID 或者好友代码, 可以留空", "确认", "跳过")
                    .done(text => {
                        try {
                            let steamId = BigInt(text.trim());
                            if (steamId == 0) {
                                throw "输入数值有误";
                            }

                            if (steamId > steamIdConvert) {
                                steamId -= steamIdConvert;
                            }
                            const giftInfo = {
                                accountid_giftee: steamId.toString(),
                                gift_message: {
                                    gifteename: steamId.toString(),
                                    message: "Send by Fast_Add_Cart",
                                    sentiment: nickname,
                                    signature: nickname
                                },
                                time_scheduled_send: 0
                            };
                            resolve(giftInfo);
                        } catch (err) {
                            ShowAlertDialog("提示", "输入数值有误")
                                .then(() => resolve(null));
                        }
                    })
                    .fail(() => {
                        resolve(null);
                    });
            }
        });
    }

    // 获取游戏详情
    function getStoreItem(appIds = null, subIds = null, bundleIds = null) {
        return new Promise((resolve, reject) => {
            const ids = [];
            if (appIds) {
                for (let x of appIds) {
                    ids.push({ appid: x });
                }
            }
            if (subIds) {
                for (let x of subIds) {
                    ids.push({ packageid: x });
                }
            }
            if (bundleIds) {
                for (let x of bundleIds) {
                    ids.push({ bundleid: x });
                }
            }

            if (ids.length === 0) {
                reject([false, "未提供有效ID"]);
            }

            const payload = {
                ids,
                context: {
                    language: storeLanguage,
                    country_code: userCountry,
                    steam_realm: "1"
                },
                data_request: {
                    include_all_purchase_options: true
                }
            };
            const json = encodeURI(JSON.stringify(payload));
            fetch(
                `https://api.steampowered.com/IStoreBrowseService/GetItems/v1/?input_json=${json}`,
                {
                    method: "GET",
                }
            )
                .then(async (response) => {
                    if (response.ok) {
                        const { response: { store_items } } = await response.json();
                        resolve(store_items);
                    } else {
                        reject(t("networkRequestError"));
                    }
                })
                .catch((err) => {
                    reject(err);
                });
        });
    }

    //读取购物车
    function getAccountCart() {
        return new Promise((resolve, reject) => {
            fetch(
                `https://api.steampowered.com/IAccountCartService/GetCart/v1/?access_token=${accessToken}`,
                {
                    method: "GET",
                }
            )
                .then(async (response) => {
                    if (response.ok) {
                        const { response: { cart } } = await response.json();
                        resolve(cart);
                    } else {
                        reject(t("networkRequestError"));
                    }
                })
                .catch((err) => {
                    reject(err);
                });
        });
    }

    //添加购物车
    function addItemsToAccountCart(subIds = null, bundleIds = null, isPrivate = false) {
        return new Promise((resolve, reject) => {
            const items = [];
            if (subIds) {
                for (let x of subIds) {
                    items.push({ packageid: x });
                }
            }
            if (bundleIds) {
                for (let x of bundleIds) {
                    items.push({ bundleid: x });
                }
            }
            if (items.length === 0) {
                reject([false, "未提供有效ID"]);
            }

            for (let x of items) {
                x["gift_info"] = null; //giftInfo;
                x["flags"] = {
                    is_gift: false,
                    is_private: isPrivate == true
                };
            }

            const payload = {
                user_country: userCountry,
                items,
                navdata: {
                    domain: "store.steampowered.com",
                    controller: "default",
                    method: "default",
                    submethod: "",
                    feature: "spotlight",
                    depth: 1,
                    countrycode: userCountry,
                    webkey: 0,
                    is_client: false,
                    curator_data: {
                        clanid: null,
                        listid: null
                    },
                    is_likely_bot: false,
                    is_utm: false
                }
            };
            const json = JSON.stringify(payload);

            fetch(
                `https://api.steampowered.com/IAccountCartService/AddItemsToCart/v1/?access_token=${accessToken}`,
                {
                    method: "POST",
                    body: `input_json=${json}`,
                    headers: {
                        "content-type": "application/x-www-form-urlencoded; charset=UTF-8",
                    },
                }
            )
                .then(async (response) => {
                    if (response.ok) {
                        const { response: { cart } } = await response.json();
                        resolve(cart);
                    } else {
                        reject(t("networkRequestError"));
                    }
                })
                .catch((err) => {
                    reject(err);
                });
        });
    }

    //编辑购物车
    function editAccountCart(itemId, isGift, isPrivate, giftInfo = null) {
        return new Promise((resolve, reject) => {
            const payload = {
                line_item_id: itemId,
                user_country: userCountry,
                gift_info: giftInfo,
                flags: {
                    is_gift: isGift,
                    is_private: isPrivate,
                }
            };
            const json = JSON.stringify(payload);

            console.log(json);

            fetch(
                `https://api.steampowered.com/IAccountCartService/ModifyLineItem/v1/?access_token=${accessToken}`,
                {
                    method: "POST",
                    body: `input_json=${json}`,
                    headers: {
                        "content-type": "application/x-www-form-urlencoded; charset=UTF-8",
                    },
                }
            )
                .then(async (response) => {
                    if (response.ok) {
                        const { response: { line_item_ids, cart } } = await response.json();
                        resolve([line_item_ids, cart]);
                    } else {
                        reject(t("networkRequestError"));
                    }
                })
                .catch((err) => {
                    reject(err);
                });
        });
    }

    //删除购物车
    function deleteAccountCart() {
        return new Promise((resolve, reject) => {
            fetch(
                `https://api.steampowered.com/IAccountCartService/DeleteCart/v1/?access_token=${accessToken}`,
                {
                    method: "POST",
                }
            )
                .then(async (response) => {
                    if (response.ok) {
                        const { response: { line_item_ids, cart } } = await response.json();
                        resolve([line_item_ids, cart]);
                    } else {
                        reject(t("networkRequestError"));
                    }
                })
                .catch((err) => {
                    reject(err);
                });
        });
    }

    //显示提示
    function showAlert(title, text, succ = true) {
        return ShowAlertDialog(`${succ ? "✅" : "❌"}${title}`, text);
    }
})();

GM_addStyle(`
button.fac_listbtns {
    display: none;
    position: relative;
    z-index: 100;
    padding: 1px;
  }
  a.search_result_row > button.fac_listbtns {
    top: -25px;
    left: 300px;
  }
  a.tab_item > button.fac_listbtns {
    top: -40px;
    left: 330px;
  }
  a.recommendation_link > button.fac_listbtns {
    bottom: 10px;
    right: 10px;
    position: absolute;
  }
  div.wishlist_row > button.fac_listbtns {
    top: 35%;
    right: 30%;
    position: absolute;
  }
  div.game_purchase_action > button.fac_listbtns {
    right: 8px;
    bottom: 8px;
  }
  button.fac_cartbtns {
    padding: 5px 8px;
  }
  button.fac_cartbtns:not(:last-child) {
    margin-right: 4px;
  }
  button.fac_cartbtns:not(:first-child) {
    margin-left: 4px;
  }
  a.tab_item:hover button.fac_listbtns,
  a.search_result_row:hover button.fac_listbtns,
  div.recommendation:hover button.fac_listbtns,
  div.wishlist_row:hover button.fac_listbtns {
    display: block;
  }
  div.game_purchase_action:hover > button.fac_listbtns {
    display: inline;
  }
  button.fac_choose {
    padding: 1px;
    margin: 2px 5px;
  }
  textarea.fac_inputbox {
    resize: vertical;
    font-size: 12px;
    min-height: 130px;
    width: 98%;
    background: #3d4450;
    color: #fff;
    padding: 1%;
    border: gray;
    border-radius: 5px;
  }
`);