// ==UserScript== // @name mCollection资源捕获 // @namespace mCollection // @homepage https://greasyfork.org/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分 // @downloadURL https://update.greasyfork.icu/scripts/469129/mCollection%E8%B5%84%E6%BA%90%E6%8D%95%E8%8E%B7.user.js // @updateURL https://update.greasyfork.icu/scripts/469129/mCollection%E8%B5%84%E6%BA%90%E6%8D%95%E8%8E%B7.meta.js // ==/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 = `
${this.resources.length}
` 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 `
${cover} ${title}
` }, 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: `
` }) 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', `
`) 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: `
` }, { 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: `
` }], 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: `
${g_tabler.build_select({ list: ['计划中'], value: '', class: 'form-select-sm', })} ${g_tabler.build_select({ list: ['计划中'], value: '', class: 'form-select-sm', })}
全选 反选 已隐藏 取消选中 移除选中
` }, }, 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 });