2024-11-21 21:04:05 +03:00
|
|
|
// HTML Elements
|
|
|
|
|
|
|
|
const e_songDetails = {
|
2024-11-24 16:20:47 +03:00
|
|
|
title: document.getElementById("current-title"),
|
|
|
|
playlist: document.getElementById("current-playlist"),
|
|
|
|
artist: document.getElementById("current-artist"),
|
|
|
|
};
|
|
|
|
|
|
|
|
const e_playlist = document.getElementById("current-playlist");
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
const e_songListSection = document.getElementById("song-list");
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
const e_songList = {
|
2024-11-24 16:20:47 +03:00
|
|
|
next: document.getElementById("list-next"),
|
|
|
|
current: document.getElementById("list-current"),
|
|
|
|
previous: document.getElementById("list-previous"),
|
|
|
|
};
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
const e_listeners = {
|
2024-11-24 16:20:47 +03:00
|
|
|
current: document.querySelector(".listeners.current"),
|
|
|
|
total: document.querySelector(".listeners.total"),
|
|
|
|
unique: document.querySelector(".listeners.unique"),
|
|
|
|
};
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
const e_audio = document.getElementById("audio");
|
|
|
|
e_audio.volume = 0.2;
|
|
|
|
|
|
|
|
const e_warningMessage = document.getElementById("warning-message");
|
|
|
|
const e_player = document.getElementById("player");
|
|
|
|
const e_progressBar = document.getElementById("progress-bar");
|
|
|
|
const e_volumeBar = document.getElementById("volume-bar");
|
|
|
|
const e_channels = document.querySelectorAll(".channel");
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a random integer between 0 and a given limit.
|
|
|
|
* @param {int} limit - The maximum returnable integer
|
|
|
|
*/
|
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
const rRng = (limit) => Math.floor(Math.random() * limit);
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a random item from a given array.
|
|
|
|
* @param {Array} array - The maximum returnable integer
|
|
|
|
*/
|
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
const rArr = (array) => array[rRng(array.length)];
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
// :^)
|
|
|
|
|
|
|
|
const lainon = {
|
2024-11-24 16:20:47 +03:00
|
|
|
life: "https://radio.scuf.ru/",
|
|
|
|
};
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
// Set a random background image
|
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
const backgrounds = ["bg205.gif", "bg_307.gif", "bg_315.gif", "patternTV2.gif"];
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
document.body.style.setProperty(
|
2024-11-24 16:20:47 +03:00
|
|
|
"background-image",
|
|
|
|
`url("./public/asset/${rArr(backgrounds)}")`
|
2024-11-21 21:04:05 +03:00
|
|
|
);
|
|
|
|
|
|
|
|
// Default settings
|
|
|
|
|
|
|
|
let isPlaying = false;
|
|
|
|
let currentChannel = "cafe";
|
|
|
|
let progressStep = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a GET request to lainon.life, then do stuff with the reponse.
|
|
|
|
* @returns {Object} data - Information about the present radio state
|
|
|
|
*/
|
|
|
|
|
|
|
|
async function fetchData() {
|
2024-11-24 16:20:47 +03:00
|
|
|
// Since lainon.life doesn't allow CORS, the request must be
|
|
|
|
// made through a proxy. This "crosscloak" is my fork of
|
|
|
|
// cors-anywhere, and just in case you're thinking about it,
|
|
|
|
// no, it's not public and your domain is not on the whitelist.
|
|
|
|
|
|
|
|
const data = await fetch(`https://radio.scuf.ru/api/nowplaying/scuf_fm`)
|
|
|
|
.then((response) => response.json())
|
|
|
|
.catch((ERROR) => {
|
|
|
|
console.error(ERROR);
|
|
|
|
e_warningMessage.classList.remove("hidden");
|
|
|
|
});
|
|
|
|
|
|
|
|
parseData(data);
|
|
|
|
return data;
|
2024-11-21 21:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse the reponse data from lainon.life and display it on the page.
|
|
|
|
* @param {Object} radioData - Present state of the lainon.life radio.
|
|
|
|
*/
|
|
|
|
|
|
|
|
function parseData(radioData) {
|
2024-11-24 16:20:47 +03:00
|
|
|
// Update listeners
|
|
|
|
for (const listenerType in radioData.listeners) {
|
|
|
|
e_listeners[listenerType].innerText = radioData.listeners[listenerType];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Go over the radioData object.
|
|
|
|
// If the object exists, set the corresponding element's
|
|
|
|
// innerText to its value, or otherwise to say that it
|
|
|
|
// wasn't found.
|
|
|
|
|
|
|
|
for (const itemType in e_songDetails) {
|
|
|
|
e_songDetails[itemType].innerText = radioData.now_playing.song[itemType]
|
|
|
|
? radioData.now_playing.song[itemType]
|
|
|
|
: `${itemType.toUpperCase()} NOT FOUND`;
|
|
|
|
}
|
|
|
|
|
|
|
|
e_playlist.innerText = radioData.now_playing.playlist;
|
|
|
|
|
|
|
|
// Current song's length in seconds
|
|
|
|
const songLength = radioData.now_playing.duration;
|
|
|
|
|
|
|
|
// Increment for the progress bar as a percentage of 100
|
|
|
|
progressStep = (1 / songLength) * 100;
|
|
|
|
|
|
|
|
// Set the progress bar to display the current song's elapsed time
|
|
|
|
e_progressBar.style.setProperty(
|
|
|
|
"width",
|
|
|
|
progressStep * Math.floor(radioData.now_playing.elapsed) + "%"
|
|
|
|
);
|
|
|
|
|
|
|
|
e_songListSection.style = `background-image: url(${radioData.now_playing.song.art})`;
|
|
|
|
|
|
|
|
// Clear the entire song list
|
|
|
|
for (const element in e_songList) e_songList[element].innerHTML = "";
|
|
|
|
|
|
|
|
// Fill in the upcoming songs
|
|
|
|
const { song, duration } = radioData.playing_next;
|
|
|
|
const e_songEntry = document.createElement("div");
|
|
|
|
e_songEntry.className = "song-entry";
|
|
|
|
|
|
|
|
e_songEntry.innerHTML = `
|
2024-11-21 21:04:05 +03:00
|
|
|
<div class="entry-details">
|
|
|
|
<span class="entry-artist">${song.artist}</span>
|
|
|
|
-
|
|
|
|
<span class="entry-title">${song.title}</span>
|
|
|
|
</div>
|
|
|
|
<span class="entry-length">${toMinSec(duration).join(":")}</span>
|
|
|
|
`;
|
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
e_songList.next.appendChild(e_songEntry);
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Display the current song in the song list
|
|
|
|
e_songList.current.innerHTML = `
|
2024-11-21 21:04:05 +03:00
|
|
|
<div class="song-entry">
|
|
|
|
<div class="entry-details">
|
2024-11-24 16:20:47 +03:00
|
|
|
<span class="entry-artist">${
|
|
|
|
radioData.now_playing.song.artist
|
|
|
|
}</span>
|
2024-11-21 21:04:05 +03:00
|
|
|
-
|
2024-11-24 16:20:47 +03:00
|
|
|
<span class="entry-title">${
|
|
|
|
radioData.now_playing.song.title
|
|
|
|
}</span>
|
2024-11-21 21:04:05 +03:00
|
|
|
</div>
|
2024-11-24 16:20:47 +03:00
|
|
|
<span class="entry-length">${toMinSec(
|
|
|
|
radioData.now_playing.duration
|
|
|
|
).join(":")}</span>
|
2024-11-21 21:04:05 +03:00
|
|
|
</div>
|
|
|
|
`;
|
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Fill in the previous songs
|
|
|
|
for (const song of radioData.song_history) {
|
|
|
|
const e_songEntry = document.createElement("div");
|
|
|
|
e_songEntry.className = "song-entry";
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
e_songEntry.innerHTML = `
|
2024-11-21 21:04:05 +03:00
|
|
|
<div class="entry-details">
|
|
|
|
|
|
|
|
<span class="entry-artist">${song.song.artist}</span>
|
|
|
|
-
|
|
|
|
<span class="entry-title">${song.song.title}</span>
|
|
|
|
</div>
|
2024-11-24 16:20:47 +03:00
|
|
|
<span class="entry-length">${toMinSec(song.duration).join(
|
|
|
|
":"
|
|
|
|
)}</span>
|
2024-11-21 21:04:05 +03:00
|
|
|
`;
|
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
e_songList.previous.appendChild(e_songEntry);
|
|
|
|
}
|
2024-11-21 21:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert seconds to minutes.
|
|
|
|
* @param {int} totalSeconds - Any amount of seconds
|
|
|
|
* @returns {Array} [minutes, seconds]
|
|
|
|
*/
|
|
|
|
|
|
|
|
function toMinSec(totalSeconds) {
|
2024-11-24 16:20:47 +03:00
|
|
|
return [padZero(Math.floor(totalSeconds / 60)), padZero(totalSeconds % 60)];
|
2024-11-21 21:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Prefixes single digits with a "0".
|
|
|
|
* @param {} integer
|
|
|
|
* @returns {String} Input number converted to a string
|
|
|
|
*/
|
|
|
|
|
|
|
|
function padZero(integer) {
|
2024-11-24 16:20:47 +03:00
|
|
|
return integer < 10 ? "0" + integer : "" + integer;
|
2024-11-21 21:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Change to a different radio channel.
|
|
|
|
* @param {String} channelName
|
|
|
|
*/
|
|
|
|
|
|
|
|
function switchChannel(channelName) {
|
2024-11-24 16:20:47 +03:00
|
|
|
e_player.classList.remove("paused");
|
|
|
|
// Remove active class from all DOM channel elements
|
|
|
|
e_channels.forEach((el) => el.classList.remove("active"));
|
|
|
|
|
|
|
|
// Add it to the clicked one
|
|
|
|
document.getElementById(channelName).classList.add("active");
|
|
|
|
|
|
|
|
currentChannel = channelName;
|
|
|
|
e_audio.src = `https://radio.scuf.ru/listen/scuf_fm/radio.mp3`;
|
|
|
|
e_audio.play();
|
|
|
|
fetchData();
|
|
|
|
|
|
|
|
isPlaying = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
function pauseRadio() {
|
|
|
|
if (!isPlaying) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
e_player.classList.add("paused");
|
|
|
|
progressStep = 0;
|
|
|
|
e_channels.forEach((el) => el.classList.remove("active"));
|
|
|
|
document.getElementById("pause").classList.add("active");
|
|
|
|
e_audio.pause();
|
|
|
|
|
|
|
|
isPlaying = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function toggleMute() {
|
|
|
|
if (e_audio.muted) {
|
|
|
|
document.getElementById("mute").classList.remove("active");
|
|
|
|
e_volumeBar.classList.remove("muted");
|
|
|
|
e_audio.muted = false;
|
|
|
|
} else {
|
|
|
|
document.getElementById("mute").classList.add("active");
|
|
|
|
e_volumeBar.classList.add("muted");
|
|
|
|
e_audio.muted = true;
|
|
|
|
}
|
2024-11-21 21:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Change the volume
|
|
|
|
|
|
|
|
let volumeUpdate; // Interval container
|
|
|
|
|
|
|
|
function setVolume(mouseEvent) {
|
2024-11-24 16:20:47 +03:00
|
|
|
// If anything other than LMB is pressed, return early
|
|
|
|
if (mouseEvent.buttons != 1) return;
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Horizontal coordinate of the click
|
|
|
|
const clickPosition = mouseEvent.offsetX;
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Calculate new bar position and volume
|
|
|
|
let volumeBarWidth = (100 / mouseEvent.target.clientWidth) * clickPosition;
|
|
|
|
let audioVolume = (1 / mouseEvent.target.clientWidth) * clickPosition;
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Resize the volume bar to the new width
|
|
|
|
e_volumeBar.style.setProperty("width", `${volumeBarWidth}%`);
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Set the audio volume
|
|
|
|
e_audio.volume = audioVolume;
|
2024-11-21 21:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
e_player.onmousemove = setVolume;
|
|
|
|
e_player.onmousedown = setVolume;
|
|
|
|
|
|
|
|
// Refresh radio data when the tab is focused
|
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
window.onfocus = () => {
|
|
|
|
if (isPlaying) fetchData();
|
|
|
|
};
|
2024-11-21 21:04:05 +03:00
|
|
|
|
|
|
|
// Update the progress bar every second
|
2024-11-24 16:20:47 +03:00
|
|
|
const updateProgressBar = setInterval(() => {
|
|
|
|
// Current percentage
|
|
|
|
const width = parseFloat(e_progressBar.style.width);
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Future percentage
|
|
|
|
const futureWidth = width + progressStep;
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// If the future percentage doesn't exceed 100%
|
|
|
|
// increment the progress bar.
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
if (futureWidth < 100) {
|
|
|
|
e_progressBar.style.setProperty("width", futureWidth + "%");
|
|
|
|
}
|
2024-11-21 21:04:05 +03:00
|
|
|
|
2024-11-24 16:20:47 +03:00
|
|
|
// Otherwise, if it does exceed 100%, that means
|
|
|
|
// the song has changed, so fetch the new data.
|
|
|
|
else if (futureWidth >= 100) fetchData();
|
|
|
|
}, 1000);
|