Greasy Fork is available in English.

Civitai网站辅助

Civitai模型信息辅助获取

// ==UserScript==
// @name         Civitai网站辅助
// @namespace    http://tampermonkey.net/
// @version      2.7.4
// @description  Civitai模型信息辅助获取
// @author       Faded_lov
// @license      GPL
// @match        *://civitai.com/images/*
// @match        *://civitai.com/models/*
// @icon         
// @grant        none
// @require      https://cdn.bootcdn.net/ajax/libs/jquery/3.6.3/jquery.min.js
// @require      https://cdn.bootcdn.net/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js
// @require      https://cdn.bootcdn.net/ajax/libs/jszip/3.10.1/jszip.min.js

// ==/UserScript==
// 当页面准备好
$(function () {
    setTimeout(() => { // 定时一秒后执行
        check_localStorage();
        main(); // 执行主函数
    }, 1000);
});

var gallery_data_arr = new Array();  // 用于下载评论区图库中图片
var zip = new JSZip();  // 创建zip实例用于下载zip文件

/**
 * @brief 检查localStorage
 */
function check_localStorage() {
    let localStorage_list = ["settings", "png_format", "md_flag", "zip_images"];
    localStorage_list.forEach(function (item, index) {
        if (localStorage.getItem(item) == null) {
            console.log("添加存储信息:" + item);
            localStorage.setItem(item, "");
        }
    });
}

/**
 * @brief 主要运行函数
 */
