// ==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;
});
}