Use the mouse wheel to adjust the volume. Middle-click to mute or unmute the player. Volume slider always visible.
// ==UserScript==
// @name Kick Volume Wheel Control
// @namespace https://github.com/pabli24
// @version 1.0.2
// @description Use the mouse wheel to adjust the volume. Middle-click to mute or unmute the player. Volume slider always visible.
// @author Pabli
// @license MIT
// @match https://kick.com/*
// @icon 
// @grant none
// ==/UserScript==
(function() {
"use strict";
const CONFIG = {
VOLUME_STEP: 1,
SHOW_CONTROLS_ON_SCROLL: true,
SLIDER_ALWAYS_VISIBLE: true,
HIDE_CURSOR_DELAY: 4000
};
function setCookie(name, value, days) {
const date = new Date();
date.setTime(date.getTime() + days * 24 * 60 * 60 * 1000);
document.cookie = `${name}=${value}; expires=${date.toUTCString()}; path=/`;
}
function prevent(e) {
e.preventDefault();
e.stopPropagation();
e.stopImmediatePropagation();
}
const observer = new MutationObserver(mutations => {
const video = document.getElementById("video-player");
if (!video) return;
wheel(video);
});
observer.observe(document.body, { childList: true, subtree: true });
function wheel(video) {
const videoDiv = document.querySelector("#injected-embedded-channel-player-video > div");
if (videoDiv.hasAttribute("kpvolume")) return;
videoDiv.setAttribute("kpvolume", "");
videoDiv.addEventListener("wheel", (event) => {
prevent(event);
if (CONFIG.SHOW_CONTROLS_ON_SCROLL === true) {
const showEvent = new Event('mousemove');
videoDiv.dispatchEvent(showEvent);
}
if (video.muted && videoDiv.getAttribute("kpvolume")) {
video.muted = false;
setTimeout(() => {
video.volume = videoDiv.getAttribute("kpvolume");
slider();
}, 50)
} else if (event.deltaY < 0) {
video.volume = Math.min(1, video.volume + (CONFIG.VOLUME_STEP / 100)); // Increase volume
} else if (event.deltaY > 0) {
video.volume = Math.max(0, video.volume - (CONFIG.VOLUME_STEP / 100)); // Decrease volume
}
setTimeout(slider, 50);
setTimeout(() => setCookie("volume", video.volume, 365), 3000);
});
let hideCursorTimeout;
videoDiv.addEventListener("mousemove", (event) => {
setTimeout(() => {
muteBtn();
slider();
}, 50)
setTimeout(() => setCookie("volume", video.volume, 365), 3000);
if (videoDiv.contains(event.target)) {
videoDiv.style.cursor = 'default';
if (hideCursorTimeout) clearTimeout(hideCursorTimeout);
hideCursorTimeout = setTimeout(() => {
videoDiv.style.cursor = 'none';
}, CONFIG.HIDE_CURSOR_DELAY);
}
});
videoDiv.addEventListener("mouseenter", (event) => {
setTimeout(() => setCookie("volume", video.volume, 365), 100);
});
function slider() {
const controls = videoDiv.querySelector('div > div.z-controls');
if (!controls) return;
const sliderFill = controls.querySelector('span[style*="right:"]'); // style="left: 0%; right: 40%;"
const videoVolume = Math.round(video.volume * 100);
sliderFill.style.right = `${100 - videoVolume}%`;
const sliderThumb = controls.querySelector('span[style*="transform: var(--radix-slider-thumb-transform)"]'); // left: calc(40% + 1.6px);
const offset = 8 + (videoVolume / 100) * -16;
sliderThumb.style.left = `calc(${videoVolume}% + ${offset}px)`;
const sliderValuenow = controls.querySelector('span[aria-valuenow]'); // aria-valuenow="40"
sliderValuenow.setAttribute("aria-valuenow", videoVolume);
const sliderP = controls.querySelector('.group\\/volume .betterhover\\:group-hover\\/volume\\:flex');
sliderP.setAttribute("playervolume", videoVolume + "%");
}
function muteBtn() {
const muteButton = document.querySelector('#injected-embedded-channel-player-video > div > div.z-controls .group\\/volume > button');
if (!muteButton) return;
muteButton.addEventListener("click", (event) => {
prevent(event);
mute();
});
}
function mute() {
if (video.muted) {
video.muted = false;
setTimeout(() => {
video.volume = videoDiv.getAttribute("kpvolume");
slider();
}, 50)
} else {
videoDiv.setAttribute("kpvolume", video.volume);
video.muted = true;
}
}
videoDiv.addEventListener("mousedown", ({ button }) => {
if (event.button === 1) {
prevent(event);
mute();
}
});
document.addEventListener("keydown", (event) => {
if ((event.key === 'M' || event.key === 'm') && event.target.tagName !== 'INPUT' && event.target.tagName !== 'TEXTAREA' && event.target.isContentEditable !== true) {
prevent(event);
mute();
}
});
}
let styles = `
#injected-embedded-channel-player-video > div > div.z-controls .group\\/volume .betterhover\\:group-hover\\/volume\\:flex::after {
content: attr(playervolume);
font-weight: 600;
font-size: .875rem;
line-height: 1.25rem;
margin-left: .5rem;
width: 4ch;
}`
if (CONFIG.SLIDER_ALWAYS_VISIBLE === true) {
styles += `
#injected-embedded-channel-player-video > div > div.z-controls .group\\/volume .betterhover\\:group-hover\\/volume\\:flex {
display: flex;
align-items: center;
}`
}
const styleSheet = document.createElement("style");
styleSheet.textContent = styles;
document.head.appendChild(styleSheet);
})();