// ==UserScript==
// @name mCollection资源捕获
// @namespace mCollection
// @homepage https://gf.qytechs.cn/zh-CN/scripts/469129-mcollection%E8%B5%84%E6%BA%90%E6%8D%95%E8%8E%B7
// @match *://*/*
// @icon 
// @grant unsafeWindow
// @grant GM_addElement
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @version 0.2
// @author hunmer
// @license MIT
// @description 2023年7月8日 21点27分
// ==/UserScript==
// ['GM_download'].forEach(method => unsafeWindow[method] = window[method])
var g_app = unsafeWindow._g_app = {
resources: [],
cdn: 'https://neysummer2000.fun/',
// cdn: 'http://127.0.0.1:8080/',
init() {
this.timer = setInterval(() => {
let icon = this.icon = document.querySelector('#mc_float_icon')
if(icon !== null) return
GM_addStyle(`
#mc_float_icon:hover {
right: 0px !important;
}
`)
icon = this.icon = document.createElement('div')
icon.id = 'mc_float_icon'
icon.style.cssText = `
position: fixed;
right: -25px;
bottom: 50px;
width: 50px;
cursor: pointer;
height: 50px;
transition: right 0.3s ease-in-out;
`
icon.innerHTML = `
<div id="draggable-header" style="width: 100%; height: 100%;background-image: url(${GM_info.script.icon});border-radius: 20px;background-size: contain;position: relative;">
<a id="_badge" style="
position: absolute;
left: 0;
bottom: 0;
background-color: red;
border-radius: 10px;
color: white;
font-size: 10px;
min-width: 15px;
padding: 1px;
height: 15px;
text-align: center;
line-height: 11px;
">${this.resources.length}<a>
</div>
`
let lastClick
icon.onmouseup = () => Date.now() - lastClick < 150 && this.toggleShow()
icon.onmousedown = () => lastClick = Date.now()
document.body.appendChild(icon)
initDraggableEles(icon)
}, 1000)
this.observer_start()
window.onload = () => {
// this.show()
}
},
// 刷新资源展示列表
resources_refresh() {
let html = this.resources.map(item => this.buildImageContainer(item)).join('')
this.$('#list_resources').html(html)
},
// 生成图片展示div
buildImageContainer(item) {
let { type, url, title, size, width, height } = item
title ||= this.window.getFileName(url).split('?')[0]
let cover = type == 'img' ? url : this.cdn + 'public/files.png'
return `
<div class="resource_item datalist-item" style="height: 100px;display: grid;" data-action="img_select" data-size="${size}" data-width="${width}" data-height="${height}" data-type="${type}" data-url="${url}">
<img src="${cover}" alt="${cover}" class="mx-auto" height=80 data-hover="img_preview" data-out="img_unpreview" data-hoverTime=1000>
<span class="text-nowrap text-center" style="width: 100px;" title="${title}">${title}</span>
</div>`
},
resources_find(find_url, methid = 'find'){
return this.resources[methid](({ url }) => url == find_url)
},
// 添加捕获资源
resources_add(item) {
let find = this.resources_find(item.url)
if (find) {
item = Object.assign(find, item)
} else {
let cnt = this.resources.push(item)
if (this.inst.tabs) {
this.inst.tabs.getButton('resources').find('span').text('资源(' + cnt + ')')
}
if(this.icon) this.icon.querySelector('#_badge').innerHTML = cnt
}
// console.log(item)
if (this.isShowing() && this.$) {
let div = this.getImageContainer(item.url)
let el = this.$(this.buildImageContainer(item))
this.applyFilter(el) // 判断是否可通过过滤器
if (!div.length) return el.appendTo('#list_resources')
div.replaceWith(el)
}
},
// 移除资源
resources_remove(url){
this.getElement(url).remove()
let index = this.resources_find(url, 'findIndex')
if(index != -1){
this.resources.splice(index, 1)
return true
}
},
getImageContainer(url) {
return this.getImageElement(url).parents('.datalist-item')
},
getElement(url) {
return this.$(`.datalist-item[data-url="${url}"]`)
},
getImageElement(url) {
return this.$(`.datalist-item img[src="${url}"]`)
},
isShowing() {
return this?.iframe?.style?.display != 'none'
},
queue: [],
loadResources({ node, url, type, size }) {
const self = this
if (this.queue.includes(url)) return // 禁止重复请求,因为new资源对象会触发网络请求而造成死循环
this.queue.push(url)
let onLoad = function () {
let url = this.src
self.resources_add({ url, size, type, title: this.alt ?? this.title, width: this.naturalWidth, height: this.naturalHeight })
this.remove()
}
if (node) return node.addEventListener('load', onLoad)
let obj = type == 'img' ? new Image() : (type == 'video' ? new Video() : new Audio())
obj.src = url
obj.onload = onLoad
},
// 开始监听
observer_start() {
this.log('开启网络监听...')
const request_observer = new PerformanceObserver((list) => {
const entries = list.getEntries();
for (const entry of entries) {
let { entryType, initiatorType: type, name: url, transferSize: size } = entry
if (entryType === 'resource' && ['img', 'video', 'audio'].includes(type)) {
this.loadResources({ url, type, size })
// this.resources_add({ url, size, type }) // 更新资源请求的大小以及一些协议头属性
}
}
});
request_observer.observe({ entryTypes: ['resource'] });
// BUG 无法监听 insertAdjacentHTML
const dom_observer = new MutationObserver((mutationsList) => {
for (const mutation of mutationsList) {
if (mutation.type === 'childList') {
for (const node of mutation.addedNodes) {
let tagName = (node.tagName || '').toLowerCase()
let url = ''
switch (tagName) {
case 'video':
case 'audio':
url = node.src || node.querySelector('source').src
break;
case 'img':
url = node.dataset.src || node.src
break;
}
if (url.length) this.loadResources({ node })
}
}
}
});
dom_observer.observe(document.body, { childList: true, subtree: true });
},
inst: {},
show() {
const self = this
let { iframe, win, ifrmaeDoc } = createIframe({
css: `
border: 0;
right: 10px;
top: 10px;
position: fixed;
z-index: 99999;
resize: both;
max-height: 100vh;
max-width: 100vw;
width: 500px;
height: 700px;
`,
html: `
<main style="height: 100vh; width: 100vw;">
<div id='draggable-header' class="position-fixed top-0 left-0 w-full d-flex align-items-center" style="height: 30px;">
<div id="header_start" class="me-auto"></div>
<div id="header_end" class="ms-auto">
<a class="btn btn-sm btn-outline-primary" data-action="min"><i class="ti ti-minus"></i></a>
<a class="btn btn-sm btn-outline-warning" data-action="max"><i class="ti ti-square"></i></a>
<a class="btn btn-sm btn-outline-danger" data-action="close"><i class="ti ti-x"></i></a>
</div>
</div>
<div id="main_tabs"></div>
</main>
`
})
Object.assign(this, { iframe, window: win, ifrmaeDoc })
ifrmaeDoc.location.baseURL = this.cdn // 自定义远程脚本加载源头
loadRes(ifrmaeDoc, ['jquery.min.js', 'tabler.min.js', 'tabler.min.css', 'tabler-icons.min.css', 'index.css', 'until.js', 'action.js', 'preload.js', 'basedata.js', 'menu.js', 'tabler.helper.js', 'style.js', 'modal.js', 'input.js', 'toast.js', 'form.js', 'dropdown.js', 'plugins.js', 'tom-select.complete.min.js', 'tom-select.min.css', 'ping-pong.js', 'tabs1.js', 'selection.js', 'floatDiv.js'].map(url => {
if (!url.startsWith('http')) {
if (url.endsWith('.js')) {
url = 'public/js/' + url
} else
if (url.endsWith('.css')) {
url = 'public/css/' + url
}
}
return this.cdn + url
}), () => {
this.$ = win.jQuery;
this.$(() => ['initUntils', 'initSelection', 'initMenu', 'initActions', 'initTabs', 'initPlugins', 'initStyle'].map(method => self[method].call(self, win)))
})
},
unselectAll() {
this.getSelectedImage().removeClass('img_selected')
this.window.g_selection.unset('selection_img')
},
// 更新过滤器
applyFilter(imgs) {
let v = this.window.g_form.getVals('resources_filter')
v.size *= 1024
// setConfig
let ret = { hide: 0, show: 0 }
let update = imgs == undefined
imgs = update ? this.getAllImages() : [...imgs]
for (let el of imgs) {
let src = el.querySelector('img').src
let { type, size, width, height } = el.dataset
let hide = (
(v.match != '' && src.match(v.match) == null) ||
(v.exts != '' && !v.exts.split(',').some(ext => ext == 'all' || src.endsWith(ext))) ||
(!v.types.includes('all') && !v.types.includes(type)) ||
(v.size > size * 1 || v.width > width * 1 || v.height > height * 1) ||
(v.ratio > 0 && v.ratio != '' && !v.ratio.split(',').some(i => i == 'all' || Math.abs(v.ratio - width / height) <= 0.15))
)
el.classList.toggle('hide1', hide)
ret[hide ? 'hide' : 'show']++
}
update && this.window.getEle('showHidden').html(`已隐藏(${ret.hide})`)
return ret
},
// 获取所有图片
getAllImages(selector = '') {
return this.$('.resource_item' + selector)
},
// 获取选中的图片
getSelectedImage(cb) {
let imgs = this.$('.resource_item.img_selected')
if (!cb) return imgs
let ret = []
imgs.each((i, el) => {
let val = cb(el, i)
if (val !== false) ret.push(val)
})
return ret
},
// 更新选中信息
updateSelected() {
let cnt = this.getSelectedImage().length
let hide = cnt == 0
this.window.getEle('mc_send').toggleClass('hide', hide).html('添加【' + cnt + '】')
},
// 设置预览图片
setPreviewImage(src) {
const getDiv = () => document.querySelector('#mc_preview_img')
let el = getDiv()
let remove = !(src?.length)
if (!el) {
if (remove) return
document.body.insertAdjacentHTML('beforeend', `
<div id="mc_preview_img" style="
position: fixed; z-index: 99999;
bottom: 0; left: 0;
width: calc(100vw);
height: calc(100vh);
pointer-events: none;
background-color: rgba(0, 0, 0, .5)"
>
<div style="position: relative;text-align: center;height: 100%;">
<img style="height: 100%;" >
</div>
</div>`)
el = getDiv()
} else
if (remove) {
return el.remove()
}
el.querySelector('img').src = getSourceImage(src)
},
initSelection({ g_selection }) {
g_selection.register({
name: 'selection_img',
dbclickUnset: false,
container: '.datalist-items',
selector: '.datalist-item',
selectedClass: 'img_selected',
addSelect: true,
multiSelect: true,
callback: selected => this.updateSelected(),
onUnset: clear => clear && this.updateSelected()
})
},
cache_log: '',
log(text) {
text += "\n"
if(!this.window){
this.cache_log += text
}else{
text = `【${this.window.formatDate()}】` + text
this.ifrmaeDoc.querySelector('#textarea_log').value += text
}
},
log_clear() {
this.cache_log = ''
this.ifrmaeDoc.querySelector('#textarea_log').value = ''
},
initMenu({ g_menu }) {
g_menu.registerMenu({
name: 'datalist_item',
selector: '.datalist-item',
dataKey: 'data-md5',
items: [{
text: '设置封面',
icon: 'photo',
action: 'item_cover'
}],
async onShow() {
// getEle('item_trash').toggleClass('hide', trashed)
},
onHide() {
// g_preview.unpreview();
}
});
},
get_downlist(id){
return this.downlist[id]
},
add_downlist(id){
let now = Date.now()
id ??= now
if(this.get_downlist(id)) return false
return this.downlist[id] = {
start: now,
cnt: 0
}
},
remove_downlist(id){
delete this.downlist[id]
},
toggleShow(show){
if(!this.iframe) return this.show(true)
if(show === undefined) show = !this.isShowing()
this.iframe.style.display = show ? 'unset' : 'none'
},
downlist: {},
initActions(win) {
win.g_action.registerAction({
min: () => this.toggleShow(false),
show: () => this.toggleShow(true),
close: () => {
if(confirm('确定要退出吗?')){
this.iframe.remove()
this.icon.remove() & clearInterval(this.timer)
}
},
max: () => {
},
download: () => {
let info = this.add_downlist('download')
if(info === false) return this.log('请等待下载队列完成', 'danger')
this.loadScripts('filesaver', () => {
let list = this.getSelectedImage(el => () => new Promise(reslove => {
const next = args => badge.html(`[下载中] ${++info.cnt}/${info.max}`) & reslove(args)
_request(getSourceImage(el.querySelector('img').src), {
onload: ({finalUrl, response}) => next([finalUrl, response]),
onerror: next
})
}))
let max = list.length
if(!max) return
this.log('开始下载')
info.max = max
let badge = win.insertEl({ tag: 'span', text: '初始化...', props: { id: 'badge_download', class: 'badge bg-primary ms-1 me-1', 'data-action': 'stop_download' } }, { target: win.$('#header_start'), method: 'prependTo' })
awaitPromises(list).then(ret => {
let done = () => badge.remove() & this.remove_downlist('download')
let items = ret.filter(item => Array.isArray(item))
if(!items.length) return this.log('没有合法的下载内容!', 'danger') & done()
badge.html('打包中...')
let zip = new JSZip();
items.forEach(([url, blob]) => {
// TODO 文件名称
zip.file(win.getFileName(url, true), blob);
})
zip.generateAsync({type: "blob"}).then(content => {
saveAs(content, "images.zip") & done()
})
})
})
},
log_clear: () => this.log_clear(),
unsetAll: () => this.unselectAll(),
selectAll: () => {
this.getAllImages().toggleClass('img_selected', this.getSelectedImage().length == 0)
this.updateSelected()
},
reveSelect: () => {
this.getAllImages().toggleClass('img_selected')
this.updateSelected()
},
removeSelect: () => {
this.getSelectedImage().each((i, el) => this.resources_remove(el.dataset.url))
this.updateSelected()
},
showHidden: () => {
this.getAllImages('.hide1')
},
img_preview: dom => this.setPreviewImage(dom.src),
img_unpreview: () => this.setPreviewImage(),
mc_send: dom => {
let btn = this.window.getEle('mc_send').addClass('btn-loading')
this.window.arrayQueue(this.getSelectedImage(el => {
let img = el.querySelector('img')
let url = img.src
let website = url
let name = img.title || win.getFileName(url, false)
let annotation = 'annotation'
return {url, website, name, annotation}
}), (item, i, max) => {
return new Promise(reslove => {
getImageBase64(item.url, getSourceImage(item.url)).then(imgData => {
btn.removeClass('btn-loading').text(`${i} / ${max}`)
reslove({...item, path: imgData})
})
})
}, queue => {
queue && Promise.all(queue).then(items => {
btn.text('导入中')
g_api.addFromPaths({ items }).then(ret => {
btn.text('导入')
ret && this.window.toast(JSON.stringify(ret, null, 2))
})
})
})
},
mc_reset: () => {
confirm('确定重置吗?').then(() => {
this.unselectAll()
})
},
// 设置过滤器
setFilter: (dom, action) => {
let { value } = dom
// TODO 一次更新所有过滤器,而不是单独更新
switch (action[1]) {
case 'size':
this.$(dom).prevUntil('.range_lable').find('.range_lable').text(this.window.renderSize(value * 1024))
break
case 'types':
case 'ratio':
let all = action[1] == 'types' ? 'all' : 0
this.$(dom).parents('.form_input').find('input[type="checkbox"]').each((i, el) => {
if (value == all) {
if (el.value != all) el.checked = false
} else
if (el.value == all) el.checked = false
})
break;
}
win.g_pp.setTimeout('apply_filter', () => this.applyFilter(), 200)
}
})
},
initTabs(win) {
const self = this
let tabs = this.inst.tabs = win.g_tabs.register({
name: 'main_tabs',
container: '#main_tabs',
class: 'show-icons',
cardBody: 'p-0',
moreItems: [],
list: [{
id: 'resources',
icon: 'activity-heartbeat',
title: '捕获',
html: `
<div class="position-relative" style="height: calc(100vh - 63px)">
<div id="list_resources" class="datalist-items w-full d-flex flex-wrap justify-content-center overflow-y-scroll" style="padding-bottom: 50px;height: calc(100vh - 280px);grid-gap: 20px;" onscroll="clearEventBubble(event)"></div>
<div id="resources_filter" style="max-height: 280px;" class=" position-absolute bottom-0 left-0 w-full overflow-y-auto">
</div>
</div>
`
},
{
id: 'import',
icon: 'database-import',
title: '导入',
html: `计划中...`
},
{
id: 'preset',
icon: 'book',
title: '预设',
html: `计划中...`
},
{
id: 'setting',
icon: 'settings',
title: '设置',
html: `计划中...`
},
{
id: 'log',
icon: 'list',
title: '日志',
html: `
<textarea id="textarea_log" class="form-control form-control-sm" readOnly rows=20 style="height: calc(100vh - 150px)">${self.cache_log}</textarea>
<div class="d-flex w-full mt-2" style="height: 50px;">
<div class="ms-auto">
<button class="btn btn-primary" data-action="log_clear">清空</button>
</div>
</div>
`
}],
event_init() {
this.setActive('resources')
},
event_shown({ tab }) {
switch (tab) {
case 'resources':
return self.initResourcesTab(win)
case 'log':
self.cache_log = ''
return
}
}
}).refresh()
},
initResourcesTab({ g_form, g_tabler }) {
const self = this
if (self.resources_inited) return
self.resources_inited = true
self.resources_refresh()
// tom-select
const toTomList = list => list.map(item => {
let [value, text] = Array.isArray(item) ? item : [item]
text ??= value
return { value, text }
})
const onInit = function () {
const removeValue = (search, val) => {
let arr = (val ?? this.getValue()).split(',')
let index = arr.indexOf(search)
if (index != -1) {
arr.splice(index, 1)
this.setValue(arr)
}
}
let last
this.on('item_select', el => removeValue(el.dataset.value));
this.on('change', val => {
if (val == last) return
last = val
if (val == '') return this.setValue(['all'])
let arr = val.split(',')
if (arr.pop() == 'all') {
this.setValue(['all'])
} else {
removeValue('all', val)
}
});
}
g_form.build('resources_filter', {
class: 'p-0 m-0 pb-2 h-full align-content-center align-items-center',
element_class: 'text-center align-self-center',
element_bodyClass: 'mt-0 mb-0 p',
elements: {
types: {
class: 'col-4',
title: '',
type: 'checkbox_list',
list: { all: '全部', img: '图片', video: '视频', audio: '音频' },
value: 'all',
props: 'data-change="setFilter,types"'
},
ratio: {
class: 'col-4',
title: '',
type: 'tom_select',
size: 'sm',
list: toTomList([['all', '所有尺寸'], [0.66, '2:3'], [0.75, '3:4'], [1, '1:1'], [1.77, '16:9']]),
value: ['all'],
onInit,
props: 'data-change="setFilter,ratio"'
},
exts: {
class: 'col-4',
title: '',
size: 'sm',
type: 'tom_select',
list: toTomList([['all', '所有格式'], 'jpg', 'png', 'webp', 'gif', 'svg', 'mp4', 'mp3', 'wav', 'webm']),
value: ['all'],
onInit,
props: 'data-change="setFilter,types"'
},
match: {
class: 'col-12 mb-1',
title: '',
rows: 2,
placeHolder: '网址过滤',
type: 'textarea',
size: 'sm',
props: 'data-input="setFilter,match"'
},
width: {
class: 'col-4',
title: '宽',
type: 'range',
opts: { min: 0, max: 4000, step: 1, format: '%spx' },
value: 0,
props: 'data-input="setFilter,width"'
},
height: {
class: 'col-4',
title: '高',
type: 'range',
opts: { min: 0, max: 4000, step: 1, format: '%spx' },
value: 0,
props: 'data-input="setFilter,height"'
},
size: {
class: 'col-4',
title: '大小',
type: 'range',
opts: { min: 0, max: 1024 * 4, step: 1 }, // TODO 变更选择范围选项
value: 0,
props: 'data-input="setFilter,size"'
},
actions: {
class: 'col-12',
type: 'html',
bodyClass: 'mt-2',
value: `
<div class="row align-items-center">
<div class="col-3">
${g_tabler.build_select({
list: ['计划中'],
value: '',
class: 'form-select-sm',
})}
${g_tabler.build_select({
list: ['计划中'],
value: '',
class: 'form-select-sm',
})}
</div>
<div class="col-auto">
<small>
<a href='javascript: void(0)' data-action="selectAll">全选</a>
<a href='javascript: void(0)' data-action="reveSelect">反选</a>
<a href='javascript: void(0)' data-action="showHidden">已隐藏</a>
<a href='javascript: void(0)' data-action="unsetAll">取消选中</a>
<a href='javascript: void(0)' data-action="removeSelect">移除选中</a>
</small>
</div>
<div class="col-12 text-end d-flex justify-content-end" style="gap: 10px;">
<div class="dropdown">
<a class="btn btn-sm dropdown-toggle" data-bs-toggle="dropdown">...</a>
<div class="dropdown-menu">
<a data-action="download" class="dropdown-item"><i class="ti ti-download me-1"></i>下载</a>
<a class="dropdown-item">...</a>
</div>
</div>
<button class="btn btn-sm btn-primary" data-action="mc_send">导入</button>
<button class="btn btn-sm btn-danger" data-action="mc_reset">重置</button>
</div>
</div>
`
},
},
target: self.ifrmaeDoc.querySelector('#resources_filter')
})
},
initStyle({ g_style }) {
g_style.addStyle('image', `
.img_selected img {
border: 4px solid #206bc4;
}
`)
},
initPlugins(win) {
let { g_plugin } = win
win.assignInstance(g_plugin, {
homepage: 'https://github.com/hunmer/mCollection/issues',
init: () => g_plugin.initPlugins()
})
},
loadScripts(name, cb){
let urls
if(name == 'filesaver'){
urls = ['https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js', 'https://cdnjs.cloudflare.com/ajax/libs/jszip/3.10.1/jszip.min.js']
}else return
loadRes(document, urls, cb)
},
initUntils(win) {
// iframe不支持使用eval函数的替代方案
Object.assign(win, {
getObjVal(obj, key, def) {
let val = obj
key.split('.').some(k => {
if (typeof (val[k]) == 'undefined') return true
val = val[k]
})
return val ?? def
},
setObjVal(obj, key, value) {
let last
let val = obj
let keys = key.split('.')
if (keys.every(k => {
if (typeof (val[k]) != 'undefined') {
last = val
val = val[k]
return true
}
})) {
last[keys.pop()] = value
return true
}
}
})
},
}
g_app.init()
var g_api = {
api: 'http://127.0.0.1:41597/',
fetch(url, opts) {
return GM_xmlhttpRequest({
url, ...opts
})
},
addFromPaths(data) {
let items = data.items
let all = items.length
delete data.items
if(all > 50) g_app.inst.tabs.setActive('log')
g_app.log(`准备发送数据给mCollection...(${ items.length})`) & console.log(data)
return new Promise(reslove => {
const next = () => {
let list = items.splice(0, 10)
let len = list.length
if(!len) return reslove({msg: 'OK'})
g_app.log(`发送中...(${len} / ${items.length})`)
this.fetch(this.api + "api/item/addFromPaths", {
method: 'POST',
responseType: 'JSON',
headers: {
'Content-Type': 'application/json'
// 'Content-Type': 'application/x-www-form-urlencoded',
},
onerror: () => alert('导入失败,请确保mCollection在后台运行!') & reslove(),
onload: ({status, statusText, responseText}) => {
if (status == 200 && statusText == 'OK') {
g_app.log(`收到mCollection回复...(${responseText}})`)
let ret = JSON.parse(responseText)
// reslove(ret)
console.log(ret)
next()
}
},
data: JSON.stringify({...data, items: list})
})
}
next()
})
}
}
function getImageBase64(url, source ) {
// TODO 进度显示
let target = source || url
return new Promise(reslove => {
if (target.startsWith('data:image')) return reslove(target)
_request(target, {
// onprogress: progress =>
onload: ({response}) => {
let img = new Image();
img.src = URL.createObjectURL(response);
img.onload = () => {
let canvas = document.createElement('canvas');
canvas.width = img.width;
canvas.height = img.height;
canvas.getContext('2d').drawImage(img, 0, 0);
reslove(canvas.toDataURL());
}
},
onerror: reason => {
console.error({msg: '获取图片失败!', url, source, reason})
if(source != url) return getImageBase64(url).then(reslove) // 尝试获取小图片
}
})
})
}
function _request(opts, callbacks = {}){
let {onerror, onload, onstatechange} = callbacks
if(typeof(opts) != 'object') opts = {url: opts}
return GM_xmlhttpRequest(Object.assign({
timeout: 1000 * 10,
responseType: 'blob',
anonymous: true,
onprogress({ loaded, total }) {
callbacks.onprogress && callbacks.onprogress(parseInt(loaded / total * 100))
},
onload(...args) {
onload && onload.apply(this, args)
},
onreadystatechange({readyState, status}) {
if(readyState == 4 && status != 200){
onerror && onerror('error code ' + status)
}
onstatechange && onstatechange.apply(this, args)
},
// 下面的好像不触发...
onerror: () => onerror && onerror('error'),
ontimeout: () => onerror && onerror('timeout'),
onabort: () => onerror && onerror('abort'),
}, opts))
}
function createIframe(opts) {
let iframe = this.iframe = document.createElement('iframe');
iframe.src = 'about:blank';
iframe.sandbox = 'allow-scripts allow-same-origin allow-modals';
iframe.style.cssText = opts.css || ''
document.body.appendChild(iframe);
let { contentWindow: win, contentDocument } = iframe
let ifrmaeDoc = contentDocument || win.document;
ifrmaeDoc.body.innerHTML = opts.html
initDraggableEles(ifrmaeDoc, iframe)
return { iframe, win, ifrmaeDoc }
}
function initDraggableEles(container, parentEle){
parentEle ??= container
let isDragging = false;
let lastX, lastY;
let header = container.querySelector('#draggable-header')
header.addEventListener('mousedown', function ({ offsetX, offsetY }) {
lastX = offsetX;
lastY = offsetY;
isDragging = true;
})
header.addEventListener('mouseenter', () => header.style.cursor = 'move')
header.addEventListener('mouseleave', () => header.style.cursor = 'none')
container.addEventListener('mousemove', event => {
if (isDragging) {
let { left, top, width, height } = parentEle != container ? parentEle.getBoundingClientRect() : {left: 0, top: 0, width: 0, height: 0}
let x = Math.min(Math.max(0, left + event.clientX - lastX), unsafeWindow.innerWidth - width);
let y = Math.min(Math.max(0, top + event.clientY - lastY), unsafeWindow.innerHeight - height);
parentEle.style.left = `${x}px`;
parentEle.style.top = `${y}px`;
event.preventDefault()
}
})
container.addEventListener('mouseup', () => isDragging = false)
container.addEventListener('mouseleave', () => isDragging = false)
}
function getSourceImage(url) {
let args = url.split('/')
if (url.includes('i.pinimg.com')) { // pinterest
args[3] = 'originals'
}else
if(url.includes('.sinaimg.cn')){ // weibo
// https://wx1.sinaimg.cn/orj360/006YezONly1hfn358xn4lj30jg0jgwgp.jpg
// https://wx1.sinaimg.cn/large/006YezONly1hfn358xn4lj30jg0jgwgp.jpg
args[3] = 'large'
}
return args.join('/')
}
const _loadedScripts = []
function loadRes(doc, files, callback, cache = true) {
files = [...files]
const load = url => {
return new Promise(reslove => {
if(_loadedScripts.includes(url)) reslove()
_request({url, responseType: undefined}, {
onload: ({ responseText }) => {
_loadedScripts.push(url)
let isCss = url.endsWith('.css')
if (isCss) {
let arr = url.split('/')
arr.pop()
responseText = responseText.replaceAll('./', arr.join('/') + '/') // 替换相对资源地址
}
reslove(GM_addElement(doc.head, isCss ? 'style' : 'script', {textContent: responseText}))
}
})
})
}
const next = () => {
let url = files.shift()
if (url == undefined) return callback && callback()
let ext = url.split('.').pop().toLowerCase()
if (ext == "js") {
if (!cache || !doc.querySelector('script[src="' + url + '"]')) {
return load(url).then(next)
}
} else
if (ext == "css") {
if (!cache || !doc.querySelector('link[href="' + url + '"]')) {
return load(url).then(next)
}
}
next()
}
next()
}
// 类似primise.all, 但是它也能接受函数对象,让promise初始化保持顺序进行
function awaitPromises(promises) {
return new Promise(reslove=>{
let ret = []
const next = ()=>{
let promise = promises.shift()
if (promise == undefined) return reslove(ret)
if(typeof(promise) == 'function') promise = promise()
promise.then(val => ret.push(val) & next())
}
next()
})
}
Object.defineProperty(Array.prototype, 'map1', {
value: function(cb) {
let ret = [], val
this.forEach(item => {
if((val = cb(item)) !== undefined) ret.push(val)
})
return ret
},
enumerable: false
});