// ==UserScript==
// @name Holodex for YouTube
// @description A userscript that adds a player in your YouTube page when current video has valid timeline on Holodex.
// @author Allen Wei
// @namespace https://github.com/Alllen95Wei
// @license MIT
// @match https://www.youtube.com/*
// @match https://holodex.net/*
// @version 1.0.3
// @grant GM_getValues
// @grant GM_setValues
// ==/UserScript==
let HOLODEX_API_KEY = GM_getValues({holodex_api_key: null}).holodex_api_key
const STYLE_TEXT = `
/*
* Below is the stylesheet injected by "Holodex for YouTube" and is for elements created by the userscript.
* Don't worry; other elements shouldn't be affected!
*/
table, th, td {
border: 1px solid #000000;
border-collapse: collapse;
}
th {
background-color: #f06292;
}
td {
background-color: #9cc9fd;
}
button:hover {
cursor: pointer;
}
#song-list-btn {
border: none;
background-image: linear-gradient(to bottom right, #5da2f2, #f06292);
}
#song-list-btn:hover {
background-image: linear-gradient(to bottom right, #7bb1f3, #ed82a8);
}
#song-list-div {
font-size: small;
white-space: pre-line;
}
#song-list-table-div {
max-height: 75vh;
margin: 0.5vh;
overflow: auto;
border-radius: 12px;
}
#autoplay-div {
width: 100%;
background-color: white;
}
#autoplay-switch-label {
margin-left: 5px;
}
#song-info-div {
width: 100%;
overflow: auto;
display: flex;
align-items: center;
}
#hd-thumbnail {
margin: 5px;
width: 80px;
height: 80px;
border-radius: 10px;
box-shadow: 0 0 5px 0 #A5A5A5;
transition: width 0.1s linear, height 0.1s linear;
}
#hd-thumbnail:hover {
width: 110px;
height: 110px;
transition: width 0.1s linear, height 0.1s linear;
}
#song-detail-div {
display: inline-grid;
}
#song-info-name {
font-size: large;
display: inline-block;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
margin-right: 5px;
margin-bottom: 0.05em;
}
#song-info-artist {
font-size: small;
display: inline-block;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
margin-right: 5px;
margin-bottom: 0.1em;
}
#autoplay-div {
position: sticky;
top: 0;
}
#control-div {
display: flex;
align-items: center;
justify-content: center;
}
#progress-div {
width: 100%;
display: flex;
}
#progress-bar {
-webkit-appearance: none;
width: 80%;
display: flex;
appearance: none;
background: transparent;
cursor: pointer;
}
#progress-bar:focus {
outline: none;
}
/******** Chrome, Safari, Opera and Edge Chromium styles ********/
/* slider track */
#progress-bar::-webkit-slider-runnable-track {
background-image: linear-gradient(to right, #81b7f3, #ef86aa);
border-radius: 0.5rem;
height: 0.25rem;
}
/* slider thumb */
#progress-bar::-webkit-slider-thumb {
-webkit-appearance: none; /* Override default look */
appearance: none;
margin-top: -4px; /* Centers thumb on the track */
background-color: #808080;
border-radius: 0.1rem;
height: 1.2rem;
width: 0.3rem;
}
#progress-bar:focus::-webkit-slider-thumb {
outline: 3px solid #808080;
outline-offset: 0.125rem;
}
/*********** Firefox styles ***********/
/* slider track */
#progress-bar::-moz-range-track {
background-image: linear-gradient(to right, #81b7f3, #ef86aa);
border-radius: 0.5rem;
height: 0.25rem;
}
/* slider thumb */
#progress-bar::-moz-range-thumb {
background-color: #808080;
border: none; /*Removes extra border that FF applies*/
border-radius: 0.1rem;
height: 1rem;
width: 0.2rem;
}
#progress-bar:focus::-moz-range-thumb{
outline: 3px solid #808080;
outline-offset: 0.125rem;
}
.song-name-div {
height: 40px;
max-height: 40px;
overflow: auto;
align-content: center;
}
.song-artist {
font-size: 10px;
font-style: italic;
}
.hd-button {
border: none;
width: 100%;
height: auto;
padding: 5px;
transition-duration: 0.15s;
}
.hd-button:hover {
background-color: #b6b6b6;
}
.play-button {
border: none;
width: 40px;
height: 40px;
background-size: cover;
background-color: #7f7f7f;
}
.progress-span {
width: 10%;
display: flex;
justify-content: center;
}
.control-buttons {
border: none;
margin: 2px 4px;
width: 3.5vw;
background-color: darkgray;
}
/* ============= Google Material Symbols Stylesheet ============= */
.material-symbols-rounded {
font-variation-settings:
'FILL' 1,
'wght' 400,
'GRAD' 200,
'opsz' 24
}
`
const DEFAULT_ART_BASE64 = ""
const GOOGLE_ICONS_URL = [
// "https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@40,500,1,200&icon_names=skip_previous",
// "https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@40,500,1,200&icon_names=play_arrow",
// "https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@40,500,1,200&icon_names=skip_next"
"https://fonts.googleapis.com/css2?family=Material+Symbols+Rounded:opsz,wght,FILL,[email protected],100..700,0..1,-50..20"
]
const escapeHTMLPolicy = trustedTypes.createPolicy("forceInner", {createHTML: (to_escape) => to_escape})
let UPDATE_TRY = 0
let LAST_SONG_ID = ""
let LAST_SONG_INDEX = 0
let IS_RANDOM = false
/*
* 0 = no repeat \
* 1 = repeat one \
* 2 = repeat all
*/
let REPEAT_MODE = 0
let QUEUE = []
let SONG_INDEX = []
// Utils
function secondsToFormattedString(seconds) {
return `${Math.floor(seconds / 60)}:${seconds % 60 < 10 ? "0" : ""}${Math.floor(seconds % 60)}`
}
function compareSongSequence(a, b) {
if (a["start"] > b["start"]) {
return 1
}
if (a["start"] < b["start"]) {
return -1
}
return 0
}
function shuffle(array) {
for (let i = array.length - 1; i > 0; i--) {
let j = Math.floor(Math.random() * (i + 1));
[array[i], array[j]] = [array[j], array[i]];
}
return array;
}
function autoplay(videoPlayer, updateNow, goNext, goPrevious) {
UPDATE_TRY++
// check every 4 triggers (see https://developer.mozilla.org/en-US/docs/Web/API/HTMLMediaElement/timeupdate_event)
if (UPDATE_TRY >= 4 || updateNow) {
UPDATE_TRY = 0
const currentTime = videoPlayer.currentTime
let nextSong = null
for (const [start, end, i] of SONG_INDEX) {
if (start <= currentTime && currentTime <= end) { // now playing
const nowPlaying = QUEUE[i]
console.log(`Now playing: ${nowPlaying["name"]}, by ${nowPlaying["original_artist"]}`)
LAST_SONG_INDEX = i
if (goNext) {
nextSong = QUEUE[i + 1] || null
} else if (goPrevious) {
nextSong = QUEUE[i - 1] || null
} else {
nextSong = "PLAYING"
}
break
} else if (!IS_RANDOM && start > currentTime) { // finds the next song only when "shuffle" is not enabled
// if (nextSong === songList[i]) { // the next song is confirmed and will NOT change
// break
// }
nextSong = QUEUE[i]
break
} else if (IS_RANDOM) { // the playlist is random and no song is playing
nextSong = QUEUE[LAST_SONG_INDEX + 1] || null
break
}
}
if (nextSong !== null && nextSong !== "PLAYING") {
console.log("Next song:", nextSong)
console.log(`Seeking to ${nextSong["start"]}`)
videoPlayer.currentTime = nextSong["start"]
} else if (nextSong !== "PLAYING") {
console.log(nextSong)
console.log("The last song in the list has finished, pausing the player")
videoPlayer.pause()
}
}
}
function getNowPlayingSong(videoPlayer) {
const currentTime = videoPlayer.currentTime
let i = 0
for (const song of QUEUE) {
if (song.start <= currentTime && currentTime <= song.end) {
return [song, i]
}
i++
}
return [{
"id": null,
"name": "未在播放任何歌曲",
"original_artist": "啟用自動播放以使用播放器功能",
"art": DEFAULT_ART_BASE64,
"start": 0,
"end": 0
}, -1]
}
function updateNowPlayingInfo(videoPlayer, forceUpdate) {
const [nowSong, index] = getNowPlayingSong(videoPlayer)
const currentTime = videoPlayer.currentTime
const progressBar = document.getElementById("progress-bar")
if (nowSong.id !== LAST_SONG_ID || forceUpdate) {
// Update song info
document.getElementById("hd-thumbnail").src = (nowSong.art || DEFAULT_ART_BASE64).replace("100x100", "500x500")
document.getElementById("song-info-name").textContent = nowSong.name
document.getElementById("song-info-name").title = nowSong.name
document.getElementById("song-info-artist").textContent = nowSong.original_artist
document.getElementById("song-info-artist").title = nowSong.original_artist
// Update time length of the song
const duration = nowSong.end - nowSong.start
document.getElementById("progress-duration").textContent = secondsToFormattedString(duration)
LAST_SONG_ID = nowSong.id
const controlBtns = document.getElementsByClassName("control-buttons")
if (nowSong.id === null) {
for (const btn of controlBtns) {
btn.disabled = true
}
progressBar.disabled = true
} else {
for (const btn of controlBtns) {
btn.disabled = false
}
progressBar.disabled = false
// Disable skip buttons on the first/last one
if (index === QUEUE.length - 1) {
document.getElementById("skip-next-btn").disabled = true
}
if (index === 0) {
document.getElementById("skip-previous-btn").disabled = true
}
}
}
// Update progress bar
if (nowSong.id === null) {
progressBar.min = 0
progressBar.max = 1
progressBar.value = 0
document.getElementById("progress-nowtime").textContent = "0:00"
document.getElementById("progress-nowtime").title = "-0:00"
} else {
progressBar.min = `${nowSong.start}`
progressBar.max = `${nowSong.end}`
progressBar.value = currentTime
document.getElementById("progress-nowtime").textContent = secondsToFormattedString(currentTime - nowSong.start)
document.getElementById("progress-nowtime").title = `-${secondsToFormattedString(nowSong.end - currentTime)}`
}
}
function renderQueue(songListTable, videoPlayer) {
const tableLength = songListTable.rows.length
for (let i = 0; i < (tableLength - 1); i++) {
songListTable.deleteRow(-1)
}
let index = 1
for (const song of QUEUE) {
const row = songListTable.insertRow()
// Song no.
const songNoCell = row.insertCell()
songNoCell.style.textAlign = "center"
songNoCell.textContent = `${index}`
// Song name
const songNameCell = row.insertCell()
songNameCell.style.paddingLeft = "0.3vw"
const songNameDiv = document.createElement("div")
songNameDiv.className = "song-name-div"
const songName = document.createElement("p")
songName.className = "song-name"
songName.textContent = song["name"]
const songArtist = document.createElement("p")
songArtist.className = "song-artist"
songArtist.textContent = song["original_artist"]
songNameDiv.append(songName, songArtist)
songNameCell.append(songNameDiv)
// Song duration
const songDurationCell = row.insertCell()
songDurationCell.style.textAlign = "center"
songDurationCell.textContent = secondsToFormattedString(song["end"] - song["start"])
// Play button
const playButton = document.createElement("button")
playButton.id = `play-button-${index}`
playButton.className = "play-button"
playButton.textContent = "▶️"
playButton.style.backgroundImage = `url(${song["art"]})`
// Left click: seek to the start of the song
playButton.addEventListener("click", () => {
videoPlayer.currentTime = song["start"]
})
// Right click: open in Musicdex
playButton.addEventListener("contextmenu", () => {
window.open(`https://music.holodex.net/song/${song["id"]}`, "_blank")
})
row.insertCell().appendChild(playButton)
index++
}
}
function generateQueue(songListTable, videoPlayer, start, isRandom) {
if (isRandom) {
QUEUE = shuffle(QUEUE)
} else {
QUEUE = QUEUE.sort(compareSongSequence)
}
console.log("New queue:", QUEUE)
renderQueue(songListTable, videoPlayer)
let index = 0
SONG_INDEX = []
for (const song of QUEUE) {
// Append to index
SONG_INDEX.push([song["start"], song["end"], index])
index++
}
}
// Main function
function main() {
// Try to remove previous ones
LAST_SONG_ID = ""
LAST_SONG_INDEX = 0
IS_RANDOM = false
QUEUE = []
SONG_INDEX = []
try {
document.getElementById("song-list-div").remove()
} catch (e) {} // Ignore error when nothing found
const vId = new URLSearchParams(window.location.search).get("v")
if (vId === null) {
console.log("No video ID found, not a video page.")
return
}
let data = {}
if (HOLODEX_API_KEY === null || HOLODEX_API_KEY === undefined) {
console.warn("No API key found. Visit https://holodex.net/ and login to update your API key.")
return
} else {
console.log(`Holodex API key found: ${HOLODEX_API_KEY.slice(0, 9)}****-****-****-************`)
}
fetch(`https://holodex.net/api/v2/videos/${vId}`, {headers: {"X-APIKEY": HOLODEX_API_KEY}})
.then((response) => {
console.log("API responded with status code: ", response.status)
return response.json()
})
.then(response => {
data = response
if (!("songs" in data)) {
console.log("No songs found.")
return
}
// Get song list
QUEUE = data.songs
console.log(QUEUE)
// Inject CSS
const stylesheet = document.createElement("style");
stylesheet.textContent = STYLE_TEXT;
document.head.append(stylesheet)
for (const url of GOOGLE_ICONS_URL) {
const googleStylesheet = document.createElement("link")
googleStylesheet.rel = "stylesheet"
googleStylesheet.href = url
document.head.append(googleStylesheet)
}
// Fetch YouTube elements
const minorColumn = document.getElementById("secondary-inner")
const videoPlayer = document.getElementsByTagName("video")[0]
// Create song list div
const songListDiv = document.createElement("div")
songListDiv.id = "song-list-div"
// Create div for table to prevent super long table
const songListTableDiv = document.createElement("div")
songListTableDiv.id = "song-list-table-div"
songListTableDiv.style.height = "0px"
songListTableDiv.style.visibility = "hidden"
// Create "autoplay" switch
const autoplayDiv = document.createElement("div")
autoplayDiv.id = "autoplay-div"
const autoplaySwitchLabel = document.createElement("label")
autoplaySwitchLabel.id = "autoplay-switch-label"
autoplaySwitchLabel.htmlFor = "autoplay-switch"
autoplaySwitchLabel.textContent = "啟用自動播放"
const autoplaySwitch = document.createElement("input")
autoplaySwitch.id = "autoplay-switch"
autoplaySwitch.type = "checkbox"
autoplayDiv.append(autoplaySwitchLabel, autoplaySwitch)
songListTableDiv.append(autoplayDiv)
// Create player
const playerDiv = document.createElement("div")
playerDiv.id = "player-div"
// Create info div
const songInfoDiv = document.createElement("div")
songInfoDiv.id = "song-info-div"
const thumbnail = document.createElement("img")
thumbnail.id = "hd-thumbnail"
const songDetailDiv = document.createElement("div")
songDetailDiv.id = "song-detail-div"
const songName = document.createElement("span")
songName.id = "song-info-name"
const songArtist = document.createElement("span")
songArtist.id = "song-info-artist"
songDetailDiv.append(songName, songArtist)
songInfoDiv.append(thumbnail, songDetailDiv)
// Create control div
const controlDiv = document.createElement("div")
controlDiv.id = "control-div"
controlDiv.style.display = "flex"
const progressDiv = document.createElement("div")
progressDiv.id = "progress-div"
const nowTime = document.createElement("span")
nowTime.id = "progress-nowtime"
nowTime.className = "progress-span"
nowTime.textContent = "0:00"
const duration = document.createElement("span")
duration.id = "progress-duration"
duration.className = "progress-span"
duration.textContent = "0:00"
const progressBar = document.createElement("input")
progressBar.id = "progress-bar"
progressBar.type = "range"
progressBar.addEventListener("input", () => {
videoPlayer.currentTime = parseInt(progressBar.value)
})
progressDiv.append(nowTime, progressBar, duration)
const repeatBtn = document.createElement("button")
repeatBtn.classList.add("material-symbols-rounded", "control-buttons")
repeatBtn.id = "repeat-btn"
repeatBtn.textContent = "repeat"
repeatBtn.addEventListener("click", () => {
if (REPEAT_MODE === 2) {
REPEAT_MODE = 0
} else {
REPEAT_MODE++
}
if (REPEAT_MODE === 0) {
repeatBtn.textContent = "repeat"
} else if (REPEAT_MODE === 1) {
repeatBtn.textContent = "repeat_one_on"
} else if (REPEAT_MODE === 2) {
repeatBtn.textContent = "repeat_on"
}
})
const previousBtn = document.createElement("button")
previousBtn.classList.add("material-symbols-rounded", "control-buttons")
previousBtn.id= "skip-previous-btn"
previousBtn.textContent = "skip_previous"
const playBtn = document.createElement("button")
playBtn.classList.add("material-symbols-rounded", "control-buttons")
playBtn.id = "play-pause-btn"
playBtn.textContent = "pause"
playBtn.addEventListener("click", () => {
if (videoPlayer.paused) {
videoPlayer.play().then()
} else {
videoPlayer.pause()
}
})
const nextBtn = document.createElement("button")
nextBtn.classList.add("material-symbols-rounded", "control-buttons")
nextBtn.id = "skip-next-btn"
nextBtn.textContent = "skip_next"
const shuffleBtn = document.createElement("button")
shuffleBtn.classList.add("material-symbols-rounded", "control-buttons")
shuffleBtn.id = "shuffle-btn"
shuffleBtn.textContent = "shuffle"
shuffleBtn.addEventListener("click", () => {
let [_, nowSongIndex] = getNowPlayingSong(videoPlayer)
if (IS_RANDOM) {
shuffleBtn.textContent = "shuffle"
} else {
shuffleBtn.textContent = "shuffle_on"
}
IS_RANDOM = !IS_RANDOM
generateQueue(songListTable, videoPlayer, nowSongIndex + 1, IS_RANDOM)
updateNowPlayingInfo(videoPlayer, true)
})
controlDiv.append(repeatBtn, previousBtn, playBtn, nextBtn, shuffleBtn)
playerDiv.append(songInfoDiv, progressDiv, controlDiv)
autoplayDiv.append(playerDiv)
// Create song list table
const songListTable = document.createElement("table")
songListTable.id = "song-list-table"
songListTable.style.width = "100%"
songListTable.style.backgroundColor = "#FFFFFF"
songListTable.innerHTML = escapeHTMLPolicy.createHTML(`
<tr style="height: fit-content">
<th style="width: 5%">#</th>
<th style="width: 75%">歌名</th>
<th style="width: 10%">時長</th>
<th style="width: 40px">播放</th>
</tr>
`)
generateQueue(songListTable, videoPlayer, 0, false)
// Set up autoplay after loop
// Wrap autoplay as a lambda object, so that we can remove it when needed
function apl() {
autoplay(videoPlayer, false, false, false)
}
autoplaySwitch.addEventListener("change", () => {
console.log("Autoplay enabled: ", autoplaySwitch.checked)
if (autoplaySwitch.checked) {
videoPlayer.addEventListener("timeupdate", apl)
} else {
videoPlayer.removeEventListener("timeupdate", apl)
}
})
// Remove the old autoplay instance before switching to other videos
document.addEventListener("yt-navigate-start", () => {
try {
videoPlayer.removeEventListener("timeupdate", apl)
} catch (e) {}
})
// Set up song info updater
// Same reason as autoplay
function unpil() {
updateNowPlayingInfo(videoPlayer, false)
}
videoPlayer.addEventListener("timeupdate", unpil)
document.addEventListener("yt-navigate-start", () => {
try {
videoPlayer.removeEventListener("timeupdate", unpil)
} catch (e) {}
})
videoPlayer.addEventListener("play", () => {
playBtn.textContent = "pause"
})
videoPlayer.addEventListener("pause", () => {
playBtn.textContent = "play_arrow"
})
// Set up control buttons
function appl() {
autoplay(videoPlayer, true, false, true)
}
previousBtn.addEventListener("click", appl)
function apnl() {
autoplay(videoPlayer, true, true, false)
}
nextBtn.addEventListener("click", apnl)
// Append the table into its div before it appends into the main div
songListTableDiv.append(songListTable)
songListDiv.append(songListTableDiv)
// Create "toggle song list" button
const songListBtn = document.createElement("button")
songListBtn.id = "song-list-btn"
songListBtn.classList.add(
"hd-button",
"yt-spec-button-shape-next",
"yt-spec-button-shape-next--outline",
"yt-spec-button-shape-next--mono",
"yt-spec-button-shape-next--size-m",
"yt-spec-button-shape-next--enable-backdrop-filter-experiment"
)
songListBtn.textContent = "開啟歌曲清單"
songListBtn.addEventListener("click", () => {
if (songListTableDiv.style.visibility !== "visible") {
songListBtn.textContent = "關閉歌曲清單"
songListTableDiv.style.visibility = "visible"
songListTableDiv.style.height = ""
} else {
songListBtn.textContent = "開啟歌曲清單"
songListTableDiv.style.visibility = "hidden"
songListTableDiv.style.height = "0px"
}
})
songListDiv.prepend(songListBtn)
minorColumn.prepend(songListDiv)
})
.catch((error) => {
console.error(error)
})
}
(function () {
"use strict";
console.log("Extension starts")
if (window.location.hostname === "holodex.net") {
console.log("We're in Holodex!")
const rawData = JSON.parse(window.localStorage.getItem("holodex-v2"))
if (rawData.userdata.user === null) {
console.warn("Not logged in. Login to update your API key.")
return
}
if (HOLODEX_API_KEY === null || HOLODEX_API_KEY === undefined) {
console.log("API key is missing, fetching")
} else if (HOLODEX_API_KEY !== rawData.userdata.user.api_key) {
console.log("New API key found, updating")
} else {
console.log("Your API key is good to go!")
return
}
HOLODEX_API_KEY = rawData.userdata.user.api_key
GM_setValues({holodex_api_key: HOLODEX_API_KEY})
console.log(`Your new API key: ${HOLODEX_API_KEY.slice(0, 9)}****-****-****-************`)
}
// 程式碼開始
// if (new URLSearchParams(window.location.search).get("v") !== null) {
// document.addEventListener("load", main)
// }
// document.addEventListener("yt-navigate-finish", main)
// main()
document.addEventListener("yt-navigate-finish", () => {
console.log("yt-navigate-finish")
main()
});
})()