// ==UserScript==
// @name Saima customizer
// @name:ru Saima: кастомизация
// @namespace http://tampermonkey.net/
// @version 1.0.0
// @description Saima extension customization ( https://saima.ai/ ): fast speed control WPM (words per minute); correct placement of the Saima button at the same time as the SponsorBlock button.
// @description:ru Настройка расширения Saima ( https://saima.ai/ ): быстрый контроль скорости WPM (слов в минуту); правильное размещение кнопки Saima одновременно с кнопкой SponsorBlock.
// @author Igor Lebedev
// @license MIT
// @icon 
// @match http://*.youtube.com/*
// @match https://*.youtube.com/*
// @require https://openuserjs.org/src/libs/sizzle/GM_config.js
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @run-at document-idle
// ==/UserScript==
//debugger;
/* global GM_config */
(() => {
'use strict';
GM_config.init({
id: 'sc_config',
title: GM_info.script.name + ' Settings',
fields: {
DEBUG_MODE: {
label: 'Debug mode',
type: 'checkbox',
default: false,
title: 'Log debug messages to the console'
},
CHECK_FREQUENCY_LAUNCH_SAIMA: {
label: 'Check frequency (ms) launch of the Saima extension',
type: 'number',
min: 1000,
default: 5000,
title: 'The number of milliseconds to wait between checking the launch of the Saima extension'
},
CHECK_FREQUENCY_LAUNCH_SPONSOR_BLOCK: {
label: 'Check frequency (ms) launch of the SponsorBlock extension',
type: 'number',
min: 1000,
default: 5000,
title: 'The number of milliseconds to wait between checking the launch of the Saima extension'
},
WPM_SPEED: {
label: 'WPM speed with one scroll of the wheel',
type: 'number',
min: 1,
max: 400,
default: 30,
title: 'Changing the WPM (words per minute) speed with one scroll of the wheel'
},
SPEED_INDICATOR_TRANSPARENT: {
label: 'Transparency of the speed indicator',
type: 'number',
min: 0.1,
max: 1,
default: 0.3,
title: 'Transparency of the speed indicator on the button'
},
},
events: {
init: onInit
}
})
GM_registerMenuCommand('Settings', () => {
GM_config.open()
})
class Debugger {
constructor (name, enabled) {
this.debug = {}
if (!window.console) {
return () => { }
}
Object.getOwnPropertyNames(window.console).forEach(key => {
if (typeof window.console[key] === 'function') {
if (enabled) {
this.debug[key] = window.console[key].bind(window.console, name + ': ')
} else {
this.debug[key] = () => { }
}
}
})
return this.debug
}
}
var DEBUG
const SELECTORS = {
PLAYER: '#movie_player',
StartSegmentButton: '#startSegmentButton',
SaimaButtonContainer: '#saima-button-container',
SaimaButtonContainer_button: '#saima-button-container > div > button',
SaimaButtonContainer_button_canvas: '#saima-button-container > div > button > canvas',
SaimaTextField: '.__saima__text-field',
}
const WPM_Min = 107 // Минимальная скорость (слов в минуту)
const WPM_Max = 498 // Максимальная скорость (слов в минуту)
const WPM_IntervalLenght = WPM_Max - WPM_Min // величина рабочего диапазаона скоростей WPM
let CanvasMaskTransparent = 0.3 // прозрачность маски
let WheelWPM = 30 // Изменение скорости WPM за одну прокрутку колёсика
function onInit() {
DEBUG = new Debugger(GM_info.script.name, GM_config.get('DEBUG_MODE'))
let CanvasMaskTransparent = GM_config.get('SPEED_INDICATOR_TRANSPARENT')
WheelWPM = GM_config.get('WPM_SPEED')
Move_after_SponsorBlock()
Quick_WPM_speed_control()
}
// Перестановка на плеере кнопки Saima правее кнокпи SponsorBlock - во избежание смещений кнопки Saima при наведении на кнопку SponsorBlock
function Move_after_SponsorBlock(){
let jsInitChecktimer = null
function isDownloadElements(evt) {
function wait () {
//ожидание загрузки страницы до необходимого значения
if (watchThresholdReached()) {
try {
const StartSegmentButton = document.querySelector(SELECTORS.StartSegmentButton)
const SaimaButtonContainer = document.querySelector(SELECTORS.SaimaButtonContainer)
if (StartSegmentButton && SaimaButtonContainer) {
// Перестановка
StartSegmentButton.after(SaimaButtonContainer)
clearInterval(jsInitChecktimer)
jsInitChecktimer = null
}
} catch (e) {
DEBUG.info(`Failed to like video: ${e}. `)
}
}
}
jsInitChecktimer = setInterval(wait, GM_config.get('CHECK_FREQUENCY_LAUNCH_SPONSOR_BLOCK'))
}
isDownloadElements()
}
// Быстрое изменение скорости WPM
function Quick_WPM_speed_control() {
let jsInitChecktimer2 = null
function isDownloadStartSegmentButton(evt) {
function wait () {
//ожидание загрузки страницы до необходимого значения
if (watchThresholdReached()) {
try {
const SaimaButtonContainer = document.querySelector(SELECTORS.SaimaButtonContainer)
const SaimaTextField = document.querySelector(SELECTORS.SaimaTextField)
if (SaimaButtonContainer && SaimaTextField) {
const SaimaButtonContainer_button = document.querySelector(SELECTORS.SaimaButtonContainer_button)
if (SaimaButtonContainer_button) {
let SaimaButtonContainer_spanIco_SpeedProp = 1 - (SaimaTextField._value - WPM_Min) / WPM_IntervalLenght
let canvas = document.createElement('canvas')
let ctx = canvas.getContext('2d')
// Предполагая, что размеры контейнера уже установлены
canvas.width = SaimaButtonContainer_button.offsetWidth
canvas.height = SaimaButtonContainer_button.offsetHeight
canvas.style.position = 'absolute'
// Добавляем эффект маски
ctx.fillStyle = `rgba(0, 0, 0, ${CanvasMaskTransparent})`
// ctx.fillRect(0, 0, canvas.width, canvas.height / 2)
ctx.fillRect(0, 0, canvas.width, canvas.height * SaimaButtonContainer_spanIco_SpeedProp)
// Добавляем canvas в контейнер
SaimaButtonContainer_button.appendChild(canvas)
}
SaimaButtonContainer.addEventListener('wheel', function(event) {
// Предотвращаем стандартное поведение прокрутки страницы
event.preventDefault()
let RepeatedPresses = 1 // Количество нажатий на одну прокрутку колеса мыши
let TimeOutMS = 1 // Интервал между нажатиями
// A function to simulate a button click
function simulateClickWrap(sign) {
function simulateClick() {
let valueNew = SaimaTextField.valueAsNumber + sign * WheelWPM
switch(sign) {
case 1:
if (valueNew > WPM_Max) valueNew = WPM_Max
break
case -1:
if (valueNew < WPM_Min) valueNew = WPM_Min
break
}
SaimaTextField.value = valueNew
SaimaTextField._value = valueNew
// Создаем новое событие 'input'
let eventInput = new Event('input', {
bubbles: true, // Событие будет всплывать вверх по DOM-дереву
cancelable: false // Событие 'input' обычно не предполагает отмену его действий
});
// Искусственно вызываем событие 'input' на элементе input
SaimaTextField.dispatchEvent(eventInput)
}
if ((event.deltaY < 0 && SaimaTextField.valueAsNumber < WPM_Max) || (event.deltaY > 0 && SaimaTextField.valueAsNumber > WPM_Min)) {
for (let i = 0; i < RepeatedPresses; i++) {
setTimeout((function(index) {
return function() {
simulateClick()
// console.log('Button clicked:', index);
}
})(i), i * TimeOutMS) // Задержка TimeOutMS между кликами
}
}
const SaimaButtonContainer_button_canvas = document.querySelector(SELECTORS.SaimaButtonContainer_button_canvas)
if (SaimaButtonContainer_button_canvas) {
let SaimaButtonContainer_spanIco_SpeedProp = 1 - (SaimaTextField._value - WPM_Min) / WPM_IntervalLenght
function updateMaskHeight(newHeightFraction) {
let ctx = SaimaButtonContainer_button_canvas.getContext('2d')
// Очистка холста
ctx.clearRect(0, 0, SaimaButtonContainer_button_canvas.width, SaimaButtonContainer_button_canvas.height)
// Перерисовка маски с новой высотой
ctx.fillStyle = `rgba(0, 0, 0, ${CanvasMaskTransparent})`
ctx.fillRect(0, 0, SaimaButtonContainer_button_canvas.width, SaimaButtonContainer_button_canvas.height * newHeightFraction)
}
// Обновление высоты маски от высоты холста
updateMaskHeight(SaimaButtonContainer_spanIco_SpeedProp)
}
}
// event.deltaY содержит значение, которое указывает направление прокрутки:
// положительное значение для прокрутки вниз, отрицательное - вверх.
if (event.deltaY < 0) {
// console.log('Прокрутка вверх')
// if (SaimaTextField.valueAsNumber < WPM_Max) simulateClickWrap(1)
simulateClickWrap(1)
} else {
// console.log('Прокрутка вниз')
// if (SaimaTextField.valueAsNumber > WPM_Min) simulateClickWrap(-1)
simulateClickWrap(-1)
}
})
clearInterval(jsInitChecktimer2)
jsInitChecktimer2 = null
}
} catch (e) {
DEBUG.info(`Failed to like video: ${e}. `)
}
}
}
jsInitChecktimer2 = setInterval(wait, GM_config.get('CHECK_FREQUENCY_LAUNCH_SAIMA'))
}
isDownloadStartSegmentButton()
}
function watchThresholdReached () {
const player = document.querySelector(SELECTORS.PLAYER)
if (player) {
return true
}
return false
}
// onInit()
window.addEventListener("yt-navigate-start", e => { onInit() }) // переиницализация при обновлении страницы без перезагрузки скрипта
})();