// ==UserScript==
// @name Instagram: 图片,视频批量下载器
// @name:en Instagram: pictures, video batch downloader
// @namespace http://tampermonkey.net/
// @version 4.0
// @description Instagram下载器,支持图片和视频批量下载
// @description:en Downloader for Instagram, support batch download pictures and videos
// @author jaywang
// @match https://www.instagram.com/*
// @require https://cdn.jsdelivr.net/npm/jquery@3.2.1/dist/jquery.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js
// @icon 
// @grant unsafeWindow
// @run-at document-idle
// @license MIT
// @downloadURL https://update.greasyfork.icu/scripts/435655/Instagram%3A%20%E5%9B%BE%E7%89%87%EF%BC%8C%E8%A7%86%E9%A2%91%E6%89%B9%E9%87%8F%E4%B8%8B%E8%BD%BD%E5%99%A8.user.js
// @updateURL https://update.greasyfork.icu/scripts/435655/Instagram%3A%20%E5%9B%BE%E7%89%87%EF%BC%8C%E8%A7%86%E9%A2%91%E6%89%B9%E9%87%8F%E4%B8%8B%E8%BD%BD%E5%99%A8.meta.js
// ==/UserScript==
(function () {
'use strict';
// Your code here...
/** 更多选项列表的选择器 */
const selectionListSelector = 'div.RnEpo.Yx5HN > div > div > div > div';
/** 当前post下的用户名称选择器 */
const usernameASelector = 'header.Ppjfr a.ZIAjV';
/** 复制图片按钮选择器 */
const copyURLSelector = `${selectionListSelector} > button:nth-last-child(3)`;
/** 打开帖子按钮选择器 */
const openPostSelector = `${selectionListSelector} > button:nth-last-child(2)`;
/** 单个下载按钮 */
const singleDownloadBtn = '';
/** 批量下载图片按钮 */
const batchDownloadBtn = '';
/**
* 当前post的下标
* -1: 代表单图
*/
let currentIndex = -1;
/**
* 当前post
*/
let currentPost;
/**
* 下载图片按钮事件
* 点击三个小点更多按钮
*/
$('body').click(async (el) => {
/** 下载按钮事件 */
if (el.target.getAttribute('jaywangdownload') === 'single') {
const index = currentIndex;
const post = currentPost;
const username = post.querySelector(usernameASelector).text;
const isPrivate = isPrivateUser();
let initSrc;
let src;
if (isPrivate) {
const container = post.querySelector('._97aPb');
src = getPrivateSrc(container, index);
} else {
$(copyURLSelector).click();
initSrc = await navigator.clipboard.readText();
src = await getResource(initSrc, index);
}
save(src, getName(username, index));
}
/** 下载合集按钮事件 */
if (el.target.getAttribute('jaywangdownload') === 'batch') {
let initSrc;
if (isPrivateUser() && withoutOpenPostBtn()) {
//console.log(`location`, location)
} else {
$(`${selectionListSelector} > button:nth-last-child(${getOpenPostLastLocation()})`).click();
}
initSrc = location.href;
const data = await fetchPostInformation(initSrc);
const batchInfo = getBatchInformation(data);
batchSaveAs(batchInfo);
}
/** 点击三个小点更多按钮 */
if (el.target.closest('button.wpO6b')) {
/** 获取当前post */
currentPost = el.target.closest('article');
/** 容纳点点点的容器 */
const container = currentPost.querySelector('._97aPb');
const dots = currentPost.querySelector('._3eoV-');
const isMulti = isMultiplePost(container);
currentIndex = isMulti ? getPostIndex(dots) : 0;
}
});
/** */
/** DOM变动的回调函数 */
const callback = function (mutationRecord) {
for (const record of mutationRecord) {
const nodeList = record.addedNodes;
if (nodeList.length === 1 && isMoreOptionButton(nodeList[0])) {
$(selectionListSelector).prepend(singleDownloadBtn);
$(selectionListSelector).prepend(batchDownloadBtn);
return;
}
}
};
/** 检测DOM变动 */
const observer = new MutationObserver(callback);
observer.observe(document.body, {
childList: true,
subtree: false
});
/**
* 判断是否是更多选项按钮
* @param {Node} node
* @return {boolean}
*/
function isMoreOptionButton(node) {
return node.querySelector('.mt3GC');
}
/**
* 获取Post信息
* @param uri
* @returns {Promise}
*/
async function fetchPostInformation(uri) {
let formatedUri = uri;
if (uri.includes('?utm_source')) {
formatedUri = uri.match(/.*(?=\?utm_source)/);
}
formatedUri += '?__a=1';
const result = await fetch(formatedUri);
const data = await result.json();
return data;
}
/**
* 获取资源链接
* @param {string} uri
* @param {number} index
*/
async function getResource(uri, index) {
const data = await fetchPostInformation(uri);
const isSingle = data.graphql.shortcode_media.edge_sidecar_to_children === undefined;
const node = isSingle
? data.graphql.shortcode_media
: data.graphql.shortcode_media.edge_sidecar_to_children.edges[index].node;
const isVideo = node.is_video;
const src = isVideo
? node.video_url
: node.display_resources[node.display_resources.length - 1].src;
return src;
}
/**
* 获取第i个信息
* @param data
* @param index
* @returns {*|string}
*/
function getBatchInformation(data) {
const isSingle = data.graphql.shortcode_media.edge_sidecar_to_children === undefined;
let edges;
if (isSingle) {
edges = [
{
node: data.graphql.shortcode_media
}
];
} else {
edges = data.graphql.shortcode_media.edge_sidecar_to_children.edges;
}
const username = data.graphql.shortcode_media.owner.username;
const infoList = [];
for (const i in edges) {
const node = edges[i].node;
const isVideo = node.is_video;
const src = isVideo
? node.video_url
: node.display_resources[node.display_resources.length - 1].src;
infoList.push({
name: getName(username, i),
src,
suffix: isVideo ? 'mp4' : 'jpg'
});
}
return infoList;
}
/**
* 在浏览器里面打开
* @param {string} src
*/
function openInBrowser(src) {
const a = document.createElement('a');
a.target = '_blank';
a.href = src;
document.body.append(a);
a.click();
a.remove();
}
/**
* 另存为图片或视频
* @param {string} src 下载源
* @param {string} name 图片名称
*/
async function save(src, name) {
const data = await fetch(src);
const blob = await data.blob();
downloadBlob(blob, name);
}
/**
* 下载blob
* @param {blob} blob
*/
function downloadBlob(blob) {
const domString = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = domString;
a.setAttribute('download', name);
a.click();
}
/**
* 批量下载
*/
async function batchSaveAs(fileList) {
let promiseList = [];
let filenameList = [];
let filename;
for (const { src, name, suffix } of fileList) {
filename = name;
const promise = fetch(src);
promiseList.push(promise);
filenameList.push(`${name}.${suffix}`);
}
const resultList = await Promise.all(promiseList);
const blobList = resultList.map((result) => result.blob());
const zip = new JSZip();
const folder = zip.folder(filename);
for(let i = 0; i < filenameList.length; i++) {
folder.file(filenameList[i], blobList[i]);
}
const content = await folder.generateAsync({type:"blob"});
saveAs(content, `${filename}.zip`);
}
/**
* 定位当前图片下标
* 多个元素_97aPb内有一个rQDP3,内有_3eoV-
* 单个元素_97aPb内没有rQDP3
* @param {Element} node
*/
function getPostIndex(node) {
const nodeList = node.childNodes;
for (let i = 0; i < nodeList.length; i++) {
const classList = nodeList[i].classList;
if (classList && classList.contains('XCodT')) {
return i;
}
}
return 0;
}
/**
* 判断post图片/视频数量是多个还是单个
* @param {Element} el container
* @return {boolean}
*/
function isMultiplePost(el) {
return el.querySelector('._3eoV-') !== null;
}
/**
* 格式化的日期
* @returns {string}
*/
function getFormatedDate() {
const d = new Date();
return '' + d.getHours() + d.getMinutes() + d.getSeconds();
}
/**
* 判断是否是私人用户
* @returns {boolean}
*/
function isPrivateUser() {
const nodeList = document.querySelector(selectionListSelector).querySelectorAll('.HoLwm');
return nodeList.length <= 3;
}
/**
* 判断是否有打开帖子按钮
* @returns {boolean}
*/
function withoutOpenPostBtn() {
const nodeList = document.querySelector(selectionListSelector).querySelectorAll('.HoLwm');
return nodeList.length === 1;
}
/**
* 获取打开帖子按钮倒数的位置
* @returns {number}
*/
function getOpenPostLastLocation() {
const nodeList = document.querySelector(selectionListSelector).querySelectorAll('.HoLwm');
return nodeList.length;
}
/**
* 获取图片,视频资源链接
* @param {Element} container
* @param {number} index
* @returns {string}
*/
function getPrivateSrc(container, index) {
let resourceContainer = container;
if (isMultiplePost(container)) {
/**
* 图片在post开始是第一个有效的li
* post结尾是第二个li
* post中间是中间一个li(中间)
*/
const hasPrevBtn = container.querySelectorAll('.POSa_').length !== 0;
const hasNextBtn = container.querySelectorAll('._6CZji').length !== 0;
let nth = 3;
if (hasNextBtn && !hasPrevBtn) {
nth = 2;
}
resourceContainer = container.querySelector(`ul.vi798 li:nth-child(${nth})`);
}
const video = resourceContainer.querySelector('video');
const img = resourceContainer.querySelector('img');
if (video) {
return video.src;
}
if (img) {
const sets = img.srcset.split(',');
const lastSet = sets[0];
return lastSet.split(' ')[0];
}
}
/**
* 根据名称获取下标
* @param {string} username
* @param {number} index
* @returns {string}
*/
function getName(username, index) {
return `${username.split('.').join('')}_${index + 1}_${getFormatedDate()}`;
}
})();