function main() {
    // ^ 8. 保存https://civitai.com/images/*中的图片和信息
    // * 8.1 控件添加
    let $pic_down_btn = $(`
            <button class="mantine-UnstyledButton-root mantine-Button-root mantine-b0t5u6 down_pic" type="button" data-button="true" id="">
                <div class="mantine-3xbgk5 mantine-Button-inner">
                    <span>Download !</span>
                </div>
            </button>
            `);
    let timely = setInterval(function () {
        if ($('.down_pic').length == 0) { // 控件还没加上
            if (location.href.match(/images/g) != null) // 页面仍在正确页面(没回到模型页面)
            {
                console.log('add control...' + $('.mantine-UnstyledButton-root.mantine-Button-root.mantine-b0t5u6').length);
                // $('.mantine-ButtonGroup-root.mantine-n93y9k').append($btn);
                $('.mantine-UnstyledButton-root.mantine-Button-root.mantine-b0t5u6').after($pic_down_btn);
            }
        }
    }, 1000);

    // * 8.2 控件事件绑定
    // 以委派方式实现新增元素的事件绑定
    $(document).on('mousedown', '.down_pic', function () {
        let picture_name = location.href.match(/\d+/g)[0]; // 获取文件名
        $('.mantine-UnstyledButton-root.mantine-Button-root.mantine-b0t5u6').click(); // 点击复制按钮到剪切板
        readClipText(picture_name); // 读取剪切板并保存到本地
        $('.down_pic span').text('Downloading...');
        let picture_url = $(this).parents('.mantine-Paper-root.mantine-eljy0r').find('.mantine-8od8ev img').attr('src'); // 获取链接
        console.log(picture_url);
        downloadImg(picture_url, picture_name)
            .then(function () {
                // 这里可以执行下载完成后需要执行的代码
                $('.down_pic span').text('Download !');
            })
            .catch(function (error) {
                console.error(error);
                alert('图片下载出现错误,请检查控制台中相关信息\n' + error.message);
                // 这里可以执行下载失败后需要执行的代码
                $('.down_pic span').text('Error !!');
            });; // 下载图片
    });


    // ^ 0. 预处理程序
    // * 0.1 当前主题读取:light or dark
    let theme = document.cookie.match(/(?<=mantine-color-scheme=)\w(?=[^;]+)/);  // l 或 d

    // * 0.2 LocalStorage数据读取
    let md_flag = false;
    if (localStorage.getItem("md_flag") == "1") {
        md_flag = true;
    }

    // * 0.3 设计交互按钮区域的底框
    $Interaction_region = $(`<div id="interaction-region"></div>`);
    $Interaction_region_style = $(`
    <style>
    #interaction-region {
        height: auto;
        width: auto;
        background-color: #f8f9fa;
        border: 1px solid #dee2e6;
        padding: 5px;
        border-radius: 5px;
    }
    </style>
    `);
    if (theme == 'd') {
        $('.mantine-agabl4:contains("Details")').parent().before($Interaction_region);
        $('#interaction-region').css({
            'background-color': '#25262b',
            'border': '1px solid #373a40'
        });
    } else {
        $('.mantine-9dt51j:contains("Details")').parent().before($Interaction_region);
    }
    $('head').append($Interaction_region_style);

    // * 0.4 创建一个公用的类(主要是对背景色等的处理)
    $public_css = $(`
    <style>
        .public-downloading {
            color: white !important;
            background-color: #FFA500 !important;
        }

        .public-error {
            color: white !important;
            background-color: red !important;
        }
    </style>
    `);
    $('head').append($public_css);

    // ^ 1. 文件格式切换处理
    // * 1.1 样式、控件添加
    $file_format = $(`
    <label class="switch" title="txt格式">
        <input type="checkbox" id="switch-checkbox"/>
        <span class="slider round"></span>
    </label>
    `);
    $file_format_style = $(`
    <style>
    /* 样式化复选框 */
    .switch {
        position: relative;
        display: inline-block;
        width: 60px;
        height: 34px;
    }

    .switch input {
        opacity: 0;
        width: 0;
        height: 0;
    }

    .slider {
        position: absolute;
        cursor: pointer;
        top: 0;
        left: 0;
        right: 0;
        bottom: 0;
        background-color: #ccc;
        transition: .2s;
    }

    .slider:before {
        position: absolute;
        content: "";
        height: 26px;
        width: 26px;
        left: 4px;
        bottom: 4px;
        background-color: white;
        transition: .3s;
    }

    input:checked+.slider {
        background-color: #2196F3;
    }

    input:checked+.slider:before {
        transform: translateX(26px);
    }

    .slider.round {
        border-radius: 34px;
    }

    .slider.round:before {
        border-radius: 50%;
    }

    /* 隐藏复选框 */
    .switch input {
        display: none;
    }
</style>

    `);
    $('.mantine-Group-root.mantine-1eod8xg').before($file_format);
    $('head').append($file_format_style);

    // * 1.2 文件格式切换事件监听、处理
    // 1.2.1 检查localStorage状态,初始化页面控件显示
    if (md_flag) {
        $("#switch-checkbox").prop("checked", true).parent().prop("title", "markdown格式");
    }
    // 1.2.2 监听复选框状态变化,状态变化时更新localStorage,并刷新页面
    $("#switch-checkbox").on("change", () => {
        if ($("#switch-checkbox").prop("checked")) {
            localStorage.setItem("md_flag", "1");
        } else {
            localStorage.removeItem("md_flag");
        }
        location.reload();
    });

    // ^ 2. 模型信息API数据处理:文件名显示、.info下载、文档下载、所有图片下载
    // * 2.1 添加控件、样式(文本信息下载按钮、图片下载按钮、备注框、文件名显示等)
    $filename = $(`
    <tr class="mantine-1avyp1d">
        <td class="mantine-vp2u6p">
            <div class="mantine-Text-root mantine-152y8ed">File Name</div>
        </td>
        <td class="mantine-1avyp1d">
            <div class="mantine-Group-root mantine-1u5ck20 filename">loading...</div>
        </td>
    </tr>
    <tr class="mantine-1avyp1d">
        <td class="mantine-vp2u6p">
            <div class="mantine-Text-root mantine-152y8ed">File Size</div>
        </td>
        <td class="mantine-1avyp1d">
            <div class="mantine-Group-root mantine-1u5ck20 filesize">loading...</div>
        </td>
    </tr>
    `);
    $filename_dark = $(`
    <tr class="mantine-1avyp1d">
        <td class="mantine-1lnluqq">
            <div class="mantine-Text-root mantine-1363hig">File Name</div>
        </td>
        <td class="mantine-1avyp1d">
            <div class="mantine-Text-root mantine-ljqvxq filename">loading...</div>
        </td>
    </tr>
    <tr class="mantine-1avyp1d">
        <td class="mantine-1lnluqq">
            <div class="mantine-Text-root mantine-1363hig">File Size</div>
        </td>
        <td class="mantine-1avyp1d">
            <div class="mantine-Text-root mantine-ljqvxq filesize">loading...</div>
        </td>
    </tr>
    `);
    $down_msg_div = $(`
    <div class="container">
        <div class="btn-down" id="downInfo" style="margin-right:3px;" title="下载.info文件">Loading...</div>
        <div class="btn-down" id="downMsg" style="margin-left:3px;" title="下载说明文档">Loading...</div>
    </div>
    <div class="container">
        <div class="png-set-btn" onclick="png_set()" title="图片格式设置"><span>图片格式</span></div>
        <div class="png-set-menu">
            <div id="png-set">
                <form>
                    <div><input type="checkbox" name="preview" id="png-set-1"><label for="png-set-1" title="文件后缀">.preview.png</label>
                    </div>
                    <div><input type="checkbox" name="info_tips" id="png-set-2"><label for="png-set-2" title="判断图片是否含有参数">_noInfo</label>
                    </div>
                </form>
            </div>
        </div>
        <script>
            function png_set() {
                if (!$(".png-set-menu").width()) {
                    // 打开动画
                    $('.png-set-btn').addClass('open-png-menu');
                    $('.png-set-menu').width(402);
                    // 加载设置
                    let settings = localStorage.getItem('png_format');
                    if (!settings) settings = '';
                    $('#png-set input').each(function (i, ele) {
                        if (settings.includes($(ele).prop('name'))) {
                            $(ele).prop('checked', true);
                        } else {
                            $(ele).prop('checked', false);
                        }
                    });
                } else {
                    // 关闭动画
                    $('.png-set-btn').removeClass('open-png-menu');
                    $('.png-set-menu').width(0);
                    // 保存设置
                    let settings = '';
                    $('#png-set input:checked').each(function (i, ele) {
                        settings += $(ele).prop('name') + ',';
                    });
                    localStorage.setItem('png_format', settings);
                }
            }
        </script>
        <div class="btn-down" id="downPic" title="下载左侧所有预览图">Loading...</div>
    </div>
    `);
    $textarea = $(`
    <textarea id="tips" rows=15 placeholder='备注...' ></textarea>
    `);
    $style = $(`
    <style>
        .filename,.filesize {
            font-weight: 600;
            color: #d400ff;
        }

        .container {
            display: flex;
            position: relative;
            padding: 0;
        }

        .btn-down {
            flex-grow: 1;
            width: auto;
            height: 35px;
            margin: 3px auto;
            text-align: center;
            line-height: 35px;
            font-size: 15px;
            font-weight: 500;
            color: #fff;
            background-color: #40c057;
            border-radius: 5px;
            cursor: pointer;
            user-select: none;
            transition: 0.3s;
        }

        .btn-down:hover {
            background-color: #37b24d;
        }

        .png-set-btn {
            z-index: 2;
            width: 80px;
            height: 35px;
            margin: 3px 5px 0 0;
            line-height: 35px;
            text-align: center;
            background-color: #40c057;
            color: white;
            border-radius: 5px;
            user-select: none;
            cursor: pointer;
            transition: all 0.5s;
        }

        .png-set-btn:hover {
            background-color: #37b24d;
        }

        .open-png-menu {
            height: 25px;
            line-height: 25px;
            margin-top: 8px;
            margin-left: 3px;
            font-size: 14px;
        }

        .png-set-menu {
            position: absolute;
            z-index: 1;
            width: 0px;
            height: 35px;
            margin-top: 3px;
            background-color: #f8f9fa;
            overflow: hidden;
            border-radius: 5px;
            transition: all 0.5s;
        }

        #png-set form div {
            float: right;
        }

        #png-set input {
            display: none;
        }

        #png-set label {
            display: inline-block;
            width: 150px;
            height: 25px;
            margin: 5px 5px 5px 0;
            background-color: #868e96;
            color: white;
            text-align: center;
            line-height: 25px;
            border-radius: 5px;
            transition: all .3s;
            user-select: none;
            cursor: pointer;
        }

        #png-set input:checked+label {
            background-color: #40c057;
        }

        #tips {
            width: auto;
            font-size: 15px;
            background-color: #f8f9fa;
            border: 2px solid #999;
            border-radius: 5px;
            resize: none;
        }
    </style>
    `);
    if (theme == 'd') { // 深色模式
        $('#interaction-region~.mantine-1avyp1d tbody').append($filename_dark);
    } else {  // 浅色模式
        $('#interaction-region~.mantine-1avyp1d tbody').append($filename);
    }
    // 加载交互区域控件
    $('#interaction-region').append($down_msg_div);
    // markdown格式不加载备注栏,txt格式加载
    if (!md_flag) {
        $('.mantine-1h0gy36:last').after($textarea);
    }
    if (theme == 'd') { // 深色模式
        $('.png-set-menu').css('background-color', '#25262b');
        $('#tips').css('backgroundColor', '#25262b');
    }
    // 样式加载
    $('head').append($style);

    // * 2.2 图片下载格式切换事件处理
    if (localStorage.getItem('png_format') == 'preview') {
        $('.png-format-checkbox input').prop('checked', true);
    }
    $('.png-format-checkbox input').on('click', function () {
        if ($('.png-format-checkbox input').prop('checked')) {
            localStorage.setItem('png_format', 'preview');
        } else {
            localStorage.removeItem('png_format');
        }
    });

    // * 2.3 通过api配合模型id获取模型信息数据
    let requestUrl = 'https://civitai.com/api/v1/models/' + location.href.match(/\d+/);
    console.log('请求链接: ' + requestUrl);
    fetch(requestUrl)
        .catch(error => {
            alert('API数据不正常,请检查或重试\n' + error.message);
        })
        .then(response => {
            console.log(response);
            if (!response.ok) {
                throw new Error('Network response was not ok');
            }
            return response.json();
        })
        .then(data => {
            console.log(data);

            // * 2.4 显示当前选择的模型的版本和文件名
            let model_version = 0;
            let filename;
            let version_class = theme == 'd' ? 'mantine-z8ikjj' : 'mantine-14vhbbe';
            console.log('当前版本: ' + $(`.mantine-lw13s0 .mantine-lw13s0 button[class*="${version_class}"]`).text());
            // 判断是否存在相同版本名
            let versions_name = new Array();
            for (let i = 0; i < data.modelVersions.length; i++) {
                let name = data.modelVersions[i].name;
                if (versions_name.includes(name)) {  // 存在相同版本名
                    alert('当前模型存在相同版本名,请自行确认下载的信息正确性');
                    let flag = false;
                    // 不确定直接通过API索引号判断版本是否正确
                    $(`.mantine-lw13s0 .mantine-lw13s0 button`).each(function (index, ele) {
                        if ($(ele).hasClass(version_class)) {
                            model_version = index;
                            flag = true;
                            return false;
                        }
                    });
                    if (flag) {
                        break;
                    }
                } else {
                    versions_name.push(name);
                    if (name == $(`.mantine-lw13s0 .mantine-lw13s0 button[class*="${version_class}"]`).text()) {
                        model_version = i;
                    }
                }
            }
            console.log('模型版本序号(从0开始): ' + model_version);
            filename = getFilename(data, model_version);

            // 模型切换时更新文件名显示
            $('.mantine-lw13s0 .mantine-lw13s0 button').on('click', function () {
                let btn_text = $(this).text();
                setTimeout(function () {
                    console.log('当前版本: ' + btn_text);
                    // 判断是否存在相同版本名
                    let versions_name = new Array();
                    for (let i = 0; i < data.modelVersions.length; i++) {
                        let name = data.modelVersions[i].name;
                        if (versions_name.includes(name)) {  // 存在相同版本名
                            // alert('当前模型存在相同版本名,请自行确认下载的信息正确性');
                            let flag = false;
                            // 不确定直接通过API索引号判断版本是否正确
                            $(`.mantine-lw13s0 .mantine-lw13s0 button`).each(function (index, ele) {
                                if ($(ele).hasClass(version_class)) {
                                    model_version = index;
                                    flag = true;
                                    return false;
                                }
                            });
                            if (flag) {
                                break;
                            }
                        } else {
                            versions_name.push(name);
                            if (name == btn_text) {
                                model_version = i;
                            }
                        }
                    }
                    console.log('模型版本序号: ' + model_version);
                    filename = getFilename(data, model_version);
                }, 500);
            });
            // // 手动触发模型切换函数以代替模型版本初次获取
            // $('.mantine-lw13s0 .mantine-lw13s0 button:eq(0)').click();

            // * 2.5 下载.info文件的按钮事件处理
            $('#downInfo').text('下载.info文件(C站助手)')
                .click(function () {
                    $(this).text('Downloading......').addClass('public-downloading');
                    downInfo(data.modelVersions[model_version].id, filename);
                });

            // * 2.6 下载说明文档的按钮的事件处理
            $('#downMsg').text(`下载说明文档 (${md_flag ? "markdown" : "txt"}格式)`)
                .click(function () {
                    downloadDoc(data, model_version, md_flag); // 下载文件
                    // 点击like
                    if (theme == 'd') { // 深色模式
                        $('.mantine-1tkveyv .mantine-3xbgk5:eq(1)').click();
                    } else { // 浅色模式
                        $('.mantine-1ayq7x2 .mantine-3xbgk5:eq(1)').click();
                    }
                });

            // * 2.7 批量下载图片的按钮事件处理
            // 2.7.1 获取当前模型版本预览图数量
            let pictures_count = $('.mantine-ContainerGrid-col:eq(1) .mantine-7aj0so').length;
            $('#downPic').text(`下载所有预览图(${pictures_count})`);
            if (!pictures_count) {
                // alert('绑定图片下载失败,请手动点击切换版本(当前版本就行)以重新绑定');
                $('#downPic').text('信息加载失败,请点击当前版本按钮重新绑定')
                    .prop('title', '下载左侧所有预览图\n版本按钮即图片区域左上角按钮');
            }
            // 切换版本时更新当前版本预览图提示
            $('.mantine-lw13s0 .mantine-lw13s0 button').on('click', function () {
                $('#downPic').text(`Loading...`);
                let update_status = setInterval(function () {
                    if ($('.picbtn').length != 0) {  // 页面加载完成
                        pictures_count = $('.mantine-ContainerGrid-col:eq(1) .mantine-7aj0so').length
                        $('#downPic').text(`下载所有预览图(${pictures_count})`);
                        clearInterval(update_status);
                    }
                }, 500);
            });

            // 2.7.2 下载所有预览图
            $('#downPic').on({
                click: () => {
                    console.log('=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=');
                    $('#downPic').text('Downloading...').addClass('public-downloading');
                    let images_arr = new Array();
                    $('.mantine-ContainerGrid-col:eq(1) .mantine-7aj0so').each(function (index, domEle) {
                        images_arr.push($(domEle).prop('src'));
                    });

                    // 下载图片
                    console.log(images_arr);
                    if (localStorage.getItem('zip_images') == '1') { // 以zip格式下载图片
                        downloadZip(data, model_version, images_arr, filename, 0);
                    } else { // 直接下载图片
                        downloadImages(data, model_version, images_arr, filename, 0);
                    }
                }
            });

            // ^ 3. 图片下载(左侧单张下载)
            // * 3.1 相关控件、样式添加
            $btn = $('<button type="button" style="position:absolute;bottom:37px;left:6px" class="picbtn" title="下载当前图片">Download</button>');
            $btn_style = $(`
            <style>
                .picbtn {
                    z-index: 2;
                    width: auto;
                    height: 30px;
                    align: center;
                    text-align: center;
                    border: 0;
                    background: #d5d5d3;
                    color: gray;
                    border-radius: 5px;
                    transition: all 0.5s;
                }

                .picbtn:hover {
                    background: #40c057;
                    color: white;
                }
            </style>
            `);
            $('head').append($btn_style);
            $("a[href^='/images/']").after($btn);
            // 切换版本时重新绑定按钮
            $('.mantine-lw13s0 .mantine-lw13s0 button').on('click', function () {
                $('.picbtn').remove();
                let add_btn = setInterval(function () {
                    if ($('.picbtn').length == 0) {
                        $("a[href^='/images/']").after($btn);
                        console.log('绑定图片下载按钮ing...');
                    } else {
                        console.log('绑定图片下载按钮成功');
                        clearInterval(add_btn);
                    }
                }, 500);
            });
            // 过滤按钮切换时重新绑定按钮
            $('.mantine-xickm0').on('mouseout', function () {
                $('.picbtn').remove();
                $("a[href^='/images/']").after($btn);
            });
            // 右上角切换按钮
            $('.mantine-6v7rx2 .mantine-1g4q40w button:eq(0)').on('mouseout', function () {
                setTimeout(function () {
                    $('.picbtn').remove();
                    $("a[href^='/images/']").after($btn);
                }, 2000);
            });

            // * 3.2 单张图片下载按钮事件处理
            // 以委派方式实现新增元素的事件绑定
            $(document).on('mousedown', '.picbtn', function () {
                $(this).text('Downloading...').addClass('public-downloading');

                // 3.2.1 获取下载的图片文件名
                let pic_name = filename;  // 当前选择的模型版本文件名
                console.log('=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=');
                console.log('picture name        : ' + pic_name);

                let pic_url = $(this).parent().find('img,video').prop('src'); // 图片链接(或者动图)

                // 判断API中是否有图片(理论上没有图片不会有按钮能点击,为了保险增加该检测)
                if (!data.modelVersions[model_version].images.length) {
                    alert('未查找到该模型版本图片的信息,请自行检查错误');
                    return;
                }

                // 3.2.2 通过遍历请求的数据中图片链接获取宽度信息
                let pic_width = 0;
                // (1). 在模型API中查询
                for (let index = 0; index < data.modelVersions[model_version].images.length; index++) {  // 当前模型版本内所有图片遍历
                    if (data.modelVersions[model_version].images[index].url.includes(pic_url.split('/')[4].replace(/\s+/g, ""))) {
                        pic_width = data.modelVersions[model_version].images[index].width;
                        console.log('picture width       : ' + pic_width);
                        break;
                    }
                }
                // (2). 在图库API中查询
                if (pic_width == 0) {
                    for (let index = 0; index < gallery_data_arr.length; index++) { // 数组遍历
                        for (let img_index = 0; img_index < gallery_data_arr[index].images.length; img_index++) { // 数据元素内部数据遍历
                            if (gallery_data_arr[index].images[img_index].url.includes(pic_url.split('/')[4].replace(/\s+/g, ""))) {
                                pic_width = gallery_data_arr[index].images[img_index].width;
                                console.log('picture width(gal..): ' + pic_width);
                                break;
                            }
                        }
                        if (pic_width) {  // 查询到宽度
                            break;
                        }
                    }
                }
                // (3). 获取模型API中最大宽度
                if (pic_width == 0) {// 图片不在api返回数据内(api只返回10张图片)
                    alert('未查询到该图片宽度,以模型预览图中最大宽度为准\r\n获取到的极大可能不是原图');
                    pic_width = getMaxWidth(data.modelVersions[model_version].images);
                    console.log('picture width (max) : ' + pic_width);
                }

                // 3.2.3 图片下载
                console.log('original picture url: ' + pic_url);
                pic_url = pic_url.replace(/width=\d+/, 'width=' + pic_width);
                console.log('new picture url     : ' + pic_url);

                let clicked_btn = $(this);  // 存储被点击的按钮的索引,方便在图片下载完成时进行操作
                downloadImg(pic_url, pic_name)
                    .then(function () {
                        // 图片下载完成
                        clicked_btn.text('Download').removeClass('public-downloading');
                    })
                    .catch(function (error) {
                        alert('图片下载出现错误,请检查控制台中相关信息\n' + error.message);
                        clicked_btn.text('Download').addClass('public-error');
                    });
                console.log('=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=');
            });

            // ^ 4. 懒人一键下载
            // * 4.1 相关控件、样式添加
            $easy_download = $(`
            <div class="auto_download">
                <div class="setting-btn" onclick="open_menu()" title="设置一键下载内容"><span>配置</span></div>
                <div class="setting-menu">
                    <div id="setting" style="display: none;">
                        <div class="close-btn" onclick="close_menu()" title="save">save</div>
                        <form>
                            <div><input type="checkbox" name="model_down" id="setting_1"><label for="setting_1">下载模型</label></div>
                            <div><input type="checkbox" name="picture_down" id="setting_2"><label for="setting_2">下载第一张预览图</label></div>
                            <div><input type="checkbox" name="text_down" id="setting_3"><label for="setting_3">下载说明文件</label></div>
                            <div><input type="checkbox" name="info_down" id="setting_4"><label for="setting_4">下载.info文件</label></div>
                        </form>
                    </div>
                </div>
                <script>
                    function open_menu() {
                        // 加载设置
                        let settings = localStorage.getItem('settings');
                        if (!settings) settings = '';
                        $('#setting input').each(function (i, ele) {
                            if (settings.includes($(ele).prop('name'))) {
                                $(ele).prop('checked', true);
                            }
                        });

                        // 展开动画
                        $('.setting-btn span').fadeOut(10);
                        $('.setting-menu').animate({
                            width: '402',
                            height: '125'
                        }, 500);
                        $('#setting').delay(500).fadeIn(500);
                    }

                    function close_menu() {
                        // 保存设置
                        let settings = '';
                        $('#setting input:checked').each(function (i, ele) {
                            settings += $(ele).prop('name') + ',';
                        });
                        localStorage.setItem('settings', settings);

                        // 关闭动画
                        $('#setting').fadeOut(500);
                        $('.setting-menu').delay(300).animate({
                            width: '0',
                            height: '0'
                        }, 500);
                        $('.setting-btn span').delay(850).fadeIn(50);
                    }
                </script>
                <div class="easy_download" title="一键下载选定的文件"> 一键下载 !</div>
            </div>
            `);
            $easy_download_style = $(`
            <style>
            .auto_download {
                position: relative;
                display: flex;
                align-items: center;
                justify-content: center;
            }

            .setting-btn {
                width: 60px;
                height: 36px;
                font-size: 15px;
                line-height: 36px;
                text-align: center;
                color: white;
                background-color: #228be6;
                border-radius: 5px;
                user-select: none;
                cursor: pointer;
                transition: all .2s;
            }

            .setting-btn:hover {
                background-color: #1c7ed6;
            }

            .setting-menu {
                position: absolute;
                top: 0;
                left: 0;
                z-index: 9;
                width: 0;
                height: 0;
                background-color: #f8f9fa;
                overflow: hidden;
                border-radius: 5px;
            }

            .close-btn {
                position: absolute;
                bottom: 0;
                left: 50%;
                transform: translateX(-50%);
                margin: 5px;
                width: 75px;
                height: 25px;
                color: white;
                background-color: #228be6;
                border-radius: 5px;
                text-align: center;
                line-height: 22px;
                cursor: pointer;
                user-select: none;
                transition: all 0.3s;
            }

            .close-btn:hover {
                background-color: #1c7ed6;
            }

            #setting form div {
                float: left;
            }

            #setting input {
                display: none;
            }

            #setting label {
                display: inline-block;
                width: 190px;
                height: 36px;
                margin: 5px;
                background-color: #868e96;
                text-align: center;
                line-height: 36px;
                color: white;
                border-radius: 5px;
                transition: all .3s;
                user-select: none;
                cursor: pointer;
            }

            #setting input:checked+label {
                background-color: #40c057;
            }

            .easy_download {
                flex-grow: 1;
                width: auto;
                height: 36px;
                margin: 3px 0 3px 5px;
                text-align: center;
                line-height: 36px;
                font-size: 15px;
                font-weight: 500;
                color: #fff;
                background-color: #228be6;
                border-radius: 5px;
                cursor: pointer;
                user-select: none;
                transition: 0.3s;
            }

            .easy_download:hover {
                background-color: #1c7ed6;
            }
            </style>
            `);
            $('#interaction-region').prepend($easy_download);
            $('head').append($easy_download_style);
            if (theme == 'd') { // 深色模式
                $('.setting-menu').css('background-color', '#25262b');
            }

            // * 4.3 懒人下载事件处理
            $('.easy_download').on({
                click: () => {
                    // 读取设置并给出提示信息
                    let settings = localStorage.getItem('settings');
                    if (!settings) {
                        alert('请先设置一键下载选项');
                        return;
                    } else {
                        console.log('=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=');
                        console.log(settings);
                        $('.easy_download').text(' 任务提交中......').addClass('public-downloading');
                    }

                    // 点击like
                    if (theme == 'd') { // 深色模式
                        $('.mantine-1tkveyv .mantine-3xbgk5:eq(1)').click();
                    } else { // 浅色模式
                        $('.mantine-1ayq7x2 .mantine-3xbgk5:eq(1)').click();
                    }

                    // 4.3.1 下载第一张预览图(按模型版本来)
                    if (settings.includes('picture_down')) {
                        let picture = data.modelVersions[model_version].images[0];
                        console.log(picture);

                        pic_url = picture.url.replace(/width=\d+/, 'width=' + picture.width);
                        console.log('pic_url: ' + pic_url);
                        downloadImg(pic_url, filename)
                            .then(function (status) {
                                console.log(status);
                                // 这里可以执行下载完成后需要执行的代码
                                $('.easy_download').text(' 一键下载 !').removeClass('public-downloading');
                            })
                            .catch(function (error) {
                                console.error(error);
                                alert('图片下载出现错误,请检查控制台中相关信息\n' + error.message);
                                // 这里可以执行下载失败后需要执行的代码
                                $('.easy_download').text(' 下载出现错误 !').addClass('public-error');
                            });
                    }

                    // 4.3.2 说明文档下载
                    if (settings.includes('text_down')) {
                        downloadDoc(data, model_version, md_flag);
                    }

                    // 4.3.3 info下载
                    if (settings.includes('info_down')) {
                        downInfo(data.modelVersions[model_version].id, filename);
                    }

                    // 4.3.4 模型下载
                    if (settings.includes('model_down')) {
                        // 获取下载链接
                        let down_url = data.modelVersions[model_version].files[0].downloadUrl;  // 默认链接
                        // 查找最大的文件
                        if (data.modelVersions[model_version].files.length > 1) {  // 存在多个文件
                            let max_sizeKB = data.modelVersions[model_version].files[0].sizeKB;
                            for (let i = 1; i < data.modelVersions[model_version].files.length; i++) {
                                if (data.modelVersions[model_version].files[i].sizeKB > max_sizeKB) {  // 有更大的文件
                                    down_url = data.modelVersions[model_version].files[i].downloadUrl;
                                }
                            }
                        }
                        console.log('下载链接: ' + down_url);
                        let link = $("<a>");
                        link.attr("href", down_url);
                        link.attr("download", "");
                        // 将 a 标签添加到页面中
                        $("body").append(link);
                        // 模拟点击操作,从而触发文件的下载
                        link[0].click();
                        // 将 a 标签从页面中移除
                        link.remove();
                    }

                    //不下载图片时直接恢复按钮样式
                    if (!settings.includes('picture_down')) {
                        $('.easy_download').text(' 一键下载 !').removeClass('public-downloading');
                    }
                    console.log('=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=');
                }
            });

            // ^ 5. 触发词下载格式切换(\n)
            // * 5.1 相关控件添加
            $trigger_words = $(`
            <div class="trigger-words-checkbox" id="trigger_words" title="是否在触发词中添加回车以分隔tag(适合如不同tag代表不同服装、发型)">
                <input type="checkbox">
                <label>\\n</label>
            </div>
            `);
            $trigger_words_style = $(`
            <style>
            .trigger-words-checkbox {
                width: 36px;
                height: 36px;
                position: relative;
                margin-left: 5px;
                border-radius: 5px;
                box-sizing: border-box;
            }

            .trigger-words-checkbox input[type="checkbox"] {
                position: absolute;
                top: 0;
                left: 0;
                width: 100%;
                height: 100%;
                z-index: 1;
                opacity: 0;
                cursor: pointer;
            }

            .trigger-words-checkbox label {
                position: absolute;
                width: 36px;
                height: 36px;
                border-radius: 5px;
                background-color: #868e96;
                box-sizing: border-box;
                transition: all 0.3s ease-out;
                color: white;
                font-size: 18px;
                font-weight: 500;
                line-height: 36px;
                text-align: center;
            }

            .trigger-words-checkbox input:hover + label {
                background-color: #565a5f;
            }

            .trigger-words-checkbox input[type="checkbox"]:checked + label {
                background-color: #228be6;
                border-color: #228be6;
            }
            </style>
            `);
            $('.easy_download').after($trigger_words);
            $('head').append($trigger_words_style);

            // ^ 6. 图库内图片下载
            // * 6.1 相关控件添加
            $add_btn = $(`
            <div id="gallery_btn" class="mantine-UnstyledButton-root mantine-Button-root mantine-16ez2wg" style="
                line-height: 34px;
                height: 34px;
                width: 125px;
                margin: auto 5px;
                text-align: center;
            " title="重新绑定当前页面中图片下载按钮">添加图片下载按钮</div>
            `);
            $('#switch-checkbox').parents('.mantine-Grid-col').prepend($add_btn);
            if (theme == 'd') {  // 暗黑模式
                $('#gallery_btn').removeClass('mantine-1tk98uh').addClass('mantine-3rhhm4');
            }

            // * 6.2 按钮事件处理
            $('#gallery_btn').on('click', function () {
                $(this).text('Loading...');
                console.log('=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=');
                $('.picbtn').remove();

                // 通过API获取信息
                getGalleryData(data.modelVersions[model_version].id, location.href.match(/\d+/));

                let complete = setInterval(() => {
                    if (gallery_data_arr.length >= $('#gallery div[id]').length - 2) {
                        clearInterval(complete);
                        console.log('=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=');
                        console.log('页面元素个数:' + $('#gallery div[id]').length);
                        console.log(gallery_data_arr);
                        // 重新绑定按钮
                        $("a[href^='/images/']").after($btn);
                        $(this).text('添加图片下载按钮');
                    }
                }, 500);
            });

            // ^ 7. ZIP打包预览图下载(zip)
            // * 7.1 相关控件添加
            $zip_images = $(`
            <!-- 懒得重新写css, 直接套用之前的 -->
            <div class="trigger-words-checkbox" id="zipImages" title="将所有预览图以zip格式打包下载,激活后点击下载所有预览图即可">
                <input type="checkbox">
                <label>ZIP</label>
            </div>
            `);
            $zip_images_styles = $(`
            <style>
            #zipImages {
                width: 48px;
                height: 35px;
                margin-top: 3px;
            }

            #zipImages label {
                width: 48px;
                height: 35px;
            }
            </style>
            `);
            $('#downPic').after($zip_images);
            $('head').append($zip_images_styles);

            // * 7.2 打包预览图控件事件处理
            // 7.2.1 控件状态加载
            if (localStorage.getItem('zip_images') == '1') {
                $('#zipImages input').prop('checked', true);
            }
            // 7.2.2 点击切换控件状态并存储
            $('#zipImages input').on('click', function () {
                if ($('#zipImages input').prop('checked')) {
                    localStorage.setItem('zip_images', '1');
                } else {
                    localStorage.removeItem('zip_images');
                }
            });
            // 7.2.3 重新访问页面时加载显示以确保显示正确
            window.onfocus = function () {  // 当前页面得到焦点
                if (localStorage.getItem('zip_images') == '1') {
                    $('#zipImages input').prop('checked', true);
                } else {
                    $('#zipImages input').prop('checked', false);
                }
            }
            // * 7.3 打包预览图下载事件触发在 2.7.2(下载所有预览图) 中
        });
}

