Instagram Download

download instagram image and video

当前为 2018-05-12 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

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

;(function () {
  'use strict'
  
  const $ = (selector) => document.querySelector(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)
      }
    }
  }
  
  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;
          }`
      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 wrap = $(this.setting.wrap)
      if (!wrap) return
      if (wrap.querySelector(`.${this.setting.btnClass}`)) return
      wrap.classList.add('hover-img')
      const img = wrap.querySelector('img')
      const video = wrap.querySelector('video')
      if (img) {
        this.downloadImage(wrap, img)
      } else if (video) {
        this.downloadVideo(wrap, video)
      }
    }
    
    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',
    wrap: '._gxii9'
  })
})()