531 lines
23 KiB
HTML
Executable File
531 lines
23 KiB
HTML
Executable File
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
|
|
<html>
|
|
<head>
|
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
|
|
<title>Directory listing for /</title>
|
|
</head>
|
|
|
|
<script>
|
|
// [[file:m3u-player.org::*The script][The script:1]]
|
|
// @license magnet:?xt=urn:btih:cf05388f2679ee054f2beb29a391d25f4e673ac3&dn=gpl-2.0.txt GPL-v2-or-Later
|
|
const nodes = document.querySelectorAll("audio,video");
|
|
const playlists = {};
|
|
const prefetchedTracks = new Map(); // use a map for insertion order, so we can just blow away old entries.
|
|
// maximum prefetched blobs that are kept.
|
|
const MAX_PREFETCH_KEEP = 10;
|
|
// maximum allowed number of entries in a playlist to prevent OOM attacks against the browser with self-referencing playlists
|
|
const MAX_PLAYLIST_LENGTH = 1000;
|
|
const PLAYLIST_MIME_TYPES = ["audio/x-mpegurl", "audio/mpegurl", "application/vnd.apple.mpegurl","application/mpegurl","application/x-mpegurl"];
|
|
function stripUrlParameters(link) {
|
|
const url = new URL(link, window.location);
|
|
url.search = "";
|
|
url.hash = "";
|
|
return url.href;
|
|
}
|
|
function isPlaylist(link) {
|
|
const linkHref = stripUrlParameters(link);
|
|
return linkHref.endsWith(".m3u") || linkHref.endsWith(".m3u8");
|
|
}
|
|
function isBlob(link) {
|
|
return new URL(link, window.location).protocol == 'blob';
|
|
}
|
|
function parsePlaylist(textContent) {
|
|
return textContent.match(/^(?!#)(?!\s).*$/mg)
|
|
.filter(s => s); // filter removes empty strings
|
|
}
|
|
/**
|
|
* Download the given playlist, parse it, and store the tracks in the
|
|
* global playlists object using the url as key.
|
|
*
|
|
* Runs callback once the playlist downloaded successfully.
|
|
*/
|
|
function fetchPlaylist(url, onload, onerror) {
|
|
const playlistFetcher = new XMLHttpRequest();
|
|
playlistFetcher.open("GET", url, true);
|
|
playlistFetcher.responseType = "blob"; // to get a mime type
|
|
playlistFetcher.onload = () => {
|
|
if (PLAYLIST_MIME_TYPES.includes(playlistFetcher.response.type)) { // security check to ensure that filters have run
|
|
const reader = new FileReader();
|
|
const load = onload; // propagate to inner scope
|
|
reader.addEventListener("loadend", e => {
|
|
playlists[url] = parsePlaylist(reader.result);
|
|
onload();
|
|
});
|
|
reader.readAsText(playlistFetcher.response);
|
|
} else {
|
|
console.error("playlist must have one of the playlist MIME type '" + PLAYLIST_MIME_TYPES + "' but it had MIME type '" + playlistFetcher.response.type + "'.");
|
|
onerror();
|
|
}
|
|
};
|
|
playlistFetcher.onerror = onerror;
|
|
playlistFetcher.abort = onerror;
|
|
playlistFetcher.send();
|
|
}
|
|
function servedPartialDataAndCanRequestAll (xhr) {
|
|
if (xhr.status === 206) {
|
|
if (xhr.getResponseHeader("content-range").includes("/")) {
|
|
if (!xhr.getResponseHeader("content-range").includes("/*")) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
function prefetchTrack(url, onload) {
|
|
if (prefetchedTracks.has(url)) {
|
|
return;
|
|
}
|
|
// first cleanup: kill the oldest entries until we're back at the allowed size
|
|
while (prefetchedTracks.size > MAX_PREFETCH_KEEP) {
|
|
const key = prefetchedTracks.keys().next().value;
|
|
const track = prefetchedTracks.get(key);
|
|
prefetchedTracks.delete(key);
|
|
}
|
|
// first set the prefetched to the url so we will never request twice
|
|
prefetchedTracks.set(url, url);
|
|
// now start replacing it with a blob
|
|
const xhr = new XMLHttpRequest();
|
|
xhr.open("GET", url, true);
|
|
xhr.responseType = "blob";
|
|
xhr.onload = () => {
|
|
if (servedPartialDataAndCanRequestAll(xhr)) {
|
|
const endRange = Number(xhr.getResponseHeader("content-range").split("/")[1]) - 1;
|
|
const rangeXhr = new XMLHttpRequest();
|
|
rangeXhr.open("GET", url, true);
|
|
rangeXhr.responseType = "blob";
|
|
rangeXhr.setRequestHeader("range", "bytes=0-" + endRange);
|
|
rangeXhr.onload = () => {
|
|
prefetchedTracks.set(url, rangeXhr.response);
|
|
if (onload) {
|
|
onload();
|
|
}
|
|
};
|
|
rangeXhr.send();
|
|
} else {
|
|
prefetchedTracks.set(url, xhr.response);
|
|
if (onload) {
|
|
onload();
|
|
}
|
|
}
|
|
};
|
|
xhr.send();
|
|
}
|
|
|
|
function showStaticOverlay(mediaTag, canvas) {
|
|
if (mediaTag instanceof Audio) {
|
|
return;
|
|
}
|
|
// take screenshot of video and overlay it to mask short-term flicker.
|
|
const realWidth = mediaTag.getBoundingClientRect().width;
|
|
const realHeight = mediaTag.getBoundingClientRect().height;
|
|
canvas.width = realWidth;
|
|
canvas.height = realHeight;
|
|
// need the actual video size
|
|
const videoAspectRatio = mediaTag.videoHeight / mediaTag.videoWidth;
|
|
const tagAspectRatio = realHeight / realWidth;
|
|
const videoIsPartialHeight = tagAspectRatio > (videoAspectRatio * 1.01); // avoid rounding errors
|
|
const videoIsPartialWidth = videoAspectRatio > (tagAspectRatio * 1.01); // avoid rounding errors
|
|
if (videoIsPartialHeight) {
|
|
canvas.height = realWidth * videoAspectRatio;
|
|
} else if (videoIsPartialWidth) {
|
|
canvas.width = realHeight / videoAspectRatio;
|
|
}
|
|
const context = canvas.getContext("2d");
|
|
context.scale(canvas.width / mediaTag.videoWidth, canvas.height / mediaTag.videoHeight);
|
|
context.drawImage(mediaTag, 0, 0);
|
|
canvas.hidden = true;
|
|
mediaTag.parentNode.insertBefore(canvas, mediaTag.nextSibling);
|
|
canvas.style.position = "absolute";
|
|
// shift canvas to cover only the space where the video actually is
|
|
if (videoIsPartialWidth) {
|
|
canvas.style.marginLeft = "-" + ((realWidth + canvas.width) / 2.) + "px";
|
|
} else {
|
|
canvas.style.marginLeft = "-" + realWidth + "px";
|
|
}
|
|
if (videoIsPartialHeight) {
|
|
canvas.style.marginTop = ((realHeight - canvas.height) / 2.) + "px";
|
|
}
|
|
canvas.hidden = false;
|
|
}
|
|
|
|
function updateSrc(mediaTag, callback) {
|
|
const playlistUrl = mediaTag.getAttribute("playlist");
|
|
const trackIndex = mediaTag.getAttribute("track-index");
|
|
// deepcopy playlists to avoid shared mutation
|
|
let playlist = [...playlists[playlistUrl]];
|
|
let trackUrl = playlist[trackIndex];
|
|
// download and splice in playlists as needed
|
|
if (isPlaylist(trackUrl)) {
|
|
if (playlist.length >= MAX_PLAYLIST_LENGTH) {
|
|
// skip playlist if we already have too many tracks
|
|
changeTrack(mediaTag, +1);
|
|
} else {
|
|
// do not use the cached playlist here, though it is tempting: it might genuinely change to allow for updates
|
|
fetchPlaylist(
|
|
trackUrl,
|
|
() => {
|
|
playlist.splice(trackIndex, 1, ...playlists[trackUrl]);
|
|
playlists[playlistUrl] = playlist;
|
|
updateSrc(mediaTag, callback);
|
|
},
|
|
() => callback());
|
|
}
|
|
} else {
|
|
let url = prefetchedTracks.has(trackUrl)
|
|
? prefetchedTracks.get(trackUrl) instanceof Blob
|
|
? URL.createObjectURL(prefetchedTracks.get(trackUrl))
|
|
: trackUrl : trackUrl;
|
|
const oldUrl = mediaTag.getAttribute("src");
|
|
// prevent size flickering by setting height before src change
|
|
const canvas = document.createElement("canvas");
|
|
if (!isNaN(mediaTag.duration) // already loaded a valid file
|
|
&& document.fullscreen !== true) { // overlay does not work for fullscreen
|
|
// mask flickering with a static overlay
|
|
try {
|
|
showStaticOverlay(mediaTag, canvas);
|
|
} catch (error) {
|
|
console.log(error);
|
|
}
|
|
}
|
|
// force sizes to stay constant during loading of the next segment
|
|
mediaTag.style.height = mediaTag.getBoundingClientRect().height.toString() + 'px';
|
|
mediaTag.style.width = mediaTag.getBoundingClientRect().width.toString() + 'px';
|
|
// swich to the next segment
|
|
mediaTag.setAttribute("src", url);
|
|
mediaTag.oncanplaythrough = () => {
|
|
if (!isNaN(mediaTag.duration)) { // already loaded a valid file
|
|
// unset element styles to allow recomputation if sizes changed
|
|
mediaTag.style.height = null;
|
|
mediaTag.style.width = null;
|
|
}
|
|
// remove overlay
|
|
canvas.hidden = true;
|
|
canvas.remove(); // to allow garbage collection
|
|
};
|
|
setTimeout(() => canvas.remove(), 300); // fallback
|
|
// replace the url when done, because a blob from an xhr request
|
|
// is more reliable in the media tag;
|
|
// the normal URL caused jumping prematurely to the next track.
|
|
if (url == trackUrl) {
|
|
prefetchTrack(trackUrl, () => {
|
|
if (mediaTag.paused) {
|
|
if (url == mediaTag.getAttribute("src")) {
|
|
if (mediaTag.currentTime === 0) {
|
|
mediaTag.setAttribute("src", URL.createObjectURL(
|
|
prefetchedTracks.get(url)));
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
// allow releasing memory
|
|
if (isBlob(oldUrl)) {
|
|
URL.revokeObjectURL(oldUrl);
|
|
}
|
|
// update title
|
|
mediaTag.parentElement.querySelector(".m3u-player--title").title = trackUrl;
|
|
mediaTag.parentElement.querySelector(".m3u-player--title").textContent = trackUrl;
|
|
// start prefetching the next three tracks.
|
|
for (const i of [1, 2, 3]) {
|
|
if (playlist.length > Number(trackIndex) + i) {
|
|
prefetchTrack(playlist[Number(trackIndex) + i]);
|
|
}
|
|
}
|
|
callback();
|
|
}
|
|
}
|
|
function changeTrack(mediaTag, diff) {
|
|
const currentTrackIndex = Number(mediaTag.getAttribute("track-index"));
|
|
const nextTrackIndex = currentTrackIndex + diff;
|
|
const tracks = playlists[mediaTag.getAttribute("playlist")];
|
|
if (nextTrackIndex >= 0) { // do not collapse the if clauses with double-and, that does not survive inlining
|
|
if (tracks.length > nextTrackIndex) {
|
|
mediaTag.setAttribute("track-index", nextTrackIndex);
|
|
updateSrc(mediaTag, () => mediaTag.play());
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Turn a media tag into playlist player.
|
|
*/
|
|
function initPlayer(mediaTag) {
|
|
mediaTag.setAttribute("playlist", mediaTag.getAttribute("src"));
|
|
mediaTag.setAttribute("track-index", 0);
|
|
const url = mediaTag.getAttribute("playlist");
|
|
const wrapper = mediaTag.parentElement.insertBefore(document.createElement("div"), mediaTag);
|
|
const controls = document.createElement("div");
|
|
const left = document.createElement("span");
|
|
const title = document.createElement("span");
|
|
const right = document.createElement("span");
|
|
controls.appendChild(left);
|
|
controls.appendChild(title);
|
|
controls.appendChild(right);
|
|
left.classList.add("m3u-player--left");
|
|
right.classList.add("m3u-player--right");
|
|
title.classList.add("m3u-player--title");
|
|
title.style.overflow = "hidden";
|
|
title.style.textOverflow = "ellipsis";
|
|
title.style.whiteSpace = "nowrap";
|
|
title.style.opacity = "0.3";
|
|
title.style.direction = "rtl"; // for truncation on the left
|
|
title.style.paddingLeft = "0.5em";
|
|
title.style.paddingRight = "0.5em";
|
|
controls.style.display = "flex";
|
|
controls.style.justifyContent = "space-between";
|
|
const styleTag = document.createElement("style");
|
|
styleTag.innerHTML = ".m3u-player--left:hover, .m3u-player--right:hover {color: wheat; background-color: DarkSlateGray}";
|
|
wrapper.appendChild(styleTag);
|
|
wrapper.appendChild(controls);
|
|
controls.style.width = mediaTag.getBoundingClientRect().width.toString() + "px";
|
|
// appending the media tag to the wrapper removes it from the outer scope but keeps the event listeners
|
|
wrapper.appendChild(mediaTag);
|
|
left.innerHTML = "<"; // not textContent, because we MUST escape
|
|
// the tag here and textContent shows the
|
|
// escaped version
|
|
left.onclick = () => changeTrack(mediaTag, -1);
|
|
right.innerHTML = ">";
|
|
right.onclick = () => changeTrack(mediaTag, +1);
|
|
fetchPlaylist(
|
|
url,
|
|
() => {
|
|
updateSrc(mediaTag, () => null);
|
|
mediaTag.addEventListener("ended", event => {
|
|
if (mediaTag.currentTime >= mediaTag.duration) {
|
|
changeTrack(mediaTag, +1);
|
|
}
|
|
});
|
|
},
|
|
() => null);
|
|
// keep the controls aligned to the media tag
|
|
mediaTag.resizeObserver = new ResizeObserver(entries => {
|
|
controls.style.width = entries[0].contentRect.width.toString() + "px";
|
|
});
|
|
mediaTag.resizeObserver.observe(mediaTag);
|
|
}
|
|
function processTag(mediaTag) {
|
|
const canPlayClaim = mediaTag.canPlayType('audio/x-mpegurl');
|
|
let supportsPlaylists = !!canPlayClaim;
|
|
if (canPlayClaim == 'maybe') { // yes, seriously: specced as you only know when you try
|
|
supportsPlaylists = false;
|
|
}
|
|
if (!supportsPlaylists) {
|
|
if (isPlaylist(mediaTag.getAttribute("src"))) {
|
|
initPlayer(mediaTag);
|
|
}
|
|
}
|
|
}
|
|
document.addEventListener('DOMContentLoaded', () => {
|
|
const nodes = document.querySelectorAll("audio,video");
|
|
nodes.forEach(processTag);
|
|
});
|
|
// @license-end
|
|
// The script:1 ends here
|
|
|
|
</script>
|
|
|
|
<style>
|
|
body {background: #222; padding: 2% 10%; margin: auto;}
|
|
p {color: #ccc;}
|
|
.grid { display: grid; grid-template-columns: repeat(4, 360px);}
|
|
audio {padding: 6px 0%;}
|
|
</style>
|
|
|
|
<body>
|
|
|
|
<div class="grid">
|
|
|
|
60s.m3u <audio src="http://www.radio.pervii.com/top_radio_60s.m3u" controls="controls"></audio>
|
|
|
|
70s.m3u <audio src="http://www.radio.pervii.com/top_radio_70s.m3u" controls="controls"></audio>
|
|
|
|
80s.m3u <audio src="http://www.radio.pervii.com/top_radio_80s.m3u" controls="controls"></audio>
|
|
|
|
90s.m3u <audio src="http://www.radio.pervii.com/top_radio_90s.m3u" controls="controls"></audio>
|
|
|
|
acid_jazz.m3u <audio src="http://www.radio.pervii.com/top_radio_acid_jazz.m3u" controls="controls"></audio>
|
|
|
|
african.m3u <audio src="http://www.radio.pervii.com/top_radio_african.m3u" controls="controls"></audio>
|
|
|
|
alternative.m3u <audio src="http://www.radio.pervii.com/top_radio_alternative.m3u" controls="controls"></audio>
|
|
|
|
ambient.m3u <audio src="http://www.radio.pervii.com/top_radio_ambient.m3u" controls="controls"></audio>
|
|
|
|
americana.m3u <audio src="http://www.radio.pervii.com/top_radio_americana.m3u" controls="controls"></audio>
|
|
|
|
anime.m3u <audio src="http://www.radio.pervii.com/top_radio_anime.m3u" controls="controls"></audio>
|
|
|
|
arabic.m3u <audio src="http://www.radio.pervii.com/top_radio_arabic.m3u" controls="controls"></audio>
|
|
|
|
asian.m3u <audio src="http://www.radio.pervii.com/top_radio_asian.m3u" controls="controls"></audio>
|
|
|
|
big_band.m3u <audio src="http://www.radio.pervii.com/top_radio_big_band.m3u" controls="controls"></audio>
|
|
|
|
bluegrass.m3u <audio src="http://www.radio.pervii.com/top_radio_bluegrass.m3u" controls="controls"></audio>
|
|
|
|
blues.m3u <audio src="http://www.radio.pervii.com/top_radio_blues.m3u" controls="controls"></audio>
|
|
|
|
breakbeat.m3u <audio src="http://www.radio.pervii.com/top_radio_breakbeat.m3u" controls="controls"></audio>
|
|
|
|
chillout.m3u <audio src="http://www.radio.pervii.com/top_radio_chillout.m3u" controls="controls"></audio>
|
|
|
|
christian.m3u <audio src="http://www.radio.pervii.com/top_radio_christian.m3u" controls="controls"></audio>
|
|
|
|
classical.m3u <audio src="http://www.radio.pervii.com/top_radio_classical.m3u" controls="controls"></audio>
|
|
|
|
club.m3u <audio src="http://www.radio.pervii.com/top_radio_club.m3u" controls="controls"></audio>
|
|
|
|
college.m3u <audio src="http://www.radio.pervii.com/top_radio_college.m3u" controls="controls"></audio>
|
|
|
|
comedy.m3u <audio src="http://www.radio.pervii.com/top_radio_comedy.m3u" controls="controls"></audio>
|
|
|
|
country.m3u <audio src="http://www.radio.pervii.com/top_radio_country.m3u" controls="controls"></audio>
|
|
|
|
dance.m3u <audio src="http://www.radio.pervii.com/top_radio_dance.m3u" controls="controls"></audio>
|
|
|
|
deutsch.m3u <audio src="http://www.radio.pervii.com/top_radio_deutsch.m3u" controls="controls"></audio>
|
|
|
|
disco.m3u <audio src="http://www.radio.pervii.com/top_radio_disco.m3u" controls="controls"></audio>
|
|
|
|
discofox.m3u <audio src="http://www.radio.pervii.com/top_radio_discofox.m3u" controls="controls"></audio>
|
|
|
|
downtempo.m3u <audio src="http://www.radio.pervii.com/top_radio_downtempo.m3u" controls="controls"></audio>
|
|
|
|
drum_and_bass.m3u <audio src="http://www.radio.pervii.com/top_radio_drum_and_bass.m3u" controls="controls"></audio>
|
|
|
|
easy_listening.m3u <audio src="http://www.radio.pervii.com/top_radio_easy_listening.m3u" controls="controls"></audio>
|
|
|
|
ebm.m3u <audio src="http://www.radio.pervii.com/top_radio_ebm.m3u" controls="controls"></audio>
|
|
|
|
electronic.m3u <audio src="http://www.radio.pervii.com/top_radio_electronic.m3u" controls="controls"></audio>
|
|
|
|
eurodance.m3u <audio src="http://www.radio.pervii.com/top_radio_eurodance.m3u" controls="controls"></audio>
|
|
|
|
film.m3u <audio src="http://www.radio.pervii.com/top_radio_film.m3u" controls="controls"></audio>
|
|
|
|
folk.m3u <audio src="http://www.radio.pervii.com/top_radio_folk.m3u" controls="controls"></audio>
|
|
|
|
france.m3u <audio src="http://www.radio.pervii.com/top_radio_france.m3u" controls="controls"></audio>
|
|
|
|
funk.m3u <audio src="http://www.radio.pervii.com/top_radio_funk.m3u" controls="controls"></audio>
|
|
|
|
goa.m3u <audio src="http://www.radio.pervii.com/top_radio_goa.m3u" controls="controls"></audio>
|
|
|
|
gospel.m3u <audio src="http://www.radio.pervii.com/top_radio_gospel.m3u" controls="controls"></audio>
|
|
|
|
gothic.m3u <audio src="http://www.radio.pervii.com/top_radio_gothic.m3u" controls="controls"></audio>
|
|
|
|
greek.m3u <audio src="http://www.radio.pervii.com/top_radio_greek.m3u" controls="controls"></audio>
|
|
|
|
hardcore.m3u <audio src="http://www.radio.pervii.com/top_radio_hardcore.m3u" controls="controls"></audio>
|
|
|
|
hardrock.m3u <audio src="http://www.radio.pervii.com/top_radio_hardrock.m3u" controls="controls"></audio>
|
|
|
|
hip_hop.m3u <audio src="http://www.radio.pervii.com/top_radio_hip_hop.m3u" controls="controls"></audio>
|
|
|
|
house.m3u <audio src="http://www.radio.pervii.com/top_radio_house.m3u" controls="controls"></audio>
|
|
|
|
india.m3u <audio src="http://www.radio.pervii.com/top_radio_india.m3u" controls="controls"></audio>
|
|
|
|
indie.m3u <audio src="http://www.radio.pervii.com/top_radio_indie.m3u" controls="controls"></audio>
|
|
|
|
industrial.m3u <audio src="http://www.radio.pervii.com/top_radio_industrial.m3u" controls="controls"></audio>
|
|
|
|
instrumental.m3u <audio src="http://www.radio.pervii.com/top_radio_instrumental.m3u" controls="controls"></audio>
|
|
|
|
italian.m3u <audio src="http://www.radio.pervii.com/top_radio_italian.m3u" controls="controls"></audio>
|
|
|
|
jazz.m3u <audio src="http://www.radio.pervii.com/top_radio_jazz.m3u" controls="controls"></audio>
|
|
|
|
jpop.m3u <audio src="http://www.radio.pervii.com/top_radio_jpop.m3u" controls="controls"></audio>
|
|
|
|
jungle.m3u <audio src="http://www.radio.pervii.com/top_radio_jungle.m3u" controls="controls"></audio>
|
|
|
|
latin.m3u <audio src="http://www.radio.pervii.com/top_radio_latin.m3u" controls="controls"></audio>
|
|
|
|
lounge.m3u <audio src="http://www.radio.pervii.com/top_radio_lounge.m3u" controls="controls"></audio>
|
|
|
|
metal.m3u <audio src="http://www.radio.pervii.com/top_radio_metal.m3u" controls="controls"></audio>
|
|
|
|
mixed.m3u <audio src="http://www.radio.pervii.com/top_radio_mixed.m3u" controls="controls"></audio>
|
|
|
|
musical.m3u <audio src="http://www.radio.pervii.com/top_radio_musical.m3u" controls="controls"></audio>
|
|
|
|
oldies.m3u <audio src="http://www.radio.pervii.com/top_radio_oldies.m3u" controls="controls"></audio>
|
|
|
|
opera.m3u <audio src="http://www.radio.pervii.com/top_radio_opera.m3u" controls="controls"></audio>
|
|
|
|
polish.m3u <audio src="http://www.radio.pervii.com/top_radio_polish.m3u" controls="controls"></audio>
|
|
|
|
polka.m3u <audio src="http://www.radio.pervii.com/top_radio_polka.m3u" controls="controls"></audio>
|
|
|
|
pop.m3u <audio src="http://www.radio.pervii.com/top_radio_pop.m3u" controls="controls"></audio>
|
|
|
|
portugal.m3u <audio src="http://www.radio.pervii.com/top_radio_portugal.m3u" controls="controls"></audio>
|
|
|
|
progressive.m3u <audio src="http://www.radio.pervii.com/top_radio_progressive.m3u" controls="controls"></audio>
|
|
|
|
punk.m3u <audio src="http://www.radio.pervii.com/top_radio_punk.m3u" controls="controls"></audio>
|
|
|
|
quran.m3u <audio src="http://www.radio.pervii.com/top_radio_quran.m3u" controls="controls"></audio>
|
|
|
|
rap.m3u <audio src="http://www.radio.pervii.com/top_radio_rap.m3u" controls="controls"></audio>
|
|
|
|
reggae.m3u <audio src="http://www.radio.pervii.com/top_radio_reggae.m3u" controls="controls"></audio>
|
|
|
|
retro.m3u <audio src="http://www.radio.pervii.com/top_radio_retro.m3u" controls="controls"></audio>
|
|
|
|
rnb.m3u <audio src="http://www.radio.pervii.com/top_radio_rnb.m3u" controls="controls"></audio>
|
|
|
|
rock.m3u <audio src="http://www.radio.pervii.com/top_radio_rock.m3u" controls="controls"></audio>
|
|
|
|
romanian.m3u <audio src="http://www.radio.pervii.com/top_radio_romanian.m3u" controls="controls"></audio>
|
|
|
|
russian.m3u <audio src="http://www.radio.pervii.com/top_radio_russian.m3u" controls="controls"></audio>
|
|
|
|
salsa.m3u <audio src="http://www.radio.pervii.com/top_radio_salsa.m3u" controls="controls"></audio>
|
|
|
|
schlager.m3u <audio src="http://www.radio.pervii.com/top_radio_schlager.m3u" controls="controls"></audio>
|
|
|
|
ska.m3u <audio src="http://www.radio.pervii.com/top_radio_ska.m3u" controls="controls"></audio>
|
|
|
|
smooth_jazz.m3u <audio src="http://www.radio.pervii.com/top_radio_smooth_jazz.m3u" controls="controls"></audio>
|
|
|
|
soul.m3u <audio src="http://www.radio.pervii.com/top_radio_soul.m3u" controls="controls"></audio>
|
|
|
|
soundtrack.m3u <audio src="http://www.radio.pervii.com/top_radio_soundtrack.m3u" controls="controls"></audio>
|
|
|
|
spain.m3u <audio src="http://www.radio.pervii.com/top_radio_spain.m3u" controls="controls"></audio>
|
|
|
|
spiritual.m3u <audio src="http://www.radio.pervii.com/top_radio_spiritual.m3u" controls="controls"></audio>
|
|
|
|
sport.m3u <audio src="http://www.radio.pervii.com/top_radio_sport.m3u" controls="controls"></audio>
|
|
|
|
swing.m3u <audio src="http://www.radio.pervii.com/top_radio_swing.m3u" controls="controls"></audio>
|
|
|
|
symphonic.m3u <audio src="http://www.radio.pervii.com/top_radio_symphonic.m3u" controls="controls"></audio>
|
|
|
|
talk.m3u <audio src="http://www.radio.pervii.com/top_radio_talk.m3u" controls="controls"></audio>
|
|
|
|
techno.m3u <audio src="http://www.radio.pervii.com/top_radio_techno.m3u" controls="controls"></audio>
|
|
|
|
top_40.m3u <audio src="http://www.radio.pervii.com/top_radio_top_40.m3u" controls="controls"></audio>
|
|
|
|
trance.m3u <audio src="http://www.radio.pervii.com/top_radio_trance.m3u" controls="controls"></audio>
|
|
|
|
turk.m3u <audio src="http://www.radio.pervii.com/top_radio_turk.m3u" controls="controls"></audio>
|
|
|
|
urban.m3u <audio src="http://www.radio.pervii.com/top_radio_urban.m3u" controls="controls"></audio>
|
|
|
|
usa.m3u <audio src="http://www.radio.pervii.com/top_radio_usa.m3u" controls="controls"></audio>
|
|
|
|
various.m3u <audio src="http://www.radio.pervii.com/top_radio_various.m3u" controls="controls"></audio>
|
|
|
|
wave.m3u <audio src="http://www.radio.pervii.com/top_radio_wave.m3u" controls="controls"></audio>
|
|
|
|
world.m3u <audio src="http://www.radio.pervii.com/top_radio_world.m3u" controls="controls"></audio>
|
|
|
|
|
|
</body>
|
|
</html>
|