/**
 * 对请求到的数据根据需要下载的版本返回文件名并更新显示
 * TIPS: 增加了对文件大小的显示
 * @param {*} fetch_data 请求的数据
 * @param {*} model_version 模型版本号
 * @returns 文件名
 */
function getFilename(fetch_data, model_version) {

    let file_name = fetch_data.modelVersions[model_version].files[0].name;
    let file_size = fetch_data.modelVersions[model_version].files[0].sizeKB;
    if (fetch_data.modelVersions[model_version].files.length > 1) {  // 存在多个文件
        for (let i = 1; i < fetch_data.modelVersions[model_version].files.length; i++) {
            if (fetch_data.modelVersions[model_version].files[i].sizeKB > file_size) {  // 有更大的文件
                file_size = fetch_data.modelVersions[model_version].files[i].sizeKB;
                file_name = fetch_data.modelVersions[model_version].files[i].name;
            }
        }
    }

    file_name = file_name.slice(0, file_name.lastIndexOf('.'));
    $('.filename').text(file_name);
    $('.filesize').text((file_size / 1024).toFixed(2) + "MB");
    return file_name;
}

/**
 * 下载.info文件
 * @param {*} model_version_id 模型版本id
 * @param {*} file_name 文件名
 */
function downInfo(model_version_id, file_name) {
    let url = 'https://civitai.com/api/v1/model-versions/' + model_version_id;
    console.log('请求链接: ' + url);
    fetch(url)
        .catch(error => {
            alert('API数据不正常,请检查或重试\n' + error.message);
        })
        .then(response => {
            if (!response.ok) {
                throw new Error('Network response was not ok');
            }
            return response.json();
        })
        .then(data => {
            console.log(data);

            let encoded = encodeChinese(JSON.stringify(data, null, 4));
            let blob = new Blob([encoded], {
                type: "text/plain;charset=utf-8"
            });
            saveAs(blob, `${file_name}.civitai.info`);
            $('#downInfo').text('下载.info文件(C站助手)').removeClass('public-downloading');
        });
}

