// ==UserScript==
// @name [Unsupported] IG Helper: download Instagram pic & vids
// @name:zh-CN [不再维护] IG Helper: 下载 Instagram 图片和视频
// @description Unsupported
// @description:zh-cn 不再维护
// @version 1.9.9
// @namespace InstagramHelper
// @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/1.3.3/FileSaver.min.js
// @match https://www.instagram.com/*
// @match https://*.cdninstagram.com/*
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @grant GM_download
// @license MIT License
// ==/UserScript==
(function() {
'use strict';
/*
Common function
*/
Element.prototype.parents = function(selector) {
// Vanilla JS jQuery.parents() realisation
// https://gist.github.com/ziggi/2f15832b57398649ee9b
var elements = [];
var elem = this;
var ishaveselector = selector !== undefined;
while ((elem = elem.parentElement) !== null) {
if (elem.nodeType !== Node.ELEMENT_NODE) {
continue;
}
if (!ishaveselector || elem.matches(selector)) {
elements.push(elem);
}
}
return elements;
};
var GM_download_extra = function(url, name) {
// https://gist.github.com/ccloli/832a8350b822f3ff5094
if (url == null) return;
var data = {
method: 'GET',
responseType: 'arraybuffer',
onload: function(res) {
var blob = new Blob([res.response], {
type: 'application/octet-stream'
});
var url = URL.createObjectURL(blob); // blob url
var a = document.createElement('a');
a.setAttribute('href', url);
a.setAttribute('download', data.name != null ? data.name : 'filename');
document.documentElement.appendChild(a);
// call download
// a.click() or CLICK the download link can't modify filename in Firefox (why?)
// Solution from FileSaver.js, https://github.com/eligrey/FileSaver.js/
var e = new MouseEvent('click');
a.dispatchEvent(e);
document.documentElement.removeChild(a);
setTimeout(function() {
// reduce memory usage
URL.revokeObjectURL(url);
if ('close' in blob) blob.close(); // File Blob.close() API, not supported by all the browser right now
blob = undefined;
}, 1000);
if (typeof data.onafterload === 'function') data.onafterload(); // call onload function
}
// error object of onerror function is not supported right now
};
if (typeof url === 'string') {
data.url = url;
data.name = name;
} else {
if (url instanceof Object === false) return;
// as documentation, you can only use [url, name, headers, saveAs, onload, onerror] function, but we won't check them
// Notice: saveAs is not supported
if (url.url == null) return;
for (var i in url) {
if (i === 'onload') data.onafterload = url.onload; // onload function support
else data[i] = url[i];
}
}
// it returns this GM_xhr, thought not mentioned in documentation
return GM_xmlhttpRequest(data);
};
var GM_download_extra_img = function(src, title) {
var img = new Image();
img.crossOrigin = 'Anonymous';
img.onload = function() {
var canvas = document.createElement('CANVAS');
var ctx = canvas.getContext('2d');
var dataURL;
canvas.height = this.height;
canvas.width = this.width;
ctx.drawImage(this, 0, 0);
// Save image
canvas.toBlob(function(blob) {
saveAs(blob, title);
}, 'image/jpeg', 0.8);
canvas = null;
};
img.src = src;
if (img.complete || img.complete === undefined) {
img.src = '';
img.src = src;
}
};
var GM_download_extra_video = function(src, title) {
window.open(src);
// TODO
/* fetch(src).then(function(res) {
res.blob().then(function(blob) {
var _link = document.createElement('a');
_link.download = title;
_link.style.display = 'none';
var _blob = new Blob([blob]);
_link.href = URL.createObjectURL(_blob);
document.body.appendChild(_link);
_link.click();
document.body.removeChild(_link);
});
}); */
};
var GM_addStyle_extra = function(css) {
var _head = document.getElementsByTagName('head')[0];
if (_head) {
var _style = document.createElement('style');
_style.setAttribute('type', 'text/css');
_style.textContent = css;
_head.appendChild(_style);
return _style;
}
return null;
};
var ig_helper_style = '.downloadBtn {' +
'position:absolute; width:46px; height:28px; opacity:0; right:25px; top:25px; z-index:1; text-align:center;' +
'font-size:14px; line-height:26px; padding:0 8px; font-weight:600; color:#fff; white-space:nowrap; outline:0;' +
'cursor:pointer; -webkit-user-select:none; -moz-user-select:none; user-select:none;' +
'transition:opacity .2s ease-out; transition-delay:.1s; border-radius:3px; border:1px solid #db2d74;' +
'background-color:#db2d74; background-size:22px; background-position:center; background-repeat:no-repeat;' +
'background-image:url("")' +
'}' +
'.downloadBtn.inStories {width:28px; top:10px; right:10px; border-radius:50%; font-size:12px; background-size:18px;}' +
'.KL4Bh:hover .downloadBtn,.OAXCp:hover .downloadBtn, .qbCDp:hover ~ .downloadBtn {opacity:1} ' +
'._9AhH0 {display:none !important}' +
'._2us5i:hover .downloadBtn {opacity:1}' +
'._lz6s {z-index:3 !important}';
if (typeof GM_addStyle !== 'undefined') {
GM_addStyle(ig_helper_style);
} else {
// Greasemonkey 4.0 remove the GM_addStyle function.
GM_addStyle_extra(ig_helper_style);
}
/*
Main
*/
var observer = new MutationObserver(init);
var config = {
'childList': true,
'subtree': true
};
observer.observe(document.body, config);
function init() {
/* Home page */
if (window.location.pathname === '/') {
var _box_home = document.querySelector('#react-root > section > main > section > div > div > div');
// Logged in
if (_box_home) {
findMedia(_box_home);
}
}
/* Detail page */
if (window.location.pathname.match('/p/')) {
var _box_detail = '';
/*
Absolute
box class: QBXjJ
*/
if (document.querySelector('article.QBXjJ')) {
_box_detail = document.querySelector('article.QBXjJ');
findMedia(_box_detail);
}
/*
Dialog
*/
else {
setTimeout(function() {
if (document.querySelector('div[role="dialog"]')) {
if (document.querySelector('div[role="dialog"]').querySelector('article')) {
_box_detail = document.querySelector('div[role="dialog"]').querySelector('article');
findMedia(_box_detail);
}
}
}, 1000); // TODO: first click can't find dialog
}
}
/* Stories page */
if (window.location.pathname.match('/stories/')) {
setTimeout(function() {
var _box_story = document.querySelector('#react-root > section div.yS4wN');
if (_box_story) {
findMedia(_box_story, 'stories');
}
}, 50);
}
}
function findMedia(box, way) {
var _box = box, _way = way;
var _parent, _url, _username, _title;
_box.addEventListener('mouseover', function(event) {
event.stopPropagation();
/*
Picture
img class: FFVAD
*/
if (event.target.className === 'FFVAD') {
_parent = event.target.parentNode;
_url = event.target.src;
_title = _url.match(/[a-zA-Z0-9_]+.jpg/g);
_username = '';
// title class: FPmhX
if (_parent.parents('article')[0].querySelector('.FPmhX')) {
// TODO: 此内容不应在缩略图页面出现,但不明原因出现了。暂时做判断,如果在缩略图页面时不下载图片。同时避免报错。
_username = _parent.parents('article')[0].querySelector('.FPmhX').title;
addBtn(_parent, _url, _username, _title);
}
}
/*
Video
video class: tWeCl
video play button class: QvAa1
*/
if (event.target.className.indexOf('QvAa1') >= 0) {
_parent = event.target.parentNode;
_url = _parent.querySelector('.tWeCl').src;
_title = _url.match(/[a-zA-Z0-9_]+.mp4/g);
_username = _parent.parents('article')[0].querySelector('.FPmhX').title;
addBtn(_parent, _url, _username, _title);
}
/*
Stories Picture & Video
_8XqED: #react-root > div > div > section._8XqED
z6Odz parent: cover box (when autoplay videos disable, user click the cover box to play the video)
Debug: click more button stop auto video
*/
if (event.target.className.indexOf('_8XqED') >= 0 && _way === 'stories') {
var _current_target = document.querySelector('.z6Odz').parentNode;
_parent = _current_target.parentNode.parentNode;
// Stories Video: video 'if' in front of the image
if (_parent.querySelector('video')) {
_url = _parent.querySelector('video > source').src;
_title = _url.match(/[a-zA-Z0-9_]+.mp4/g);
_username = _parent.parents('section')[0].querySelector('.FPmhX').title;
addBtn(_parent, _url, _username, _title);
return false;
}
// Stories Picture
if (_parent.querySelector('img')) {
_url = _parent.querySelector('img').src;
_title = _url.match(/[a-zA-Z0-9_]+.jpg/g);
_username = _parent.parents('section')[0].querySelector('.FPmhX').title;
addBtn(_parent, _url, _username, _title);
return false;
}
}
});
}
function addBtn(parent, url, username, title) {
if (!parent.querySelector('.downloadBtn')) {
var _parent = parent;
var _url = url;
var _url_param = _url.indexOf('?') >= 0 ? _url.substring(0, _url.indexOf('?')) : _url;
_url_param = _url_param.replace(/\?ig_cache_key=[a-zA-Z0-9%.]+/, '');
var _title = title[0];
var _filename = username + '_' + _url_param.substring(_url_param.lastIndexOf('/') + 1, _url_param.length);
var _btn = document.createElement('button');
var _ua = navigator.userAgent.toLowerCase();
var removeBtn = function() {
if (_parent.querySelector('.downloadBtn')) {
_parent.removeChild(_parent.querySelector('.downloadBtn'));
}
};
if (window.location.pathname.match('/stories/')) {
_btn.className = 'downloadBtn inStories';
} else {
_btn.className = 'downloadBtn';
}
// Download
_btn.addEventListener('click', function(event) {
event.stopPropagation();
// Support DM_download
if (typeof GM_download !== 'undefined') {
// Safari GM_download unavailable
if (_ua.match(/version\/([\d.]+)/)) {
if (_title.indexOf('.mp4') >= 0) {
GM_download_extra_video(_url, _filename);
} else {
GM_download_extra_img(_url, _filename);
}
}
// Other browser
else {
// Chrome
if (_ua.match(/chrome\/\d./g)) {
GM_download(_url, _filename);
}
// Firefox
else if (_ua.match(/firefox\/\d./g)) {
GM_download_extra(_url, _filename);
}
}
}
// Firefox Greasemonkey not support DM_download
else {
if (_title.indexOf('.mp4') >= 0) {
GM_download_extra_video(_url, _filename);
} else {
GM_download_extra_img(_url, _filename);
}
}
}, false);
_parent.appendChild(_btn);
// Show stories btn
if (document.querySelector('.z6Odz')) {
document.querySelector('.z6Odz').addEventListener('mouseover', function(event) {
event.stopPropagation();
document.querySelector('.downloadBtn').style.opacity = '1';
}, false);
document.querySelector('.GHEPc').addEventListener('mouseleave', function(event) {
event.stopPropagation();
document.querySelector('.downloadBtn').style.opacity = '0';
}, false);
}
// More media on one box, ignore stories page
if (!window.location.pathname.match('/stories/')) {
var _left_btn = '._2Igxi';
var _right_btn = '.Zk-Zb';
if (_parent.querySelector(_right_btn) || _parent.parents('article')[0].querySelector(_right_btn)) {
var _btn_right = _parent.querySelector(_right_btn) ? _parent.querySelector(_right_btn) : _parent.parents('article')[0].querySelector(_right_btn);
_btn_right.addEventListener('click', removeBtn, false);
}
if (_parent.querySelector(_left_btn) || _parent.parents('article')[0].querySelector(_left_btn)) {
var _btn_left = _parent.querySelector(_left_btn) ? _parent.querySelector(_left_btn) : _parent.parents('article')[0].querySelector(_left_btn);
_btn_left.addEventListener('click', removeBtn, false);
}
}
}
}
})();