// ==UserScript==
// @name PixivUserBatchDownload
// @name:zh-CN P站画师个人作品批量下载工具
// @name:zh-TW P站畫師個人作品批量下載工具
// @name:zh-HK P站畫師個人作品批量下載工具
// @description Batch download pixiv user's images in one key.(Based on Aria2)
// @description:zh-CN 配合Aria2,一键批量下载P站画师的全部作品
// @description:zh-TW 配合Aria2,一鍵批量下載P站畫師的全部作品
// @description:zh-HK 配合Aria2,一鍵批量下載P站畫師的全部作品
// @version 5.21.152
// @author Mapaler <mapaler@163.com>
// @copyright 2016~2024+, Mapaler <mapaler@163.com>
// @namespace http://www.mapaler.com/
// @icon https://www.pixiv.net/favicon.ico
// @homepage https://github.com/Mapaler/PixivUserBatchDownload
// @supportURL https://github.com/Mapaler/PixivUserBatchDownload/issues
// @match *://www.pixiv.net/*
// @exclude *://www.pixiv.net/upload.php*
// @exclude *://www.pixiv.net/messages.php*
// @exclude *://www.pixiv.net/ranking.php*
// @exclude *://www.pixiv.net/info.php*
// @exclude *://www.pixiv.net/ranking_report_user.php*
// @exclude *://www.pixiv.net/setting*
// @exclude *://www.pixiv.net/stacc*
// @exclude *://www.pixiv.net/premium*
// @exclude *://www.pixiv.net/discovery*
// @exclude *://www.pixiv.net/howto*
// @exclude *://www.pixiv.net/idea*
// @exclude *://www.pixiv.net/ads*
// @exclude *://www.pixiv.net/terms*
// @exclude *://www.pixiv.net/novel*
// @exclude *://www.pixiv.net/cate_r18*
// @exclude *://www.pixiv.net/manage*
// @exclude *://www.pixiv.net/report*
// @resource pubd-style https://github.com/Mapaler/PixivUserBatchDownload/raw/master/PixivUserBatchDownload%20ui.css?v=5.20.146
// @require https://cdn.jsdelivr.net/npm/crypto-js@4.2.0/core.min.js
// @require https://cdn.jsdelivr.net/npm/crypto-js@4.2.0/md5.min.js
// @require https://cdn.jsdelivr.net/npm/crypto-js@4.2.0/sha256.min.js
// @require https://cdn.jsdelivr.net/npm/crypto-js@4.2.0/enc-base64.min.js
// @grant window.close
// @grant window.focus
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_addStyle
// @grant GM_getResourceText
// @grant GM_addValueChangeListener
// @grant GM_registerMenuCommand
// @grant GM_unregisterMenuCommand
// @connect pixiv.net
// @connect pximg.net
// @connect localhost
// @connect 127.0.0.1
// @noframes
// ==/UserScript==
(function() {
'use strict';
//获取当前是否是本地开发状态,自己用 localStorage.setItem("pubd-dev", "1") 既可以开启,localStorage.removeItem("pubd-dev") 关闭
//注意开发状态下为了方便随时调整预览,css 不会自动加载,需要自己用 Stylus 扩展添加 PixivUserBatchDownload ui.css 的内容。
const mdev = Boolean(localStorage.getItem("pubd-dev"));
if (mdev) console.log("GM_info信息:",GM_info); //开发模式时显示meta数据
/*
* 公共变量区
*/
//储存vue框架下P站页面主要内容的DIV位置,现在由程序自行搜索判断,搜索依据为 mainDivSearchCssSelectorArray。
//#root vue的root,源代码中固定存在
//#root>div:nth-of-type(2) 真正存储页面内容的动态 div,搜索条件为root下新增的没有 id 的 node(P站目前是这样)。
//#root>div:nth-of-type(2)>div 再往下由于只有一个单独div,因此再往继续找到有 2 个以上的 node 的时候,记为子 root,其实后面的代码基本不会用到。
let subRoot = null;
//子 root 下面能找到按钮插入点的 node,暨主要内容的 div 而不是顶栏,记为 mainDiv
let mainDiv = null;
//不同页面开始按钮的插入位点
//哪天P站位置改版了,可能就需要手动调整这些路径
//下面的 :scope 指的是 mainDiv,2023年8月9日 当前路径为 #root>div:nth-of-type(2)>div>div:nth-of-type(3)
// 先使用 selectors 字符串搜索,获取失败时使用 fallcack 处理函数
const mainDivSearchCssSelector = [
{selectors: '#spa-contents .user-stats'}, // 手机版用户页
{selectors: '#spa-contents .user-details-card'}, // 手机版作品页
// PC版 单个作品页
{selectors: ':scope aside section:has(button[data-gtm-user-id][data-click-label])',
fallcack: (node) => {
const ele = node.querySelector(':scope main+aside>section div[title]');
return ele?.closest('section');
},
},
//2024年10月11日 目前火狐更新到128后,支持 :has 了,定位全部被导向这一条了,上面一条不起作用了。
// PC版 用户资料页
{selectors: ':scope :has(>div>h1,>div>button[data-gtm-user-id][data-click-label])',
fallcack: (node) => {
const ele = node.querySelector(':scope div[title]:not(a [title])');
return ele?.parentElement?.parentElement?.nextElementSibling;
},
},
];
//搜索页,列表的ul位置(用来显示收藏状态)
const searchListCssPath = ':scope>div>div:nth-of-type(6)>div>section>div:nth-of-type(2)>ul';
//作者页面“主页”按钮的CSS位置(用来获取作者ID)
const userMainPageCssPath = ":scope nav>a";
//作品页,收藏按钮的CSS位置(用来获取当前作品ID)
const artWorkStarCssPath = ":scope main>section>div>div>figcaption>div>div>ul>li:nth-of-type(2)>a";
//作品页,作者头像链接的CSS位置(用来获取作者ID)
const artWorkUserHeadCssPath = ":scope aside>section>h2>div>a";
const scriptVersion = GM_info.script.version.trim(); //本程序的版本
const scriptIcon = GM_info.script.icon64 || GM_info.script.icon; //本程序的图标
const scriptName = (defaultName=>{ //本程序的名称
if (typeof(GM_info) != "undefined") //使用了扩展
{
if (GM_info.script.name_i18n)
{
return GM_info.script.name_i18n[navigator.language.replaceAll("-","_")]; //支持Tampermonkey
} else
{
return GM_info.script.localizedName || //支持Greasemonkey 油猴子 3.x
GM_info.script.name; //支持Violentmonkey 暴力猴
}
}
return defaultName;
})('PixivUserBatchDownload');
const pubd = { //储存程序设置
configVersion: 2, //当前设置版本,用于处理老版本设置的改变
touch: false, //是否为手机版
loggedIn: false, //登录了(未启用)
start: null, //开始按钮指针
menu: null, //菜单指针
dialog: { //窗口们的指针
config: null, //设置窗口
login: null, //登录窗口
refresh_token: null, //刷新token窗口
downthis: null, //下载当前窗口
downillust: null, //下载当前作品窗口
importdata: null, //导入数据窗口(还未开发)
multiple: null, //多画师批量下载窗口(还未开发)
},
oAuth: null, //储存账号密码
downSchemes: [], //储存下载方案
downbreak: false, //是否停止发送Aria2的flag
ajaxTimes: 0, //已经从P站获取信息的次数(用来判断是否要减速)
fastStarList: null, //储存快速收藏的简单数字
starUserlists: [], //储存完整的下载列表
};
//匹配P站内容的正则表达式
const illustPathRegExp = /(\/.+\/\d{4}\/\d{2}\/\d{2}\/\d{2}\/\d{2}\/\d{2}\/((\d+)(?:-([0-9a-zA-Z]+))?(?:_p|_ugoira)))\d+(?:_\w+)?\.([\w\d]+)/i; //P站画作地址 path部分 正则匹配式
const limitingPathRegExp = /(\/common\/images\/(limit_(?:mypixiv|unknown)_\d+))\.([\w\d]+)/i; //P站无权访问作品地址 path部分 正则匹配式
const limitingFilenameExp = /limit_(mypixiv|unknown)/ig; //P站上锁图片文件名正则匹配式
//Header使用
const PixivAppVersion = "6.127.0"; //Pixiv APP的版本
const AndroidVersion = "15.0.0"; //安卓的版本
const UA = `PixivAndroidApp/${PixivAppVersion} (Android ${AndroidVersion}; Android SDK built for x64)`; //向P站请求数据时的UA
//X_Client加密的salt,目前是固定值
const X_Client_Hash_Salt = "28c1fdd170a5204386cb1313c7077b34f83e4aaf4aa829ce78c231e05b0bae2c";
/*
const X_Client_Hash_Salt = [ //数值写法
0x28,0xC1,0xFD,0xD1,
0x70,0xA5,0x20,0x43,
0x86,0xCB,0x13,0x13,
0xC7,0x07,0x7B,0x34,
0xF8,0x3E,0x4A,0xAF,
0x4A,0xA8,0x29,0xCE,
0x78,0xC2,0x31,0xE0,
0x5B,0x0B,0xAE,0x2C
].map(n=>n.toString(16).toLowerCase()).join('');
*/
const Referer = "https://app-api.pixiv.net/"; //手机app的 Referer,不是这个无法下载图片
const ContentType = "application/x-www-form-urlencoded; charset=UTF-8"; //重要,不要轻易修改
//登录时的固定参数
const client_id = "MOBrBDS8blbauoSck0ZfDbtuzpyT"; //安卓版固定数据
const client_secret = "lsACyCD94FhDUtGTXi3QzcFE2uU1hqtDaKeqrdwj"; //安卓版固定数据
let thisPageUserid = null, //当前页面的画师ID
thisPageIllustid = null, //当前页面的作品ID
downIllustMenuId = null, //下载当前作品的菜单的ID(Tampermonker菜单内的指针)
recommendList = null; //推荐作品列表Dom位置
const startDelayAjaxTimes = 100; //开始执行延迟的ajax次数
const ajaxDelayDuration = 1000; //每次延迟的时间
const changeTermwiseCount = 6000; //图片数量大于这个值就从按作者发送切换为按图片发送
/*
* 初始化数据库,这个功能没继续开发了的
*/
if (mdev)
{
const dbName = "PUBD";
var db;
const DBOpenRequest = indexedDB.open(dbName);
DBOpenRequest.onsuccess = function(event) {
db = event.target.result; //DBOpenRequest.result;
console.log("PUBD:数据库已可使用");
};
DBOpenRequest.onerror = function(event) {
// 错误处理
console.log("PUBD:数据库无法启用",event);
};
DBOpenRequest.onupgradeneeded = function(event) {
let db = event.target.result;
// 建立一个对象仓库来存储用户的相关信息,我们选择 user.id 作为键路径(key path)
// 因为 user.id 可以保证是不重复的
let usersStore = db.createObjectStore("users", { keyPath: "user.id" });
// 建立一个索引来通过姓名来搜索用户。名字可能会重复,所以我们不能使用 unique 索引
usersStore.createIndex("name", "user.name", { unique: false });
// 使用账户建立索引,我们确保用户的账户不会重复,所以我们使用 unique 索引
usersStore.createIndex("account", "user.account", { unique: true });
let illustsStore = db.createObjectStore("illusts", { keyPath: "id" });
illustsStore.createIndex("type", "type", { unique: false });
illustsStore.createIndex("userid", "user.id", { unique: false });
// 使用事务的 oncomplete 事件确保在插入数据前对象仓库已经创建完毕
illustsStore.transaction.oncomplete = function(event) {
console.log("PUBD:数据库建立完毕");
};
};
}
/*
* 获取初始状态
*/
//尝试获取旧版网页对象
/*if (typeof(unsafeWindow) != "undefined")
{ //原来的信息-除少部分页面外已失效2020年7月9日
const pixiv = unsafeWindow.pixiv;
}*/
if (typeof(pixiv) != "undefined")
{
if (mdev) console.log("PUBD:本页面存在 pixiv 对象:",pixiv);
thisPageUserid = parseInt(pixiv.context.userId);
if (pixiv.user.loggedIn)
{
pubd.loggedIn = true;
}
if (/touch/i.test(pixiv.touchSourcePath))
{
pubd.touch = true; //新版的手机页面也还是老板结构-2020年7月9日
document.body.classList.add('pubd-touch');
}
}
//尝试获取当前页面画师ID
const metaPreloadData = document.querySelector('#meta-preload-data'); //HTML源代码里有,会被前端删掉的数据
if (metaPreloadData != undefined) //更加新的存在于HTML元数据中的页面信息
{
pubd.loggedIn = true;
if (mdev) console.log("PUBD:本页面抢救出 metaPreloadData 对象:",metaPreloadData);
const preloadData = JSON.parse(metaPreloadData.content);
if (mdev) console.log("PUBD:metaPreloadData 中的 preloadData 元数据:",preloadData);
if (preloadData.user) thisPageUserid = parseInt(Object.keys(preloadData.user)[0]);
if (preloadData.illust) thisPageIllustid = parseInt(Object.keys(preloadData.illust)[0]); //必须判断是否存在,否则会出现can't convert undefined to object错误
}
//获取是否为老的手机版
if (location.host.includes("touch")) //typeof(pixiv.AutoView)!="undefined"
{
pubd.touch = true;
document.body.classList.add('pubd-touch');
}
/*
* 自定义对象区
*/
Date.PixivTimezoneOffset = (date=>{ //国内为 +08:00
const o = date.getTimezoneOffset(); //本地时区差分钟数,正时区为负数
//时区 +差时:差分
return `${Math.trunc(-o/60).toLocaleString(void 0,{
style: "decimal",
minimumIntegerDigits: 2,
signDisplay: "always"
})}:${(o%60).toLocaleString(void 0,{
style: "decimal",
minimumIntegerDigits: 2,
signDisplay: "never"
})}`;
//return `${o<=0?'+':'-'}${[o/60,o%60].map(n=>Math.trunc(Math.abs(n)).toString().padStart(2,'0')).join(':')}`;
})(new Date());
/**
* 生成P站需要的时间格式
* @returns P站时间格式如 "2019-09-03T18:51:40+08:00"
*/
Date.prototype.toPixivString = function() {
let s = this.toJSON();
return s.substring(0,s.indexOf('.')) + Date.PixivTimezoneOffset;
};
//
/**
* 日期格式化
* @param {string} format 格式 y:年,M:月,d:日,h:时,m:分,s:秒
* @returns {string} 格式化的日期字符串
* @see {@link https://www.cnblogs.com/tugenhua0707/p/3776808.html|代码来源}
* @example
* // returns "2023-08-29 21:22:51"
* new Date().format("yyyy-MM-dd hh:mm:ss");
*/
Date.prototype.format = function(format) {
let fmt = format; //复制到新的字符串内
const o = [
["M+" , this.getMonth()+1], //月份
["d+" , this.getDate()], //日
["h+" , this.getHours()], //小时
["m+" , this.getMinutes()], //分
["s+" , this.getSeconds()], //秒
["q+" , Math.floor((this.getMonth()+3)/3)], //季度
["S" , this.getMilliseconds()] //毫秒
];
let regRes;
if(regRes = /(y+)/.exec(fmt)) {
fmt = fmt.replace(regRes[1], (this.getFullYear()+"").substr(4 - regRes[1].length));
}
for (let i = 0; i < o.length; i++) {
let [k, v] = o[i];
if(regRes = new RegExp("("+ k +")").exec(fmt)) {
fmt = fmt.replace(regRes[1], (regRes[1].length===1) ? v : v.toString().padStart(2,'0'));
}
}
return fmt;
}
/*
//一个被收藏的画师
class StarUser{
constructor(id){
const user = this;
user.id = id;
user.infoDone = false;
user.downDone = false;
user.userinfo = null;
user.illusts = null;
}
}
*/
//一个画师收藏列表
class UsersStarList{
constructor(title,userArr = []){
this.title = title;
this.users = new Set(Array.from(userArr));
this.users.delete(null);
}
add(userid)
{
if (isNaN(userid)) userid = parseInt(userid,10);
if (!isNaN(userid) && userid != null) this.users.add(userid);
}
delete(userid)
{
if (isNaN(userid)) userid = parseInt(userid,10);
this.users.delete(userid);
}
has(userid)
{
if (isNaN(userid)) userid = parseInt(userid,10);
return this.users.has(userid);
}
toggle(userid)
{ //切换有无
if (isNaN(userid)) userid = parseInt(userid,10);
const _users = this.users;
if (_users.has(userid) || isNaN(userid) || userid == null)
{
_users.delete(userid);
return false;
}else
{
_users.add(userid);
return true;
}
}
importArray(arr)
{
const arrMaxLength = 500000;
arr = arr.filter(uid=>!isNaN(uid));
if (arr.length>arrMaxLength)
{
alert(`PUBD:收藏用户最多仅允许添加 ${arrMaxLength.toLocaleString()} 个数据。`);
arr = arr.splice(500000); //删除50万以后的
}
const _users = this.users;
arr.forEach(uid=>_users.add(uid));
}
exportArray()
{
return Array.from(this.users);
}
}
//一个本程序使用的headers数据
class HeadersObject{
constructor(importObj){
const header = this;
const timeStr = new Date().toPixivString();
header["App-OS"] = "android";
header["App-OS-Version"] = AndroidVersion;
header["App-Version"] = PixivAppVersion;
header["User-Agent"] = UA;
header["Content-Type"] = ContentType; //重要
header["Referer"] = Referer; // jshint ignore:line
header["X-Client-Hash"] = CryptoJS.MD5(timeStr + X_Client_Hash_Salt).toString();
header["X-Client-Time"] = timeStr;
if (typeof(obj) == "object") Object.assign(this, importObj);
}
}
//储存一项图片列表分析数据的对象
class Works{
constructor(){
this.done = false; //是否分析完毕
this.item = []; //储存图片数据
this.break = false; //储存停止分析的Flag
this.runing = false; //是否正在运行的Flasg
this.next_url = ""; //储存下一页地址(断点续传)
}
};
Math.randomInteger = function(max, min = 0)
{
let _max = Math.max(max, min),
_min = Math.min(max, min);
return this.floor(this.random()*(_max-_min+1)+_min);
}
//认证方案
class oAuth2
{
constructor(existAuth){
this.code_verifier = this.constructor.random_code_verifier();
this.login_time = null;
this.authorization_code = null;
this.auth_data = null;
this.idp_urls = { //默认的综合网址集
"account-edit": "https://accounts.pixiv.net/api/v2/account/edit",
"auth-token": "https://oauth.secure.pixiv.net/auth/token",
"auth-token-redirect-uri": "https://app-api.pixiv.net/web/v1/users/auth/pixiv/callback",
};
if (typeof(existAuth) == "object" && existAuth.auth_data)
{
Object.assign(this, existAuth);
}
}
static random_code_verifier()
{
//OAuth 2.0 客户端应用生成一个43~128位的随机字符串 (code_verifier)并查找它的 SHA256 哈希,这称为 code_challenge。
const codeLen = Math.randomInteger(43, 128); //产生43~128位
const passArray = new Uint8Array(codeLen);
window.crypto.getRandomValues(passArray); //获取符合密码学要求的安全的随机值
const unreservedChars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-._~';
const charsLength = unreservedChars.length - 1;
const codeArray = Array.from(passArray).map(x => unreservedChars.charAt(x/0xFF * charsLength));
return codeArray.join('');
}
get_code_challenge(code_challenge_method)
{
if (code_challenge_method == "S256") //S256
{
const bytes = CryptoJS.SHA256(this.code_verifier);
const base64 = bytes.toString(CryptoJS.enc.Base64);
const base64url = this.constructor.base64_to_base64url(base64);
return base64url;
}
else //plain
{
return this.code_verifier;
}
}
static base64_to_base64url(base64)
{
let base64url = base64
.replace(/\+/g, '-')
.replace(/\//g, '_')
.replace(/=+$/, '');
return base64url;
}
refresh_idp_urls(options = {})
{
const thisAuth = this;
//登录的Auth API
GM_xmlhttpRequest({
url: "https://app-api.pixiv.net/idp-urls",
method: "get",
responseType: "text",
headers: new HeadersObject(),
onload: function(response) {
let jo;
try {
jo = JSON.parse(response.responseText);
} catch (e) {
console.error("获取综合网址集失败,返回可能不是JSON格式。", e, response);
if(options.onload_notJson) options.onload_notJson(response.responseText);
return;
}
if (jo.has_error || jo.errors) {
console.error("获取综合网址集失败,返回错误消息", jo);
if(options.onload_hasError) options.onload_hasError(jo);
return;
} else { //登录成功
Object.assign(thisAuth.idp_urls, jo);
console.info("获取综合网址集成功", jo);
if(options.onload) options.onload(jo);
return;
}
},
onerror: function(response) {
console.error("获取登录重定向网址失败,网络请求发生错误", response);
if(options.onerror) options.onerror(response);
return;
}
});
}
get_login_url()
{
const code_challenge_method = "S256"; //P站不支持plain
const loginURL = new URL("https://app-api.pixiv.net/web/v1/login");
loginURL.searchParams.set("code_challenge", this.get_code_challenge(code_challenge_method));
loginURL.searchParams.set("code_challenge_method", code_challenge_method);
loginURL.searchParams.set("client","pixiv-android");
return loginURL;
}
login(authorization_code, options = {})
{
this.authorization_code = authorization_code;
const thisAuth = this;
const postObj = new URLSearchParams();
postObj.set("code_verifier", thisAuth.code_verifier);
postObj.set("code", authorization_code);
postObj.set("grant_type","authorization_code");
postObj.set("redirect_uri",thisAuth.idp_urls["auth-token-redirect-uri"]);
postObj.set("client_id", client_id);
postObj.set("client_secret", client_secret);
postObj.set("include_policy","true");
this.refresh_idp_urls({
onload: function(){
//登录的Auth API
GM_xmlhttpRequest({
url: thisAuth.idp_urls["auth-token"],
method: "post",
responseType: "text",
headers: new HeadersObject(),
data: postObj.toString(),
onload: function(response) {
let jo;
try {
jo = JSON.parse(response.responseText);
} catch (e) {
console.error("登录失败,返回可能不是JSON格式,或本程序异常。", e, response);
if(options.onload_notJson) options.onload_notJson(response.responseText);
return;
}
if (jo.has_error || jo.errors) {
console.error("登录失败,返回错误消息", jo);
if(options.onload_hasError) options.onload_hasError(jo);
return;
} else { //登录成功
thisAuth.auth_data = jo;
thisAuth.login_time = new Date().getTime();
console.info("登录成功", jo);
if(options.onload) options.onload(jo);
return;
}
},
onerror: function(response) {
console.error("登录失败,网络请求发生错误", response);
if(options.onerror) options.onerror(response);
return;
}
});
}
});
}
refresh_token(options = {})
{
const thisAuth = this;
const postObj = new URLSearchParams();
postObj.set("client_id", client_id);
postObj.set("client_secret", client_secret);
postObj.set("grant_type","refresh_token");
postObj.set("refresh_token",thisAuth.auth_data.refresh_token);
postObj.set("include_policy","true");
//登录的Auth API
GM_xmlhttpRequest({
url: thisAuth.idp_urls["auth-token"],
method: "post",
responseType: "text",
headers: new HeadersObject(),
data: postObj.toString(),
onload: function(response) {
let jo;
try {
jo = JSON.parse(response.responseText);
} catch (e) {
console.error("刷新Token失败,返回可能不是JSON格式,或本程序异常。", e, response);
if(options.onload_notJson) options.onload_notJson(response.responseText);
return;
}
if (jo.has_error || jo.errors) {
console.error("刷新Token失败,返回错误消息", jo);
if(options.onload_hasError) options.onload_hasError(jo);
return;
} else { //登录成功
thisAuth.auth_data = jo;
thisAuth.login_time = new Date().getTime();
console.info("刷新Token成功", jo);
if(options.onload) options.onload(jo);
return;
}
},
onerror: function(response) {
console.error("刷新Token失败,网络请求发生错误", response);
if(options.onerror) options.onerror(response);
return;
}
});
}
save()
{
GM_setValue("pubd-oauth", this);
}
}
//一个掩码
var Mask = function(name, logic, content){
this.name = name;
this.logic = logic;
this.content = content;
};
//一个下载方案
var DownScheme = function(name) {
//默认值
this.name = name ? name : "默认方案";
this.rpcurl = "http://localhost:6800/jsonrpc";
this.proxyurl = "";
this.downloadurl = "%{illust.parsedURL.protocol}//%{illust.parsedURL.host}%{illust.parsedURL.path_before_page}%{page}.%{illust.extention}";
this.downfilter = "";
this.savedir = "D:/PixivDownload/";
this.savepath = "%{illust.user.id}/%{illust.filename}%{page}.%{illust.extention}";
this.textout = "%{illust.url_without_page}%{page}.%{illust.extention}\n";
this.masklist = [];
};
DownScheme.prototype.maskAdd = function(name, logic, content) {
var mask = new Mask(name, logic, content);
this.masklist.push(mask);
return mask;
};
DownScheme.prototype.maskRemove = function(index) {
this.masklist.splice(index, 1);
};
DownScheme.prototype.loadFromJson = function(json) {
if (typeof(json) == "string") {
try {
json = JSON.parse(json);
} catch (e) {
console.error("读取的方案数据是字符串,但非JSON。",e);
return false;
}
}
const _this = this;
Object.keys(_this).forEach(function(key){
if (key=="masklist")
{
_this.masklist.length = 0; //清空之前的
json.masklist.forEach(function(mask){
_this.masklist.push(new Mask(mask.name, mask.logic, mask.content));
});
}else if(json[key] != undefined)
{
_this[key] = json[key];
}
});
return true;
};
//创建菜单类
var pubdMenu = function(classname) {
//生成菜单项
function buildMenuItem(title, classname, callback, submenu) {
var item = document.createElement("li");
if (title == 0) //title为0时,只添加一条菜单分割线
{
item.className = "pubd-menu-line" + (classname ? " " + classname : "");
return item;
}
item.className = "pubd-menu-item" + (classname ? " " + classname : "");
//如果有子菜单则添加子菜单
if (typeof(submenu) == "object") {
item.classList.add("pubd-menu-includesub"); //表明该菜单项有子菜单
submenu.classList.add("pubd-menu-submenu"); //表明该菜单是子菜单
//a.addEventListener("mouseenter",function(){callback.show()});
//a.addEventListener("mouseleave",function(){callback.hide()});
item.appendChild(submenu);
item.subitem = submenu;
}else
{
item.subitem = null; //子菜单默认为空
}
//添加链接
var a = item.appendChild(document.createElement("a"));
a.className = "pubd-menu-item-a";
//添加图标
var icon = a.appendChild(document.createElement("i"));
icon.className = "pubd-icon";
//添加文字
var span = a.appendChild(document.createElement("span"));
span.className = "text";
span.innerHTML = title;
//添加菜单操作
if (typeof(callback) == "string") { //为字符串时,当作链接处理
a.target = "_blank";
a.href = callback;
} else if (typeof(callback) == "function") { //为函数时,当作按钮处理
item.addEventListener("click", callback);
//a.onclick = callback;
}
return item;
}
var menu = document.createElement("ul");
menu.className = "pubd-menu display-none" + (classname ? " " + classname : "");
menu.item = [];
//显示该菜单
menu.show = function() {
menu.classList.remove("display-none");
};
menu.hide = function() {
menu.classList.add("display-none");
};
//添加菜单项
menu.add = function(title, classname, callback, submenu) {
var itm = buildMenuItem(title, classname, callback, submenu);
this.appendChild(itm);
this.item.push(itm);
return itm;
};
//鼠标移出菜单时消失
menu.addEventListener("mouseleave", function(e) {
this.hide();
});
return menu;
};
//创建通用对话框类
var Dialog = function(caption, classname, id) {
//构建标题栏按钮
function buildDlgCptBtn(text, classname, callback) {
if (!callback) classname = "";
const btn = document.createElement("a");
btn.className = "dlg-cpt-btn" + (classname ? " " + classname : "");
if (typeof(callback) == "string") {
btn.target = "_blank";
btn.href = callback;
} else {
if (callback)
btn.addEventListener("click", callback);
}
var btnTxt = btn.appendChild(document.createElement("span"));
btnTxt.className = "dlg-cpt-btn-text";
btnTxt.innerHTML = text;
return btn;
}
var dlg = document.createElement("div");
if (id) dlg.id = id;
dlg.className = "pubd-dialog display-none" + (classname ? " " + classname : "");
//添加图标与标题
var cpt = dlg.appendChild(document.createElement("div"));
cpt.className = "caption";
dlg.icon = cpt.appendChild(document.createElement("i"));
dlg.icon.className = "pubd-icon";
var captionDom = cpt.appendChild(document.createElement("span"));
Object.defineProperty(dlg , "caption", {
get() {
return captionDom.textContent;
},
set(str) {
captionDom.innerHTML = str;
}
});
dlg.caption = caption;
//添加标题栏右上角按钮 captionButtons
var cptBtns = dlg.cptBtns = dlg.appendChild(document.createElement("div"));
cptBtns.className = "dlg-cpt-btns";
//添加按钮的函数
cptBtns.add = function(text, classname, callback) {
var btn = buildDlgCptBtn(text, classname, callback);
this.insertBefore(btn, this.firstChild);
return btn;
};
//添加关闭按钮
cptBtns.close = cptBtns.add("X", "dlg-btn-close", (function() {
dlg.classList.add("display-none");
}));
//添加内容区域
var content = dlg.content = dlg.appendChild(document.createElement("div"));
content.className = "dlg-content";
//窗口激活
dlg.active = function() {
if (!this.classList.contains("pubd-dlg-active")) { //如果没有激活的话才执行
var dlgs = document.querySelectorAll(".pubd-dialog"); //获取网页已经载入的所有的窗口
for (var dlgi = 0; dlgi < dlgs.length; dlgi++) { //循环所有窗口
if (dlgs[dlgi] != this)
{
dlgs[dlgi].classList.remove("pubd-dlg-active"); //取消激活
dlgs[dlgi].style.zIndex = parseInt(window.getComputedStyle(dlgs[dlgi], null).getPropertyValue("z-index")) - 1; //从当前网页最终样式获取该窗体z级,并-1.
}
}
this.classList.add("pubd-dlg-active"); //添加激活
this.style.zIndex = ""; //z级归零
}
};
//窗口初始化
dlg.initialise = function() { //窗口初始化默认情况下什么也不做,具体在每个窗口再设置
return;
};
//窗口显示
dlg.show = function(posX, posY, arg) {
if (posX) dlg.style.left = posX + "px"; //更改显示时初始坐标
if (posY) dlg.style.top = posY + "px";
dlg.initialise(arg); //对窗体进行初始化(激活为可见前提前修改窗体内容)
dlg.classList.remove("display-none");
dlg.active(); //激活窗口
};
//窗口隐藏
dlg.hide = function() { //默认情况下等同于关闭窗口
dlg.cptBtns.close.click();
};
//添加鼠标拖拽移动
var drag = dlg.drag = [0, 0]; //[X,Y] 用以储存窗体开始拖动时的鼠标相对窗口坐标差值。
//startDrag(cpt, dlg);
cpt.addEventListener("mousedown", function(e) { //按下鼠标则添加移动事件
var eX = e.pageX>0?e.pageX:0, eY = e.pageY>0?e.pageY:0; //不允许鼠标坐标向上、左超出网页。
drag[0] = eX - dlg.offsetLeft;
drag[1] = eY - dlg.offsetTop;
var handler_mousemove = function(e) { //移动鼠标则修改窗体坐标
var eX = e.pageX>0?e.pageX:0, eY = e.pageY>0?e.pageY:0; //不允许鼠标坐标向上、左超出网页。
dlg.style.left = (eX - drag[0]) + 'px';
dlg.style.top = (eY - drag[1]) + 'px';
};
var handler_mouseup = function(e) { //抬起鼠标则取消移动事件
document.removeEventListener("mousemove", handler_mousemove);
};
document.addEventListener("mousemove", handler_mousemove);
document.addEventListener("mouseup", handler_mouseup, { once: true });
});
//点击窗口任何区域激活窗口
dlg.addEventListener("mousedown", function(e) {
dlg.active();
});
return dlg;
};
//创建框架类
var Frame = function(title, classname) {
var frame = document.createElement("fieldset");
frame.className = "pubd-frame" + (classname ? " " + classname : "");
var caption = frame.appendChild(document.createElement("legend"));
caption.className = "pubd-frame-caption";
caption.textContent = title;
var content = frame.content = frame.appendChild(document.createElement("div"));
content.className = "pubd-frame-content";
frame.rename = function(newName) {
if (typeof(newName) == "string" && newName.length > 0) {
this.querySelector("legend").textContent = newName;
return true;
} else
return false;
};
return frame;
};
//创建带Label的Input类
const LabelInput = function(text, classname, name, type, value, beforeText, title) {
var label = document.createElement("label");
if (text) label.appendChild(document.createTextNode(text));
label.className = classname;
if (title) label.title = title;
var ipt = label.input = document.createElement("input");
ipt.name = name;
ipt.id = ipt.name;
ipt.type = type;
ipt.value = value;
if (beforeText && label.childNodes.length>0)
label.insertBefore(ipt, label.firstChild);
else
label.appendChild(ipt);
return label;
};
//构建错误信息显示模块
const ErrorMsg = function() {
const error_msg_list = document.createElement("ul");
error_msg_list.className = "error-msg-list";
//添加错误显示功能
error_msg_list.clear = function() {
this.innerHTML = ""; //清空当前信息
};
error_msg_list.add = function(arg) {
function addLine(str)
{
const li = document.createElement("li");
li.className = "error-msg-list-item";
li.appendChild(document.createTextNode(str));
return li;
}
const fragment = document.createDocumentFragment();
if (Array.isArray(arg)) //数组
{
arg.forEach(str=>fragment.appendChild(addLine(str)));
}
else //单文本
{
fragment.appendChild(addLine(arg));
}
this.appendChild(fragment);
};
error_msg_list.replace = function(arg) {
this.clear();
this.add(arg);
};
return error_msg_list;
}
//创建进度条类
const Progress = function(classname, align_right) {
const progress = document.createElement("div");
progress.className = "pubd-progress" + (classname ? " " + classname : "");
if (align_right) progress.classList.add("pubd-progress-right");
progress.scaleNum = 0;
const bar = progress.appendChild(document.createElement("div"));
bar.className = "pubd-progress-bar";
const txt = progress.appendChild(document.createElement("span"));
txt.className = "pubd-progress-text";
progress.set = function(scale, pos = 2, str = null) {
const percentStr = (scale * 100).toFixed(pos) + "%"; //百分比的数字
this.scaleNum = Math.min(Math.max(scale, 0), 1);
bar.style.width = percentStr;
txt.textContent = str || percentStr;
};
Object.defineProperty(progress , "scale", {
get() {
return this.scaleNum;
},
set(num) {
this.set(num);
}
});
return progress;
};
//创建 卡片类
function InfoCard(datas) {
var cardDiv = this.dom = document.createElement("div");
cardDiv.className = "pubd-infoCard";
var thumbnailDiv = cardDiv.appendChild(document.createElement("div"));
thumbnailDiv.className = "pubd-infoCard-thumbnail";
var thumbnailImgDom = thumbnailDiv.appendChild(document.createElement("img"));
var infosDlDom = cardDiv.appendChild(document.createElement("dl"));
infosDlDom.className = "pubd-infoCard-dl";
Object.defineProperty(this , "thumbnail", {
get() {
return thumbnailImgDom.src;
},
set(url) {
thumbnailImgDom.src = url;
}
});
var infoObj;
this.reload = function() //重构Card文本区域
{
infosDlDom.classList.add('display-none');
for (let ci = infosDlDom.childNodes.length-1;ci >= 0;ci--)
{ //删掉所有老子元素
var x = infosDlDom.childNodes[ci];
x.remove();
x = null;
}
const fragment = document.createDocumentFragment();
Object.entries(infoObj).forEach(entry=>{
const dt = fragment.appendChild(document.createElement("dt"));
const dd = fragment.appendChild(document.createElement("dd"));
dt.appendChild(document.createTextNode(entry[0]));
if (entry[1]) dd.appendChild(document.createTextNode(entry[1]));
});
infosDlDom.appendChild(fragment);
infosDlDom.classList.remove('display-none');
};
Object.defineProperty(this , "infos", {
get() {
return infoObj;
},
set(obj) {
infoObj = obj;
this.reload();
}
});
this.infos = datas || {}; //使用传入data进行初始设定
}
//创建下拉框类
var Select = function(classname, name) {
var select = document.createElement("select");
select.className = "pubd-select" + (classname ? " " + classname : "");
select.name = name;
select.id = select.name;
select.add = function(text, value) {
var opt = new Option(text, value);
this.options.add(opt);
};
select.remove = function(index) {
var x = this.options.remove(index);
x = null;
};
return select;
};
//创建Aria2类
var Aria2 = (function() {
var jsonrpc_version = '2.0';
function get_auth(url) {
return url.match(/^(?:(?![^:@]+:[^:@\/]*@)[^:\/?#.]+:)?(?:\/\/)?(?:([^:@]*(?::[^:@]*)?)?@)?/)[1];
}
function request(jsonrpc_path, method, params, callback, priority) {
if (callback == undefined) callback = ()=>{};
var auth = get_auth(jsonrpc_path);
jsonrpc_path = jsonrpc_path.replace(/^((?![^:@]+:[^:@\/]*@)[^:\/?#.]+:)?(\/\/)?(?:(?:[^:@]*(?::[^:@]*)?)?@)?(.*)/, '$1$2$3'); // auth string not allowed in url for firefox
var request_obj = {
jsonrpc: jsonrpc_version,
method: method,
id: priority ? 1 : Date.now(),
};
if (params) request_obj.params = params;
if (auth && auth.indexOf('token:') == 0)
{
if (method == "system.multicall")
{ //多项目操作时单独设置token
params.forEach(function(param){
param.forEach(function(method){
method.params.unshift(auth);
});
});
}else
{
params.unshift(auth);
}
}
var headers = { "Content-Type": ContentType };
if (auth && auth.indexOf('token:') != 0) {
headers.Authorization = "Basic " + btoa(auth);
}
GM_xmlhttpRequest({
url: jsonrpc_path + "?tm=" + (new Date()).getTime().toString(),
method: "POST",
responseType: "text",
data: JSON.stringify(request_obj),
headers: headers,
onload: function(response) {
try {
var JSONreq = JSON.parse(response.response);
callback(JSONreq);
} catch (e) {
console.error("Aria2发送信息错误", e, response);
callback(false);
}
},
onerror: function(response) {
console.error(response);
callback(false);
}
});
}
return function(jsonrpc_path) {
const _this = this;
_this.jsonrpc_path = jsonrpc_path;
_this.addUri = function(uri, options, callback) {
request(_this.jsonrpc_path, 'aria2.addUri', [
[uri, ], options
], callback);
};
_this.addTorrent = function(base64txt, options, callback) {
request(_this.jsonrpc_path, 'aria2.addTorrent', [base64txt, [], options], callback);
};
_this.getVersion = function(callback) {
request(_this.jsonrpc_path, 'aria2.getVersion', [], callback, true);
};
_this.getGlobalOption = function(callback) {
request(_this.jsonrpc_path, 'aria2.getGlobalOption', [], callback, true);
};
_this.system = {
multicall:function(params,callback){
request(_this.jsonrpc_path, 'system.multicall', params, callback);
},
};
return this;
};
})();
/*
* 自定义函数区
*/
//仿GM_notification函数v1.2,发送网页通知。
//此函数非Debug用,为了替换选项较少但是兼容其格式的GM_notification插件
function GM_notification(text, title, image, onclick) {
const options = {};
let rTitle, rText;
let ondone, onclose;
const dataMode = Boolean(typeof(text) == "string"); //GM_notification有两种模式,普通4参数模式和option对象模式
if (dataMode)
{ //普通模式
rTitle = title;
rText = text;
options.body = text;
options.icon = image;
}else
{ //选项模式
const details = text;
rTitle = details.title;
rText = details.text;
if (details.text) options.body = details.text;
if (details.image) options.icon = details.image;
if (details.timeout) options.timestamp = details.timeout;
ondone = title;
onclose = image;
//if (details.highlight) options.highlight = details.highlight; //没找到这个功能
}
function sendNotification(general){
const n = new Notification(rTitle, options);
if (general)
{ //普通模式
if (onclick) n.onclick = onclick;
}else
{ //选项模式,这里和TamperMonkey API不一样,区分了关闭和点击。
if (ondone) n.onclick = ondone;
if (onclose) n.onclose = onclose;
}
}
// 先检查浏览器是否支持
if (!("Notification" in window)) {
alert(rTitle + "\r\n" + rText);
// 检查用户是否同意接受通知
} else if (Notification.permission === "granted") {
Notification.requestPermission(function(permission) {
sendNotification(dataMode);
});
}
// 否则我们需要向用户获取权限
else if (Notification.permission !== 'denied') {
Notification.requestPermission(function(permission) {
// 如果用户同意,就可以向他们发送通知
if (permission === "granted") {
sendNotification(dataMode);
}
});
}
}
//有默认值的获取设置
function getValueDefault(name, defaultValue) {
var value = GM_getValue(name);
if (value != null)
return value;
else
return defaultValue;
}
//加入了Auth的网络请求函数
function xhrGenneral(url, onload_suceess_Cb, onload_hasError_Cb, onload_notJson_Cb, onerror_Cb, dlog=str=>str) {
const headersObj = new HeadersObject();
const auth = pubd.oAuth.auth_data;
if (auth) {
headersObj.Authorization = auth.token_type[0].toUpperCase() + auth.token_type.substring(1) + " " + auth.access_token;
} else {
console.info(dlog("未登录账户,尝试以非登录状态获取信息"));
}
GM_xmlhttpRequest({
url: url,
method: "get",
responseType: "text",
headers: headersObj,
onload: function(response) {
let jo;
try {
jo = JSON.parse(response.responseText);
} catch (e) {
console.error(dlog("错误:返回可能不是JSON格式,或本程序异常"), e, response);
onload_notJson_Cb(response);
return;
}
if (jo)
{
if (mdev) console.log("请求URL %s,结果 %o",url,JSON.parse(response.responseText));
//jo.error.message 是JSON字符串的错误信息,Token错误的时候返回的又是普通字符串
//jo.error.user_message 是单行文本的错误信息
if (jo.error) {
if (jo.error.message.includes("Error occurred at the OAuth process.")) {
if (auth) {
console.warn(dlog("授权 Token 过期,开始自动更新。"),jo);
//自动重新登录
pubd.dialog.refresh_token.show(
(document.body.clientWidth - 370)/2,
window.scrollY+300,
{
onload: ()=>{
pubd.dialog.refresh_token.hide();
xhrGenneral(url, onload_suceess_Cb, onload_hasError_Cb, onload_notJson_Cb, onerror_Cb);
},
onload_hasError: onload_hasError_Cb,
onload_notJson: onload_notJson_Cb,
onerror: onerror_Cb,
}
);
} else {
console.info(dlog("非登录模式尝试获取信息失败"),jo);
onload_hasError_Cb(jo);
}
return;
}else if (jo.error.message.includes("Rate Limit")) {
console.warn(dlog("获取信息速度太快,触发P站速度限制,1分钟后自动重试。"),jo);
setTimeout(()=>{
xhrGenneral(url, onload_suceess_Cb, onload_hasError_Cb, onload_notJson_Cb, onerror_Cb);
}, 1000 * 60)
return;
}else
{
onload_hasError_Cb(jo);
return;
}
} else { //登录成功
//console.info("JSON返回成功",jo);
onload_suceess_Cb(jo);
return;
}
}
},
onerror: function(response) {
console.error(dlog("错误:网络请求发送失败"), response);
onerror_Cb(response);
}
});
}
//用id来获取动画帧数据
function getUgoiraMeta(iid, onload_suceess_Cb, onload_hasError_Cb, onload_notJson_Cb, onerror_Cb)
{
xhrGenneral(
"https://app-api.pixiv.net/v1/ugoira/metadata?illust_id=" + iid,
onload_suceess_Cb,
onload_hasError_Cb,
onload_notJson_Cb,
onerror_Cb
);
}
//为了区分设置窗口和保存的设置,产生一个新的下载方案数组
function NewDownSchemeArrayFromJson(jsonarr) {
if (typeof(jsonarr) == "string") {
try {
jsonarr = JSON.parse(jsonarr);
} catch (e) {
console.error("PUBD:拷贝新下载方案数组时失败(是字符串,但不是JSON)", e);
return false;
}
}
let sarr = [];
if (Array.isArray(jsonarr)) {
sarr = jsonarr.map(json=>{
let scheme = new DownScheme();
scheme.loadFromJson(json);
return scheme;
});
}
return sarr;
}
//获取URL参数
function getQueryString(name, url = document.location) {
const urlObj = new URL(url);
return urlObj.searchParams.get(name);
}
//从图片URL获取图片属性
function parseIllustUrl(url) {
let src
try {
src = new URL(url);
} catch (error) {
return null;
}
const obj = {
domain: src.host, //为了兼容老的
parsedURL: { //目前用到的不多,只保留这两个值
host: src.host, //域(即主机名)后跟端口
protocol: src.protocol, //URL 协议名
}
};
const parsedURL = obj.parsedURL;
let regRes = new RegExp(illustPathRegExp.source, illustPathRegExp.flags).exec(src.pathname);
if (regRes)
{
//为了兼容老的
obj.url_without_page = `${src.origin}${regRes[1]}`;
obj.filename = regRes[2];
//id直接在原始数据有
obj.token = regRes[4];
obj.extention = regRes[5];
parsedURL.path_before_page = regRes[1];
parsedURL.filename = regRes[2];
parsedURL.id = regRes[3];
parsedURL.token = regRes[4];
parsedURL.extention = regRes[5];
}else if (regRes = new RegExp(limitingPathRegExp.source, limitingPathRegExp.flags).exec(src.pathname)) //上锁图片
{
//为了兼容老的
obj.url_without_page = `${src.origin}${regRes[1]}`;
obj.filename = regRes[2];
//id直接在原始数据有
obj.extention = regRes[3];
parsedURL.path_before_page = regRes[1];
parsedURL.limited = true;
parsedURL.filename = regRes[2];
parsedURL.extention = regRes[3];
}else
{
parsedURL.unknown = true;
}
return obj;
}
//从一个作品数据得到原始图片的下载地址
function getIllustDownUrl(scheme, userInfo, illust, page)
{
return showMask(scheme.downloadurl, scheme.masklist, userInfo, illust, page);
//return `${illust.parsedURL.protocol}//${illust.parsedURL.host}${illust.parsedURL.path_before_page}${page}.${illust.extention}`;
}
//获取当前用户ID
function getCurrentUserId()
{
//从URL获取作者ID
function getUserIdFromUrl(url) {
let userid = parseInt(getQueryString("id",url),10); //老地址:https://www.pixiv.net/member_illust.php?id=3896348
if (!userid)
{
const regSrc = new RegExp("users/(\\d+)", "ig"); //新地址:https://www.pixiv.net/users/3896348
const regRes = regSrc.exec(url.pathname);
if (regRes) {
return parseInt(regRes[1],10);
}
}
return userid;
}
let userid = getUserIdFromUrl(document.location);
if(!userid)
{
userid = thisPageUserid;
if (mainDiv)
{
const userMainPageLink = mainDiv.querySelector(userMainPageCssPath); //作者主页的“主页”按钮
//var artWorkLink = mainDiv.querySelector(artWorkStarCssPath);
const userHeadLink = mainDiv.querySelector(artWorkUserHeadCssPath);
if (userMainPageLink) //如果是作者页面
{
userid = getUserIdFromUrl(userMainPageLink);
}
if (userHeadLink) //如果是作品页面
{
userid = getUserIdFromUrl(userHeadLink);
}
if(pubd.touch)
{
const touch_userHeadLink = mainDiv.querySelector('.user-details-card .user-details-icon'); //如果是作品页面
if (touch_userHeadLink) //如果是作品页面
{
userid = getUserIdFromUrl(touch_userHeadLink);
}
}
}
}
return userid;
}
//检查并快速添加画师收藏的函数
function toggleStar(userid)
{
userid = userid || getCurrentUserId();
const res = pubd.fastStarList.toggle(userid);
if (res)
{ //添加
pubd.start.star.classList.add("stars");
}else
{ //删除
pubd.start.star.classList.remove("stars");
}
GM_setValue("pubd-faststar-list",pubd.fastStarList.exportArray());
}
//检查是否有画师并改变星星状态
function checkStar()
{
const userid = getCurrentUserId();
const res = pubd.fastStarList.has(userid);
if (res)
{ //存在,则标记
pubd.start.star.classList.add("stars");
return true;
}else
{ //不存在,则去掉标记
pubd.start.star.classList.remove("stars");
return false;
}
}
//更改推荐列表里的收藏显示状态
function refreshRecommendListState() {
if (!recommendList) return;
const liNodes = recommendList.getElementsByTagName("li");
for (const liNode of liNodes) {
const imgLink = liNode.querySelector(":scope a[data-gtm-user-id]");
const uid = parseInt(imgLink.dataset.gtmUserId, 10); //得到这个作品的作者ID
liNode.classList.toggle("pubd-stared", pubd.fastStarList.has(uid));
}
}
//构建开始按钮
function buildbtnStart() {
const btnStart = document.createElement("div");
btnStart.id = "pubd-start";
btnStart.className = "pubd-start";
//添加图标
const star = btnStart.star = btnStart.appendChild(document.createElement("i"));
star.className = "pubd-icon star";
star.title = "快速收藏当前画师(开发中功能,目前没用)";
//添加文字
const caption = btnStart.caption = btnStart.appendChild(document.createElement("div"));
caption.className = "text";
caption.innerHTML = "使用PUBD扒图";
caption.title = "快速下载当前画师";
//添加文字
const menu = btnStart.menu = btnStart.appendChild(document.createElement("i"));
menu.className = "pubd-icon menu";
menu.title = "PUBD菜单";
//鼠标移入和按下都起作用
//btnStart.addEventListener("mouseenter",function(){pubd.menu.show()});
star.onclick = function(){toggleStar();};
menu.onclick = function(){pubd.menu.classList.toggle("display-none");};
caption.onclick = function(){pubd.menu.downthis.click();};
return btnStart;
}
//构建开始菜单
function buildbtnMenu() {
/*
var menu2 = new pubdMenu();
menu2.add("子菜单1","",function(){alert("子菜单1")});
menu2.add("子菜单2","",function(){alert("子菜单2")});
var menu1 = new pubdMenu();
menu1.add("子菜单1","",function(){alert("子菜单1")});
menu1.add("子菜单2","",null,menu2);
var menu3 = new pubdMenu();
menu3.add("子菜单1","",function(){alert("子菜单1")});
menu3.add("子菜单2","",function(){alert("子菜单2")});
menu3.add("子菜单2","",function(){alert("子菜单3")});
menu3.add("子菜单2","",function(){alert("子菜单4")});
var menu4 = new pubdMenu();
menu4.add("子菜单1","",null,menu3);
menu4.add("子菜单2","",function(){alert("子菜单2")});
menu4.add("子菜单2","",function(){alert("子菜单5")});
menu4.add("子菜单2","",function(){alert("子菜单6")});
*/
var menu = new pubdMenu("pubd-menu-main");
menu.id = "pubd-menu";
menu.downillust = menu.add("下载当前作品", "pubd-menu-this-illust", function(e) {
pubd.dialog.downillust.show(
(document.body.clientWidth - 500)/2,
window.scrollY+150,
{id:getQueryString('illust_id',
pubd.touch ?
mainDiv.querySelector('.illust-details-content .work-stats>a') : //手机版
mainDiv.querySelector(artWorkStarCssPath) //新版Vue结构
)}
);
menu.hide();
});
menu.downthis = menu.add("下载该画师所有作品", "pubd-menu-this-user", function(e) {
pubd.dialog.downthis.show(
(document.body.clientWidth - 440)/2,
window.scrollY+100,
{id:getCurrentUserId()}
);
menu.hide();
});
/*
menu.add("占位用","",null,menu1);
menu.add("没功能","",null,menu4);
menu.add("多个画师下载",null,function()
{//做成“声音”的设备样子
alert("这个功能也没有开发")
}
);
*/
menu.add(0);
if (mdev) menu.downmult = menu.add("多画师下载", "pubd-menu-multiple", function(e) {
pubd.dialog.multiple.show(
(document.body.clientWidth - 440)/2,
window.scrollY+100
);
menu.hide();
});
menu.add("选项", "pubd-menu-setting", function(e) {
pubd.dialog.config.show(
(document.body.clientWidth - 400)/2,
window.scrollY+50
);
menu.hide();
});
return menu;
}
//构建Token剩余时间进度条
function buildProgressToken()
{
const progress = new Progress("pubd-token-expires", true);
progress.animateHook = null; //储存Token进度条动画句柄
progress.token_animate = function(){
const _this = progress;
if (!pubd.oAuth.auth_data)
{
_this.set(0, 2, "尚未登录");
clearInterval(_this.animateHook);
return;
}
const nowdate = new Date();
const olddate = new Date(pubd.oAuth.login_time);
const expires_in = parseInt(pubd.oAuth.auth_data.expires_in);
const differ = expires_in - (nowdate - olddate) / 1000;
if (differ > 0) {
const scale = differ / expires_in;
_this.set(scale, 2, "Token有效剩余 " + parseInt(differ) + " 秒");
} else {
_this.set(0, 2, "Token已失效,请刷新");
clearInterval(_this.animateHook);
}
//console.log("Token有效剩余" + differ + "秒"); //检测动画后台是否停止
}
//开始动画
progress.start_token_animate = function(){
const _this = progress;
_this.stop_token_animate();
requestAnimationFrame(_this.token_animate);
_this.animateHook = setInterval(()=>requestAnimationFrame(_this.token_animate), 1000);
};
//停止动画
progress.stop_token_animate = function(){
const _this = progress;
clearInterval(_this.animateHook);
};
return progress;
}
//构建设置对话框
function buildDlgConfig() {
const dlg = new Dialog("PUBD选项 v" + scriptVersion, "pubd-config", "pubd-config");
dlg.cptBtns.add("反馈", "dlg-btn-debug", "https://github.com/Mapaler/PixivUserBatchDownload/issues");
dlg.cptBtns.add("?", "dlg-btn-help", "https://github.com/Mapaler/PixivUserBatchDownload/wiki");
dlg.token_ani = null; //储存Token进度条动画句柄
var dl = dlg.content.appendChild(document.createElement("dl"));
var dt = dl.appendChild(document.createElement("dt"));
var dd = dl.appendChild(document.createElement("dd"));
dlg.frmLogin = dd.appendChild(new Frame("Pixiv访问权限", "pubd-token"));
var dl_t = dlg.frmLogin.content.appendChild(document.createElement("dl"));
var dd_t = dl_t.appendChild(document.createElement("dd"));
var ul_t = dd_t.appendChild(document.createElement("ul"));
ul_t.className = "horizontal-list";
var li_t = ul_t.appendChild(document.createElement("li"));
const userAvatar = li_t.appendChild(document.createElement("div"));
userAvatar.className = "user-avatar";
userAvatar.img = userAvatar.appendChild(document.createElement("img"));
userAvatar.img.className = "avatar-img";
var li_t = ul_t.appendChild(document.createElement("li"));
const userName = li_t.appendChild(document.createElement("div"));
userName.className = "user-name";
const userAccount = li_t.appendChild(document.createElement("div"));
userAccount.className = "user-account";
var li_t = ul_t.appendChild(document.createElement("li"));
//登录/退出
const btnLogin = li_t.appendChild(document.createElement("button"));
btnLogin.className = "pubd-tologin";
btnLogin.onclick = function(){
if (dlg.frmLogin.classList.contains("logged-in"))
{
//退出
pubd.oAuth = new oAuth2();
pubd.oAuth.save();
dlg.refreshLoginState();
}else
{
//登录
pubd.dialog.login.show(
(document.body.clientWidth - 370)/2,
window.scrollY+200
);
}
}
const tokenInfo = dlg.tokenInfo = dl_t.appendChild(document.createElement("dd"));
tokenInfo.className = "pubd-token-info";
const progress = dlg.tokenExpires = tokenInfo.appendChild(buildProgressToken());
const btnRefresh = tokenInfo.appendChild(document.createElement("button"));
btnRefresh.className = "pubd-open-refresh-token";
btnRefresh.appendChild(document.createTextNode("刷新许可"));
btnRefresh.onclick = function() {
//刷新许可
pubd.dialog.refresh_token.show(
(document.body.clientWidth - 370)/2,
window.scrollY+300
);
};
dlg.refreshLoginState = function() {
if (!pubd.oAuth) return;
const auth_data = pubd.oAuth.auth_data;
if (auth_data)
{
userAvatar.img.src = auth_data.user.profile_image_urls.px_50x50;
userAvatar.img.alt = userAvatar.title = auth_data.user.name;
userName.textContent = auth_data.user.name;
userAccount.textContent = auth_data.user.account;
btnLogin.textContent = "退出";
progress.start_token_animate();
btnRefresh.disabled = false;
dlg.frmLogin.classList.add("logged-in");
}else
{
userAvatar.img.src = "";
userAvatar.img.alt = userAvatar.title = "";
userName.textContent = "未登录";
userAccount.textContent = "Not logged in";
btnLogin.textContent = "登录";
progress.token_animate();
progress.stop_token_animate();
btnRefresh.disabled = true;
dlg.frmLogin.classList.remove("logged-in");
}
}
//“通用分析选项”窗口选项
var dt = document.createElement("dt");
dl.appendChild(dt);
var dd = document.createElement("dd");
var frm = new Frame("通用分析选项", "pubd-commonanalyseoptions");
var chk_getugoiraframe = new LabelInput("获取动图帧数", "pubd-getugoiraframe", "pubd-getugoiraframe", "checkbox", "1", true);
dlg.getugoiraframe = chk_getugoiraframe.input;
frm.content.appendChild(chk_getugoiraframe);
dd.appendChild(frm);
dl.appendChild(dd);
//“下载该画师”窗口选项
var dt = document.createElement("dt");
dl.appendChild(dt);
var dd = document.createElement("dd");
var frm = new Frame("下载窗口", "pubd-frm-downthis");
var chk_autoanalyse = new LabelInput("打开窗口自动获取数据", "pubd-autoanalyse", "pubd-autoanalyse", "checkbox", "1", true);
dlg.autoanalyse = chk_autoanalyse.input;
var chk_autodownload = new LabelInput("获取完成自动发送下载", "pubd-autodownload", "pubd-autodownload", "checkbox", "1", true);
dlg.autodownload = chk_autodownload.input;
frm.content.appendChild(chk_autoanalyse);
frm.content.appendChild(chk_autodownload);
dd.appendChild(frm);
dl.appendChild(dd);
//向Aria2的发送模式
var dt = dl.appendChild(document.createElement("dt"));
var dd = dl.appendChild(document.createElement("dd"));
var frm = dd.appendChild(new Frame("向Aria2逐项发送模式", "pubd-frm-termwisetype"));
var radio0 = frm.content.appendChild(new LabelInput("完全逐项(按图片)", "pubd-termwisetype", "pubd-termwisetype", "radio", "0", true));
var radio1 = frm.content.appendChild(new LabelInput("半逐项(按作品)", "pubd-termwisetype", "pubd-termwisetype", "radio", "1", true));
var radio2 = frm.content.appendChild(new LabelInput("不逐项(按作者)", "pubd-termwisetype", "pubd-termwisetype", "radio", "2", true));
dlg.termwiseType = [radio0.input, radio1.input, radio2.input];
//“发送完成后,点击通知”窗口选项
var dt = dl.appendChild(document.createElement("dt"));
var dd = dl.appendChild(document.createElement("dd"));
var frm = dd.appendChild(new Frame("发送完成通知", "pubd-frm-clicknotification"));
var radio0 = frm.content.appendChild(new LabelInput("点击通知什么也不做", "pubd-clicknotification", "pubd-clicknotification", "radio", "0", true));
var radio1 = frm.content.appendChild(new LabelInput("点击通知激活该窗口", "pubd-clicknotification", "pubd-clicknotification", "radio", "1", true));
var radio2 = frm.content.appendChild(new LabelInput("点击通知关闭该窗口", "pubd-clicknotification", "pubd-clicknotification", "radio", "2", true));
var radio3 = frm.content.appendChild(new LabelInput("通知自动消失关闭该窗口", "pubd-clicknotification", "pubd-clicknotification", "radio", "3", true));
dlg.noticeType = [radio0.input, radio1.input, radio2.input, radio3.input];
//配置方案储存
dlg.schemes = null;
dlg.reloadSchemes = function() { //重新读取所有下载方案
if (dlg.schemes.length < 1) {
alert("目前本程序没有任何下载方案,需要正常使用请先新建方案。");
}
dlg.downSchemeDom.options.length = 0;
dlg.schemes.forEach(function(item, index) {
dlg.downSchemeDom.add(item.name, index);
});
if (dlg.downSchemeDom.options.length > 0)
dlg.selectScheme(0);
};
dlg.loadScheme = function(scheme) { //读取一个下载方案
if (scheme == undefined) {
dlg.rpcurl.value = "";
dlg.proxyurl.value = "";
dlg.downloadurl.value = "";
dlg.downfilter.value = "";
dlg.savedir.value = "";
dlg.savepath.value = "";
dlg.textout.value = "";
dlg.loadMasklistFromArray([]);
} else {
dlg.rpcurl.value = scheme.rpcurl;
dlg.proxyurl.value = scheme.proxyurl;
dlg.downloadurl.value = scheme.downloadurl;
dlg.downfilter.value = scheme.downfilter;
dlg.savedir.value = scheme.savedir;
dlg.savepath.value = scheme.savepath;
dlg.textout.value = scheme.textout;
dlg.loadMasklistFromArray(scheme.masklist);
}
};
dlg.addMask = function(name, logic, content, value) { //向掩码列表添加一个新的掩码
if (value == undefined)
value = dlg.masklist.options.length;
var text = name + " : " + logic + " : " + content;
var opt = new Option(text, value);
dlg.masklist.options.add(opt);
};
dlg.loadMask = function(mask) { //读取一个掩码到三个文本框,只是用来查看
dlg.mask_name.value = mask.name;
dlg.mask_logic.value = mask.logic;
dlg.mask_content.value = mask.content;
};
dlg.loadMasklistFromArray = function(masklist) { //从掩码数组重置掩码列表
dlg.masklist.length = 0;
masklist.forEach(function(item, index) {
dlg.addMask(item.name, item.logic, item.content, index);
});
};
//选择一个方案,同时读取设置
dlg.selectScheme = function(index) {
if (index == undefined) index = 0;
if (dlg.downSchemeDom.options.length < 1 || dlg.downSchemeDom.selectedOptions.length < 1) { return; }
var scheme = dlg.schemes[index];
dlg.loadScheme(scheme);
dlg.downSchemeDom.selectedIndex = index;
};
//选择一个掩码,同时读取设置
dlg.selectMask = function(index) {
if (dlg.downSchemeDom.options.length < 1 || dlg.downSchemeDom.selectedOptions.length < 1) { return; }
if (dlg.masklist.options.length < 1 || dlg.masklist.selectedOptions.length < 1) { return; }
var scheme = dlg.schemes[dlg.downSchemeDom.selectedIndex];
var mask = scheme.masklist[index];
dlg.loadMask(mask);
dlg.masklist.selectedIndex = index;
};
//配置方案选择
var dt = dl.appendChild(document.createElement("dt"));
dt.textContent = "默认下载方案";
var dd = dl.appendChild(document.createElement("dd"));
var slt = dlg.downSchemeDom = dd.appendChild(new Select("pubd-downscheme"));
slt.onchange = function() {
dlg.selectScheme(this.selectedIndex);
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-downscheme-new";
ipt.value = "新建";
ipt.onclick = function() {
var schemName = prompt("请输入方案名", "我的方案");
if (schemName)
{
var scheme = new DownScheme(schemName);
var length = dlg.schemes.push(scheme);
dlg.downSchemeDom.add(scheme.name, length - 1);
dlg.downSchemeDom.selectedIndex = length - 1;
dlg.loadScheme(scheme);
//dlg.reloadSchemes();
}
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-downscheme-remove";
ipt.value = "删除";
ipt.onclick = function() {
if (dlg.downSchemeDom.options.length < 1) { alert("已经没有方案了"); return; }
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中方案"); return; }
var index = dlg.downSchemeDom.selectedIndex;
var c = confirm("你确定要删除“" + dlg.schemes[index].name + "”方案吗?");
if (c)
{
var x = dlg.schemes.splice(index, 1);
x = null;
dlg.downSchemeDom.remove(index);
var index = dlg.downSchemeDom.selectedIndex;
if (index < 0) dlg.reloadSchemes(); //没有选中的,重置
else dlg.loadScheme(dlg.schemes[index]);
}
};
//配置方案详情设置
var dt = dl.appendChild(document.createElement("dt"));
var dd = dl.appendChild(document.createElement("dd"));
dd.className = "pubd-selectscheme-bar";
var frm = dd.appendChild(new Frame("当前方案设置", "pubd-selectscheme"));
var dl_ss = frm.content.appendChild(document.createElement("dl"));
//Aria2 URL
var dt = dl_ss.appendChild(document.createElement("dt"));
dt.textContent = "Aria2 JSON-RPC 路径";
var rpcchk = dlg.rpcchk = dt.appendChild(document.createElement("span")); //显示检查状态用
rpcchk.className = "pubd-rpcchk-info";
rpcchk.runing = false;
var dd = dl_ss.appendChild(document.createElement("dd"));
var rpcurl = dlg.rpcurl = dd.appendChild(document.createElement("input"));
rpcurl.type = "url";
rpcurl.className = "pubd-rpcurl";
rpcurl.name = "pubd-rpcurl";
rpcurl.id = rpcurl.name;
rpcurl.placeholder = "Aria2的信息接收路径";
rpcurl.onchange = function() {
dlg.rpcchk.innerHTML = "";
dlg.rpcchk.runing = false;
if (dlg.downSchemeDom.selectedOptions.length < 1) { return; }
var schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].rpcurl = rpcurl.value;
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-rpcchk";
ipt.value = "检查路径";
ipt.onclick = function() {
if (rpcchk.runing) return;
if (rpcurl.value.length < 1) {
rpcchk.textContent = "路径为空";
return;
}
rpcchk.textContent = "正在连接...";
rpcchk.runing = true;
var aria2 = new Aria2(rpcurl.value);
aria2.getVersion(function(rejo) {
if (rejo)
rpcchk.textContent = "发现Aria2 ver" + rejo.result.version;
else
rpcchk.textContent = "Aria2连接失败";
rpcchk.runing = false;
});
};
var dt = dl_ss.appendChild(document.createElement("dt"));
dt.textContent = "Aria2 代理服务器地址";
var dta = dt.appendChild(document.createElement("a"));
dta.className = "pubd-help-link";
dta.textContent = "(?)";
dta.href = "https://github.com/Mapaler/PixivUserBatchDownload/wiki/Aria2%e9%80%9a%e8%bf%87%e4%bb%a3%e7%90%86%e4%b8%8b%e8%bd%bd";
dta.target = "_blank";
var dd = dl_ss.appendChild(document.createElement("dd"));
var proxyurl = dlg.proxyurl = dd.appendChild(document.createElement("input"));
proxyurl.type = "text";
proxyurl.className = "pubd-proxyurl";
proxyurl.name = "pubd-proxyurl";
proxyurl.id = proxyurl.name;
proxyurl.placeholder = "[http://][USER:PASSWORD@]HOST[:PORT]";
proxyurl.onchange = function() {
if (dlg.downSchemeDom.selectedOptions.length < 1) { return; }
const schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].proxyurl = this.value;
};
var dt = dl_ss.appendChild(document.createElement("dt"));
dt.textContent = "作品下载地址";
var dd = dl_ss.appendChild(document.createElement("dd"));
var downloadurl = dlg.downloadurl = dd.appendChild(document.createElement("input"));
downloadurl.type = "text";
downloadurl.className = "pubd-downloadurl";
downloadurl.name = "pubd-downloadurl";
downloadurl.readOnly = true;
downloadurl.id = downloadurl.name;
downloadurl.onclick = function() {
if (this.readOnly)
{
if (confirm("警告!\n修改下载地址可能导致无法下载图片,您确定要修改吗?\n\n若确需修改,建议先在文本输出模式测试。"))
{
this.readOnly = false;
}
}
};
downloadurl.onchange = function() {
if (dlg.downSchemeDom.selectedOptions.length < 1) { return; }
const schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].downloadurl = this.value;
};
//下载过滤
var dt = dl_ss.appendChild(document.createElement("dt"));
dt.textContent = "下载过滤器";
var dta = dt.appendChild(document.createElement("a"));
dta.className = "pubd-help-link";
dta.textContent = "(?)";
dta.href = "https://github.com/Mapaler/PixivUserBatchDownload/wiki/%E4%B8%8B%E8%BD%BD%E8%BF%87%E6%BB%A4%E5%99%A8";
dta.target = "_blank";
var dd = document.createElement("dd");
var downfilter = document.createElement("input");
downfilter.type = "text";
downfilter.className = "pubd-downfilter";
downfilter.name = "pubd-downfilter";
downfilter.id = downfilter.name;
downfilter.placeholder = "符合条件的图片将不会被发送到Aria2";
downfilter.onchange = function() {
if (dlg.downSchemeDom.selectedOptions.length < 1) { return; }
var schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].downfilter = downfilter.value;
};
dlg.downfilter = downfilter;
dd.appendChild(downfilter);
dl_ss.appendChild(dd);
//下载目录
var dt = document.createElement("dt");
dl_ss.appendChild(dt);
dt.textContent = "下载目录";
var dd = document.createElement("dd");
var savedir = document.createElement("input");
savedir.type = "text";
savedir.className = "pubd-savedir";
savedir.name = "pubd-savedir";
savedir.id = savedir.name;
savedir.placeholder = "文件下载到的目录";
savedir.onchange = function() {
if (dlg.downSchemeDom.selectedOptions.length < 1) { return; }
var schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].savedir = savedir.value;
};
dlg.savedir = savedir;
dd.appendChild(savedir);
dl_ss.appendChild(dd);
//保存路径
var dt = dl_ss.appendChild(document.createElement("dt"));
dt.textContent = "保存路径";
var dta = dt.appendChild(document.createElement("a"));
dta.className = "pubd-help-link";
dta.textContent = "(?)";
dta.href = "https://github.com/Mapaler/PixivUserBatchDownload/wiki/%E6%8E%A9%E7%A0%81";
dta.target = "_blank";
var dd = document.createElement("dd");
var savepath = document.createElement("input");
savepath.type = "text";
savepath.className = "pubd-savepath";
savepath.name = "pubd-savepath";
savepath.id = savepath.name;
savepath.placeholder = "分组保存的文件夹和文件名";
savepath.onchange = function() {
if (dlg.downSchemeDom.selectedOptions.length < 1) { return; }
var schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].savepath = savepath.value;
};
dlg.savepath = savepath;
dd.appendChild(savepath);
dl_ss.appendChild(dd);
//输出文本
var dt = dl_ss.appendChild(document.createElement("dt"));
dt.textContent = "文本输出模式格式";
var dta = dt.appendChild(document.createElement("a"));
dta.className = "pubd-help-link";
dta.textContent = "(?)";
dta.href = "https://github.com/Mapaler/PixivUserBatchDownload/wiki/%e9%80%89%e9%a1%b9%e7%aa%97%e5%8f%a3#%E6%96%87%E6%9C%AC%E8%BE%93%E5%87%BA%E6%A8%A1%E5%BC%8F%E6%A0%BC%E5%BC%8F";
dta.target = "_blank";
var dd = document.createElement("dd");
dd.className = "pubd-textout-bar";
var textout = document.createElement("textarea");
textout.className = "pubd-textout";
textout.name = "pubd-textout";
textout.id = textout.name;
textout.placeholder = "直接输出文本信息时的格式";
textout.wrap = "off";
textout.onchange = function() {
if (dlg.downSchemeDom.selectedOptions.length < 1) { return; }
var schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].textout = textout.value;
};
dlg.textout = textout;
dd.appendChild(textout);
dl_ss.appendChild(dd);
//自定义掩码
var dt = dl_ss.appendChild(document.createElement("dt"));
dt.textContent = "自定义掩码";
var dta = dt.appendChild(document.createElement("a"));
dta.className = "pubd-help-link";
dta.textContent = "(?)";
dta.href = "https://github.com/Mapaler/PixivUserBatchDownload/wiki/%E8%87%AA%E5%AE%9A%E4%B9%89%E6%8E%A9%E7%A0%81";
dta.target = "_blank";
var dd = document.createElement("dd");
dl_ss.appendChild(dd);
//▼掩码名
var ipt = document.createElement("input");
ipt.type = "text";
ipt.className = "pubd-mask-name";
ipt.name = "pubd-mask-name";
ipt.id = ipt.name;
ipt.placeholder = "自定义掩码名";
dlg.mask_name = ipt;
dd.appendChild(ipt);
//▲掩码名
//▼执行条件
var ipt = document.createElement("input");
ipt.type = "text";
ipt.className = "pubd-mask-logic";
ipt.name = "pubd-mask-logic";
ipt.id = ipt.name;
ipt.placeholder = "执行条件";
dlg.mask_logic = ipt;
dd.appendChild(ipt);
//▲执行条件
var ipt = document.createElement("input");
ipt.type = "button";
ipt.className = "pubd-mask-add";
ipt.value = "+";
ipt.onclick = function() { //增加自定义掩码
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中下载方案"); return; }
if (dlg.mask_name.value.length < 1) { alert("掩码名称为空"); return; }
if (dlg.mask_logic.value.length < 1) { alert("执行条件为空"); return; }
if (dlg.mask_content.value.includes("%{" + dlg.mask_logic.value + "}")) { alert("该掩码调用自身,会形成死循环。"); return; }
var schemeIndex = dlg.downSchemeDom.selectedIndex;
dlg.schemes[schemeIndex].maskAdd(dlg.mask_name.value, dlg.mask_logic.value, dlg.mask_content.value);
dlg.addMask(dlg.mask_name.value, dlg.mask_logic.value, dlg.mask_content.value);
dlg.mask_name.value = dlg.mask_logic.value = dlg.mask_content.value = "";
};
dd.appendChild(ipt);
var mask_remove = document.createElement("input");
mask_remove.type = "button";
mask_remove.className = "pubd-mask-remove";
mask_remove.value = "-";
mask_remove.onclick = function() { //删除自定义掩码
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中下载方案"); return; }
if (dlg.masklist.options.length < 1) { alert("已经没有掩码了"); return; }
if (dlg.masklist.selectedOptions.length < 1) { alert("没有选中掩码"); return; }
var schemeIndex = dlg.downSchemeDom.selectedIndex;
var maskIndex = dlg.masklist.selectedIndex;
dlg.schemes[schemeIndex].maskRemove(maskIndex);
dlg.masklist.remove(maskIndex);
for (var mi = maskIndex; mi < dlg.masklist.options.length; mi++) {
dlg.masklist.options[mi].value = mi;
}
};
dd.appendChild(mask_remove);
//▼掩码内容
var ipt = document.createElement("input");
ipt.type = "text";
ipt.className = "pubd-mask-content";
ipt.name = "pubd-mask-content";
ipt.id = ipt.name;
ipt.placeholder = "掩码内容";
dlg.mask_content = ipt;
dd.appendChild(ipt);
//▲掩码内容
dl_ss.appendChild(dd);
//▼掩码列表
var dd = document.createElement("dd");
dd.className = "pubd-mask-list-bar";
var masklist = new Select("pubd-mask-list", "pubd-mask-list");
masklist.size = 5;
masklist.onchange = function() { //读取选中的掩码
dlg.selectMask(this.selectedIndex);
};
dlg.masklist = masklist;
dd.appendChild(masklist);
//▲掩码列表
dl_ss.appendChild(dd);
//保存按钮栏
var dt = document.createElement("dt");
dl.appendChild(dt);
var dd = document.createElement("dd");
dd.className = "pubd-config-savebar";
var ipt = document.createElement("input");
ipt.type = "button";
ipt.className = "pubd-reset";
ipt.value = "清空选项";
ipt.onclick = function() {
if (confirm("您确定要将PUBD保存的所有设置,以及方案全部删除吗?\n(⚠️不可恢复)")==true){
dlg.reset();
return true;
}else{
return false;
}
};
dd.appendChild(ipt);
var ipt = document.createElement("input");
ipt.type = "button";
ipt.className = "pubd-save";
ipt.value = "保存选项";
ipt.onclick = function() {
dlg.save();
};
dd.appendChild(ipt);
dl.appendChild(dd);
//保存设置函数
dlg.save = function() {
//作品发送完成后,如何处理通知
var noticeType = 0;
dlg.noticeType.some(function(item){
if (item.checked) noticeType = parseInt(item.value);
return item.checked;
});
//逐项发送模式
var termwiseType = 2;
dlg.termwiseType.some(function(item){
if (item.checked) termwiseType = parseInt(item.value);
return item.checked;
});
GM_setValue("pubd-getugoiraframe", dlg.getugoiraframe.checked); //获取动图帧数
GM_setValue("pubd-autoanalyse", dlg.autoanalyse.checked); //自动分析
GM_setValue("pubd-autodownload", dlg.autodownload.checked); //自动下载
GM_setValue("pubd-noticeType", noticeType); //处理通知
GM_setValue("pubd-termwiseType", termwiseType); //逐项发送
GM_setValue("pubd-downschemes", dlg.schemes); //下载方案
GM_setValue("pubd-defaultscheme", dlg.downSchemeDom.selectedIndex); //默认方案
GM_setValue("pubd-configversion", pubd.configVersion); //设置版本
GM_notification({text:"设置已保存", title:scriptName, image:scriptIcon});
pubd.downSchemes = NewDownSchemeArrayFromJson(dlg.schemes);
pubd.dialog.downthis.reloadSchemes();
pubd.dialog.downillust.reloadSchemes();
};
//重置设置函数
dlg.reset = function() {
GM_deleteValue("pubd-auth"); //登录相关信息
GM_deleteValue("pubd-getugoiraframe"); //获取动图帧数
GM_deleteValue("pubd-autoanalyse"); //自动分析
GM_deleteValue("pubd-autodownload"); //自动下载
GM_deleteValue("pubd-noticeType"); //处理通知
GM_deleteValue("pubd-termwiseType"); //逐项发送
GM_deleteValue("pubd-downschemes"); //下载方案
GM_deleteValue("pubd-defaultscheme"); //默认方案
GM_deleteValue("pubd-configversion"); //设置版本
GM_notification({text:"已清空重置设置", title:scriptName, image:scriptIcon});
};
//窗口关闭
dlg.close = function() {
progress.stop_token_animate();
};
//关闭窗口按钮
dlg.cptBtns.close.addEventListener("click", dlg.close);
//窗口初始化
dlg.initialise = function() {
dlg.getugoiraframe.checked = getValueDefault("pubd-getugoiraframe", true);
dlg.autoanalyse.checked = getValueDefault("pubd-autoanalyse", false);
dlg.autodownload.checked = getValueDefault("pubd-autodownload", false);
(dlg.noticeType[parseInt(getValueDefault("pubd-noticeType", 0))] || dlg.noticeType[0]).checked = true;
(dlg.termwiseType[parseInt(getValueDefault("pubd-termwiseType", 2))] || dlg.termwiseType[2]).checked = true;
dlg.schemes = NewDownSchemeArrayFromJson(pubd.downSchemes);
dlg.reloadSchemes();
dlg.selectScheme(getValueDefault("pubd-defaultscheme", 0));
dlg.refreshLoginState();
};
return dlg;
}
//构建登录对话框
function buildDlgLogin() {
const dlg = new Dialog("登录账户", "pubd-login", "pubd-login");
dlg.newAuth = null;
var frm = dlg.content.appendChild(new Frame("1.做好获取 APP 登录连接的准备", "pubd-auth-help"));
const aHelp = frm.content.appendChild(document.createElement("a"));
aHelp.appendChild(document.createTextNode("如何获取 APP 登录连接?"));
aHelp.target = "_blank";
aHelp.href = "https://github.com/Mapaler/PixivUserBatchDownload/wiki/%E8%8E%B7%E5%8F%96APP%E7%99%BB%E9%99%86%E9%93%BE%E6%8E%A5";
var frm = dlg.content.appendChild(new Frame("2.进行官方 APP 登录", "pubd-auth-weblogin"));
const aLogin = frm.content.appendChild(document.createElement("a"));
aLogin.appendChild(document.createTextNode("访问官方登录页面"));
aLogin.className = "pubd-login-official-link";
aLogin.target = "_blank";
var frm = dlg.content.appendChild(new Frame("3.填写 APP 登录连接", "pubd-auth-applogin"));
dlg.content.appendChild(frm);
var div = frm.content.appendChild(document.createElement("div"));
const pixivLink = div.appendChild(document.createElement("input"));
pixivLink.type = "url";
pixivLink.className = "pubd-pixiv-app-link";
pixivLink.placeholder = "例如:pixiv://account/login?code=xxxxxx&via=login";
const btnLogin = div.appendChild(document.createElement("button"));
btnLogin.className = "pubd-login-auth";
btnLogin.appendChild(document.createTextNode("登录"));
//登录按钮
btnLogin.onclick = function() {
if (/^pixiv:\/\//i.test(pixivLink.value))
{
const loginLink = new URL(pixivLink.value);
const authorization_code = loginLink.searchParams.get("code");
if (authorization_code)
{
//使用token登录
dlg.error.replace("登录中···");
const options = {
onload:function(jore) { //onload_suceess_Cb
dlg.error.replace("登录成功");
dlg.newOAuth.save(); //保存新的认证
pubd.oAuth = dlg.newOAuth; //使用新的认证替换原来的认证
pubd.dialog.config.refreshLoginState();
},
onload_hasError:function(jore) { //onload_haserror_Cb //返回错误消息
dlg.error.replace(["错误代码:" + jore.errors.system.code, jore.errors.system.message]);
},
onload_notJson:function(re) { //onload_notjson_Cb //返回不是JSON
dlg.error.replace(["服务器返回不是 JSON 格式", re]);
},
onerror:function(re) { //onerror_Cb //网络请求发生错误
dlg.error.replace("网络请求发生错误");
},
}
dlg.newOAuth.login(authorization_code, options);
}else
{
alert("PUBD:登录链接中未找到 code");
}
}else
{
alert("PUBD:输入的链接格式不正确");
}
};
//错误信息
dlg.error = dlg.content.appendChild(new ErrorMsg());
dlg.content.appendChild(document.createElement("hr"));
var frm = dlg.content.appendChild(new Frame("使用现有刷新许可证登录", "pubd-refresh_token-login"));
dlg.content.appendChild(frm);
var div = frm.content.appendChild(document.createElement("div"));
const iptRefreshToken = div.appendChild(document.createElement("input"));
iptRefreshToken.type = "text";
iptRefreshToken.className = "pubd-refresh-token";
iptRefreshToken.placeholder = "refresh_token";
const btnRefreshToken = div.appendChild(document.createElement("button"));
btnRefreshToken.className = "pubd-login-refresh_token";
btnRefreshToken.appendChild(document.createTextNode("登录"));
//登录按钮
btnRefreshToken.onclick = function() {
if (!pubd.oAuth.auth_data)
{
pubd.oAuth.auth_data = {};
}
pubd.oAuth.auth_data.refresh_token = iptRefreshToken.value;
//刷新许可
pubd.dialog.refresh_token.show(
(document.body.clientWidth - 370)/2,
window.scrollY+300
);
};
//窗口初始化
dlg.initialise = function() {
this.error.clear();
//每次打开这个窗口,都创建一个新的认证
this.newOAuth = new oAuth2();
aLogin.href = this.newOAuth.get_login_url();
};
return dlg;
}
//构建token刷新对话框
function buildDlgRefreshToken() {
const dlg = new Dialog("刷新许可", "pubd-refresh-token pubd-dialog-transparent", "pubd-refresh-token");
//Logo部分
const logo_box = dlg.content.appendChild(document.createElement("div"));
logo_box.className = "logo-box";
const logo = logo_box.appendChild(document.createElement("img"));
logo.className = "pixiv-logo";
logo.src = "https://s.pximg.net/accounts/assets/6bea8becc71d27cd20649ffbc047e456.svg";
logo.alt = "pixiv logo";
const progress = dlg.tokenExpires = dlg.content.appendChild(buildProgressToken());
const lblRefreshToken = dlg.content.appendChild(document.createElement("label"));
lblRefreshToken.textContent = "刷新用许可证代码(refresh_token)";
const iptRefreshToken = lblRefreshToken.appendChild(document.createElement("input"));
iptRefreshToken.className = "pubd-refresh-token";
iptRefreshToken.type = "text";
iptRefreshToken.readOnly = true;
//错误信息
dlg.error = dlg.content.appendChild(new ErrorMsg());
//窗口关闭
dlg.close = function() {
progress.stop_token_animate();
};
//关闭窗口按钮
dlg.cptBtns.close.addEventListener("click", dlg.close);
//窗口初始化
dlg.initialise = function(arg = {}) {
this.error.clear();
iptRefreshToken.value = pubd.oAuth.auth_data.refresh_token;
progress.start_token_animate();
dlg.error.replace("刷新许可中···");
const options = {
onload:function(jore) { //onload_suceess_Cb
pubd.oAuth.save();
dlg.error.replace("成功更新");
iptRefreshToken.value = jore.refresh_token;
progress.start_token_animate();
pubd.dialog.config.refreshLoginState();
if (arg.onload) arg.onload(jore);
},
onload_hasError:function(jore) { //onload_haserror_Cb //返回错误消息
dlg.error.replace(["错误代码:" + jore.errors.system.code, jore.errors.system.message]);
if (arg.onload_hasError) arg.onload_hasError(jore);
},
onload_notJson:function(re) { //onload_notjson_Cb //返回不是JSON
dlg.error.replace(["服务器返回不是 JSON 格式", re]);
if (arg.onload_notJson) arg.onload_notJson(re);
},
onerror:function(re) { //onerror_Cb //网络请求发生错误
dlg.error.replace("网络请求发生错误");
if (arg.onerror) arg.onerror(re);
},
}
pubd.oAuth.refresh_token(options);
};
return dlg;
}
//构建通用下载对话框
function buildDlgDown(caption, classname, id) {
var dlg = new Dialog(caption, classname, id);
var dl = dlg.content.appendChild(document.createElement("dl"));
var dt = document.createElement("dt");
dl.appendChild(dt);
dt.innerHTML = ""; //用户头像等信息
var dd = document.createElement("dd");
dlg.infoCard = new InfoCard(); //创建信息卡
dd.appendChild(dlg.infoCard.dom);
dl.appendChild(dd);
var dt = document.createElement("dt");
dl.appendChild(dt);
dt.innerHTML = "进程日志";
var dd = document.createElement("dd");
var ipt = document.createElement("textarea");
ipt.readOnly = true;
ipt.className = "pubd-down-log";
ipt.wrap = "off";
dlg.logTextarea = ipt;
dd.appendChild(ipt);
dl.appendChild(dd);
//下载方案
dlg.schemes = null;
dlg.reloadSchemes = function() { //重新读取所有下载方案
dlg.schemes = pubd.downSchemes;
dlg.downSchemeDom.options.length = 0;
dlg.schemes.forEach(function(item, index) {
dlg.downSchemeDom.add(item.name, index);
});
if (getValueDefault("pubd-defaultscheme",0) >= 0)
dlg.selectScheme(getValueDefault("pubd-defaultscheme",0));
else if (dlg.downSchemeDom.options.length > 0)
dlg.selectScheme(0);
};
//选择一个方案,同时读取设置
dlg.selectScheme = function(index) {
if (index == undefined) index = 0;
if (dlg.downSchemeDom.options.length < 1 || dlg.downSchemeDom.selectedOptions.length < 1) { return; }
dlg.downSchemeDom.selectedIndex = index;
};
var dt = document.createElement("dt");
dl.appendChild(dt);
dt.innerHTML = "选择下载方案";
var dd = document.createElement("dd");
var slt = new Select("pubd-downscheme");
dlg.downSchemeDom = slt;
dd.appendChild(slt);
dl.appendChild(dd);
//下载按钮栏
var dt = document.createElement("dt");
dl.appendChild(dt);
var dd = document.createElement("dd");
dd.className = "pubd-downthis-downbar";
var textdown = document.createElement("input");
textdown.type = "button";
textdown.className = "pubd-textdown";
textdown.value = "输出\n文本";
textdown.onclick = function(event) {
dlg.textdownload(event);
};
textdown.disabled = true;
dlg.textdown = textdown;
dd.appendChild(textdown);
var startdown = document.createElement("input");
startdown.type = "button";
startdown.className = "pubd-startdown";
startdown.value = "发送到Aria2";
startdown.onclick = function() {
dlg.startdownload();
};
startdown.disabled = true;
dlg.startdown = startdown;
dd.appendChild(startdown);
dl.appendChild(dd);
//文本输出栏
var dt = document.createElement("dt");
dl.appendChild(dt);
var dd = document.createElement("dd");
dd.className = "pubd-down-textout-bar";
dl.appendChild(dd);
var ipt = document.createElement("textarea");
ipt.readOnly = true;
ipt.className = "pubd-down-textout display-none";
ipt.wrap = "off";
dlg.textoutTextarea = ipt;
dd.appendChild(ipt);
//显示日志相关
dlg.logArr = []; //用于储存一行一行的日志信息。
dlg.logClear = function() {
dlg.logArr.length = 0;
this.logTextarea.value = "";
};
dlg.log = function(text) {
dlg.logArr.push(text);
this.logTextarea.value = this.logArr.join("\n");
this.logTextarea.scrollTop = this.logTextarea.scrollHeight;
};
return dlg;
}
//构建当前画师下载对话框
function buildDlgDownThis(userid) {
//一个用户的信息
var UserInfo = function() {
this.done = false; //是否已完成用户信息获取
this.info = {
profile: null,
user: null,
};
this.illusts = new Works();
this.bookmarks = new Works();
};
var dlg = new buildDlgDown("下载当前画师", "pubd-down pubd-downthis", "pubd-downthis");
dlg.infoCard.infos = {"ID":userid};
dlg.user = new UserInfo();
dlg.works = null; //当前处理对象
var dt = document.createElement("dt");
var dd = document.createElement("dd");
dlg.infoCard.dom.insertAdjacentElement("afterend",dt);
dt.insertAdjacentElement("afterend",dd);
var frm = dd.appendChild(new Frame("下载内容"));
var radio1 = frm.content.appendChild(new LabelInput("他的作品", "pubd-down-content", "pubd-down-content", "radio", "0", true));
var radio2 = frm.content.appendChild(new LabelInput("他的收藏", "pubd-down-content", "pubd-down-content", "radio", "1", true));
dlg.dcType = [radio1.input, radio2.input];
radio1.input.onclick = function() { reAnalyse(this); };
radio2.input.onclick = function() { reAnalyse(this); };
function reAnalyse(radio) {
if (radio.checked == true) {
if (radio.value == 0)
dlg.user.bookmarks.break = true; //radio值为0,使收藏中断
else
dlg.user.illusts.break = true; //radio值为1,使作品中断
dlg.analyse(radio.value, dlg.infoCard.infos.ID);
}
}
var dt = document.createElement("dt");
dd.insertAdjacentElement("afterend",dt);
dt.innerHTML = "信息获取进度";
var dd = document.createElement("dd");
dt.insertAdjacentElement("afterend",dd);
var progress = new Progress();
dlg.progress = progress;
dd.appendChild(progress);
var btnBreak = document.createElement("input");
btnBreak.type = "button";
btnBreak.className = "pubd-breakdown";
btnBreak.value = "中断操作";
btnBreak.onclick = function() {
dlg.user.illusts.break = true; //使作品中断
dlg.user.bookmarks.break = true; //使收藏中断
pubd.downbreak = true; //使下载中断
};
dlg.logTextarea.parentNode.previousElementSibling.appendChild(btnBreak);
//分析
dlg.analyse = function(contentType, userid, callbackAfterAnalyse) {
if (!userid) {dlg.log("错误:没有用户ID。"); return;}
contentType = contentType == undefined ? 0 : parseInt(contentType);
var works = contentType == 0 ? dlg.user.illusts : dlg.user.bookmarks; //将需要分析的数据储存到works里
dlg.works = works;
if (works.runing) {
dlg.log("已经在进行分析操作了");
return;
}
works.break = false; //暂停flag为false
works.runing = true; //运行状态为true
pubd.ajaxTimes = 0; //ajax提交次数恢复为0
dlg.textdown.disabled = true; //禁用下载按钮
dlg.startdown.disabled = true; //禁用输出文本按钮
dlg.progress.set(0); //进度条归零
dlg.logClear(); //清空日志
//根据用户信息是否存在,决定分析用户还是图像
if (!dlg.user.done) {
startAnalyseUser(userid, contentType);
} else {
dlg.log("ID:" + userid + " 用户信息已存在");
startAnalyseWorks(dlg.user, contentType); //开始获取第一页
}
function startAnalyseUser(userid, contentType) {
dlg.log("开始获取ID为 " + userid + " 的用户信息");
++pubd.ajaxTimes;
xhrGenneral(
"https://app-api.pixiv.net/v1/user/detail?user_id=" + userid,
function(jore) { //onload_suceess_Cb
works.runing = true;
dlg.user.done = true;
dlg.user.info = Object.assign(dlg.user.info, jore);
if (mdev)
{
const usersStore = db.transaction("users", "readwrite").objectStore("users");
let usersStoreRequest = usersStore.get(jore.user.id);
usersStoreRequest.onsuccess = function(event) {
// 获取我们想要更新的数据
let data = event.target.result;
if (data)
console.log("上次的头像",data.user.profile_image_urls);
if (!data || //没有老数据
!data.avatarBlob || //没有头像
data.user.profile_image_urls.medium != jore.user.profile_image_urls.medium //换了头像
)
{
console.debug("需要更新头像图片",jore.user.profile_image_urls);
GM_xmlhttpRequest({
url: jore.user.profile_image_urls.medium,
method: "get",
responseType: "blob",
headers: new HeadersObject(),
onload: function(response) {
console.info("用户头像Blob结果", response.response);
var obj_url = URL.createObjectURL(response.response);
var newImg = new Image();
newImg.src = obj_url;
URL.revokeObjectURL(obj_url);
document.body.appendChild(newImg);
var newData = data ? Object.assign(data,jore) : jore;
newData.avatarBlob = response.response;
// 把更新过的对象放回数据库
const usersStore = db.transaction("users", "readwrite").objectStore("users");
var requestUpdate = usersStore.put(newData);
requestUpdate.onerror = function(event) {// 错误处理
console.error(`${newData.user.name} 更新数据库头像发生错误`,newData);
};
requestUpdate.onsuccess = function(event) {// 完成,数据已更新!
console.debug(`${newData.user.name} 已${data?"更新":"添加"}到头像用户数据库`,newData);
};
return;
},
onerror: function(response) {
console.error("抓取头像失败", response);
return;
}
});
}else
{
var newData = data ? Object.assign(data,jore) : jore;
// 把更新过的对象放回数据库
var requestUpdate = usersStore.put(newData);
requestUpdate.onerror = function(event) {// 错误处理
console.error(`${newData.user.name} 发生错误`,newData);
};
requestUpdate.onsuccess = function(event) {// 完成,数据已更新!
console.debug(`${newData.user.name} 已${data?"更新":"添加"}到用户数据库`,newData);
};
}
};
usersStoreRequest.onerror = function(event) {// 错误处理
console.error(`${jore.user.name} 数据库里没有?`,jore);
};
}
dlg.infoCard.thumbnail = jore.user.profile_image_urls.medium;
dlg.infoCard.infos = Object.assign(dlg.infoCard.infos, {
"昵称": jore.user.name,
"作品投稿数": jore.profile.total_illusts + jore.profile.total_manga,
"公开收藏数": jore.profile.total_illust_bookmarks_public,
});
startAnalyseWorks(dlg.user, contentType); //分析完成后开始获取第一页
},
function(jore) { //onload_haserror_Cb //返回错误消息
works.runing = false;
dlg.log("错误信息:" + (jore.error.message || jore.error.user_message));
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
return;
},
function(re) { //onload_notjson_Cb //返回不是JSON
dlg.log("错误:返回不是JSON,或本程序异常");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
},
function(re) { //onerror_Cb //网络请求发生错误
dlg.log("错误:网络请求发生错误");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
},
function(str) { //dlog,推送错误消息
dlg.log(str);
return str;
}
);
}
//开始分析作品的前置操作
function startAnalyseWorks(user, contentType) {
var uInfo = user.info;
var works, total, contentName, apiurl;
//获取作品,contentType == 0,获取收藏,contentType == 1
if (contentType == 0) {
works = user.illusts;
total = uInfo.profile.total_illusts + uInfo.profile.total_manga;
contentName = "作品";
apiurl = "https://app-api.pixiv.net/v1/user/illusts?user_id=" + uInfo.user.id;
} else {
works = user.bookmarks;
total = uInfo.profile.total_illust_bookmarks_public;
contentName = "收藏";
apiurl = "https://app-api.pixiv.net/v1/user/bookmarks/illust?user_id=" + uInfo.user.id + "&restrict=public";
}
if (works.item.length > 0) { //断点续传
dlg.log(`${contentName} 断点续传进度 ${works.item.length}/${total}`);
dlg.progress.set(works.item.length / total); //设置当前下载进度
}
analyseWorks(user, contentType, apiurl); //开始获取第一页
}
//分析作品递归函数
function analyseWorks(user, contentType, apiurl) {
var uInfo = user.info;
var works, total, contentName;
if (contentType == 0) {
works = user.illusts;
total = uInfo.profile.total_illusts + uInfo.profile.total_manga;
contentName = "作品";
} else {
works = user.bookmarks;
total = uInfo.profile.total_illust_bookmarks_public;
contentName = "收藏";
}
if (works.done) {
//返回所有动图
var ugoiras = works.item.filter(function(item) {
return item.type == "ugoira";
});
dlg.log(`共存在 共 ${ugoiras.length} 件动图`);
if (ugoiras.some(function(item) { //如果有没有帧数据的动图
return item.ugoira_metadata == undefined;
})) {
if (!getValueDefault("pubd-getugoiraframe",true)) {
dlg.log("由于用户设置,跳过获取动图帧数。");
} else {
analyseUgoira(works, ugoiras, function() { //开始分析动图
analyseWorks(user, contentType, apiurl); //开始获取下一页
});
return;
}
}//没有动图则继续
if (works.item.length < total)
dlg.log("可能因为权限原因,无法获取到所有 " + contentName);
//计算一下总页数
works.picCount = works.item.reduce(function(pV,cItem){
var page = cItem.page_count;
if (cItem.type == "ugoira" && cItem.ugoira_metadata) //动图
{
page = cItem.ugoira_metadata.frames.length;
}
return pV+=page;
},0);
dlg.log(`${contentName} 共 ${works.item.length} 件(约 ${works.picCount} 张图片)已获取完毕。`);
dlg.progress.set(1);
works.runing = false;
works.next_url = "";
dlg.textdown.disabled = false;
dlg.startdown.disabled = false;
if (callbackAfterAnalyse) callbackAfterAnalyse();
return;
}
if (works.break) {
dlg.log("检测到 " + contentName + " 中断进程命令");
works.break = false;
works.runing = false;
dlg.textdown.disabled = false; //启用按钮,中断暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
return;
}
setTimeout(()=>{
xhrGenneral(
apiurl,
function(jore) { //onload_suceess_Cb
works.runing = true;
var illusts = jore.illusts;
illusts.forEach(function(work) {
const original = work.page_count > 1 ?
work.meta_pages[0].image_urls.original : //漫画多图
work.meta_single_page.original_image_url; //单张图片或动图,含漫画单图
//取得解析后的网址
const parsedUrl = parseIllustUrl(original);
//合并到work里
Object.assign(work, parsedUrl);
if (parsedUrl.parsedURL.limited)
{
dlg.log(`${contentName} ${work.id} 非公开,无权获取下载地址。`);
}else if(parsedUrl.parsedURL.unknown)
{
dlg.log(`${contentName} ${work.id} 未知的原图网址格式。`);
}
works.item.push(work);
if (mdev)
{
const illustsStore = db.transaction("illusts", "readwrite").objectStore("illusts");
const illustsStoreRequest = illustsStore.put(work);
illustsStoreRequest.onsuccess = function(event) {
//console.debug(`${work.title} 已添加到作品数据库`);
};
}
});
dlg.log(`${contentName} 获取进度 ${works.item.length}/${total}`);
if (works == dlg.works) dlg.progress.set(works.item.length / total); //如果没有中断则设置当前下载进度
if (jore.next_url) { //还有下一页
works.next_url = jore.next_url;
} else { //没有下一页
works.done = true;
}
analyseWorks(user, contentType, jore.next_url); //开始获取下一页
},
function(jore) { //onload_haserror_Cb //返回错误消息
works.runing = false;
dlg.log("错误信息:" + (jore.error.message || jore.error.user_message));
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
return;
},
function(re) { //onload_notjson_Cb //返回不是JSON
dlg.log("错误:返回不是JSON,或本程序异常");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
},
function(re) { //onerror_Cb //网络请求发生错误
dlg.log("错误:网络请求发生错误");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
}
);
},pubd.ajaxTimes++ > startDelayAjaxTimes ? ajaxDelayDuration : 0);
}
function analyseUgoira(works, ugoirasItems, callback) {
var dealItems = ugoirasItems.filter(function(item) {
return (item.type == "ugoira" && item.ugoira_metadata == undefined);
});
if (dealItems.length < 1) {
dlg.log("动图获取完毕");
dlg.progress.set(1); //设置当前下载进度
callback();
return;
}
if (works.break) {
dlg.log("检测到中断进程命令");
works.break = false;
works.runing = false;
dlg.textdown.disabled = false; //中断暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
return;
}
var work = dealItems[0]; //当前处理的图
setTimeout(()=>{
if (pubd.ajaxTimes == startDelayAjaxTimes) dlg.log(`已提交超过 ${startDelayAjaxTimes} 次请求,为避免被P站限流,现在开始每次请求将间隔 ${ajaxDelayDuration/1000} 秒。`);
getUgoiraMeta(
work.id,
function(jore) { //onload_suceess_Cb
works.runing = true;
//var illusts = jore.illusts;
work = Object.assign(work, jore);
if (mdev)
{
const illustsStore = db.transaction("illusts", "readwrite").objectStore("illusts");
const illustsStoreRequest = illustsStore.put(work);
illustsStoreRequest.onsuccess = function(event) {
console.debug(`${work.title} 已更新动画帧数据到数据库`);
};
}
dlg.log("动图信息 获取进度 " + (ugoirasItems.length - dealItems.length + 1) + "/" + ugoirasItems.length);
dlg.progress.set(1 - dealItems.length / ugoirasItems.length); //设置当前下载进度
analyseUgoira(works, ugoirasItems, callback); //开始获取下一项
},
function(jore) { //onload_haserror_Cb //返回错误消息
if(work.restrict > 0) //非公共权限
{ //添加一条空信息
work.ugoira_metadata = {
frames: [],
zip_urls: {
medium: "",
},
};
dlg.log("无访问权限,跳过本条。");
analyseUgoira(works, ugoirasItems, callback); //开始获取下一项
}else
{
works.runing = false;
dlg.log("错误信息:" + (jore.error.message || jore.error.user_message));
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
}
return;
},
function(re) { //onload_notjson_Cb //返回不是JSON
dlg.log("错误:返回不是JSON,或本程序异常");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
},
function(re) { //onerror_Cb //网络请求发生错误
dlg.log("错误:网络请求发生错误");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
}
);
},pubd.ajaxTimes++ > startDelayAjaxTimes ? ajaxDelayDuration : 0);
}
};
//输出文本按钮
dlg.textdownload = function(event) {
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中方案"); return; }
var scheme = dlg.schemes[dlg.downSchemeDom.selectedIndex];
var contentType = dlg.dcType[1].checked ? 1 : 0;
var userInfo = dlg.user.info;
var illustsItems = contentType == 0 ? dlg.user.illusts.item : dlg.user.bookmarks.item; //将需要分析的数据储存到works里
dlg.log("正在生成文本信息");
try {
var outTxtArr;
if (event.ctrlKey)
{
outTxtArr = showMask(scheme.textout, scheme.masklist, userInfo, null, 0);
}else
{
outTxtArr = illustsItems.map(function(illust) {
var page_count = illust.page_count;
if (illust.type == "ugoira" && illust.ugoira_metadata) //动图
{
page_count = illust.ugoira_metadata.frames.length;
}
var outArr = []; //输出内容
for (var pi = 0; pi < page_count; pi++) {
if (returnLogicValue(scheme.downfilter, userInfo, illust, pi) || limitingFilenameExp.test(illust.filename)) {
//跳过此次输出
continue;
}else{
outArr.push(showMask(scheme.textout, scheme.masklist, userInfo, illust, pi));
}
}
return outArr.join("");
}).join("");
}
dlg.textoutTextarea.value = outTxtArr;
dlg.textoutTextarea.classList.remove("display-none");
dlg.log("文本信息输出成功");
} catch (error) {
console.log(error);
}
};
//开始下载按钮
dlg.startdownload = function() {
dlg.textoutTextarea.classList.add("display-none");
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中方案"); return; }
var scheme = dlg.schemes[dlg.downSchemeDom.selectedIndex];
var contentType = dlg.dcType[1].checked ? 1 : 0;
var userInfo = dlg.user.info;
var works = (contentType == 0 ? dlg.user.illusts : dlg.user.bookmarks);
var illustsItems = works.item.concat(); //为了不改变原数组,新建一个数组
let termwiseType = parseInt(getValueDefault("pubd-termwiseType", 2));
if (works.picCount > changeTermwiseCount && termwiseType ==2)
{
dlg.log(`图片总数超过${changeTermwiseCount}张,自动切换为使用按作品逐项发送模式。`);
termwiseType = 1;
}
if (termwiseType == 0)
dlg.log("开始按图片逐项发送(约 "+works.picCount+" 次请求),⏳请耐心等待。");
else if (termwiseType == 1)
dlg.log("开始按作品逐项发送(约 "+illustsItems.length+" 次请求),⏳请耐心等待。");
else if (termwiseType == 2)
dlg.log("开始按作者发送,数据量较大时有较高延迟。\n⏳请耐心等待完成通知,勿多次点击。");
else
{
alert("错误:未知的逐项模式" + termwiseType);
console.error("PUBD:错误:未知的逐项模式:", termwiseType);
return;
}
var downP = { progress: dlg.progress, current: 0, max: 0 };
downP.max = works.picCount; //获取总需要下载发送的页数
var aria2 = new Aria2(scheme.rpcurl); //生成一个aria2对象
sendToAria2_illust(aria2, termwiseType, illustsItems, userInfo, scheme, downP, function() {
aria2 = null;
dlg.log("😄 " + userInfo.user.name + " 下载信息发送完毕");
var ntype = parseInt(getValueDefault("pubd-noticeType", 0)); //获取结束后如何处理通知
var bodyText = "" + userInfo.user.name + " 的相关插画已全部发送到指定的Aria2";
if (ntype == 1)
bodyText += "\n\n点击此通知 🔙返回 页面。";
else if (ntype == 2)
bodyText += "\n\n点击此通知 ❌关闭 页面。";
else if (ntype == 3)
bodyText += "\n\n通知结束时页面将 🅰️自动❌关闭。";
GM_notification(
{
text:bodyText,
title:"下载信息发送完毕",
image:userInfo.user.profile_image_urls.medium
},
function(){ //点击了通知
var ntype = parseInt(getValueDefault("pubd-noticeType", 0));
if (ntype == 1)
window.focus();
else if (ntype == 2)
window.close();
},
function(){ //关闭了通知
var ntype = parseInt(getValueDefault("pubd-noticeType", 0));
if (ntype == 3)
window.close();
}
);
});
};
//启动初始化
dlg.initialise = function(arg) {
var dcType = 0;
if (dlg.user.bookmarks.runing) //如果有程序正在运行,则覆盖设置。
dcType = 1;
else if (dlg.user.illusts.runing)
dcType = 0;
dlg.dcType[dcType].checked = true;
let uid = arg.id;
if (arg && arg.id>0) //提供了ID
{
if (arg.id != dlg.infoCard.infos.ID)
{ //更换新的id
dlg.infoCard.thumbnail = "";
dlg.infoCard.infos = {"ID":arg.id}; //初始化窗口id
dlg.user = new UserInfo(); //重置用户数据
}
}else if(!dlg.infoCard.infos.ID) //没有ID
{
uid = parseInt(prompt("没有用户ID,请手动输入。", "ID缺失"),10);
dlg.infoCard.infos = {"ID":uid}; //初始化窗口id
}
if (getValueDefault("pubd-autoanalyse",false)) {
//开始自动分析的话,也自动添加到快速收藏
if (!pubd.fastStarList.has(userid)) { //不存在,则添加
pubd.fastStarList.add(uid);
pubd.start.star.classList.add("stars");
GM_setValue("pubd-faststar-list",pubd.fastStarList.exportArray());
console.debug(`已将 ${uid} 添加到快速收藏`);
}else if (mdev)
{
console.debug(`快速收藏中已存在 ${uid}`);
}
dlg.analyse(dcType, uid, function(){
if (getValueDefault("pubd-autodownload",false)) { //自动开始
dlg.log("🅰️自动开始发送");
dlg.startdownload();
}
});
}
dlg.reloadSchemes();
};
return dlg;
}
//构建当前作品下载对话框
function buildDlgDownIllust(illustid) {
var dlg = new buildDlgDown("下载当前作品", "pubd-down pubd-downillust", "pubd-downillust");
dlg.infoCard.infos = {"ID":illustid};
dlg.work = null; //当前处理对象
//分析
dlg.analyse = function(illustid,callbackAfterAnalyse) {
if (!illustid) {dlg.log("错误:没有作品ID。"); return;}
dlg.textdown.disabled = true; //禁用下载按钮
dlg.startdown.disabled = true; //禁用输出文本按钮
dlg.logClear(); //清空日志
if (dlg.work != undefined)
{
dlg.textdown.disabled = false;
dlg.startdown.disabled = false;
console.log("当前作品JSON数据:",dlg.work);
dlg.log("图片信息获取完毕");
if (callbackAfterAnalyse) callbackAfterAnalyse();
}else
{
dlg.log("开始获取作品信息");
analyseWork(illustid); //开始获取第一页
}
//分析作品递归函数
function analyseWork(illustid) {
xhrGenneral(
"https://app-api.pixiv.net/v1/illust/detail?illust_id=" + illustid,
function(jore) { //onload_suceess_Cb
var work = dlg.work = jore.illust;
const original = work.page_count > 1 ?
work.meta_pages[0].image_urls.original : //漫画多图
work.meta_single_page.original_image_url; //单张图片或动图,含漫画单图
//取得解析后的网址
const parsedUrl = parseIllustUrl(original);
//合并到work里
Object.assign(work, parsedUrl);
if (parsedUrl.parsedURL.limited)
{
dlg.log(`${contentName} ${work.id} 非公开,无权获取下载地址。`);
}else if(parsedUrl.parsedURL.unknown)
{
dlg.log(`${contentName} ${work.id} 未知的原图网址格式。`);
}
if (mdev)
{
const illustsStoreRequest = db.transaction("illusts", "readwrite").objectStore("illusts").put(work);
illustsStoreRequest.onsuccess = function(event) {
console.debug(`${work.title} 已添加到作品数据库`);
};
}
dlg.infoCard.thumbnail = work.image_urls.square_medium;
var iType = "插画";
if (work.type == "ugoira")
iType = "动画";
else if (work.type == "manga")
iType = "漫画";
if (work.page_count>1)
iType += "(多图)";
dlg.infoCard.infos = Object.assign(dlg.infoCard.infos, {
"作品名称": work.title,
"作品类型": iType,
"作品页数": work.page_count,
});
if (work.type == "ugoira" && work.ugoira_metadata == undefined && getValueDefault("pubd-getugoiraframe",true))
{
analyseUgoira(work, function() { //开始分析动图
dlg.textdown.disabled = false;
dlg.startdown.disabled = false;
dlg.infoCard.infos["作品页数"] = work.ugoira_metadata.frames.length;
dlg.infoCard.reload(); //必须要reload
dlg.log("图片信息获取完毕");
console.log("当前作品JSON数据:",work);
if (callbackAfterAnalyse) callbackAfterAnalyse();
});
return;
}else
{
if (!getValueDefault("pubd-getugoiraframe",true)) {
dlg.log("由于用户设置,跳过获取动图帧数。");
}
dlg.textdown.disabled = false;
dlg.startdown.disabled = false;
dlg.log("图片信息获取完毕");
console.log("当前作品JSON数据:",work);
if (callbackAfterAnalyse) callbackAfterAnalyse();
}
},
function(jore) { //onload_haserror_Cb //返回错误消息
dlg.log("错误信息:" + (jore.error.message || jore.error.user_message));
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
return;
},
function(re) { //onload_notjson_Cb //返回不是JSON
dlg.log("错误:返回不是JSON,或本程序异常");
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
},
function(re) { //onerror_Cb //网络请求发生错误
dlg.log("错误:网络请求发生错误");
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
}
);
}
function analyseUgoira(work, callback) {
getUgoiraMeta(
work.id,
function(jore) { //onload_suceess_Cb
work = Object.assign(work, jore);
if (mdev)
{
const illustsStoreRequest = db.transaction("illusts", "readwrite").objectStore("illusts").put(work);
illustsStoreRequest.onsuccess = function(event) {
console.debug(`${work.title} 已更新动画帧数据到数据库`);
};
}
dlg.log("动图信息获取完成");
callback(); //开始获取下一项
},
function(jore) { //onload_haserror_Cb //返回错误消息
if(work.restrict > 0) //非公共权限
{ //添加一条空信息
work.ugoira_metadata = {
frames: [],
zip_urls: {
medium: "",
},
};
dlg.log("无访问权限,跳过本条。");
callback(); //开始获取下一项
}else
{
works.runing = false;
dlg.log("错误信息:" + (jore.error.message || jore.error.user_message));
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
}
return;
},
function(re) { //onload_notjson_Cb //返回不是JSON
dlg.log("错误:返回不是JSON,或本程序异常");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
},
function(re) { //onerror_Cb //网络请求发生错误
dlg.log("错误:网络请求发生错误");
works.runing = false;
dlg.textdown.disabled = false; //错误暂停时,可以操作目前的进度。
dlg.startdown.disabled = false;
}
);
}
};
//输出文本按钮
dlg.textdownload = function(event) {
var illust = dlg.work;
if (illust == undefined) {dlg.log("没有获取作品数据。"); return;}
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中方案"); return; }
var scheme = dlg.schemes[dlg.downSchemeDom.selectedIndex];
dlg.log("正在生成文本信息");
try {
var page_count = illust.page_count;
if (illust.type == "ugoira" && illust.ugoira_metadata) //动图
{
page_count = illust.ugoira_metadata.frames.length;
}
var outArr = []; //输出内容
for (var pi = 0; pi < page_count; pi++) {
if (returnLogicValue(scheme.downfilter, null, illust, pi) || limitingFilenameExp.test(illust.filename)) {
//跳过此次输出
continue;
}else{
outArr.push(showMask(scheme.textout, scheme.masklist, null, illust, pi));
}
}
var outTxt = outArr.join("");
dlg.textoutTextarea.value = outTxt;
dlg.textoutTextarea.classList.remove("display-none");
dlg.log("文本信息输出成功");
} catch (error) {
console.log(error);
}
};
//开始下载按钮
dlg.startdownload = function() {
dlg.textoutTextarea.classList.add("display-none");
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中方案"); return; }
var scheme = dlg.schemes[dlg.downSchemeDom.selectedIndex];
var termwiseType = parseInt(getValueDefault("pubd-termwiseType", 2));
if (termwiseType == 0)
dlg.log("开始按图片逐项发送,⏳请耐心等待。");
else if (termwiseType == 1 || termwiseType == 2)
dlg.log("一次性发送整个作品,⏳请耐心等待。");
else
{
alert("错误:未知的逐项模式" + termwiseType);
console.error("PUBD:错误:未知的逐项模式:", termwiseType);
return;
}
var aria2 = new Aria2(scheme.rpcurl); //生成一个aria2对象
sendToAria2_illust(aria2, termwiseType, [dlg.work], null, scheme, null, function() {
aria2 = null;
dlg.log("😄 当前作品下载信息发送完毕");
});
};
//启动初始化
dlg.initialise = function(arg) {
if (arg && arg.id>0) //提供了ID
{
if (arg.id != dlg.infoCard.infos.ID)
{ //更换新的id
dlg.infoCard.thumbnail = "";
dlg.infoCard.infos = {"ID":arg.id}; //初始化窗口id
dlg.work = null; //重置作品数据
}
}else if(!dlg.infoCard.infos.ID) //没有ID
{
dlg.infoCard.infos = {"ID":parseInt(prompt("没有作品ID,请手动输入。", "ID缺失"))}; //初始化窗口id
}
dlg.analyse(dlg.infoCard.infos.ID, function(){
if (getValueDefault("pubd-autodownload",false)) { //自动开始
dlg.log("🅰️自动开始发送");
dlg.startdownload();
}
});
dlg.reloadSchemes();
};
return dlg;
}
//构建导入数据对话框
function buildDlgImportData() {
var dlg = new Dialog("导入数据", "pubd-import", "pubd-import");
var dl = dlg.content.appendChild(document.createElement("dl"));
var dt = dl.appendChild(document.createElement("dt"));
dt.innerHTML = "导入内容";
var dd = dl.appendChild(document.createElement("dd"));
dd.className = "pubd-import-textarea-bar";
var ipt = dd.appendChild(document.createElement("textarea"));
ipt.className = "pubd-import-textarea";
dlg.importTxt = ipt;
var dd = dl.appendChild(document.createElement("dd"));
var btn = dd.appendChild(document.createElement("input"));
btn.type = "button";
btn.className = "pubd-import-done";
btn.value = "导入";
//启动初始化
dlg.initialise = function(arg) {
ipt.value = "";
if (arg)
{
btn.onclick = function()
{//返回文本框的内容
arg.callback(ipt.value);
dlg.hide();
};
}else
{
btn.onclick = function()
{
alert("窗口异常启动,未提供回调函数");
};
}
};
return dlg;
}
//构建多画师下载管理对话框
function buildDlgMultiple() {
var dlg = new Dialog("多画师下载管理", "pubd-multiple", "pubd-multiple");
var dl = dlg.content.appendChild(document.createElement("dl"));
var dt = dl.appendChild(document.createElement("dt"));
var dd = dl.appendChild(document.createElement("dd"));
var frm = dd.appendChild(new Frame("导出Pivix账号关注", "pubd-frm-userlist"));
var dl_input_frm = frm.content.appendChild(document.createElement("dl"));
var dt = dl_input_frm.appendChild(document.createElement("dt"));
var dd = dl_input_frm.appendChild(document.createElement("dd"));
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-inputstar-public";
ipt.value = "导出公开关注";
ipt.onclick = function() {
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-inputstar-public";
ipt.value = "导出非公开关注";
ipt.onclick = function() {
};
/*
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-backup";
ipt.value = "备份列表JSON"
ipt.onclick = function() {
}
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-restore";
ipt.value = "导入备份"
ipt.onclick = function() {
}
*/
var dt = dl.appendChild(document.createElement("dt"));
dt.innerHTML = "选择收藏列表";
var dd = dl.appendChild(document.createElement("dd"));
var slt = dd.appendChild(new Select("pubd-staruserlists"));
slt.onchange = function() {
dlg.loadTheList(this.selectedIndex);
};
//每次脚本预加载的时候事先生成列表
slt.options.add(new Option('快速收藏',0));
//重新读取所有收藏列表
dlg.reloadStarList = function() {
while (slt.length>0)
{
const x = slt.options[0];
x.remove();
x = null;
}
slt.options.length = 0;
pubd.starUserlists.forEach((ulist,idx) => slt.options.add(new Option(ulist.title,idx)));
};
dlg.loadTheList = function(listIdx) {
const listArr = listIdx > 0 ? pubd.starUserlists[listIdx].exportArray() : pubd.fastStarList.exportArray();
const ulDom = dlg.ulDom;
ulDom.classList.add("display-none");
while (ulDom.childNodes.length)
{
const x = ulDom.childNodes[0];
if (x.nodeName == 'li')
{
const l = x.querySelector('label');
l.ipt.remove();
delete l.ipt;
l.card.dom.remove();
delete l.card.dom
delete l.card;
l.remove();
l = null;
}
x.remove();
x = null;
}
const fragment = document.createDocumentFragment();
console.log(listArr)
listArr.forEach(uid=>{ //添加每一个作者的信息
const uli = fragment.appendChild(document.createElement('li'));
uli.className = 'user-card-li';
uli.setAttribute('data-user-id',uid);
const lbl = uli.appendChild(new LabelInput(null,'user-card-lbl',`user-${uid}`,'checkbox',uid));
const card = lbl.card = new InfoCard({
"ID": uid,
"昵称": null,
"作品获取程度": null,
"数据更新时间": null,
});
lbl.appendChild(card.dom);
});
ulDom.appendChild(fragment);
ulDom.classList.remove("display-none");
};
dlg.userListDom = slt;
var dd = dl.appendChild(document.createElement("dd"));
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-new";
ipt.value = "新建";
ipt.onclick = function() {
var schemName = prompt("请输入方案名", "我的方案");
if (schemName)
{
var scheme = new DownScheme(schemName);
var length = dlg.schemes.push(scheme);
dlg.downSchemeDom.add(scheme.name, length - 1);
dlg.downSchemeDom.selectedIndex = length - 1;
dlg.loadScheme(scheme);
//dlg.reloadSchemes();
}
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-rename";
ipt.value = "重命名列表";
ipt.onclick = function() {
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-remove";
ipt.value = "删除";
ipt.onclick = function() {
if (dlg.downSchemeDom.options.length < 1) { alert("已经没有方案了"); return; }
if (dlg.downSchemeDom.selectedOptions.length < 1) { alert("没有选中方案"); return; }
var index = dlg.downSchemeDom.selectedIndex;
dlg.schemes.splice(index, 1);
dlg.downSchemeDom.remove(index);
var index = dlg.downSchemeDom.selectedIndex;
if (index < 0) dlg.reloadSchemes(); //没有选中的,重置
else dlg.loadScheme(dlg.schemes[index]);
};
var dd = dl.appendChild(document.createElement("dd"));
var frm = dd.appendChild(new Frame("当前列表", "pubd-frm-userlist"));
var dl_ul_frm = frm.content.appendChild(document.createElement("dl"));
var dt = dl_ul_frm.appendChild(document.createElement("dt"));
var dd = dl_ul_frm.appendChild(document.createElement("dd"));
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-this-add";
ipt.value = "添加画师ID";
ipt.onclick = function() {
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-this-remove";
ipt.value = "删除选中画师";
ipt.onclick = function() {
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-this-reset-getdata";
ipt.value = "重置数据获取状态";
ipt.onclick = function() {
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-this-reset-downloaded";
ipt.value = "重置下载状态";
ipt.onclick = function() {
};
var dt = dl_ul_frm.appendChild(document.createElement("dt"));
dt.innerHTML = "画师列表";
var ipt = dt.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-break";
ipt.value = "中断操作";
ipt.onclick = function() {
};
var dd = dl_ul_frm.appendChild(document.createElement("dd"));
var dl_ul = dd.appendChild(document.createElement("ul"));
dlg.ulDom = dl_ul;
dl_ul.className = "pubd-userlist-ul";
var dt = dl.appendChild(document.createElement("dt"));
var dd = dl.appendChild(document.createElement("dd"));
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-this-getdata";
ipt.value = "获取画师数据";
ipt.onclick = function() {
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-textdown";
ipt.value = "输出文本";
ipt.onclick = function() {
};
var ipt = dd.appendChild(document.createElement("input"));
ipt.type = "button";
ipt.className = "pubd-userlist-download";
ipt.value = "下载列表内画师作品";
ipt.onclick = function() {
};
//启动初始化
dlg.initialise = function(arg) {
dlg.loadTheList(0); //加载快速收藏列表
};
return dlg;
}
//作品循环递归输出
function sendToAria2_illust(aria2, termwiseType, illusts, userInfo, scheme, downP, callback) {
if (illusts.length < 1) //做完了
{
callback();
return;
}
if (pubd.downbreak)
{
GM_notification({text:"已中断向Aria2发送下载信息。但Aria2本身仍未停止下载已添加内容,请手动停止。", title:scriptName, image:scriptIcon});
pubd.downbreak = false;
return;
}
if (termwiseType == 0) //完全逐项
{
var illust = illusts.shift(); //读取首个作品
sendToAria2_Page(aria2, illust, 0, userInfo, scheme, downP, function() {
sendToAria2_illust(aria2, termwiseType, illusts, userInfo, scheme, downP, callback); //发送下一个作品
});
return; //不再继续执行
}else if (termwiseType == 1) //部分逐项(每作品合并)
{
var illust = illusts.shift(); //读取首个作品
var page_count = illust.page_count; //作品页数
if (illust.type == "ugoira" && illust.ugoira_metadata) //修改动图的页数
{
page_count = illust.ugoira_metadata.frames.length;
}
if (limitingFilenameExp.test(illust.filename)) //无权查看的文件
{
if (downP) downP.progress.set((downP.current += page_count) / downP.max); //直接加上一个作品所有页数
sendToAria2_illust(aria2, termwiseType, illusts, userInfo, scheme, downP, callback); //调用自身
return;
}
var aria2_params = [];
for (let page=0;page<page_count;page++)
{
if (returnLogicValue(scheme.downfilter, userInfo, illust, page)) {
//跳过此次下载
//console.info("符合下载过滤器定义,跳过下载:", illust);
continue;
} else {
var aria2_method = {'methodName':'aria2.addUri','params':[]};
var url = getIllustDownUrl(scheme, userInfo, illust, page);
aria2_method.params.push([url]); //添加下载链接
var options = {
"out": pathSafe(showMask(scheme.savepath, scheme.masklist, userInfo, illust, page), 'pathWithoutDriver'),
"referer": Referer,
"user-agent": UA,
};
if (scheme.savedir.length > 0) {
options.dir = pathSafe(showMask(scheme.savedir, scheme.masklist, userInfo, illust, page), 'path');
}
if (scheme.proxyurl.length > 0) {
options["all-proxy"] = scheme.proxyurl;
}
aria2_method.params.push(options);
aria2_params.push(aria2_method);
}
}
if (aria2_params.length>0)
{
aria2.system.multicall([aria2_params],function(res){
if (res === false) {
alert("发送到指定的Aria2失败,请检查到Aria2连接是否正常。");
return;
}
if (downP) downP.progress.set((downP.current += page_count) / downP.max); //直接加上一个作品所有页数
sendToAria2_illust(aria2, termwiseType, illusts, userInfo, scheme, downP, callback); //调用自身
});
}else
{ //这个作品全部跳过的时候
if (downP) downP.progress.set((downP.current += page_count) / downP.max); //直接加上一个作品所有页数
sendToAria2_illust(aria2, termwiseType, illusts, userInfo, scheme, downP, callback); //调用自身
}
return;
}else if(termwiseType == 2) //不逐项,每作者合并
{
var aria2_params = [];
for (var illustIndex = 0; illustIndex < illusts.length; illustIndex++)
{
var illust = illusts[illustIndex];
if (limitingFilenameExp.test(illust.filename)) continue; //无权查看的文件,直接继续
var page_count = illust.page_count; //作品页数
if (illust.type == "ugoira" && illust.ugoira_metadata) //修改动图的页数
{
page_count = illust.ugoira_metadata.frames.length;
}
for (let page=0;page<page_count;page++)
{
if (returnLogicValue(scheme.downfilter, userInfo, illust, page)) {
//跳过此次下载
//console.info("符合下载过滤器定义,跳过下载:", illust);
continue;
} else {
var aria2_method = {'methodName':'aria2.addUri','params':[]};
var url = getIllustDownUrl(scheme, userInfo, illust, page);
aria2_method.params.push([url]); //添加下载链接
var options = {
"out": pathSafe(showMask(scheme.savepath, scheme.masklist, userInfo, illust, page), 'pathWithoutDriver'),
"referer": Referer,
"user-agent": UA,
};
if (scheme.savedir.length > 0) {
options.dir = pathSafe(showMask(scheme.savedir, scheme.masklist, userInfo, illust, page), 'path');
}
if (scheme.proxyurl.length > 0) {
options["all-proxy"] = scheme.proxyurl;
}
aria2_method.params.push(options);
aria2_params.push(aria2_method);
}
}
}
if (aria2_params.length>0)
{
aria2.system.multicall([aria2_params],function(res){
if (res === false) {
alert("发送到指定的Aria2失败,请检查到Aria2连接是否正常。不排除数据过大,可考虑临时使用逐项或半逐项模式。");
var l= JSON.stringify(aria2_params).length/1024;
console.error("Aria2接受失败。数据量在未添加token的情况下有" + (
(l>1024)?
((l/1024)+"MB"):
(l+"KB")
),aria2_params);
return;
}
if (downP) downP.progress.set((downP.current = downP.max) / downP.max); //直接加上所有页数
sendToAria2_illust(aria2, termwiseType, [], userInfo, scheme, downP, callback); //调用自身
});
}else
{ //这个作品全部跳过的时候
if (downP) downP.progress.set((downP.current = downP.max) / downP.max); //直接加上所有页数
sendToAria2_illust(aria2, termwiseType, [], userInfo, scheme, downP, callback); //调用自身
}
return;
}
}
//作品每页循环递归输出
function sendToAria2_Page(aria2, illust, page, userInfo, scheme, downP, callback) {
if (pubd.downbreak) {
GM_notification({text:"已中断向Aria2发送下载信息。但Aria2本身仍未停止下载已添加内容,请手动停止。", title:scriptName, image:scriptIcon});
pubd.downbreak = false;
return;
}
var page_count = illust.page_count;
if (illust.type == "ugoira" && illust.ugoira_metadata) //动图的帧数当页数
{
page_count = illust.ugoira_metadata.frames.length;
}
if (limitingFilenameExp.test(illust.filename)) //无法查看的文件,直接把page加到顶
{
page = page_count;
downP.progress.set((downP.current += page_count) / downP.max); //直接加上所有页数
}
if (page >= page_count) //本作品页数已经完毕
{
callback();
return;
}
var url = getIllustDownUrl(scheme, userInfo, illust, page);
if (returnLogicValue(scheme.downfilter, userInfo, illust, page)) {
//跳过此次下载
downP.progress.set(++downP.current / downP.max); //设置进度
sendToAria2_Page(aria2, illust, ++page, userInfo, scheme, downP, callback); //递归调用自身
//console.info("符合下载过滤器定义,跳过下载:", illust);
} else {
var options = {
"out": pathSafe(showMask(scheme.savepath, scheme.masklist, userInfo, illust, page), 'pathWithoutDriver'),
"referer": Referer,
"user-agent": UA,
};
if (scheme.savedir.length > 0) {
options.dir = pathSafe(showMask(scheme.savedir, scheme.masklist, userInfo, illust, page), 'path');
}
if (scheme.proxyurl.length > 0) {
options["all-proxy"] = scheme.proxyurl;
}
aria2.addUri(url, options, function(res) {
if (res === false) {
alert("发送到指定的Aria2失败,请检查到Aria2连接是否正常。");
return;
}
downP.progress.set(++downP.current / downP.max); //设置进度
sendToAria2_Page(aria2, illust, ++page, userInfo, scheme, downP, callback); //递归调用自身
});
}
}
//返回掩码值
function showMask(oldStr, maskList, user, illust, page) {
//ES6原生模式,将来再启用
/*const cm = function(maskName) //customMask
{
const cusMask = maskList.find(mask=>mask.name == maskName);
if (cusMask) { //如果有对应的自定义掩码
if (returnLogicValue(cusMask.logic, user, illust, page)) //mask的逻辑判断
return eval("`" + cusMask.content +"`"); //递归
else
return "";
}
}
let newStr = eval("`" + oldStr +"`"); //需要解决旧有路径里\右斜杠的问题*/
//以下均为传统掩码
var newStr = oldStr;
//var pattern = "%{([^}]+)}"; //旧的,简单匹配
var regPattern = "%{(.*?(?:[^\\\\](?:\\\\{2})+|[^\\\\]))}"; //新的,支持转义符
var regResult = null;
/* jshint ignore:start */
//不断循环直到没有掩码
while ((regResult = new RegExp(regPattern).exec(newStr)) != null) {
var mskO = regResult[0], //包含括号的原始掩码
mskN = regResult[1]; //去掉掩码括号
if (mskN != undefined) {
//去掉转义符的掩码名
mskN = (mskN != undefined) ? mskN.replace(/\\{/ig, "{").replace(/\\}/ig, "}").replace(/\\\\/ig, "\\") : null;
//搜寻自定义掩码
var cusMask = maskList.find(mask=>mask.name == mskN);
if (cusMask) { //如果有对应的自定义掩码
try {
if (returnLogicValue(cusMask.logic, user, illust, page)) //mask的逻辑判断
newStr = newStr.replace(mskO, cusMask.content);
else
newStr = newStr.replace(mskO, "");
} catch (e) {
console.error(mskO + " 自定义掩码出现了异常情况", e);
}
} else { //普通掩码
try {
var evTemp = eval(mskN);
if (evTemp != undefined)
newStr = newStr.replace(mskO, evTemp.toString());
else
newStr = newStr.replace(mskO, "");
} catch (e) {
newStr = newStr.replace(mskO, "");
console.error(mskO + " 掩码出现了异常情况", e);
}
}
}
}
/* jshint ignore:end */
return newStr;
}
//返回逻辑值
function returnLogicValue(logic, user, illust, page) {
try {
if (logic.length == 0) return false;
/* jshint ignore:start */
const evTemp = Boolean(eval(logic));
/* jshint ignore:end */
return evTemp;
} catch (e) {
console.error("逻辑运算出现了异常情况,逻辑内容:","(" + logic + ")", e);
return false;
}
}
/**
* 去除不可用的字符,替换为可以使用的安全路径字符串
* @param {string} str 输入路径字符串
* @param {'basic' | 'path' | 'pathWithoutDriver' | 'filename' | 'fn'} type 去除不可用字符串的形式,是路径还是文件名
* @param {string} newChar 被替换为的字符
* @returns {string} 安全路径字符串
*/
function pathSafe(str = "", type = "path", newChar = "") { //去除Windows下无法作为文件名的字符,目前为了支持Linux暂不替换两种斜杠吧。
let nstr = str.toString(); //新字符
nstr = nstr.replace(/\u0000-\u001F\u007F-\u00A0/ig, ""); //一定去除所有的控制字符,已包含\r \n
switch(type) {
case "path": { //只替换路径中不能出现的字符,包括除了第一个盘符以外的其他任何
nstr = nstr.replace(/["<>\|\*\?]|(?<!^\w):/ig, newChar);
break;
}
case "pathWithoutDriver": { //只替换路径中不能出现的字符,包括除了第一个盘符以外的其他任何
nstr = nstr.replace(/["<>\|\*\?:]/ig, newChar);
break;
}
case "fn": case "filename": { //替换所有Windows名内不能出现的字符
nstr = nstr.replace(/["<>\|:\*\?\\/]/ig, newChar); //只替换路径中完全不能出现的特殊字符
break;
}
}
return nstr;
}
//主引导程序
function Main(touch) {
if (!mdev) { //不是开发模式时加载CSS资源
let css = GM_getResourceText("pubd-style");
if (css.includes('@-moz-document')) {
let cssStart = css.indexOf("{", css.indexOf('domain("www.pixiv.net")'))+1,
cssEnd = css.lastIndexOf("}");
css = css.substring(cssStart, cssEnd);
}
GM_addStyle(css);
}
//删除以前储存的账号密码
let cfgVer = GM_getValue("pubd-configversion");
if (cfgVer && cfgVer < pubd.configVersion)
{
GM_deleteValue("pubd-auth");
}
//载入设置
pubd.oAuth = new oAuth2(GM_getValue("pubd-oauth"));
pubd.downSchemes = NewDownSchemeArrayFromJson(getValueDefault("pubd-downschemes",[]));
//对下载方案的修改添加监听
GM_addValueChangeListener("pubd-downschemes", function(name, old_value, new_value, remote) {
pubd.downSchemes = NewDownSchemeArrayFromJson(new_value); //重新读取下载方案(可能被其他页面修改的)
});
//快速收藏列表的监听修改
//pubd.fastStarList = getValueDefault("pubd-faststar-list",[]);
pubd.fastStarList = new UsersStarList("快速收藏",getValueDefault("pubd-faststar-list",[]));
GM_addValueChangeListener("pubd-faststar-list", function(name, old_value, new_value, remote) {
pubd.fastStarList = null;
pubd.fastStarList = new UsersStarList("快速收藏",getValueDefault("pubd-faststar-list",[]));
if (mdev) console.log('收藏有变化',pubd.fastStarList.users);
checkStar();
//更改推荐列表里的收藏显示状态
refreshRecommendListState();
//将来还需要在更改收藏时,就自动刷新所有的其他推荐列表
//put my code
});
//登录信息的监听修改
GM_addValueChangeListener("pubd-oauth", function(name, old_value, new_value, remote) {
pubd.oAuth = new oAuth2(new_value);
});
//预先添加所有视窗,即便没有操作按钮也能通过菜单打开
let fragment = document.createDocumentFragment();
pubd.dialog.config = fragment.appendChild(buildDlgConfig());
pubd.dialog.login = fragment.appendChild(buildDlgLogin());
pubd.dialog.refresh_token = fragment.appendChild(buildDlgRefreshToken());
pubd.dialog.downthis = fragment.appendChild(buildDlgDownThis(thisPageUserid));
pubd.dialog.downillust = fragment.appendChild(buildDlgDownIllust(thisPageIllustid));
pubd.dialog.importdata = fragment.appendChild(buildDlgImportData());
pubd.dialog.multiple = fragment.appendChild(buildDlgMultiple());
let btnDlgInsertPlace = document.body; //视窗插入点,直接插入到body就行
btnDlgInsertPlace.appendChild(fragment);
//添加Tampermonkey扩展菜单内的入口
GM_registerMenuCommand("PUBD-选项", function(){
pubd.dialog.config.show(
(document.body.clientWidth - 400)/2,
window.scrollY+50
);
});
GM_registerMenuCommand("PUBD-下载该画师", function(){
pubd.dialog.downthis.show(
(document.body.clientWidth - 440)/2,
window.scrollY+100,
{id:getCurrentUserId()}
);
});
if (mdev)
GM_registerMenuCommand("PUBD-导入窗口测试", function(){
pubd.dialog.importdata.show(
(document.body.clientWidth - 370)/2,
window.scrollY+200,
{callback:function(txt){
const importArr = txt.split("\n");
const needAddArr = importArr.map(str=>{
let res = null;
if (
Boolean(res = new RegExp("^(\\d+)$","ig").exec(str)) ||
Boolean(res = new RegExp("member.+?\\?id=(\\d+)","ig").exec(str)) ||
Boolean(res = new RegExp("users/(\\d+)","ig").exec(str))
)
{
return parseInt(res[1],10);
}else
{
if (str.length>0)
console.log("未知的字符串",str);
return null;
}
}).filter(Boolean);
console.log(needAddArr);
if (needAddArr.length>0)
{
console.log(`新增了${needAddArr.length}个收藏`);
pubd.fastStarList.importArray(needAddArr);
GM_setValue("pubd-faststar-list",pubd.fastStarList.exportArray());
}
}}
);
});
//建立开始按钮
const btnStartBox = document.createElement("div");
btnStartBox.className = "pubd-btnStartInsertPlace";
pubd.start = btnStartBox.appendChild(buildbtnStart());
pubd.menu = btnStartBox.appendChild(buildbtnMenu());
//添加开始按钮,开始按钮内容直接固定为 btnStartBox
function insertStartBtn(btnStartInsertPlace)
{
if (btnStartInsertPlace == undefined)
{
console.error("PUBD:未找到开始按钮插入点。");
return false;
}else
{
if (/^\/artworks\//i.test(location.pathname)) //如果是作品页面,显示下载当前作品按钮
{
pubd.menu.downillust.classList.remove("display-none");
downIllustMenuId = GM_registerMenuCommand("PUBD-下载该作品", function(){
pubd.dialog.downillust.show(
(document.body.clientWidth - 500)/2,
window.scrollY+150,
{id:getQueryString('illust_id',
pubd.touch ?
mainDiv.querySelector('.illust-details-content .work-stats>a') : //手机版
mainDiv.querySelector(artWorkStarCssPath) //新版Vue结构
)}
);
});
}else
{
pubd.menu.downillust.classList.add("display-none");
GM_unregisterMenuCommand(downIllustMenuId);
}
checkStar(); //检查是否有收藏
//插入开始操作按钮
btnStartInsertPlace.appendChild(btnStartBox);
console.log("PUBD:网页发生变动,已重新呈现开始按钮。 %o", btnStartBox);
return true;
}
}
/*
手机版网页的root
#spa-contents 会被删掉重新添加,所以只能用更上一层
*/
const vueRoot = document.querySelector("#root"); //vue框架的root div
const wrapper = document.querySelector("#wrapper"); //仍然少量存在的老板页面
const touchRoot = wrapper ? wrapper.querySelector("#contents") : null;
if (window.MutationObserver && (vueRoot || touch)) //如果支持MutationObserver,且是vue框架
{
let reInsertStart = true; //是否需要重新插入开始按钮
let changeIllustUser = new MutationObserver(function(mutationsList, observer) {
if (mdev) console.log("作者链接 href 改变了",mutationsList);
checkStar();
});
let observerLoop = new MutationObserver(function(mutationsList, observer) {
const removedNodes = mutationsList.flatMap(mutation=>[...mutation.removedNodes]);
//const addNodes = mutationsList.flatMap(mutation=>[...mutation.addNodes]);
//当在P站首页的时候,不需要生效
if (location.pathname.substring(1).length == 0) {
console.log("PUBD:P站首页不需要执行。");
return;
}
//如果被删除的节点里有我们的开始按钮,就重新插入;或者搜索列表被删除
if (removedNodes.some(node=>node.contains(btnStartBox)))
{
console.log('已经添加的开始按钮因为页面改动被删除了');
mainDiv = null;
reInsertStart = true;
}
//搜索新的主div并插入开始按钮
if (reInsertStart)
{
for (const node of (touch ? touchRoot : subRoot).children) {
if (recommendList = node.querySelector(searchListCssPath)) {//如果是搜索结果界面而非用户/作品界面
mainDiv = node; //重新选择主div
if (mdev) console.debug("mainDiv 为 %o,搜索列表为 %o,", mainDiv, recommendList);
reInsertStart = false;
break;
} else {
const foundStartBtn = mainDivSearchCssSelector.some(entry => {
let btnStartInsertPlace,
cssS = entry.selectors,
fallcack = entry.fallcack;
try {
btnStartInsertPlace = node.querySelector(cssS);
} catch (e) {
if (mdev) console.error(`${cssS} 获取开始按钮容器异常`, e);
if (typeof fallcack === 'function') {
if (mdev) console.debug('尝试使用配置的 fallcack 重新获取');
btnStartInsertPlace = fallcack(node);
} else {
if (mdev) console.debug('未配置 fallcack 无法获取');
}
}
if(btnStartInsertPlace) {
mainDiv = node; //重新选择主div
if (mdev) console.debug("mainDiv 为 %o ,始按钮插入点条目为 %o",mainDiv,entry);
reInsertStart = !insertStartBtn(btnStartInsertPlace); //插入开始按钮
const userHeadLink = mainDiv.querySelector(artWorkUserHeadCssPath);
if (userHeadLink) //如果是作品页面
{
changeIllustUser.observe(userHeadLink, {attributeFilter:["href"]});
}
return true;
}else return false;
})
if (foundStartBtn) break; //如果插入了开始按钮,就退出循环
}
}
}
//作品页面显示推荐的部分
let otherWorks;
if (!recommendList && (otherWorks = subRoot.querySelector(".gtm-illust-recommend-zone") || subRoot.querySelector("section h3")?.parentElement?.parentElement?.parentElement))
{ //已发现推荐列表大部位
if (recommendList = otherWorks.querySelector(":scope ul"))
{
if (mdev) console.log("发现推荐列表 %o", recommendList);
refreshRecommendListState();
}
}
if (recommendList)
{
//如果有新增,就重新刷新已收藏选中状态
if (mutationsList.some(mutation=>mutation.target==recommendList && mutation.addedNodes.length))
refreshRecommendListState();
if (removedNodes.some(node=>node.contains(recommendList)))
{ //如果被删除的节点里有推荐列表,重新标空
if (mdev) console.log('推荐列表被删除了');
recommendList = null;
}
}
});
//只执行一次的,插找P站新的根节点的位置
let observerFindSubRoot = new MutationObserver(function(mutationsList, observer) {
for (const mutation of mutationsList) {
for (const node of mutation.addedNodes) {
if(!node.id){ //如果 root 下新增没有 id 的 node,就开始处理
//一直循环到下面有多个 node 时,当作子root,否则继续往下。
subRoot = node;
while (subRoot.childNodes.length == 1) {
subRoot = subRoot.childNodes[0];
}
if (mdev) console.log("subRoot 为 %o", subRoot);
observer.disconnect();
observerLoop.observe(subRoot, {childList:true, subtree:true});
return;
}else continue;
}
}
});
if (vueRoot) {
observerFindSubRoot.observe(vueRoot, {childList:true, subtree:false});
} else {
observerLoop.observe(touchRoot, {childList:true, subtree:true});
}
}else if(vueRoot == undefined)
{
if (wrapper) //仍然少量存在的老板页面
{
console.log('PUBD:你访问的是仍然少量存在的老板页面。');
insertStartBtn(document.querySelector("._user-profile-card")) || //老版用户资料页
insertStartBtn(document.querySelector(".ui-layout-west aside")) || //老版作品页
insertStartBtn(document.querySelector(".introduction")) //老版未登录页面
;
}else
{
console.log('PUBD:未找到 root div,可能P站又改版了,程序得修改。');
}
}else
{
alert('PUBD:您的浏览器不支持 MutationObserver,请使用最新浏览器。');
}
}
Main(pubd.touch); //开始主程序
})();