1287 lines
62 KiB
HTML
1287 lines
62 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样式 */
|
||
.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;
|
||
}
|
||
.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: flex;
|
||
justify-content: center;
|
||
align-items: center;
|
||
z-index: 1000;
|
||
display: none;
|
||
}
|
||
.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;
|
||
white-space: normal;
|
||
word-wrap: break-word;
|
||
overflow-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>
|
||
|
||
<!-- 居中Logo容器 -->
|
||
<div class="logo-container">
|
||
<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" style="display: none;">
|
||
<div class="loading-spinner"></div>
|
||
<div id="loading-text">加载中...</div>
|
||
</div>
|
||
<div class="media-content">
|
||
<video id="videoPlayer" controls autoplay 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">
|
||
当前目录:<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 = 'last_visited_dir';
|
||
let mediaList = [];
|
||
let dirList = [];
|
||
let allDirs = [];
|
||
let protectedDirs = [];
|
||
let currentIndex = -1;
|
||
let currentDir = "";
|
||
let currentProtectedDir = null;
|
||
let shouldAutoPlayAfterAuth = false;
|
||
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) {
|
||
console.warn('获取本地存储失败:', e);
|
||
return "";
|
||
}
|
||
}
|
||
|
||
// 工具函数:保存上次访问目录到本地存储
|
||
function saveLastVisitedDir(dirPath) {
|
||
try {
|
||
localStorage.setItem(LAST_VISITED_DIR_KEY, dirPath);
|
||
} catch (e) {
|
||
console.warn('保存本地存储失败:', e);
|
||
}
|
||
}
|
||
|
||
// 显示加载状态
|
||
function showLoading(text = '加载中...') {
|
||
loadingText.textContent = text;
|
||
loading.style.display = 'block';
|
||
}
|
||
|
||
// 隐藏加载状态
|
||
function hideLoading() {
|
||
loading.style.display = 'none';
|
||
}
|
||
|
||
// 显示消息
|
||
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) { // 至少0.5秒更新一次
|
||
const speed = (currentBytes - uploadedBytes) / elapsed / 1024; // KB/s
|
||
speedHistory.push(speed);
|
||
|
||
if (speedHistory.length > 10) {
|
||
speedHistory.shift();
|
||
}
|
||
|
||
const avgSpeed = speedHistory.reduce((a, b) => a + b, 0) / speedHistory.length;
|
||
|
||
uploadedBytes = currentBytes;
|
||
lastUpdateTime = now;
|
||
|
||
return avgSpeed;
|
||
}
|
||
|
||
return speedHistory.length > 0 ? speedHistory[speedHistory.length - 1] : 0;
|
||
}
|
||
|
||
// 扁平化目录结构
|
||
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 populateDirSelect(selectedDir = "") {
|
||
dirSelect.innerHTML = dirList.map(dir => {
|
||
const protectedBadge = dir.protected ? '<span class="protected-badge">🔒</span>' : '';
|
||
const isSelected = dir.path === selectedDir ? 'selected' : '';
|
||
return `<option value="${dir.path}" ${isSelected}>${dir.name}${protectedBadge}</option>`;
|
||
}).join('');
|
||
}
|
||
|
||
// 加载目录列表
|
||
async function loadAllDirectories() {
|
||
try {
|
||
const response = await fetch('/api/all-directories');
|
||
if (!response.ok) throw new Error('Network response was not ok');
|
||
|
||
const data = await response.json();
|
||
allDirs = data.directories || [];
|
||
|
||
// 填充上传目录选择框
|
||
uploadTargetDir.innerHTML = allDirs.map(dir => {
|
||
const protectedBadge = dir.protected ? '<span class="protected-badge">🔒</span>' : '';
|
||
const isSelected = dir.path === currentDir ? 'selected' : '';
|
||
return `<option value="${dir.path}" ${isSelected}>${dir.name}${protectedBadge}</option>`;
|
||
}).join('');
|
||
|
||
// 填充创建目录的父目录选择框
|
||
createTargetDir.innerHTML = allDirs.map(dir => {
|
||
const protectedBadge = dir.protected ? '<span class="protected-badge">🔒</span>' : '';
|
||
return `<option value="${dir.path}">${dir.name}${protectedBadge}</option>`;
|
||
}).join('');
|
||
|
||
} catch (err) {
|
||
console.error('加载目录失败:', err);
|
||
showMessage('加载目录失败,请重试', 'error');
|
||
}
|
||
}
|
||
|
||
// 加载受保护目录列表
|
||
async function loadProtectedDirectories() {
|
||
try {
|
||
const response = await fetch('/api/protected-directories');
|
||
if (!response.ok) throw new Error('Network response was not ok');
|
||
|
||
const data = await response.json();
|
||
protectedDirs = data.protected_dirs || [];
|
||
|
||
} catch (err) {
|
||
console.error('加载受保护目录失败:', err);
|
||
}
|
||
}
|
||
|
||
// 显示密码模态框(支持上传场景)
|
||
function showPasswordModal(dirPath, autoPlay = false, isUpload = false, targetDir = "", file = null) {
|
||
console.log('showPasswordModal:', { dirPath, isUpload, targetDir, file: !!file });
|
||
|
||
currentProtectedDir = dirPath;
|
||
shouldAutoPlayAfterAuth = autoPlay;
|
||
|
||
// 保存上传信息到全局变量
|
||
if (isUpload && file && targetDir) {
|
||
pendingUploadFile = file;
|
||
pendingUploadDir = targetDir;
|
||
console.log('保存上传信息:', { fileName: file.name, targetDir });
|
||
}
|
||
|
||
modalPassword.value = '';
|
||
modalMessage.textContent = '请输入访问密码';
|
||
modalMessage.className = 'info';
|
||
modalMessage.style.display = 'block';
|
||
passwordModal.style.display = 'flex';
|
||
modalPassword.focus();
|
||
}
|
||
|
||
// 隐藏密码模态框
|
||
function hidePasswordModal() {
|
||
passwordModal.style.display = 'none';
|
||
currentProtectedDir = null;
|
||
shouldAutoPlayAfterAuth = false;
|
||
modalPassword.value = '';
|
||
}
|
||
|
||
// 验证目录密码(核心修复:保持目录状态,不刷新页面)
|
||
async function verifyDirectoryPassword(password) {
|
||
try {
|
||
console.log('验证密码:', { dirPath: currentProtectedDir, currentDir });
|
||
|
||
const formData = new FormData();
|
||
formData.append('dir_path', currentProtectedDir);
|
||
formData.append('password', password);
|
||
|
||
const response = await fetch('/api/verify-dir-password', {
|
||
method: 'POST',
|
||
body: formData,
|
||
credentials: 'include'
|
||
});
|
||
|
||
const result = await response.json();
|
||
console.log('验证结果:', result);
|
||
|
||
if (result.success) {
|
||
hidePasswordModal();
|
||
showMessage('验证成功!正在加载目录...');
|
||
|
||
// 关键修复1:保存当前目录到本地存储,持久化状态
|
||
saveLastVisitedDir(currentDir);
|
||
|
||
// 关键修复2:重新加载当前目录,强制刷新媒体列表
|
||
await loadMediaInDir(currentDir, true); // 第二个参数表示"验证后加载",跳过密码检查
|
||
|
||
// 关键修复3:更新目录选择框,确保选中当前目录
|
||
populateDirSelect(currentDir);
|
||
|
||
// 检查是否有等待上传的文件
|
||
if (pendingUploadFile && pendingUploadDir) {
|
||
console.log('开始上传加密目录文件:', {
|
||
fileName: pendingUploadFile.name,
|
||
targetDir: pendingUploadDir
|
||
});
|
||
performUpload(pendingUploadDir, pendingUploadFile);
|
||
pendingUploadFile = null;
|
||
pendingUploadDir = "";
|
||
}
|
||
|
||
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('加载媒体列表中...');
|
||
|
||
// 关键修复1:保存当前目录到全局变量和本地存储
|
||
currentDir = subdir;
|
||
saveLastVisitedDir(subdir);
|
||
|
||
const url = subdir ? `/api/media?subdir=${encodeURIComponent(subdir)}` : '/api/media';
|
||
const response = await fetch(url, {
|
||
credentials: 'include'
|
||
});
|
||
|
||
if (!response.ok) {
|
||
throw new Error(`HTTP error! status: ${response.status}`);
|
||
}
|
||
|
||
const data = await response.json();
|
||
mediaList = data.media || [];
|
||
|
||
// 关键修复2:验证后加载时,跳过密码模态框(已验证通过)
|
||
if (!isAfterAuth && data.protected) {
|
||
protectedIndicator.style.display = 'inline-block';
|
||
if (data.top_protected_dir) {
|
||
showPasswordModal(data.top_protected_dir, mediaList.length > 0);
|
||
hideLoading();
|
||
return;
|
||
}
|
||
} else {
|
||
protectedIndicator.style.display = data.protected ? 'inline-block' : 'none';
|
||
}
|
||
|
||
// 更新UI
|
||
currentDirPath.textContent = currentDir ? `/${currentDir}` : '/';
|
||
totalMediaCount.textContent = mediaList.length;
|
||
|
||
// 填充媒体选择框
|
||
mediaSelect.innerHTML = mediaList.map((media, idx) => {
|
||
let typeIcon = '📹';
|
||
if (media.type === 'image') typeIcon = '🖼️';
|
||
if (media.type === 'audio') typeIcon = '🎵';
|
||
const sizeInfo = media.size ? ` (${formatFileSize(media.size)})` : '';
|
||
return `<option value="${idx}">${typeIcon} ${media.name}${sizeInfo}</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);
|
||
showLoading('加载媒体失败,请重试', true);
|
||
showMessage(`加载失败: ${err.message}`, 'error');
|
||
}
|
||
}
|
||
|
||
// 隐藏所有媒体播放器
|
||
function hideAllMedia() {
|
||
videoPlayer.style.display = 'none';
|
||
audioPlayer.style.display = 'none';
|
||
imageViewer.style.display = 'none';
|
||
videoPlayer.pause();
|
||
audioPlayer.pause();
|
||
videoPlayer.src = "";
|
||
audioPlayer.src = "";
|
||
imageViewer.src = "";
|
||
}
|
||
|
||
// 构建媒体URL
|
||
function buildMediaUrl(media) {
|
||
let url;
|
||
if (currentDir) {
|
||
url = `/api/media/${encodeURIComponent(currentDir)}/${encodeURIComponent(media.name)}`;
|
||
} else {
|
||
url = `/api/media/${encodeURIComponent(media.name)}`;
|
||
}
|
||
// 添加时间戳防止缓存
|
||
return url + '?t=' + Date.now();
|
||
}
|
||
|
||
// 播放媒体文件
|
||
async function playMedia(idx) {
|
||
if (idx < 0 || idx >= mediaList.length) return;
|
||
|
||
showLoading('加载媒体中...');
|
||
currentIndex = idx;
|
||
const media = mediaList[currentIndex];
|
||
|
||
try {
|
||
// 更新信息显示
|
||
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();
|
||
|
||
// 构建URL
|
||
const mediaUrl = buildMediaUrl(media);
|
||
|
||
if (media.type === 'video') {
|
||
// 视频处理
|
||
videoPlayer.src = mediaUrl;
|
||
videoPlayer.style.display = 'block';
|
||
|
||
videoPlayer.oncanplay = function() {
|
||
hideLoading();
|
||
};
|
||
|
||
videoPlayer.onerror = function(e) {
|
||
console.error('Video error:', e);
|
||
showLoading('视频加载失败', true);
|
||
showMessage('视频加载失败,请检查文件', 'error');
|
||
setTimeout(hideLoading, 2000);
|
||
};
|
||
|
||
await videoPlayer.play().catch(err => {
|
||
console.warn('自动播放失败:', err);
|
||
hideLoading();
|
||
});
|
||
|
||
} else if (media.type === 'audio') {
|
||
// 音频处理
|
||
audioPlayer.src = mediaUrl;
|
||
audioPlayer.style.display = 'block';
|
||
|
||
audioPlayer.oncanplay = function() {
|
||
hideLoading();
|
||
};
|
||
|
||
audioPlayer.onerror = function(e) {
|
||
console.error('Audio error:', e);
|
||
showLoading('音频加载失败', true);
|
||
showMessage('音频加载失败,请检查文件', 'error');
|
||
setTimeout(hideLoading, 2000);
|
||
};
|
||
|
||
await audioPlayer.play().catch(err => {
|
||
console.warn('自动播放失败:', err);
|
||
hideLoading();
|
||
});
|
||
|
||
} else {
|
||
// 图片处理
|
||
imageViewer.style.display = 'block';
|
||
imageViewer.onload = function() {
|
||
hideLoading();
|
||
console.log('图片加载成功:', mediaUrl);
|
||
};
|
||
|
||
imageViewer.onerror = function(e) {
|
||
console.error('图片加载失败:', e, mediaUrl);
|
||
showLoading('图片加载失败', true);
|
||
showMessage('图片加载失败,请检查文件或路径', 'error');
|
||
setTimeout(hideLoading, 2000);
|
||
};
|
||
|
||
imageViewer.src = '';
|
||
imageViewer.src = mediaUrl;
|
||
}
|
||
|
||
mediaSelect.value = idx;
|
||
updateButtons();
|
||
|
||
} catch (err) {
|
||
console.error('播放媒体失败:', err);
|
||
showLoading('播放失败', true);
|
||
showMessage(`播放失败: ${err.message}`, 'error');
|
||
setTimeout(hideLoading, 2000);
|
||
}
|
||
}
|
||
|
||
// 更新按钮状态
|
||
function updateButtons() {
|
||
prevBtn.disabled = currentIndex <= 0;
|
||
nextBtn.disabled = currentIndex >= mediaList.length - 1;
|
||
}
|
||
|
||
// 创建目录
|
||
async function createDirectory() {
|
||
const targetPath = createTargetDir.value || "";
|
||
const dirName = newDirName.value.trim();
|
||
const password = dirPassword.value.trim();
|
||
|
||
if (!dirName) {
|
||
showMessage('请输入目录名称', 'error');
|
||
return;
|
||
}
|
||
|
||
try {
|
||
const formData = new FormData();
|
||
formData.append('target_path', targetPath);
|
||
formData.append('new_dir', dirName);
|
||
if (password) {
|
||
formData.append('dir_password', password);
|
||
}
|
||
|
||
const response = await fetch('/api/create-directory', {
|
||
method: 'POST',
|
||
body: formData
|
||
});
|
||
|
||
const result = await response.json();
|
||
|
||
if (result.success) {
|
||
showMessage(result.message);
|
||
newDirName.value = '';
|
||
dirPassword.value = '';
|
||
|
||
// 重新加载目录列表,并选中新创建的目录
|
||
await loadAllDirectories();
|
||
await loadProtectedDirectories();
|
||
|
||
const dirResponse = await fetch('/api/directories');
|
||
const dirData = await dirResponse.json();
|
||
dirList = flattenDirectories(dirData.directories);
|
||
populateDirSelect(result.path); // 选中新目录
|
||
|
||
// 加载新目录的媒体列表
|
||
await loadMediaInDir(result.path);
|
||
|
||
} else {
|
||
showMessage(result.message || '创建目录失败', 'error');
|
||
}
|
||
|
||
} catch (err) {
|
||
console.error('创建目录失败:', err);
|
||
showMessage('创建目录失败:' + err.message, 'error');
|
||
}
|
||
}
|
||
|
||
// 检查目录是否受保护
|
||
function isDirectoryProtected(dirPath) {
|
||
return protectedDirs.some(pdir => dirPath === pdir || dirPath.startsWith(pdir + '/'));
|
||
}
|
||
|
||
// 获取顶级受保护目录
|
||
function getTopProtectedDirectory(dirPath) {
|
||
for (const pdir of protectedDirs) {
|
||
if (dirPath === pdir || dirPath.startsWith(pdir + '/')) {
|
||
return pdir;
|
||
}
|
||
}
|
||
return null;
|
||
}
|
||
|
||
// 检查是否有认证Cookie
|
||
function hasAuthCookie(dirPath) {
|
||
const topDir = getTopProtectedDirectory(dirPath);
|
||
if (!topDir) return true;
|
||
const cookieName = `auth_${encodeURIComponent(topDir)}`;
|
||
return document.cookie.includes(cookieName);
|
||
}
|
||
|
||
// 上传媒体文件
|
||
async function uploadMedia() {
|
||
const file = mediaFile.files[0];
|
||
if (!file) {
|
||
showMessage('请选择要上传的媒体文件', 'error');
|
||
return;
|
||
}
|
||
|
||
const targetDir = uploadTargetDir.value;
|
||
if (!targetDir) {
|
||
showMessage('请选择目标目录', 'error');
|
||
return;
|
||
}
|
||
|
||
console.log('准备上传:', { fileName: file.name, targetDir, size: file.size });
|
||
|
||
if (isDirectoryProtected(targetDir) && !hasAuthCookie(targetDir)) {
|
||
const topDir = getTopProtectedDirectory(targetDir);
|
||
showMessage('目标目录需要密码验证,请输入密码', 'info');
|
||
showPasswordModal(topDir, false, true, targetDir, file);
|
||
return;
|
||
}
|
||
|
||
performUpload(targetDir, file);
|
||
}
|
||
|
||
// 实际执行上传函数
|
||
function performUpload(targetDir, file) {
|
||
if (!file || !targetDir) {
|
||
showMessage('上传参数缺失', 'error');
|
||
console.error('上传参数缺失:', { hasFile: !!file, hasDir: !!targetDir });
|
||
return;
|
||
}
|
||
|
||
console.log('开始上传:', {
|
||
targetDir,
|
||
fileName: file.name,
|
||
size: formatFileSize(file.size),
|
||
type: file.type
|
||
});
|
||
|
||
fileInfo.textContent = `文件名: ${file.name} | 大小: ${formatFileSize(file.size)}`;
|
||
uploadProgress.style.display = 'block';
|
||
|
||
uploadStartTime = Date.now();
|
||
uploadedBytes = 0;
|
||
totalBytes = file.size;
|
||
speedHistory = [];
|
||
lastUpdateTime = 0;
|
||
|
||
const formData = new FormData();
|
||
formData.append('target_dir', targetDir);
|
||
formData.append('file', file);
|
||
|
||
const xhr = new XMLHttpRequest();
|
||
xhr.open('POST', '/api/upload-media', true);
|
||
xhr.withCredentials = true;
|
||
|
||
xhr.upload.addEventListener('progress', (e) => {
|
||
if (e.lengthComputable) {
|
||
const percent = (e.loaded / e.total) * 100;
|
||
progressBar.style.width = `${percent}%`;
|
||
progressPercent.textContent = `${Math.round(percent)}%`;
|
||
progressDetails.textContent = `${formatFileSize(e.loaded)} / ${formatFileSize(e.total)}`;
|
||
|
||
const speed = calculateSpeed(e.loaded);
|
||
if (speed > 1024) {
|
||
uploadSpeed.textContent = `${(speed / 1024).toFixed(2)} MB/s`;
|
||
} else {
|
||
uploadSpeed.textContent = `${speed.toFixed(0)} KB/s`;
|
||
}
|
||
|
||
if (percent > 0 && percent < 100) {
|
||
const elapsed = (Date.now() - uploadStartTime) / 1000;
|
||
const totalTime = elapsed / (percent / 100);
|
||
const remaining = totalTime - elapsed;
|
||
|
||
if (remaining > 60) {
|
||
uploadSpeed.textContent += ` | 剩余: ${Math.round(remaining / 60)} 分钟`;
|
||
} else {
|
||
uploadSpeed.textContent += ` | 剩余: ${Math.round(remaining)} 秒`;
|
||
}
|
||
}
|
||
}
|
||
});
|
||
|
||
xhr.addEventListener('load', () => {
|
||
console.log('上传响应:', { status: xhr.status, response: xhr.responseText });
|
||
|
||
try {
|
||
const result = JSON.parse(xhr.responseText);
|
||
|
||
if (result.success) {
|
||
progressBar.style.width = '100%';
|
||
progressPercent.textContent = '100%';
|
||
uploadSpeed.textContent = '上传完成!';
|
||
showMessage(result.message);
|
||
mediaFile.value = '';
|
||
fileInfo.textContent = '';
|
||
|
||
if (targetDir === currentDir) {
|
||
setTimeout(() => {
|
||
loadMediaInDir(currentDir);
|
||
}, 1000);
|
||
}
|
||
|
||
} else {
|
||
showMessage(result.message || '上传失败', 'error');
|
||
}
|
||
} catch (err) {
|
||
showMessage('上传完成但处理响应失败: ' + err.message, 'error');
|
||
console.error('解析响应失败:', err, xhr.responseText);
|
||
}
|
||
|
||
setTimeout(() => {
|
||
uploadProgress.style.display = 'none';
|
||
}, 3000);
|
||
});
|
||
|
||
xhr.addEventListener('error', (e) => {
|
||
console.error('上传错误:', e);
|
||
showMessage('上传失败:网络错误', 'error');
|
||
uploadProgress.style.display = 'none';
|
||
});
|
||
|
||
xhr.addEventListener('abort', () => {
|
||
console.log('上传被取消');
|
||
showMessage('上传已取消', 'info');
|
||
uploadProgress.style.display = 'none';
|
||
});
|
||
|
||
xhr.send(formData);
|
||
}
|
||
|
||
// 初始化应用(核心修复:优先加载上次访问目录)
|
||
async function initApp() {
|
||
try {
|
||
showLoading('初始化中...');
|
||
|
||
// 关键修复1:获取上次访问的目录(优先于默认根目录)
|
||
const lastDir = getLastVisitedDir();
|
||
console.log('初始化 - 上次访问目录:', lastDir);
|
||
|
||
// 加载目录列表
|
||
const dirResponse = await fetch('/api/directories');
|
||
const dirData = await dirResponse.json();
|
||
dirList = flattenDirectories(dirData.directories);
|
||
|
||
// 关键修复2:初始化目录选择框时,选中上次访问的目录
|
||
populateDirSelect(lastDir);
|
||
|
||
// 加载所有目录(用于上传)
|
||
await loadAllDirectories();
|
||
|
||
// 加载受保护目录列表
|
||
await loadProtectedDirectories();
|
||
|
||
// 关键修复3:优先加载上次访问的目录,而不是默认根目录
|
||
const initDir = lastDir || "";
|
||
await loadMediaInDir(initDir);
|
||
|
||
hideLoading();
|
||
} catch (err) {
|
||
console.error('初始化失败:', err);
|
||
showMessage('初始化失败:' + err.message, 'error');
|
||
hideLoading();
|
||
}
|
||
}
|
||
|
||
// 事件监听器
|
||
dirSelect.addEventListener('change', (e) => {
|
||
const selectedDir = e.target.value;
|
||
loadMediaInDir(selectedDir);
|
||
});
|
||
|
||
mediaSelect.addEventListener('change', (e) => {
|
||
playMedia(parseInt(e.target.value));
|
||
});
|
||
|
||
prevBtn.addEventListener('click', () => {
|
||
if (currentIndex > 0) {
|
||
playMedia(currentIndex - 1);
|
||
}
|
||
});
|
||
|
||
nextBtn.addEventListener('click', () => {
|
||
if (currentIndex < mediaList.length - 1) {
|
||
playMedia(currentIndex + 1);
|
||
}
|
||
});
|
||
|
||
uploadBtn.addEventListener('click', uploadMedia);
|
||
createDirBtn.addEventListener('click', createDirectory);
|
||
|
||
// 密码模态框事件
|
||
confirmBtn.addEventListener('click', () => {
|
||
const password = modalPassword.value.trim();
|
||
if (!password) {
|
||
modalMessage.textContent = '请输入密码';
|
||
modalMessage.className = 'error';
|
||
return;
|
||
}
|
||
verifyDirectoryPassword(password);
|
||
});
|
||
|
||
cancelBtn.addEventListener('click', hidePasswordModal);
|
||
|
||
// ESC键关闭模态框
|
||
document.addEventListener('keydown', (e) => {
|
||
if (e.key === 'Escape') {
|
||
hidePasswordModal();
|
||
}
|
||
});
|
||
|
||
// 点击模态框外部关闭
|
||
passwordModal.addEventListener('click', (e) => {
|
||
if (e.target === passwordModal) {
|
||
hidePasswordModal();
|
||
}
|
||
});
|
||
|
||
// 媒体播放结束自动播放下一个
|
||
videoPlayer.addEventListener('ended', () => {
|
||
if (currentIndex < mediaList.length - 1) {
|
||
playMedia(currentIndex + 1);
|
||
}
|
||
});
|
||
|
||
audioPlayer.addEventListener('ended', () => {
|
||
if (currentIndex < mediaList.length - 1) {
|
||
playMedia(currentIndex + 1);
|
||
}
|
||
});
|
||
|
||
// 初始化应用
|
||
document.addEventListener('DOMContentLoaded', initApp);
|
||
</script>
|
||
</body>
|
||
</html>
|
||
|