Files
SunShineMusic/play.php

495 lines
17 KiB
PHP
Raw Normal View History

2025-09-24 14:15:23 +00:00
<?php
try {
// 从主页面获取音乐列表
$musicList = require_once __DIR__ . '/data/music.php';
// 获取要播放的音频ID
$audioId = $_GET['play'] ?? '';
$audioInfo = null;
// 查找对应的音频信息
foreach ($musicList as $music) {
if ($music['id'] === $audioId) {
$audioInfo = $music;
break;
}
}
// 如果找不到音频,跳转到主页面
if (!$audioInfo) {
header('Location: index.php');
exit;
}
} catch (Exception $e) {
// 处理错误,显示友好信息并链接到主页面
echo "<html><body>";
echo "<h1>出错了: " . htmlspecialchars($e->getMessage()) . "</h1>";
echo "<p><a href='index.php'>返回音乐主页</a></p>";
echo "</body></html>";
exit;
}
?>
<!DOCTYPE html>
<html lang="zh-cn">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title><?php echo htmlspecialchars($audioInfo['title']); ?> - 音乐播放</title>
<link rel="icon" href="./static/icon/new-icon.png" type="image/png">
<link rel="alternate icon" href="./static/icon/icon.ico" type="image/x-ico">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
<style>
@font-face {
font-family: "MyCustomFont";
src: url("./static/font/bbc.ttf") format("truetype");
font-weight: normal;
font-style: normal;
}
:root {
--main-font: "MyCustomFont", sans-serif; /* 自定义字体变量 */
}
/* 基础样式 */
body {
font-family: var(--main-font);
margin: 0;
padding: 20px;
background-color: #f5f5f7;
color: #333;
display: flex;
flex-direction: column;
align-items: center;
min-height: 100vh;
box-sizing: border-box;
}
.container {
width: 100%;
max-width: 600px;
background-color: white;
border-radius: 12px;
padding: 2rem;
box-shadow: 0 4px 20px rgba(0,0,0,0.1);
text-align: center;
}
h1 {
color: #2c3e50;
margin-bottom: 0.5rem;
}
.artist {
color: #666;
font-size: 1.1rem;
margin-bottom: 2rem;
font-style: italic;
}
.back-link {
display: inline-block;
margin-bottom: 2rem;
color: #2c3e50;
text-decoration: none;
font-size: 0.9rem;
transition: color 0.3s;
}
.back-link:hover {
color: #3498db;
text-decoration: underline;
}
.video-container {
position: relative;
width: 100%;
padding-top: 56.25%; /* 16:9比例 */
margin: 1rem 0;
overflow: hidden;
border-radius: 8px;
}
.video-container iframe {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
border: none;
}
/* 自定义音频播放器样式 */
.custom-audio-player {
width: 100%;
margin: 1.5rem 0;
box-sizing: border-box;
background-color: #f8f9fa;
border-radius: 8px;
padding: 0.8rem;
display: flex;
align-items: center;
gap: 0.8rem;
flex-wrap: wrap;
}
.audio-controls {
display: flex;
align-items: center;
gap: 0.8rem;
}
.audio-button {
background: none;
border: none;
cursor: pointer;
color: #2c3e50;
font-size: 1.2rem;
width: 36px;
height: 36px;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.2s ease;
}
.audio-button:hover {
background-color: rgba(44, 62, 80, 0.1);
}
.audio-button.active {
background-color: #2c3e50;
color: white;
}
.progress-container {
flex-grow: 1;
display: flex;
align-items: center;
gap: 0.5rem;
min-width: 200px;
}
.audio-progress {
flex-grow: 1;
height: 4px;
background-color: #ddd;
border-radius: 2px;
cursor: pointer;
position: relative;
}
.progress-fill {
height: 100%;
background-color: #2c3e50;
border-radius: 2px;
width: 0%;
transition: width 0.1s ease;
}
.progress-handle {
position: absolute;
width: 12px;
height: 12px;
border-radius: 50%;
background-color: #2c3e50;
top: 50%;
transform: translateY(-50%);
left: 0%;
display: none;
transition: left 0.1s ease;
box-shadow: 0 0 5px rgba(0,0,0,0.2);
}
.audio-progress:hover .progress-handle {
display: block;
}
.time-display {
font-size: 0.8rem;
color: #666;
width: 40px;
text-align: center;
}
.time-separator {
font-size: 0.8rem;
color: #666;
width: 10px;
text-align: center;
}
.volume-control {
display: flex;
align-items: center;
gap: 0.5rem;
margin-left: 0.5rem;
min-width: 100px;
}
.volume-button {
background: none;
border: none;
cursor: pointer;
color: #2c3e50;
font-size: 1rem;
transition: all 0.2s ease;
}
.volume-slider {
width: 80px;
height: 4px;
-webkit-appearance: none;
appearance: none;
background: #ddd;
border-radius: 2px;
outline: none;
}
.volume-slider::-webkit-slider-thumb {
-webkit-appearance: none;
appearance: none;
width: 12px;
height: 12px;
border-radius: 50%;
background: #2c3e50;
cursor: pointer;
}
.download-link {
display: inline-block;
margin-top: 1rem;
color: #2c3e50;
text-decoration: none;
padding: 0.5rem 1rem;
border: 1px solid #2c3e50;
border-radius: 20px;
transition: all 0.3s ease;
}
.download-link:hover {
background-color: #2c3e50;
color: white;
}
@media (max-width: 768px) {
.container {
padding: 1rem;
}
.volume-control {
margin-top: 0.5rem;
width: 100%;
justify-content: center;
order: 3;
}
.progress-container {
order: 2;
width: 100%;
}
}
</style>
</head>
<body>
<div class="container">
<p>
<a href="index.php#download-<?php echo $audioInfo['id']; ?>" class="download-link" target="_blank">
<i class="fas fa-download"></i> 前往主站-查看更多音乐
</a>
</p>
<h1><?php echo htmlspecialchars($audioInfo['title']); ?></h1>
<p class="artist">作者: <?php echo htmlspecialchars($audioInfo['artist']); ?></p>
<div class="custom-audio-player">
<div class="audio-controls">
<button class="audio-button play-pause" data-audio="<?php echo $audioInfo['id']; ?>"></button>
<button class="audio-button loop" data-audio="<?php echo $audioInfo['id']; ?>">🔄</button>
</div>
<div class="progress-container">
<div class="audio-progress" data-audio="<?php echo $audioInfo['id']; ?>">
<div class="progress-fill" data-audio="<?php echo $audioInfo['id']; ?>"></div>
<div class="progress-handle" data-audio="<?php echo $audioInfo['id']; ?>"></div>
</div>
<span class="time-display current-time" data-audio="<?php echo $audioInfo['id']; ?>">0:00</span>
<span class="time-separator">/</span>
<span class="time-display total-time" data-audio="<?php echo $audioInfo['id']; ?>"><?php echo $audioInfo['duration']; ?></span>
</div>
<div class="volume-control" data-audio="<?php echo $audioInfo['id']; ?>">
<button class="volume-button" data-audio="<?php echo $audioInfo['id']; ?>">🔊</button>
<input type="range" class="volume-slider" data-audio="<?php echo $audioInfo['id']; ?>" min="0" max="1" step="0.05" value="1">
</div>
</div>
<audio id="<?php echo $audioInfo['id']; ?>" class="audio-element" src="<?php echo htmlspecialchars($audioInfo['mp3']); ?>" type="audio/mpeg">
您的浏览器不支持音频播放
</audio>
<div class="video-container">
<iframe
src="https://player.bilibili.com/player.html?bvid=<?php echo $music['bvid']; ?>&page=1"
allowfullscreen="true"
loading="lazy">
</iframe>
</div>
<div class="download-section">
<p>下载:
<a href="<?php echo htmlspecialchars($music['mp3']); ?>" class="download-link" download>点我下载</a>
</p>
</div>
</div>
<script>
// 音频播放器功能
(function(d, w) {
var q = d.querySelector.bind(d);
var addEvt = function(el, evt, fn) {
el.addEventListener(evt, fn);
};
// 初始化播放器
function initAudioPlayer() {
var audioId = "<?php echo $audioInfo['id']; ?>";
var audio = q('#' + audioId);
var playBtn = q('.play-pause[data-audio="' + audioId + '"]');
var loopBtn = q('.loop[data-audio="' + audioId + '"]');
var progressBar = q('.audio-progress[data-audio="' + audioId + '"]');
var progressFill = q('.progress-fill[data-audio="' + audioId + '"]');
var progressHandle = q('.progress-handle[data-audio="' + audioId + '"]');
var currentTime = q('.current-time[data-audio="' + audioId + '"]');
var totalTime = q('.total-time[data-audio="' + audioId + '"]');
var volumeBtn = q('.volume-button[data-audio="' + audioId + '"]');
var volumeSlider = q('.volume-slider[data-audio="' + audioId + '"]');
var originalVolume = 1;
var isDragging = false;
// 音频加载完成
addEvt(audio, 'loadedmetadata', function() {
totalTime.textContent = formatTime(audio.duration);
audio.volume = volumeSlider.value;
});
// 播放/暂停
addEvt(playBtn, 'click', function() {
if (audio.paused) {
audio.play();
playBtn.textContent = '⏸';
} else {
audio.pause();
playBtn.textContent = '▶';
}
});
// 循环播放
addEvt(loopBtn, 'click', function() {
audio.loop = !audio.loop;
loopBtn.classList.toggle('active', audio.loop);
});
// 更新进度
addEvt(audio, 'timeupdate', function() {
if (!isDragging) {
var progress = (audio.currentTime / audio.duration) * 100;
progressFill.style.width = progress + '%';
progressHandle.style.left = progress + '%';
currentTime.textContent = formatTime(audio.currentTime);
}
});
// 进度条拖动
var startDrag = function(e) {
isDragging = true;
updateProgress(e);
addEvt(d, 'mousemove', dragProgress);
addEvt(d, 'touchmove', dragProgress);
addEvt(d, 'mouseup', endDrag);
addEvt(d, 'touchend', endDrag);
};
var dragProgress = function(e) {
if (isDragging) updateProgress(e);
};
var endDrag = function() {
isDragging = false;
var rect = progressBar.getBoundingClientRect();
var pos = Math.min(Math.max(0, progressHandle.offsetLeft / rect.width), 1);
audio.currentTime = pos * audio.duration;
d.removeEventListener('mousemove', dragProgress);
d.removeEventListener('touchmove', dragProgress);
};
var updateProgress = function(e) {
var rect = progressBar.getBoundingClientRect();
var clientX = e.type.includes('touch') ? e.touches[0].clientX : e.clientX;
var pos = (clientX - rect.left) / rect.width;
pos = Math.min(Math.max(0, pos), 1);
var percent = pos * 100;
progressFill.style.width = percent + '%';
progressHandle.style.left = percent + '%';
currentTime.textContent = formatTime(pos * audio.duration);
};
addEvt(progressBar, 'mousedown', startDrag);
addEvt(progressBar, 'touchstart', startDrag);
// 音量控制
addEvt(volumeSlider, 'input', function() {
audio.volume = this.value;
updateVolumeIcon();
if (audio.muted && this.value > 0) audio.muted = false;
});
// 静音切换
addEvt(volumeBtn, 'click', function() {
audio.muted = !audio.muted;
updateVolumeIcon();
if (!audio.muted) {
volumeSlider.value = originalVolume;
audio.volume = originalVolume;
} else {
originalVolume = volumeSlider.value;
volumeSlider.value = 0;
}
});
// 更新音量图标
function updateVolumeIcon() {
if (audio.muted || volumeSlider.value == 0) {
volumeBtn.textContent = '🔇';
} else if (volumeSlider.value < 0.5) {
volumeBtn.textContent = '🔈';
} else {
volumeBtn.textContent = '🔊';
}
}
// 播放结束
addEvt(audio, 'ended', function() {
playBtn.textContent = '▶';
});
}
// 格式化时间
function formatTime(seconds) {
if (isNaN(seconds)) return '0:00';
var mins = Math.floor(seconds / 60);
var secs = Math.floor(seconds % 60);
return mins + ':' + (secs < 10 ? '0' + secs : secs);
}
// 页面加载完成后初始化
addEvt(d, 'DOMContentLoaded', function() {
initAudioPlayer();
});
})(document, window);
</script>
</body>
</html>