/**
 * 对中文进行Unicode编码
 * @param {*} str 字符串
 * @returns 编码结果
 */
function encodeChinese(str) {
    let encoded = '';
    for (let i = 0; i < str.length; i++) {
        let code = str.charCodeAt(i);
        if (code < 128) {
            // ASCII 字符
            encoded += str[i];
        } else {
            // 非 ASCII 字符,转换为 Unicode 编码
            encoded += '\\u' + code.toString(16);
        }
    }
    return encoded;
}

/**
 * 对请求到的数据根据版本号组合并下载说明文件
 * @param {*} fetch_data  请求的数据
 * @param {*} model_version 模型版本号
 * @param {*} md_flag 是否下载markdown格式
 */
function downloadDoc(fetch_data, model_version, md_flag) {
    // * 1. 获取提示词
    let triggerWords = '';
    let words = fetch_data.modelVersions[model_version].trainedWords;
    if (words == undefined || words.length == 0) {
        triggerWords = '无';
    } else {
        for (let i = 0; i < words.length; i++) {
            triggerWords += words[i] + ',';
            if ($('#trigger_words input').prop('checked')) {
                triggerWords += '\n';
            }
        }
    }

    // * 2. 获取示例图参数
    let examples = '';
    let request_json_data = `{"json":{"period":"AllTime","sort":"Most Reactions",
    "modelVersionId":${fetch_data.modelVersions[model_version].id},"modelId":${fetch_data.id},"hidden":false,"limit":50,"cursor":null,"authed":true},
    "meta":{"values":{"cursor":["undefined"]}}}`;
    let request_url = 'https://civitai.com/api/trpc/image.getImagesAsPostsInfinite?input=' + encodeURIComponent(request_json_data);
    console.log('图片请求链接: ' + request_url);
    fetch(request_url)
        .catch(error => {
            alert('图片信息请求API数据不正常\n' + error.message);
        })
        .then(response => {
            console.log(response);
            if (!response.ok) {
                throw new Error('Network response was not ok');
            }
            return response.json();
        })
        .then(data => {
            console.log(data);
            let image_data = data.result.data.json.items;
            console.log(image_data);

            // 提取示例图参数
            let pictures = fetch_data.modelVersions[model_version].images;
            for (let pic_index = 0; pic_index < pictures.length; pic_index++) { // 遍历待下载的图片
                let img_url = pictures[pic_index].url; // 获取图片链接
                let hash = pictures[pic_index].hash; // 获取图片hash
                let msg = ''; // 图片信息

                for (let j = 0; j < image_data.length; j++) { // 遍历数据组
                    for (let image_info of image_data[j].images) { // 每个数据组最多10个数据
                        if (hash == image_info.hash) { // hash 匹配,查找到对应的信息
                            console.log(image_info);
                            for (let key in image_info.meta) {
                                if (key == 'hashes') {
                                    msg += '--hashes: ' + JSON.stringify(image_info.meta[key]) + '\n';
                                    continue; // 继续下一循环,不执行下方代码
                                }
                                if (key == 'resources') {
                                    for (let res in image_info.meta[key]) {
                                        msg += '--' + key + ': ' + JSON.stringify(image_info.meta[key][res]) + '\n';
                                    }
                                    continue; // 继续下一循环,不执行下方代码
                                }
                                msg += '--' + key + ': ' + image_info.meta[key] + '\n';
                            }
                            break; // 退出10个数据的循环
                        }
                    }
                    if (msg != '') {
                        break; // 找到对应的数据,退出数据组循环
                    }
                }
                if (md_flag) {  // markdown格式
                    if (msg) {
                        examples += '\n================< ' + (pic_index + 1) + ' >================\n' + '预览图: \n![](' + img_url + ')\n参数: \n' + msg;
                    } else {
                        examples += '\n================< ' + (pic_index + 1) + ' >================\n' + '预览图: \n![](' + img_url + ')\n该图无参数\n';
                    }
                } else {  // txt格式
                    if (msg) {
                        examples += '\n================< ' + (pic_index + 1) + ' >================\n' + '预览图链接: \n' + img_url + '\n参数: \n' + msg;
                    } else {
                        examples += '\n================< ' + (pic_index + 1) + ' >================\n' + '预览图链接: \n' + img_url + '\n该图无参数\n';
                    }
                }
            }

            // * 3. 数据组合
            let content = '';
            let title = $('.mantine-1k4l0d8 h1').text();  // 获取标题文本
            if (title == '') {
                title = fetch_data.name;
            }
            let url = location.href;  // 获取网页链接
            if (md_flag) {  // markdown格式
                content = '## 标题:\n' + title.replace(/([\\<>])/g, '\\$1') + '\n\n' +
                    '## 链接:\n' + url + '\n\n' +
                    '## 触发词:\n' + triggerWords.replace(/([\\<>])/g, '\\$1') + '\n\n' +
                    '## 介绍:\n' + fetch_data.description + '\n\n';
                if (fetch_data.modelVersions[model_version].description)  // 版本介绍
                    content += `## 版本(${fetch_data.modelVersions[model_version].name})介绍:\n${fetch_data.modelVersions[model_version].description}\n\n`;
                content += '## 示例:' + examples.replace(/([\\<>])/g, '\\$1');
            } else {  // txt格式
                content = '标题:\n' + title + '\n\n' +
                    '链接:\n' + url + '\n\n' +
                    '触发词:\n' + triggerWords + '\n\n' +
                    '备注:\n' + $('#tips').val() + '\n\n' +
                    '示例:' + examples;
            }

            // * 4. 下载文件
            let file_name = getFilename(fetch_data, model_version);
            let blob = new Blob([content], {
                type: "text/plain;charset=utf-8"
            });
            saveAs(blob, `${file_name}${md_flag ? '.md' : '.txt'}`);
        });
}

