// ==UserScript== // @name Download Original Picture // @name:zh-CN 下载原始图片 // @description A tool to help you download full size images from websites // @description:zh-CN 一个帮你从网站下载原始尺寸图片的工具 // @namespace https://huching.net/ // @version 0.1.1 // @license GPL-3.0 // @icon  // @author huc < ht@live.se > // @supportURL https://github.com/hz2/user-scripts-and-styles/issues/new // @require https://greasyfork.org/scripts/396752-hx-script-library/code/hx-script-library.js // @resource HxLib https://greasyfork.org/scripts/396752-hx-script-library/code/hx-script-library.js // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=your.email.here@example.com&item_name=Greasy+Fork+donation // @contributionAmount 5 // @include *://medium.com/* // @include *://twitter.com/* // @include *://mobile.twitter.com/* // @include *://weibo.com/* // @include *://*.weibo.com/* // @include *://*.vmgirls.com/* // @include *://wallpaperhub.app/* // @include *://*.bing.com/* // @include *://*.msn.cn/* // @include *://instagram.com/* // @include *://*.instagram.com/* // @include *://instagram.com/* // @include *://*.instagram.com/* // @include *://tiktok.com/* // @include *://*.tiktok.com/* // @include *://*.douyin.com/* // @include *://*.kuaishou.com/* // @include *://*.xiaohongshu.com/* // @noframes // @grant unsafeWindow // @grant GM_setClipboard // @grant GM_xmlhttpRequest // @grant GM_openInTab // @grant GM_registerMenuCommand // @grant GM_getValue // @grant GM_setValue // @grant GM_getResourceText // @grant GM_info // @grant GM_addStyle // @downloadURL none // ==/UserScript== const head = document.getElementsByTagName('head'); head[0].insertAdjacentHTML('beforeend', ``); console.warn('Welcome to %c \ud83d\ude48\ud83d\ude49\ud83d\ude4a\u0020\u0048\u007a\u00b2\u0020\u0053\u0063\u0072\u0069\u0070\u0074\u0020\u004c\u0069\u0062\u0072\u0061\u0072\u0079 %c v0.06 ', 'background-color:teal;color: white;border:1px solid teal;border-radius: 4px 0 0 4px;border-left-width:0;padding:1px;margin:2px 0;font-size:1.1em', 'background-color:#777;color: white;border:1px solid #777;border-radius: 0 4px 4px 0;border-right-width:0;padding:1px;margin:5px 0;'); const openDown = (url, e, name) => { e && e.preventDefault(); e && e.stopPropagation() const downBlobUrl = (blobUrl) => { let el = document.createElement("a"); el.setAttribute("href", blobUrl); if (name) { el.setAttribute("download", name) } if (document.createEvent) { const event = document.createEvent("MouseEvents"); event.initEvent("click", true, true); el.dispatchEvent(event); } else { el.click(); } } if (url.startsWith('blob')) { downBlobUrl(url) return } fetch(url, { mode: "cors" }) .then(resp => resp.blob()) .then(r => { const blobUrl = URL.createObjectURL(r) downBlobUrl(blobUrl) }) .catch(err => { console.log("Request failed", err); }); } const hostname = window.location.hostname const lastItem = arr => arr.length ? arr[arr.length - 1] : '' const createDomAll = (item, fn) => { let domDL = document.createElement('a'); domDL.className = 'hx-download-original-images-tool' domDL.title = '下载原始图片' item.addEventListener('load', _ => { let link = fn(item.src) domDL.href = link domDL.addEventListener('click', e => openDown(link, e)) }) item.insertAdjacentElement('afterEnd', domDL) } const createDom = (cfg) => { const { parent, link, name, className = '', style = '', target, postion = 'afterEnd' } = cfg const genDomDL = (dom) => { let domDL = dom || document.createElement('a'); Object.assign(domDL, { title: '下载原始图片', className: 'hx-download-original-images-tool ' + className, style: style, href: link, }) domDL.onclick = e => { e && e.preventDefault(); e && e.stopPropagation() const newName = name || lastItem(link.split('/')) openDown(link, e, newName) } return domDL } let parent2 = parent if (!parent && target) { parent2 = target.parentElement } // if (['afterEnd', 'beforeBegin'].includes(postion)) { // parent2 = target.parentElement.parentElement // } const exist = parent2.querySelector('.hx-download-original-images-tool') if (exist) { genDomDL(exist) } else { parent2.insertAdjacentElement(postion, genDomDL()) } } const updateLink = (dom, link) => { dom.href = link const newName = lastItem(link.split('/')) dom.onclick = e => openDown(link, e, newName) } const init = () => { if (hostname === "twitter.com" || hostname === "mobile.twitter.com") { //twitter window.addEventListener('mouseover', ({ target }) => { const src = target && target.src const parent = target.parentElement const next = parent && parent.nextElementSibling if (target.tagName == 'IMG' && !(next && next.className.includes('hx-download-original-images-tool')) && !/profile_images|video_thumb/g.test(src)) { const link = src.replace(/\&name=\w+/g, '&name=orig') const name = lastItem(link.split('/')).replace(/\?format=(\w+)\&name=orig/g, (_, b) => `.${b}`) const style = 'margin-left: 10px;margin-top: 10px;' const cfg = { parent, link, name, style } createDom(cfg) } }) } else if (hostname.includes('weibo')) { const isWeiboNode = dom => { const getNodeValue = el => el.attributes['node-type'] && el.attributes['node-type'].nodeValue if (getNodeValue(dom.parentElement) === 'artwork_box' || getNodeValue(dom) === 'img_box' || dom.className.includes('woo-picture-main') || dom.className.includes('woo-picture-slot') || dom.className.includes('imgInstance')) { return true } else { return false } } window.addEventListener('mouseover', ({ target }) => { const parent = target.parentElement const next = parent && parent.nextElementSibling if (target.tagName == 'IMG' && isWeiboNode(parent)) { const link = target.src.replace(/orj\d+|mw\d+/g, 'large') if (next && next.className.includes('hx-download-original-images-tool')) { updateLink(next, link) } else { const style = 'top: 40px;right: 10px;' const cfg = { parent, link, style } createDom(cfg) } } }) } else if (hostname === "www.vmgirls.com") { // vmgirls let domDL = document.createElement('a'); domDL.className = 'hx-download-original-images-tool ' domDL.style = 'position: relative;margin-right: 10px;display: inline-block;vertical-align: -20px;' domDL.title = '下载原始图片' domDL.onclick = e => { const list1 = [...document.querySelector('.post').querySelectorAll('a')].filter(x => x.src && x.src.indexOf('static.vmgirls.com/image') !== -1) const list2 = [...document.querySelector('.post-content').querySelectorAll('img')].filter(x => x.src && x.src.indexOf('t.cdn.ink/image') !== -1) const imgList = [...list1, ...list2].map((x, i) => ({ link: x.src && x.src.replace('-scaled', ''), name: `${x.alt || x.title}_${i}.jpg` })) domDL.title += ' ' + imgList.length imgList.forEach(x => openDown(x.link, e, x.name)) const link1 = imgList.map(x => x.link).join('\n') const link2 = imgList.map(x => `aria2c -o ${x.name} ${x.link}`).join('\n') const content = `获取链接 ` window.open(URL.createObjectURL(new Blob([content], { type: 'text/html' }))) } document.querySelector('.main-submenu').insertAdjacentElement('afterBegin', domDL) } else if (hostname === "medium.com") { // medium document.querySelector('article').querySelectorAll('img').forEach(x => (x.width > 80) && createDomAll(x, src => src.replace(/max\/\d+\//g, 'max/30000/'))) } else if (hostname === "wallpaperhub.app") { // wallpaperhub const odomList = [...document.querySelectorAll('.downloadButton')] odomList.forEach(odom => { if (odom) { let link0 = odom.href.split('downloadUrl=')[1] const link = link0 const style = 'position: relative;margin-right: 10px;display: inline-block;vertical-align: -20px;' const cfg = { parent: odom.parentElement.parentElement, link, style, postion: 'beforeBegin' } createDom(cfg) } }) } else if (hostname === "ntp.msn.cn") { // edge 首页 const link = document.querySelector('background-image')._imageSource; const style = 'position: fixed;right: 80px;top: 40px;' const cfg = { parent: document.body, link, className: 'white', style, postion: 'beforeBegin' } createDom(cfg) } else if (hostname === "www.bing.com") { // bing 首页 const orig = document.querySelector('[style*="th?id="]').style.backgroundImage const link = orig.match(/th\?id\=[\w\d\.\-\_]+/g)[0].replace('1920x1080', 'UHD') const name = link && link.split('=')[1] const style = 'position: relative;width: 42px;height: 42px;margin: 0;opacity: .9;' const cfg = { parent: document.querySelector('#id_h'), link, name, className: 'white', style, postion: 'afterBegin' } createDom(cfg) } else if (hostname === "cn.bing.com") { // bing 首页 const link = document.querySelector('#bgImgProgLoad').dataset.ultraDefinitionSrc.split('&')[0]; const name = link && link.split('=')[1] const style = 'position: fixed;right: 225px;bottom: 53px;margin: 0px;width: 64px;height: 64px;z-index: 550;opacity: .9;' const cfg = { parent: document.body, link, name, className: 'white', style, postion: 'beforeBegin' } createDom(cfg) } else if (hostname === "www.instagram.com") { window.addEventListener('mouseover', ({ target }) => { const el = target.previousElementSibling const el2 = target.parentElement const img = (el && el.querySelector('img:not([data-testid])') || el2 && el2.querySelector('video:not([data-testid])')) if (img) { const src = img.src const parent = img.parentElement const link = src const style = 'left: 10px;top: 10px;' const cfg = { parent, link, style, target: img, name: img.alt ? (img.alt + '.jpg') : src.split(/[\?\/]/g).filter(x => x.endsWith('.jpg'))[0], postion: 'beforeEnd' } createDom(cfg) } }) } else if (hostname === "www.tiktok.com") { window.addEventListener('mouseover', ({ target }) => { if (target.tagName == 'VIDEO') { const src = target.src const parent = target.parentElement const link = src const style = 'left: 10px;top: 10px;' const cfg = { parent, link, style, target, name: lastItem(src.split('?')[0].split('/').filter(x => x)), postion: 'beforeEnd' } createDom(cfg) } }) } else if (hostname === "www.douyin.com") { window.addEventListener('mouseover', ({ target }) => { // if (target && target.tagName === 'VIDEO') { // const src = (target.querySelector('source') || target).src // const link = src // const style = 'left: 10px;top: 10px;' // const cfg = { // link, // style, // target, // postion: 'afterEnd', // name: lastItem(src.split('?')[0].split('/').filter(x => x)), // } // createDom(cfg) // return // } else if (target && target.parentElement) { const container = target.parentElement.parentElement || { className: '' } if (container && container.className.includes('videoContainer') && container.querySelector('video')) { const src = (container.querySelector('source') || container.querySelector('video')).src const link = src const style = 'left: 10px;top: 10px;' const cfg = { link, style, target: container, postion: 'beforeEnd', name: lastItem(src.split('?')[0].split('/').filter(x => x)), } createDom(cfg) } } }) } else if (hostname === "www.kuaishou.com") { window.addEventListener('mouseover', ({ target }) => { if (target && target.parentElement) { const container = target.parentElement.parentElement || { className: '' } if (container && container.className.includes('kwai-player') && container.querySelector('video')) { const src = (container.querySelector('source') || container.querySelector('video')).src const link = src const style = 'left: 10px;top: 10px;' const cfg = { link, style, target: container, postion: 'beforeEnd', name: lastItem(src.split('?')[0].split('/').filter(x => x)), } createDom(cfg) } } }) } else if (hostname === "www.xiaohongshu.com") { window.addEventListener('mouseover', ({ target }) => { const container = target && target.parentElement if (container && container.className && container.className.includes('carousel')) { const inner = container.querySelector('li:not([style*=none]) .inner') const src = inner && inner.style['background-image'].replace(/^url\(\"|\"\)$/g, '') const link = src const style = 'left: 10px;top: 10px;' const cfg = { link, style, parent: container, postion: 'beforeEnd', name: lastItem(src.split('?')[0].split('/').filter(x => x)) + '.jpg', } createDom(cfg) } }) } } setTimeout(() => { init() }, 1500);