Greasy Fork is available in English.

Instagram Download

Download instagram image and video

// ==UserScript==
// @name         Instagram Download
// @namespace    null
// @version      1.0
// @description  Download instagram image and video
// @author       tgxh
// @match        *.instagram.com/*
// @grant        none
// ==/UserScript==

;(function () {
  'use strict';

  const $ = (selector) => document.querySelector(selector);
  const $$ = (selector) => document.querySelectorAll(selector);

  const map = new WeakMap();
  const internal = key => {
    if (!map.has(key)) map.set(key, {});
    return map.get(key);
  };

  let _this;

  function throttle(fn, interval) {
    let timeBefore = null;
    return () => {
      const timeNow = Date.now();
      if (timeBefore === null) timeBefore = timeNow;
      if (timeNow - timeBefore > interval) {
        timeBefore = timeNow;
        fn.apply(this, arguments);
      }
    };
  }

  function findWrap(selector = 'div[role="button"]') {
    return Array.from($$(selector)).filter(div => {
      const img = div.querySelector('img');
      const video = div.querySelector('video');
      const hasImgOrVideo = img && img.getAttribute('srcset') || video;
      return !div.querySelector(selector) && hasImgOrVideo;
    });
  }

  class Instagram {
    constructor(options) {
      const defaults = {
        delay: 200,
      };
      this.setting = Object.assign({}, defaults, options);
      this.delay = this.setting.delay;
      this.body = $('body');
      this.init();
      _this = internal(this);
      _this.render = throttle.call(this, this.animate, this.delay);
    }

    init() {
      this.observer(this.body);
      this.addStyles();
    }

    addStyles() {
      const headEle = $('head'),
        style = document.createElement('style'),
        styleText = `
          .hover-img:hover .download-btn {
          opacity: 1;
          }
          .download-btn {
          position: absolute;
          top: 50px;
          right: 50px;
          width: 40px;
          height: 40px;
          background: url('');
          transition: opacity .3s;
          opacity: 0;
          z-index: 99;
          }`;
      style.innerHTML = styleText;
      headEle.append(style);
    }

    observer(ele) {
      const config = {
        attributes: true,
        childList: true,
        subtree: true,
      };
      const observer = new MutationObserver(mutations => {
        _this.render();
      });
      observer.observe(ele, config);
    }

    animate() {
      const wraps = findWrap();
      wraps.forEach(wrap => {
        if (wrap.querySelector(`.${this.setting.btnClass}`)) return;
        wrap.parentNode.classList.add('hover-img');
        const video = wrap.querySelector('video');
        const img = Array.from(wrap.querySelectorAll('img')).filter(img => img.getAttribute('srcset'))[0];
        if (video) {
          this.downloadVideo(wrap, video);
        } else if (img) {
          this.downloadImage(wrap, img);
        }
      });
    }

    downloadImage(wrap, img = {}) {
      const srcset = img.getAttribute('srcset');
      if (srcset) {
        let href = srcset.split(',').pop();
        href = href.slice(0, href.indexOf(' '));
        this.addBtn(wrap, href);
      }
    }

    downloadVideo(wrap, video) {
      const src = video.getAttribute('src');
      if (src) {
        this.addBtn(wrap, src);
      }
    }

    addBtn(wrap, href) {
      const btn = wrap.querySelector(`.${this.setting.btnClass}`);
      if (btn) {
        wrap.removeChild(btn);
      }
      const ele = document.createElement('a');
      ele.classList.add(this.setting.btnClass);
      ele.setAttribute('href', href);
      ele.setAttribute('target', '_blank');
      ele.setAttribute('download', '');
      wrap.append(ele);
    }
  }

  window.insIntance = new Instagram({
    delay: 100,
    btnClass: 'download-btn',
  });
})();