/**
 * 获取当前模型版本示例图中最大宽度
 * @param {*} images 图片json数据
 * @returns 最大宽度
 */
function getMaxWidth(images) {
    let max_width = 0;
    for (let i = 0; i < images.length; i++) {
        if (images[i].width > max_width) {
            max_width = images[i].width;
        }
    }
    return max_width;
}

/**
 * 将 Blob 对象转换为文本内容
 * @param {*} blob blob数据
 * @param {*} callback 回调函数
 */
function blobToText(blob, callback) {
    var reader = new FileReader();
    reader.onload = function () {
        var text = reader.result;
        callback(text);
    };
    reader.readAsText(blob);
}

/**
 * 图片下载
 * @param {*} pic_url 图片链接
 * @param {*} file_name 文件名
 * @param {*} content_flag 内容返回flag
 * @returns
 */
function downloadImg(pic_url, file_name, content_flag = 0) {
    return new Promise(function (resolve, reject) {
        let x = new XMLHttpRequest();
        x.open('GET', pic_url, true);
        x.responseType = 'blob';
        x.onload = function () {
            // 将 Blob 对象转换为文本内容
            blobToText(this.response, function (text) {
                // console.log(text);

                // 判断图片是否含有参数
                if ((!text.includes('seed')) && (!text.includes('Seed')) && (!text.includes('UNICODE'))
                    && localStorage.getItem('png_format').includes('info_tips')) {  // 查找不到seed字符或UNICODE字符视为没有参数
                    file_name += '_noInfo';
                }

                // 判断是否增加预览格式
                if (localStorage.getItem('png_format').includes('preview')) {
                    file_name += '.preview';
                }

                // 获取图片格式
                if (text.startsWith('GIF89a'))  // 以该子字符串开头
                {
                    file_name += '.gif';
                } else if (pic_url.endsWith('.mp4')) { // mp4格式结尾 NOTE: 或者可以判断blob数据的type属性
                    file_name += '.mp4';
                } else {
                    file_name += '.png';
                }

                if (content_flag == 1) {  // 如果以zip格式进行打包
                    // resolve里的参数就是传入then的数据
                    resolve({ fileName: file_name, blobData: x.response });  // 返回包含文件名和 Blob 数据的对象
                } else {  // 直接下载图片
                    // 保存图片
                    let url = window.URL.createObjectURL(x.response);
                    let a = document.createElement('a');
                    a.href = url;
                    a.download = file_name;
                    a.click();
                    resolve('图片下载完成');  // 返回图片下载完成
                }

            });
        };
        x.onerror = function () {
            reject(new Error('图片下载失败'));
        };
        x.send();
    });
}

