Sponsor Block Integration | Invidious

automatically skip sponsor fragments in invidious

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name Sponsor Block Integration | Invidious
// @description automatically skip sponsor fragments in invidious
// @namespace -
// @version 1.3.0
// @author NotYou
// @include *://invidious.snopyta.org/watch?v=*
// @include *://yewtu.be/watch?v=*
// @include *://invidious.kavin.rocks/watch?v=*
// @include *://vid.puffyan.us/watch?v=*
// @include *://invidious.namazso.eu/watch?v=*
// @include *://inv.riverside.rocks/watch?v=*
// @include *://youtube.076.ne.jp/watch?v=*
// @include *://yt.artemislena.eu/watch?v=*
// @include *://invidious.flokinet.to/watch?v=*
// @include *://invidious.esmailelbob.xyz/watch?v=*
// @include *://invidious.projectsegfau.lt/watch?v=*
// @include *://inv.bp.projectsegfau.lt/watch?v=*
// @include *://y.com.sb/watch?v=*
// @include *://invidious.sethforprivacy.com/watch?v=*
// @include *://invidious.tiekoetter.com/watch?v=*
// @include *://invidious.nerdvpn.de/watch?v=*
// @include *://inv.vern.cc/watch?v=*
// @include *://invidious.slipfox.xyz/watch?v=*
// @include *://inv.privacy.com.de/watch?v=*
// @include *://invidious.rhyshl.live/watch?v=*
// @include *://invidio.xamh.de/watch?v=*
// @include *://invidious.dhusch.de/watch?v=*
// @include *://inv.odyssey346.dev/watch?v=*
// @include *://c7hqkpkpemu6e7emz5b4vyz7idjgdvgaaa3dyimmeojqbgpea3xqjoid.onion/watch?v=*
// @include *://w6ijuptxiku4xpnnaetxvnkc5vqcdu7mgns2u77qefoixi63vbvnpnqd.onion/watch?v=*
// @include *://kbjggqkzv65ivcqj6bumvp337z6264huv5kpkwuv6gu5yjiskvan7fad.onion/watch?v=*
// @include *://grwp24hodrefzvjjuccrkw3mjq4tzhaaq32amf33dzpmuxe7ilepcmad.onion/watch?v=*
// @include *://osbivz6guyeahrwp2lnwyjk2xos342h4ocsxyqrlaopqjuhwn2djiiyd.onion/watch?v=*
// @include *://u2cvlit75owumwpy4dj2hsmvkq7nvrclkpht7xgyye2pyoxhpmclkrad.onion/watch?v=*
// @include *://euxxcnhsynwmfidvhjf6uzptsmh4dipkmgdmcmxxuo7tunp3ad2jrwyd.onion/watch?v=*
// @include *://invidious.esmail5pdn24shtvieloeedh7ehz3nrwcdivnfhfcedl7gf4kwddhkqd.onion/watch?v=*
// @include *://inv.vernccvbvyi5qhfzyqengccj7lkove6bjot2xhh5kajhwvidqafczrad.onion/watch?v=*
// @include *://am74vkcrjp2d5v36lcdqgsj2m6x36tbrkhsruoegwfcizzabnfgf5zyd.onion/watch?v=*
// @include *://ng27owmagn5amdm7l5s3rsqxwscl5ynppnis5dqcasogkyxcfqn7psid.onion/watch?v=*
// @include *://verni6dr4qxjgjumnvesxerh5rvhv6oy5ddeibaqy5d7tgbiiyfa.b32.i2p/watch?v=*
// @license GPL-3.0-or-later
// @grant none
// @run-at document-end
// @icon 
// ==/UserScript==

(function init() {
  /* To enable fragment skipping change from "true" to "false" (without quotes), to disable do the opposite. */
	const categories = {
		sponsor: true,
		intro: false,
		outro: false,
		music_offtopic: false,
		selfpromo: true,
		interaction: true,
  }

	/* Write here minimal duration of the fragment, if duration will be less than this, then fragment won't be skipped. */
  const minimalDuration = 0

	/* Write here minimal votes for the fragment, if votes will be less than this, then fragment won't be skipped */
  const minimalVotes = 0 

  const colors = {
    sponsor: 'rgb(0, 212, 0)',
    intro: 'rgb(0, 255, 255)',
    outro: 'rgb(2, 2, 237)',
    music_offtopic: 'rgb(255, 153, 0)',
    selfpromo: 'rgb(255, 255, 0)',
    interaction: 'rgb(204, 0, 255)',
  }

  const videoData = document.querySelector('#video_data')

	if(!videoData) {
    setTimeout(init, 250)
  }

  const categoriesSearchParams = Object.keys(categories)
  	.filter(e => categories[e])
  	.map(e => 'category=' + e)
  	.join('&')

  const id = JSON.parse(videoData.textContent).id
	const url = 'https://sponsor.ajay.app/api/skipSegments?videoID=' + id + '&' + categoriesSearchParams
  const video = document.querySelector('video')
  const isPlayerJS = !document.querySelector('video#player')
  const segments = []
  const css = 'height: 100%;position: absolute;z-index: 1;'

  getSkipSegment().then(res => {
    for (let i = 0; i < res.length; i++) {
      const current = res[i]

      if(current.votes > minimalVotes - 1) {
      	const segment = current.segment
        const duration = current.videoDuration || video.duration
      	const left = segment[0] / duration * 100 + '%'
      	const width = (segment[1] - segment[0]) / duration * 100 + '%'
        const segmentDuration = segment[1] - segment[0]

        if(segmentDuration > minimalDuration) {
        	segments.push(segment)
      		createSegment(left, width, colors[current.category])
        }
      }
    }

    if(res.length > 0) {
      skipSponsor()

      video.addEventListener('timeupdate', skipSponsor)
    }
  })

  function skipSponsor() {
    for (let i = 0; i < segments.length; i++) {
      const segment = segments[i]
      const currentTime = video.currentTime

      if(currentTime > segment[0] && currentTime < segment[1]) {
        video.currentTime = segment[1]
      }
    }
  }

  function createSegment(left, width, color){
    if(isPlayerJS) {
      const segmentNode = document.createElement('div')
      segmentNode.style.cssText = `
			left: ${left};
			width: ${width};
			background-color: ${color};
			${css}`
      document.querySelector('.vjs-progress-holder').appendChild(segmentNode)
    }
  }

  async function getSkipSegment() {
    const result = await fetch(url).then(r => r.json()).then(c => {
      return c
    }).catch(() => {
      console.error('Sponsor Block Integration: Server Error or Intergration Not Found.')
      return []
    })

    return result
  }
})()