Files
nas-media-player/index.html
2026-04-19 11:46:40 +08:00

880 lines
46 KiB
HTML
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>NAS 轻量媒体播放器</title>
<style>
* { margin: 0; padding: 0; box-sizing: border-box; }
body { font-family: "Microsoft YaHei", Arial, sans-serif; max-width: 1400px; margin: 0 auto; padding: 20px; background: #f5f5f5; }
.logo-container { text-align: center; margin-bottom: 20px; }
.logo {
max-width: 200px; height: auto; display: inline-block;
box-shadow: 0 2px 5px rgba(0,0,0,0.1);
border-radius: 8px; padding: 5px; background: white;
}
h1 { color: #333; text-align: center; margin-bottom: 10px; }
.subtitle { color: #4285F4; text-align: center; margin-bottom: 30px; font-size: 18px; }
.subtitle a { color: #4285F4; text-decoration: none; }
.subtitle a:hover { text-decoration: underline; }
/* 媒体容器 */
.media-container {
width: 100%; background: #000; border-radius: 8px; overflow: hidden;
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
display: flex; justify-content: center; align-items: center;
min-height: 600px; position: relative;
}
.media-content {
width: 100%; height: 100%;
display: flex; justify-content: center; align-items: center; padding: 20px;
}
video, audio {
width: 100%; height: auto; max-height: 80vh; display: none; background: #000;
}
img {
max-width: 100%; max-height: 80vh; object-fit: contain; display: none; background: #000;
}
/* 加载状态 */
.loading {
position: absolute; top: 50%; left: 50%;
transform: translate(-50%, -50%);
text-align: center; color: #4285F4; z-index: 10;
background: rgba(0,0,0,0.7); padding: 20px 30px; border-radius: 8px;
display: none; /* ← 修复:只用 display:none不重复声明 */
}
.loading-spinner {
border: 4px solid rgba(66,133,244,0.3); border-radius: 50%;
border-top: 4px solid #4285F4; width: 40px; height: 40px;
animation: spin 1s linear infinite; margin: 0 auto 15px;
}
@keyframes spin { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } }
.control-panel {
margin: 20px 0; padding: 15px; background: #fff;
border-radius: 8px; box-shadow: 0 2px 5px rgba(0,0,0,0.05);
}
.select-group { display: inline-block; margin-right: 20px; margin-bottom: 15px; vertical-align: top; }
label { font-weight: bold; color: #555; margin-right: 10px; }
select, input[type="text"], input[type="file"], input[type="password"] {
padding: 10px 15px; font-size: 16px; border: 1px solid #ddd;
border-radius: 4px; outline: none; min-width: 250px;
}
select:focus, input[type="text"]:focus, input[type="password"]:focus {
border-color: #4285F4; box-shadow: 0 0 0 2px rgba(66,133,244,0.2);
}
.btn-group { display: inline-block; }
button {
padding: 10px 20px; font-size: 16px; margin-right: 10px;
border: none; border-radius: 4px; cursor: pointer; transition: all 0.3s;
}
.btn-primary { background: #4285F4; color: white; }
.btn-primary:hover { background: #3367D6; }
.btn-primary:disabled { background: #ccc; cursor: not-allowed; }
.btn-secondary { background: #28a745; color: white; }
.btn-secondary:hover { background: #218838; }
.btn-danger { background: #dc3545; color: white; }
.btn-danger:hover { background: #c82333; }
.media-info { margin-top: 15px; color: #666; font-size: 14px; }
.dir-navigation { margin: 10px 0; padding: 10px; background: #f9f9f9; border-radius: 4px; }
/* 上传区域 */
.upload-section {
margin: 30px 0; padding: 20px; background: #fff;
border-radius: 8px; box-shadow: 0 2px 5px rgba(0,0,0,0.05);
border-top: 3px solid #4285F4;
}
.upload-section h2 {
color: #333; margin-bottom: 20px; padding-bottom: 10px; border-bottom: 1px solid #eee;
}
.upload-form { display: flex; flex-wrap: wrap; gap: 15px; align-items: flex-end; }
.form-row { flex: 1; min-width: 250px; }
/* 进度条 */
.upload-progress { margin-top: 15px; display: none; }
.progress-container {
width: 100%; height: 8px; background-color: #e9ecef;
border-radius: 4px; overflow: hidden; box-shadow: inset 0 1px 2px rgba(0,0,0,0.1);
}
.progress-bar {
height: 100%; background: linear-gradient(90deg, #4285F4, #3367D6);
border-radius: 4px; width: 0%; transition: width 0.2s ease-in-out; position: relative;
}
.progress-bar::after {
content: ''; position: absolute; top: 0; left: 0; right: 0; bottom: 0;
background: linear-gradient(to right, rgba(255,255,255,0.1) 0%, rgba(255,255,255,0.3) 50%, rgba(255,255,255,0.1) 100%);
background-size: 200% 100%; animation: shimmer 1.5s infinite; opacity: 0.7;
}
@keyframes shimmer { 0% { background-position: -200% 0; } 100% { background-position: 200% 0; } }
.progress-stats {
display: flex; justify-content: space-between;
margin-top: 8px; font-size: 14px; color: #666;
}
.speed-indicator { color: #4285F4; font-weight: bold; }
/* 消息提示 */
.message {
margin-top: 15px; padding: 12px 15px; border-radius: 4px;
display: none; font-size: 14px;
}
.success { background: #d4edda; color: #155724; border: 1px solid #c3e6cb; }
.error { background: #f8d7da; color: #721c24; border: 1px solid #f5c6cb; }
.info { background: #d1ecf1; color: #0c5460; border: 1px solid #bee5eb; }
.create-dir-form {
margin-top: 15px; padding-top: 15px; border-top: 1px dashed #ddd;
}
.file-info { margin-top: 10px; font-size: 13px; color: #666; }
/* 媒体类型标签 */
.media-type-badge {
display: inline-block; padding: 3px 8px; border-radius: 4px;
font-size: 12px; margin-left: 10px;
}
.type-video { background-color: #e3f2fd; color: #0d47a1; }
.type-image { background-color: #e8f5e9; color: #1b5e20; }
.type-audio { background-color: #fff3e0; color: #e65100; }
/* 密码保护模态框 */
.modal-overlay {
position: fixed; top: 0; left: 0; right: 0; bottom: 0;
background: rgba(0,0,0,0.7);
display: none; /* ← 修复:只保留 display:none去掉多余的 display:flex */
justify-content: center; align-items: center; z-index: 1000;
}
.modal {
background: white; padding: 30px; border-radius: 8px;
width: 90%; max-width: 400px; box-shadow: 0 4px 20px rgba(0,0,0,0.2);
}
.modal h3 { margin-bottom: 20px; color: #333; text-align: center; }
.modal-form { display: flex; flex-direction: column; gap: 15px; }
.modal-footer { display: flex; justify-content: flex-end; gap: 10px; margin-top: 20px; }
.protected-badge {
display: inline-block; padding: 2px 6px; background: #ffeb3b;
color: #333; font-size: 12px; border-radius: 3px; margin-left: 5px;
}
/* 支持格式说明 */
.supported-formats {
margin: 20px 0; padding: 15px;
background: #f0f7ff; border-left: 4px solid #4285F4; border-radius: 4px;
color: #333; font-size: 14px; word-wrap: break-word;
}
.supported-formats strong { color: #4285F4; }
/* 响应式 */
@media (max-width: 768px) {
.media-container { min-height: 400px; }
.select-group { display: block; margin-right: 0; margin-bottom: 15px; }
.upload-form { flex-direction: column; align-items: stretch; }
select, input[type="text"], input[type="password"] { min-width: 100%; width: 100%; }
.supported-formats { font-size: 13px; padding: 12px; }
.logo { max-width: 150px; }
}
</style>
</head>
<body>
<div class="logo-container">
<!-- Logo SVG 内嵌 base64保持原版不变 -->
<img class="logo" src="data:image/svg+xml;base64,<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
<svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" width="142px" height="113px" viewBox="0 0 142 113" enable-background="new 0 0 142 113" xml:space="preserve">  <image id="image0" width="142" height="113" x="0" y="0"
    href="data:image/png;base64,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
OjAwc4rKegAAAABJRU5ErkJggg==" />
</svg>
" /><br>
</div>
<h1>NAS 轻量媒体播放器</h1>
<div class="subtitle"><a href="/static/zhinan.html" target="_blank">设置使用指南</a></div>
<div class="media-container">
<div id="loading" class="loading">
<div class="loading-spinner"></div>
<div id="loading-text">加载中...</div>
</div>
<div class="media-content">
<video id="videoPlayer" controls preload="metadata">
您的浏览器不支持 HTML5 视频播放,请升级浏览器
</video>
<audio id="audioPlayer" controls></audio>
<img id="imageViewer" alt="图片预览" />
</div>
</div>
<div class="control-panel">
<div class="select-group">
<label for="dirSelect">选择目录:</label>
<select id="dirSelect"></select>
</div>
<div class="select-group">
<label for="mediaSelect">选择媒体:</label>
<select id="mediaSelect"></select>
</div>
<div class="btn-group">
<button id="prevBtn" class="btn-primary" disabled>上一个</button>
<button id="nextBtn" class="btn-primary" disabled>下一个</button>
</div>
<div class="dir-navigation">
当前目录(根目录在/mnt/)<span id="currentDirPath">/</span>
<span id="protectedIndicator" class="protected-badge" style="display:none;">已加密</span>
</div>
<div class="media-info">
当前播放:<span id="currentMediaName"></span>
<span id="mediaTypeBadge" class="media-type-badge" style="display:none;"></span> |
<span id="totalMediaCount">0</span> 个媒体文件
</div>
</div>
<!-- 文件上传区域 -->
<div class="upload-section">
<h2>媒体文件上传</h2>
<div class="upload-form">
<div class="form-row">
<label for="uploadTargetDir">目标目录:</label>
<select id="uploadTargetDir"><option value="">加载中...</option></select>
</div>
<div class="form-row">
<label for="mediaFile">选择媒体文件:</label>
<input type="file" id="mediaFile"
accept=".mp4,.avi,.mkv,.webm,.mov,.flv,.wmv,.mpeg,.mpg,.m4v,.jpg,.jpeg,.png,.gif,.bmp,.webp,.tiff,.tif,.mp3,.wav,.ogg,.flac,.aac,.m4a,.wma,.ape,.alac" />
</div>
<div class="form-row">
<button id="uploadBtn" class="btn-secondary">上传文件</button>
</div>
</div>
<div id="fileInfo" class="file-info"></div>
<div class="upload-progress" id="uploadProgress">
<div class="progress-container">
<div class="progress-bar" id="progressBar"></div>
</div>
<div class="progress-stats">
<span id="progressPercent">0%</span>
<span id="progressDetails">0 MB / 0 MB</span>
<span id="uploadSpeed" class="speed-indicator">0 KB/s</span>
</div>
</div>
<div id="message" class="message"></div>
<!-- 创建新目录 -->
<div class="create-dir-form">
<h3>创建新目录</h3>
<div class="form-row">
<label for="createTargetDir">父目录:</label>
<select id="createTargetDir"><option value="">加载中...</option></select>
</div>
<div class="form-row">
<label for="newDirName">新目录名称:</label>
<input type="text" id="newDirName" placeholder="输入目录名称" />
</div>
<div class="form-row">
<label for="dirPassword">设置访问密码(可选):</label>
<input type="password" id="dirPassword" placeholder="留空则不设置密码保护" />
</div>
<div class="form-row">
<button id="createDirBtn" class="btn-primary">创建目录</button>
</div>
</div>
</div>
<!-- 密码保护模态框 -->
<div class="modal-overlay" id="passwordModal">
<div class="modal">
<h3>🔒 目录已加密保护</h3>
<div id="modalMessage" class="info" style="display:block;margin-bottom:15px;">请输入访问密码</div>
<div class="modal-form">
<div>
<label for="modalPassword">密码:</label>
<input type="password" id="modalPassword" placeholder="请输入访问密码" />
</div>
</div>
<div class="modal-footer">
<button id="cancelBtn" class="btn-danger">取消</button>
<button id="confirmBtn" class="btn-primary">确认</button>
</div>
</div>
</div>
<!-- 支持格式说明 -->
<div class="supported-formats">
<strong>支持格式:</strong>
.mp4 .avi .mkv .webm .mov .flv .wmv .mpeg .mpg .m4v
.jpg .jpeg .png .gif .bmp .webp .tiff .tif
.mp3 .wav .ogg .flac .aac .m4a .wma .ape .alac
</div>
<script>
// ── DOM 元素 ────────────────────────────────────────────────────────────────────
const videoPlayer = document.getElementById('videoPlayer');
const audioPlayer = document.getElementById('audioPlayer');
const imageViewer = document.getElementById('imageViewer');
const dirSelect = document.getElementById('dirSelect');
const mediaSelect = document.getElementById('mediaSelect');
const prevBtn = document.getElementById('prevBtn');
const nextBtn = document.getElementById('nextBtn');
const loading = document.getElementById('loading');
const loadingText = document.getElementById('loading-text');
const currentMediaName = document.getElementById('currentMediaName');
const mediaTypeBadge = document.getElementById('mediaTypeBadge');
const totalMediaCount = document.getElementById('totalMediaCount');
const currentDirPath = document.getElementById('currentDirPath');
const protectedIndicator = document.getElementById('protectedIndicator');
const uploadTargetDir = document.getElementById('uploadTargetDir');
const createTargetDir = document.getElementById('createTargetDir');
const mediaFile = document.getElementById('mediaFile');
const uploadBtn = document.getElementById('uploadBtn');
const createDirBtn = document.getElementById('createDirBtn');
const newDirName = document.getElementById('newDirName');
const dirPassword = document.getElementById('dirPassword');
const uploadProgress = document.getElementById('uploadProgress');
const progressBar = document.getElementById('progressBar');
const progressPercent = document.getElementById('progressPercent');
const progressDetails = document.getElementById('progressDetails');
const uploadSpeed = document.getElementById('uploadSpeed');
const message = document.getElementById('message');
const fileInfo = document.getElementById('fileInfo');
const passwordModal = document.getElementById('passwordModal');
const modalPassword = document.getElementById('modalPassword');
const confirmBtn = document.getElementById('confirmBtn');
const cancelBtn = document.getElementById('cancelBtn');
const modalMessage = document.getElementById('modalMessage');
// ── 全局状态 ────────────────────────────────────────────────────────────────────
const LAST_VISITED_DIR_KEY = 'nas_last_visited_dir';
let mediaList = [];
let dirList = [];
let allDirs = [];
let currentIndex = -1;
let currentDir = '';
let currentProtectedDir = null;
let pendingUploadFile = null;
let pendingUploadDir = '';
let uploadStartTime = null;
let uploadedBytes = 0;
let totalBytes = 0;
let speedHistory = [];
let lastUpdateTime = 0;
// ── 工具函数 ────────────────────────────────────────────────────────────────────
function getLastVisitedDir() {
try { return localStorage.getItem(LAST_VISITED_DIR_KEY) || ''; }
catch (e) { return ''; }
}
function saveLastVisitedDir(dirPath) {
try { localStorage.setItem(LAST_VISITED_DIR_KEY, dirPath); }
catch (e) { /* 忽略存储失败 */ }
}
function showLoading(text = '加载中...') {
loadingText.textContent = text;
loading.style.display = 'block';
}
function hideLoading() {
loading.style.display = 'none';
}
/** 显示操作结果消息(自动 3 秒后隐藏) */
function showMessage(text, type = 'success') {
message.textContent = text;
message.className = `message ${type}`;
message.style.display = 'block';
setTimeout(() => { message.style.display = 'none'; }, 3000);
}
function formatFileSize(bytes) {
if (bytes === 0) return '0 Bytes';
const k = 1024;
const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
}
function calculateSpeed(currentBytes) {
const now = Date.now();
if (lastUpdateTime === 0) { lastUpdateTime = now; uploadedBytes = currentBytes; return 0; }
const elapsed = (now - lastUpdateTime) / 1000;
if (elapsed > 0.5) {
const speed = (currentBytes - uploadedBytes) / elapsed / 1024;
speedHistory.push(speed);
if (speedHistory.length > 10) speedHistory.shift();
uploadedBytes = currentBytes;
lastUpdateTime = now;
return speedHistory.reduce((a, b) => a + b, 0) / speedHistory.length;
}
return speedHistory.length > 0 ? speedHistory[speedHistory.length - 1] : 0;
}
// ── 目录管理 ────────────────────────────────────────────────────────────────────
/**
* 修复option 元素不支持内嵌 HTML使用纯文本拼接保护标识。
*/
function populateDirSelect(selectedDir = '') {
dirSelect.innerHTML = dirList.map(dir => {
const lock = dir.protected ? ' 🔒' : '';
const isSelected = dir.path === selectedDir ? 'selected' : '';
return `<option value="${escAttr(dir.path)}" ${isSelected}>${escText(dir.name)}${lock}</option>`;
}).join('');
}
function populateUploadDirSelect() {
const makeOpt = dir => {
const lock = dir.protected ? ' 🔒' : '';
const isSel = dir.path === currentDir ? 'selected' : '';
return `<option value="${escAttr(dir.path)}" ${isSel}>${escText(dir.name)}${lock}</option>`;
};
uploadTargetDir.innerHTML = allDirs.map(makeOpt).join('');
createTargetDir.innerHTML = allDirs.map(makeOpt).join('');
}
/** 简易 HTML attribute 和 text 转义(防 XSS */
function escAttr(str) { return String(str).replace(/"/g, '&quot;'); }
function escText(str) { return String(str).replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;'); }
async function loadAllDirectories() {
try {
const resp = await fetch('/api/all-directories');
if (!resp.ok) throw new Error(`HTTP ${resp.status}`);
const data = await resp.json();
allDirs = data.directories || [];
populateUploadDirSelect();
} catch (err) {
console.error('加载目录失败:', err);
showMessage('加载目录失败,请重试', 'error');
}
}
async function loadDirectoryTree() {
try {
const resp = await fetch('/api/directories');
if (!resp.ok) throw new Error(`HTTP ${resp.status}`);
const data = await resp.json();
dirList = flattenDirectories(data.directories || []);
populateDirSelect(currentDir);
} catch (err) {
console.error('加载目录树失败:', err);
}
}
function flattenDirectories(dirs, parentPath = '', result = []) {
result.push({ name: parentPath || '主目录', path: parentPath, protected: false });
dirs.forEach(dir => {
const fullPath = parentPath ? `${parentPath}/${dir.name}` : dir.name;
result.push({ name: fullPath, path: fullPath, protected: dir.protected || false });
if (dir.children && dir.children.length > 0) {
flattenDirectories(dir.children, fullPath, result);
}
});
return result;
}
// ── 密码模态框 ──────────────────────────────────────────────────────────────────
function showPasswordModal(dirPath, autoPlay = false, isUpload = false, targetDir = '', file = null) {
currentProtectedDir = dirPath;
if (isUpload && file && targetDir) {
pendingUploadFile = file;
pendingUploadDir = targetDir;
}
modalPassword.value = '';
modalMessage.textContent = '请输入访问密码';
modalMessage.className = 'info';
modalMessage.style.display = 'block';
passwordModal.style.display = 'flex'; // ← 覆盖 display:none显示为 flex
modalPassword.focus();
}
function hidePasswordModal() {
passwordModal.style.display = 'none';
currentProtectedDir = null;
modalPassword.value = '';
}
async function verifyDirectoryPassword(password) {
try {
const fd = new FormData();
fd.append('dir_path', currentProtectedDir);
fd.append('password', password);
const resp = await fetch('/api/verify-dir-password', {
method: 'POST', body: fd, credentials: 'include'
});
const result = await resp.json();
if (result.success) {
hidePasswordModal();
showMessage('验证成功!正在加载目录...');
saveLastVisitedDir(currentDir);
await loadMediaInDir(currentDir, true);
populateDirSelect(currentDir);
if (pendingUploadFile && pendingUploadDir) {
const f = pendingUploadFile, d = pendingUploadDir;
pendingUploadFile = null; pendingUploadDir = '';
performUpload(d, f);
}
return true;
} else {
modalMessage.textContent = '密码错误,请重试';
modalMessage.className = 'error';
return false;
}
} catch (err) {
console.error('验证密码失败:', err);
modalMessage.textContent = '验证失败:' + err.message;
modalMessage.className = 'error';
return false;
}
}
// ── 媒体加载 ────────────────────────────────────────────────────────────────────
async function loadMediaInDir(subdir, isAfterAuth = false) {
try {
showLoading('加载媒体列表中...');
currentDir = subdir;
saveLastVisitedDir(subdir);
const url = subdir ? `/api/media?subdir=${encodeURIComponent(subdir)}` : '/api/media';
const resp = await fetch(url, { credentials: 'include' });
if (!resp.ok) throw new Error(`HTTP ${resp.status}`);
const data = await resp.json();
mediaList = data.media || [];
// 受保护目录且尚未认证
if (!isAfterAuth && data.protected && mediaList.length === 0) {
protectedIndicator.style.display = 'inline-block';
if (data.top_protected_dir) {
hideLoading();
showPasswordModal(data.top_protected_dir, false);
return;
}
} else {
protectedIndicator.style.display = data.protected ? 'inline-block' : 'none';
}
currentDirPath.textContent = currentDir ? `/${currentDir}` : '/';
totalMediaCount.textContent = mediaList.length;
// 填充媒体选择框
mediaSelect.innerHTML = mediaList.map((media, idx) => {
const icon = media.type === 'image' ? '🖼️' : (media.type === 'audio' ? '🎵' : '📹');
const size = media.size ? ` (${formatFileSize(media.size)})` : '';
return `<option value="${idx}">${icon} ${escText(media.name)}${size}</option>`;
}).join('');
if (mediaList.length > 0) {
await playMedia(0);
} else {
hideAllMedia();
currentMediaName.textContent = '无';
mediaTypeBadge.style.display = 'none';
currentIndex = -1;
}
updateButtons();
hideLoading();
} catch (err) {
console.error('加载媒体失败:', err);
hideLoading(); // ← 修复:错误时正确隐藏 loading
showMessage(`加载失败: ${err.message}`, 'error');
}
}
function hideAllMedia() {
videoPlayer.pause(); audioPlayer.pause();
videoPlayer.src = ''; audioPlayer.src = ''; imageViewer.src = '';
videoPlayer.style.display = 'none';
audioPlayer.style.display = 'none';
imageViewer.style.display = 'none';
}
/**
* 构建媒体 URL。
* 注意:不再对视频/音频 URL 添加时间戳,避免影响断点续传 Range 请求。
* 图片添加时间戳以防止缓存旧内容。
*/
function buildMediaUrl(media) {
const base = currentDir
? `/api/media/${encodeURIComponent(currentDir)}/${encodeURIComponent(media.name)}`
: `/api/media/${encodeURIComponent(media.name)}`;
return media.type === 'image' ? base + '?t=' + Date.now() : base;
}
async function playMedia(idx) {
if (idx < 0 || idx >= mediaList.length) return;
showLoading('加载媒体中...');
currentIndex = idx;
const media = mediaList[idx];
currentMediaName.textContent = media.name;
if (media.type === 'video') {
mediaTypeBadge.textContent = '视频'; mediaTypeBadge.className = 'media-type-badge type-video';
} else if (media.type === 'audio') {
mediaTypeBadge.textContent = '音频'; mediaTypeBadge.className = 'media-type-badge type-audio';
} else {
mediaTypeBadge.textContent = '图片'; mediaTypeBadge.className = 'media-type-badge type-image';
}
mediaTypeBadge.style.display = 'inline-block';
hideAllMedia();
const mediaUrl = buildMediaUrl(media);
try {
if (media.type === 'video') {
videoPlayer.src = mediaUrl;
videoPlayer.style.display = 'block';
videoPlayer.oncanplay = () => hideLoading();
videoPlayer.onerror = () => {
hideLoading();
showMessage('视频加载失败,请检查文件', 'error');
};
await videoPlayer.play().catch(err => { console.warn('自动播放失败:', err); hideLoading(); });
} else if (media.type === 'audio') {
audioPlayer.src = mediaUrl;
audioPlayer.style.display = 'block';
audioPlayer.oncanplay = () => hideLoading();
audioPlayer.onerror = () => {
hideLoading();
showMessage('音频加载失败,请检查文件', 'error');
};
await audioPlayer.play().catch(err => { console.warn('自动播放失败:', err); hideLoading(); });
} else {
imageViewer.style.display = 'block';
imageViewer.onload = () => hideLoading();
imageViewer.onerror = () => {
hideLoading();
showMessage('图片加载失败,请检查文件或路径', 'error');
};
imageViewer.src = mediaUrl;
}
mediaSelect.value = idx;
updateButtons();
} catch (err) {
hideLoading();
showMessage(`播放失败: ${err.message}`, 'error');
}
}
function updateButtons() {
prevBtn.disabled = currentIndex <= 0;
nextBtn.disabled = currentIndex >= mediaList.length - 1;
}
// ── 上传功能 ────────────────────────────────────────────────────────────────────
async function performUpload(targetDir, file) {
uploadBtn.disabled = true;
uploadProgress.style.display = 'block';
progressBar.style.width = '0%';
progressPercent.textContent = '0%';
progressDetails.textContent = '0 MB / 0 MB';
uploadSpeed.textContent = '0 KB/s';
speedHistory = []; lastUpdateTime = 0; uploadedBytes = 0; totalBytes = file.size;
uploadStartTime = Date.now();
return new Promise((resolve) => {
const fd = new FormData();
fd.append('target_dir', targetDir);
fd.append('file', file);
const xhr = new XMLHttpRequest();
xhr.open('POST', '/api/upload-media');
xhr.upload.onprogress = (e) => {
if (!e.lengthComputable) return;
const pct = Math.round((e.loaded / e.total) * 100);
progressBar.style.width = pct + '%';
progressPercent.textContent = pct + '%';
progressDetails.textContent = `${formatFileSize(e.loaded)} / ${formatFileSize(e.total)}`;
const spd = calculateSpeed(e.loaded);
uploadSpeed.textContent = spd > 1024
? `${(spd / 1024).toFixed(1)} MB/s`
: `${Math.round(spd)} KB/s`;
};
xhr.onload = () => {
uploadProgress.style.display = 'none';
uploadBtn.disabled = false;
try {
const result = JSON.parse(xhr.responseText);
if (result.success) {
showMessage(result.message);
mediaFile.value = '';
fileInfo.textContent = '';
// 刷新媒体列表
loadAllDirectories();
if (currentDir === targetDir) loadMediaInDir(currentDir, true);
} else {
showMessage(result.message || '上传失败', 'error');
}
} catch (e) {
showMessage('上传响应解析失败', 'error');
}
resolve();
};
xhr.onerror = () => {
uploadProgress.style.display = 'none';
uploadBtn.disabled = false;
showMessage('网络错误,上传失败', 'error');
resolve();
};
xhr.send(fd);
});
}
// ── 创建目录 ────────────────────────────────────────────────────────────────────
async function createDirectory() {
const targetPath = createTargetDir.value || '';
const dirName = newDirName.value.trim();
const password = dirPassword.value.trim();
if (!dirName) { showMessage('请输入目录名称', 'error'); return; }
try {
const fd = new FormData();
fd.append('target_path', targetPath);
fd.append('new_dir', dirName);
if (password) fd.append('dir_password', password);
const resp = await fetch('/api/create-directory', { method: 'POST', body: fd });
const result = await resp.json();
if (result.success) {
showMessage(result.message);
newDirName.value = '';
dirPassword.value = '';
// 刷新所有目录列表
await Promise.all([loadAllDirectories(), loadDirectoryTree()]);
} else {
showMessage(result.message || '创建失败', 'error');
}
} catch (err) {
console.error('创建目录失败:', err);
showMessage(`创建失败: ${err.message}`, 'error');
}
}
// ── 事件绑定 ────────────────────────────────────────────────────────────────────
dirSelect.addEventListener('change', async () => {
const selected = dirSelect.value;
currentDir = selected;
await loadMediaInDir(selected);
// 同步上传目标目录选中状态
uploadTargetDir.value = selected;
});
mediaSelect.addEventListener('change', () => {
const idx = parseInt(mediaSelect.value, 10);
if (!isNaN(idx)) playMedia(idx);
});
prevBtn.addEventListener('click', () => { if (currentIndex > 0) playMedia(currentIndex - 1); });
nextBtn.addEventListener('click', () => { if (currentIndex < mediaList.length - 1) playMedia(currentIndex + 1); });
// 视频播放结束后自动下一个
videoPlayer.addEventListener('ended', () => {
if (currentIndex < mediaList.length - 1) playMedia(currentIndex + 1);
});
audioPlayer.addEventListener('ended', () => {
if (currentIndex < mediaList.length - 1) playMedia(currentIndex + 1);
});
// 文件选择信息显示
mediaFile.addEventListener('change', () => {
if (mediaFile.files.length > 0) {
const f = mediaFile.files[0];
fileInfo.textContent = `已选择: ${f.name} (${formatFileSize(f.size)})`;
} else {
fileInfo.textContent = '';
}
});
// 上传按钮
uploadBtn.addEventListener('click', async () => {
const file = mediaFile.files[0];
if (!file) { showMessage('请选择要上传的文件', 'error'); return; }
const targetDir = uploadTargetDir.value || '';
// 检查受保护目录认证
try {
const resp = await fetch(`/api/media?subdir=${encodeURIComponent(targetDir)}`, { credentials: 'include' });
const data = await resp.json();
if (data.protected && !data.media) {
// 未认证,弹出密码框
showPasswordModal(data.top_protected_dir, false, true, targetDir, file);
return;
}
} catch (e) { /* 忽略检查失败,直接尝试上传 */ }
performUpload(targetDir, file);
});
// 创建目录按钮
createDirBtn.addEventListener('click', createDirectory);
// 模态框按钮
confirmBtn.addEventListener('click', () => {
const pwd = modalPassword.value;
if (!pwd) { modalMessage.textContent = '请输入密码'; modalMessage.className = 'error'; return; }
verifyDirectoryPassword(pwd);
});
cancelBtn.addEventListener('click', () => {
hidePasswordModal();
pendingUploadFile = null; pendingUploadDir = '';
});
modalPassword.addEventListener('keydown', (e) => {
if (e.key === 'Enter') confirmBtn.click();
});
// 键盘快捷键(左/右箭头切换媒体)
document.addEventListener('keydown', (e) => {
if (passwordModal.style.display === 'flex') return;
if (['INPUT', 'SELECT', 'TEXTAREA'].includes(e.target.tagName)) return;
if (e.key === 'ArrowLeft' && !prevBtn.disabled) prevBtn.click();
if (e.key === 'ArrowRight' && !nextBtn.disabled) nextBtn.click();
});
// ── 页面初始化 ──────────────────────────────────────────────────────────────────
async function init() {
showLoading('初始化中...');
try {
await Promise.all([loadAllDirectories(), loadDirectoryTree()]);
// 恢复上次访问的目录
const lastDir = getLastVisitedDir();
const targetDir = (lastDir && dirList.some(d => d.path === lastDir)) ? lastDir : '';
if (dirSelect.options.length > 0) {
dirSelect.value = targetDir;
currentDir = targetDir;
}
await loadMediaInDir(targetDir);
} catch (err) {
console.error('初始化失败:', err);
hideLoading();
showMessage('初始化失败,请刷新页面重试', 'error');
}
}
init();
</script>
</body>
</html>