/**
 * 下载所有预览图
 * @param {*} fetch_data 请求的数据
 * @param {*} model_version 模型版本号
 * @param {*} images_arr 图片链接数组
 * @param {*} filename 文件名
 * @param {*} index 索引号
 * @returns
 */
function downloadImages(fetch_data, model_version, images_arr, filename, index) {
    // 判断API中是否有图片信息(理论上没有图片不会有按钮能点击,为了保险增加该检测)
    if (!fetch_data.modelVersions[model_version].images.length) {
        alert('未查找到该模型版本图片的信息,请自行检查错误');
        $('#downPic').text(`下载出现错误!`).addClass('public-error');
        return;
    }

    if (index >= images_arr.length) {
        console.log('所有图片下载完成');
        console.log('=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=');
        // 这里可以执行所有下载完成后需要执行的代码
        $('#downPic').text(`下载所有预览图(${images_arr.length})`).removeClass('public-downloading');
        return;
    }

    // 通过遍历请求的数据中图片链接获取宽度信息
    let image_width = 0;
    for (let i = 0; i < fetch_data.modelVersions[model_version].images.length; i++) {  // 当前模型版本内所有图片遍历
        if (fetch_data.modelVersions[model_version].images[i].url.includes(images_arr[index].split('/')[4])) {
            image_width = fetch_data.modelVersions[model_version].images[i].width;
            break;
        }
    }
    if (image_width == 0) {// 图片不在api返回数据内(api只返回10张图片)
        image_width = getMaxWidth(fetch_data.modelVersions[model_version].images);
    }
    // let image_width = getMaxWidth(fetch_data.modelVersions[model_version].images);

    // 图片链接获取
    let image_url = '';
    if (images_arr[index]) {
        image_url = images_arr[index].replace(/width=\d+/, 'width=' + image_width);
    }

    console.log(`pic_down: ${index + 1}/${images_arr.length}`);
    console.log(`ori_url : ${images_arr[index]}`);
    console.log(`pic_url : ${image_url}`);

    // 图片下载
    downloadImg(image_url, `${filename}_${index + 1}`)
        .then(function () {
            $('#downPic').text(`Downloading(${index + 1}/${images_arr.length})`);
            downloadImages(fetch_data, model_version, images_arr, filename, index + 1);
        })
        .catch(function (error) {
            console.error(error);
            // 是否继续下载
            let continue_flag = confirm('图片下载出现错误,是否跳过本张图片并尝试下载下一张?');
            if (continue_flag) {
                downloadImages(fetch_data, model_version, images_arr, filename, index + 1);
            } else {
                $('#downPic').text(`下载出现错误!`).addClass('public-error');
                return;
            }
        });
}

