Greasy Fork is available in English.

Sponsor Block Integration | Invidious

automatically skip sponsor fragments in invidious

// ==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
  }
})()