880 lines
46 KiB
HTML
880 lines
46 KiB
HTML
<!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, '"'); }
|
||
function escText(str) { return String(str).replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>'); }
|
||
|
||
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>
|