/**
 * 获取当前页面中的gallery一栏图片的API数据
 * @param {*} model_version_id 模型版本id
 * @param {*} model_id 模型id
 * @param {*} cursor
 * @param {*} cursor_value
 */
function getGalleryData(model_version_id, model_id, cursor = null, cursor_value = 'undefined') {
    let period = 'AllTime';
    let sort = 'Most Reactions';
    let temp = JSON.parse(localStorage.getItem('model-image-filters'));
    if (temp) {
        period = temp.period;
        sort = temp.sort;
    }

    // 数据组合
    let request_data = `{"json": {"period": "${period}","sort": "${sort}","modelVersionId": ${model_version_id},"modelId": ${model_id},"limit": 50,"cursor": ${cursor},"authed": true},"meta": {"values": {"cursor": ["${cursor_value}"]}}}`;
    // 请求数据
    console.log(request_data);
    let url = 'https://civitai.com/api/trpc/image.getImagesAsPostsInfinite?input=' + encodeURIComponent(request_data);
    console.log('请求链接: ' + url);
    fetch(url)
        .catch(error => {
            alert('API数据不正常,请检查或重试\n' + error.message);
        })
        .then(response => {
            if (!response.ok) {
                throw new Error('Network response was not ok');
            }
            return response.json();
        })
        .then(data => {
            console.log(data);

            // 获取下次请求值
            let nextcursor = data.result.data.json.nextCursor;
            let nextcursor_value = data.result.data.meta.values.nextCursor;

            // 数据存储
            let gallery_info = data.result.data.json.items;
            for (let i = 0; i < gallery_info.length; i++) {
                gallery_data_arr.push(gallery_info[i]);
            }

            // 判断是否结束
            if (nextcursor != null && ($('#gallery div[id]').length - 2) > gallery_data_arr.length) {
                // 请求下一个数据
                getGalleryData(model_version_id, model_id, nextcursor, nextcursor_value);
            }

            return;
        });
}

