Civitai网站辅助(English Version)

Civitai model information assisted acquisition

// ==UserScript==
// @name         Civitai网站辅助(English Version)
// @namespace    http://tampermonkey.net/
// @version      2.5.5
// @description  Civitai model information assisted acquisition
// @author       Faded_lov
// @license      GPL
// @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

// ==/UserScript==
var gallery_data_arr = new Array();

$(function () {
    // ^ 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-325luz: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 file">
        <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-6v7rx2 .mantine-1g4q40w').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>
    `);
    $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>
    `);
    $down_msg_div = $(`
    <div class="container">
        <div class="btn-down" id="downInfo" style="margin-right:3px;" title="Download .info file (Civitai Helper)">Loading...</div>
        <div class="btn-down" id="downMsg" style="margin-left:3px;" title="Download document">Loading...</div>
    </div>
    <div class="container">
        <div class="png-set-btn" onclick="png_set()" title="Set the file name format of the downloaded image."><span>Config</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="File suffix">.preview.png</label>
                    </div>
                    <div><input type="checkbox" name="info_tips" id="png-set-2"><label for="png-set-2" title="Determine whether the picture contains metadata.">_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="Download all the preview images on the left.">Loading...</div>
    </div>
    `);
    $textarea = $(`
    <textarea id="tips" rows=15 placeholder='Remarks...' ></textarea>
    `);
    $style = $(`
    <style>
        .filename {
            font-weight: 600;
            color: #d400ff;
        }

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

        .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-1ghun0t').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('request Url: ' + requestUrl);
    fetch(requestUrl)
        .catch(error => {
            alert('API returned data is abnormal, please check the console or try again.\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('Current version: ' + $(`.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('The current model has the same version name. Ensure that the downloaded information is correct.');
                    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('Model version number (starting from 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('Current version: ' + 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 number: ' + model_version);
                    filename = getFilename(data, model_version);
                }, 500);
            });
            // // 手动触发模型切换函数以代替模型版本初次获取
            // $('.mantine-lw13s0 .mantine-lw13s0 button:eq(0)').click();

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

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

            // * 2.7 批量下载图片的按钮事件处理
            // 2.6.1 获取当前模型版本预览图数量
            let pictures_count = $('.mantine-116dglk .mantine-Carousel-container .mantine-gbunjo').length;
            $('#downPic').text(`Download all images(${pictures_count})`);
            if (!pictures_count) {
                // alert('绑定图片下载失败,请手动点击切换版本(当前版本就行)以重新绑定');
                $('#downPic').text('Error,Please rebind the button.')
                    .prop('title', 'Download all the preview images on the left.');
            }
            // 切换版本时更新当前版本预览图提示
            $('.mantine-lw13s0 .mantine-lw13s0 button').on('click', function () {
                $('#downPic').text(`Loading...`);
                let update_status = setInterval(function () {
                    if ($('.picbtn').length != 0) {  // 页面加载完成
                        pictures_count = $('.mantine-116dglk .mantine-Carousel-container .mantine-gbunjo').length
                        $('#downPic').text(`Download all images(${pictures_count})`);
                        clearInterval(update_status);
                    }
                }, 500);
            });

            // 2.6.2 下载所有预览图
            $('#downPic').on({
                click: () => {
                    console.log('=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=');
                    $('#downPic').text('Downloading...').addClass('public-downloading');
                    let images_arr = new Array();
                    $('.mantine-116dglk .mantine-Carousel-container .mantine-gbunjo').each(function (index, domEle) {
                        images_arr.push($(domEle).find('img').prop('src'));
                    });
                    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 this picture.">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('Bind image download button...');
                    } else {
                        console.log('The image download button is bound successfully.');
                        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').prop('src'); // 图片链接

                // 判断API中是否有图片(理论上没有图片不会有按钮能点击,为了保险增加该检测)
                if (!data.modelVersions[model_version].images.length) {
                    alert('No information was found for this model version picture, please check for errors.');
                    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('The image width is not found,Use the maximum width in the model preview\r\nThe retrieved image is most likely not the original.');
                    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('An image download error occurred. Please check the related information in the console.\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="Set auto download content."><span>Setting</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">model</label></div>
                            <div><input type="checkbox" name="picture_down" id="setting_2"><label for="setting_2">the first image</label></div>
                            <div><input type="checkbox" name="text_down" id="setting_3"><label for="setting_3">document</label></div>
                            <div><input type="checkbox" name="info_down" id="setting_4"><label for="setting_4">.info file</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="Download the selected file at setting."> Auto Download!</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('Please set the auto download option first.');
                        return;
                    } else {
                        console.log('=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=↓=');
                        console.log(settings);
                        $('.easy_download').text('Task submission......').addClass('public-downloading');
                    }

                    // 点击like
                    if (theme == 'd') { // 深色模式
                        $('.mantine-mwqi5l .mantine-bx8wvk:eq(1)').click();
                    } else { // 浅色模式
                        $('.mantine-1g4q40w .mantine-11d452b: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(' Auto Download!').removeClass('public-downloading');
                            })
                            .catch(function (error) {
                                console.error(error);
                                alert('An image download error occurred. Please check the related information in the console.\n' + error.message);
                                // 这里可以执行下载失败后需要执行的代码
                                $('.easy_download').text(' Download error!').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;  // 默认链接
                        // 尝试查找类型为模型的链接(如果有)
                        for (let i = 0; i < data.modelVersions[model_version].files.length; i++) {
                            if (data.modelVersions[model_version].files[i].type.indexOf("Model") > -1) {
                                down_url = data.modelVersions[model_version].files[i].downloadUrl;
                            }
                        }
                        console.log('download url: ' + 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(' Auto Download!').removeClass('public-downloading');
                    }
                    console.log('=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=');
                }
            });

            // ^ 5. 触发词下载格式切换
            // * 5.1 相关控件添加
            $trigger_words = $(`
            <div class="trigger-words-checkbox" title="Whether to add carriage return to the trigger word to separate it.">
                <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-1tk98uh" style="
                line-height: 34px;
                height: 34px;
                width: 125px;
                margin: auto 5px;
                text-align: center;
            " title="Rebind the image download button in the current page.">Rebind Button</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('Page element count:' + $('#gallery div[id]').length);
                        console.log(gallery_data_arr);
                        // 重新绑定按钮
                        $("a[href^='/images/']").after($btn);
                        $(this).text('Rebind Button');
                    }
                }, 500);
            });

        });
});

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

    let file_name = '';
    if (fetch_data.modelVersions[model_version].files.length == 1) {
        // 只有一个文件
        file_name = fetch_data.modelVersions[model_version].files[0].name;
    } else {
        // 存在多个文件,选择其中的模型文件
        for (let i = 0; i < fetch_data.modelVersions[model_version].files.length; i++) {
            if (fetch_data.modelVersions[model_version].files[i].type.indexOf("Model") > -1) {
                file_name = fetch_data.modelVersions[model_version].files[i].name;
                break;
            }
        }
    }
    file_name = file_name.slice(0, file_name.lastIndexOf('.'));
    $('.filename').text(file_name);
    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('request url: ' + url);
    fetch(url)
        .catch(error => {
            alert('API data is abnormal. Please check or try again.\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 file').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;
    for (let i = 0; i < words.length; i++) {
        triggerWords += words[i] + ',';
        if ($('.trigger-words-checkbox input').prop('checked')) {
            triggerWords += '\n';
        }
    }
    if (words.length == 0) {
        triggerWords = 'none';
    }

    // * 2. 获取示例图参数
    let examples = '';
    let pictures = fetch_data.modelVersions[model_version].images;
    for (let j = 0; j < pictures.length; j++) {
        // 获取图片链接
        let img_url = pictures[j].url;
        // 获取图片信息数据(json)
        let img_info = pictures[j].meta;
        // 存储参数的变量
        let msg = '';
        for (let key in img_info) {
            if (key == 'hashes') {
                msg += '--hashes: ' + JSON.stringify(img_info[key]) + '\n';
                continue;
            }
            if (key == 'resources') {
                for (let res in img_info[key]) {
                    msg += '--' + key + ': ' + JSON.stringify(img_info[key][res]) + '\n';
                }
                continue;
            }
            msg += '--' + key + ': ' + img_info[key] + '\n';
        }
        if (md_flag) {  // markdown格式
            if (msg) {
                examples += '\n================< ' + (j + 1) + ' >================\n' + 'Preview: \n![](' + img_url + ')\nMetadata: \n' + msg;
            } else {
                examples += '\n================< ' + (j + 1) + ' >================\n' + 'Preview: \n![](' + img_url + ')\nThere is no metadata.\n';
            }
        } else {  // txt格式
            if (msg) {
                examples += '\n================< ' + (j + 1) + ' >================\n' + 'Preview url: \n' + img_url + '\nMetadata: \n' + msg;
            } else {
                examples += '\n================< ' + (j + 1) + ' >================\n' + 'Preview url: \n' + img_url + '\nThere is no metadata.\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 = '## Title:\n' + title.replace(/([\\<>])/g, '\\$1') + '\n\n' +
            '## Url:\n' + url + '\n\n' +
            '## Trigger words:\n' + triggerWords.replace(/([\\<>])/g, '\\$1') + '\n\n' +
            '## Introduction: \n' + fetch_data.description + '\n\n';
        if (fetch_data.modelVersions[model_version].description)  // 版本介绍
            content += `## Version(${fetch_data.modelVersions[model_version].name})introduction: \n${fetch_data.modelVersions[model_version].description}\n\n`;
        content += '## Examples:' + examples.replace(/([\\<>])/g, '\\$1');
    } else {  // txt格式
        content = 'Title:\n' + title + '\n\n' +
            'Url:\n' + url + '\n\n' +
            'Trigger words:\n' + triggerWords + '\n\n' +
            'Remarks:\n' + $('#tips').val() + '\n\n' +
            'Examples:' + 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  文件名
 */
function downloadImg(pic_url, file_name) {
    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 {
                    file_name += '.png';
                }

                // 保存图片
                let url = window.URL.createObjectURL(x.response);
                let a = document.createElement('a');
                a.href = url;
                a.download = file_name;
                a.click();
                resolve('Picture download completed.');
            });
        };
        x.onerror = function () {
            reject(new Error('Image download failure.'));
        };
        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('No information was found for this model version picture, please check for errors.');
        $('#downPic').text(`Download error!`).addClass('public-error');
        return;
    }

    if (index >= images_arr.length) {
        console.log('All images have been downloaded.');
        console.log('=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=↑=');
        // 这里可以执行所有下载完成后需要执行的代码
        $('#downPic').text(`Download all images(${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('Image download error. Do you want to skip this image and try to download the next one?');
            if (continue_flag) {
                downloadImages(fetch_data, model_version, images_arr, filename, index + 1);
            } else {
                $('#downPic').text(`Download error!`).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('request url: ' + url);
    fetch(url)
        .catch(error => {
            alert('API data is abnormal. Please check or try again\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;
        });
}