/**
 * 将所有预览图打包下载
 * @param {*} fetch_data 请求的数据
 * @param {*} model_version 模型版本号
 * @param {*} images_arr 图片链接数组
 * @param {*} file_name 文件名
 * @param {*} index 索引号
 */
function downloadZip(fetch_data, model_version, images_arr, file_name, index) {
    // 判断API中是否有图片信息(理论上没有图片不会有按钮能点击,为了保险增加该检测)
    if (!fetch_data.modelVersions[model_version].images.length) {
        alert('未查找到该模型版本图片的信息,请自行检查错误');
        $('#downPic').text(`下载出现错误!`).addClass('public-error');
        return;
    }

    // * 0. 创建zip实例 (提前到程序开始处)
    // var zip = new JSZip();

    // 图片遍历结束
    if (index >= images_arr.length) {
        // * 2. ZIP压缩
        // 将Zip打包成Blob对象
        zip.generateAsync({
            type: "blob"
        }, function updateCallback(metadata) {
            // 显示压缩进度
            var percent = metadata.percent.toFixed(2);
            $('#downPic').text(percent + "% done");
        }).then(function (content) {
            // 下载Zip文件
            saveAs(content, `${file_name}.zip`);

            // * 3. 下载完成并给出提示
            console.log('ZIP打包下载完成');
            console.log('=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=');
            // 这里可以执行所有下载完成后需要执行的代码
            $('#downPic').text(`下载所有预览图(${images_arr.length})`).removeClass('public-downloading');
            return;
        });
    }
    if (index >= images_arr.length) { // 上面是异步的,需要提前退出不执行下面的语句
        return;
    }

    // * 1. 获取图片内容
    // 1.1 获取图片链接
    // 通过遍历请求的数据中图片链接获取宽度信息
    let image_width = 0;
    for (let i = 0; i < fetch_data.modelVersions[model_version].images.length; i++) {  // 当前模型版本内所有图片遍历
        if (fetch_data.modelVersions[model_version].images[i].url.includes(images_arr[index].split('/')[4])) {
            image_width = fetch_data.modelVersions[model_version].images[i].width;
            break;
        }
    }
    if (image_width == 0) {// 图片不在api返回数据内(api只返回10张图片)
        image_width = getMaxWidth(fetch_data.modelVersions[model_version].images);
    }
    // let image_width = getMaxWidth(fetch_data.modelVersions[model_version].images);

    // 图片链接获取
    let image_url = '';
    if (images_arr[index]) {
        image_url = images_arr[index].replace(/width=\d+/, 'width=' + image_width);
    }

    console.log(`pic_down: ${index + 1}/${images_arr.length}`);
    console.log(`ori_url : ${images_arr[index]}`);
    console.log(`pic_url : ${image_url}`);

    // 1.2 获取图片内容
    downloadImg(image_url, `${file_name}_${index + 1}`, 1)
        .then(({ fileName, blobData }) => {
            // 文件名:fileName,Blob 数据:blobData
            // 在此处进行需要的操作
            console.log(blobData);

            // 向zip中添加文件
            zip.file(fileName, blobData);

            // 1.3 请求下一张图片
            $('#downPic').text(`Downloading(${index + 1}/${images_arr.length})`);
            downloadZip(fetch_data, model_version, images_arr, file_name, index + 1);
        });
}

/**
 * 获取剪切板内容并保存到本地
 * @param {*} file_name 文件名
 * @note 首次需要手动确认剪贴板权限
 */
function readClipText(file_name) {
    var clipPromise = navigator.clipboard.readText();
    clipPromise.then(function (clipText) {
        // alert(clipText);
        const stringData = clipText
        // dada 表示要转换的字符串数据,type 表示要转换的数据格式
        const blob = new Blob([stringData], {
            type: "text/plain;charset=utf-8"
        })
        // 根据 blob生成 url链接
        const objectURL = URL.createObjectURL(blob)
        // 创建一个 a 标签Tag
        const aTag = document.createElement('a')
        // 设置文件的下载地址
        aTag.href = objectURL
        // 设置保存后的文件名称
        aTag.download = `${file_name}.txt`
        // 给 a 标签添加点击事件
        aTag.click()
        // 释放一个之前已经存在的、通过调用 URL.createObjectURL() 创建的 URL 对象。
        // 当你结束使用某个 URL 对象之后,应该通过调用这个方法来让浏览器知道不用在内存中继续保留对这个文件的引用了。
        URL.revokeObjectURL(objectURL)
    });
}