| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531 |
- <template>
- <el-container v-loading="loading" class="audit-container">
- <el-main class="audit-main">
- <div class="video-card">
- <div class="video-header">
- <el-tag size="small" type="info" effect="plain">ID: {{ currentVideo.videoId }}</el-tag>
- <h1 class="video-title">{{ currentVideo.title }}</h1>
- </div>
- <div class="player-wrapper">
- <video
- ref="videoPlayer"
- :src="currentVideo.videoUrl"
- controls
- autoplay
- class="main-player"
- @timeupdate="onTimeUpdate"
- />
- </div>
- <div class="timeline-fast-tracks">
- <span class="track-label"><i class="el-icon-timer" /> 进度抽审:</span>
- <el-button-group>
- <el-button size="mini" @click="jumpTo(0.1)">10%</el-button>
- <el-button size="mini" @click="jumpTo(0.3)">30%</el-button>
- <el-button size="mini" @click="jumpTo(0.5)">50%</el-button>
- <el-button size="mini" @click="jumpTo(0.8)">80%</el-button>
- </el-button-group>
- <span class="current-time-tips">当前: {{ formatTime(currentTime) }} / {{ formatTime(duration) }}</span>
- </div>
- </div>
- <div class="meta-card">
- <h3 class="section-title">视频简介描述</h3>
- <p class="description-text">{{ currentVideo.description || '该视频无文字简介' }}</p>
- <div class="meta-grid">
- <div class="meta-item">
- <span class="label">画幅方向:</span>
- <span class="value">{{ currentVideo.horizontal ? '横屏 (16:9)' : '竖屏 (9:16)' }}</span>
- </div>
- <div class="meta-item">
- <span class="label">发布时间:</span>
- <span class="value">{{ currentVideo.pubDate }}</span>
- </div>
- <div class="meta-item">
- <span class="label">可见范围:</span>
- <el-tag size="mini" type="info">{{ scopeMap[currentVideo.scope] }}</el-tag>
- </div>
- </div>
- </div>
- </el-main>
- <el-aside width="360px" class="audit-aside">
- <div class="aside-card author-card">
- <h3 class="section-title">创作者信息</h3>
- <div class="author-info">
- <el-avatar :size="46" :src="currentVideo.userCard.avatarUrl" icon="el-icon-user-solid" />
- <div class="author-detail">
- <div class="author-name">{{ currentVideo.userCard.screenName || '未知用户' }}</div>
- <div class="author-id">UID: {{ currentVideo.userCard.userIdStr }}</div>
- </div>
- </div>
- <div class="author-badge-group">
- <el-tag size="mini" type="success">历史通过率: 98%</el-tag>
- <el-tag size="mini" type="warning">违规记录: 0次</el-tag>
- </div>
- </div>
- <div class="aside-card action-card">
- <h3 class="section-title">审核决策面板</h3>
- <el-form ref="auditForm" :model="auditForm" label-position="top" size="small">
- <el-form-item label="违规标签(不通过时必选)" prop="reasonTags">
- <el-checkbox-group v-model="auditForm.reasonTags" :disabled="auditForm.decision === auditResult.REVIEW_REJECTED">
- <el-checkbox label="涉政违规" />
- <el-checkbox label="色情低俗" />
- <el-checkbox label="血腥暴力" />
- <el-checkbox label="侵权盗版" />
- <el-checkbox label="垃圾广告" />
- <el-checkbox label="画质低劣/封面党" />
- </el-checkbox-group>
- </el-form-item>
- <el-form-item label="审核批注与意见" prop="remark">
- <el-input
- v-model="auditForm.remark"
- type="textarea"
- :rows="4"
- placeholder="请输入具体的原因说明,将同步给创作者..."
- maxlength="200"
- show-word-limit
- />
- </el-form-item>
- <div class="submit-buttons">
- <el-button
- type="danger"
- icon="el-icon-close"
- class="btn-reject"
- :loading="submitting"
- @click="submitAudit(3)"
- >
- 拒绝 (驳回)
- </el-button>
- <el-button
- type="success"
- icon="el-icon-check"
- class="btn-approve"
- :loading="submitting"
- @click="submitAudit(2)"
- >
- 通过放行
- </el-button>
- </div>
- </el-form>
- </div>
- <div class="aside-card shortcut-card">
- <h4 class="shortcut-title"><i class="el-icon-info" /> 键盘盲审快捷键指南</h4>
- <div class="shortcut-grid">
- <div class="shortcut-item"><kbd>Space</kbd> <span>播放 / 暂停</span></div>
- <div class="shortcut-item"><kbd>→</kbd> <span>快进 5 秒</span></div>
- <div class="shortcut-item"><kbd>Shift + A</kbd> <span class="text-success">直接通过</span></div>
- <div class="shortcut-item"><kbd>Shift + R</kbd> <span class="text-danger">直接拒绝</span></div>
- </div>
- </div>
- </el-aside>
- </el-container>
- </template>
- <script>
- import { getAuditVideo, submitAuditVideo } from '@/api/vod'
- export default {
- name: 'VideoAudit',
- // 组件激活前:摘除全局 #app 的强制滚动行为
- beforeRouteEnter(to, from, next) {
- next(vm => {
- const appEl = document.getElementById('app')
- if (appEl) appEl.style.overflowY = 'hidden'
- })
- },
- // 组件离开前:还原还原全局 #app 的原貌,不破坏其他页面的行为
- beforeRouteLeave(to, from, next) {
- const appEl = document.getElementById('app')
- if (appEl) appEl.style.overflowY = 'scroll'
- next()
- },
- data() {
- return {
- loading: false,
- submitting: false,
- currentTime: 0,
- duration: 0,
- scopeMap: {
- 1: '本人可见',
- 2: '所有人可见',
- 3: 'VIP 可见'
- },
- auditResult: {
- PUBLISHED: 3,
- REVIEW_REJECTED: 4
- },
- // 当前待审核的视频数据模型
- currentVideo: {
- videoId: '',
- title: '',
- description: '',
- videoUrl: '',
- horizontal: null,
- pubDate: '',
- scope: 0,
- userCard: {
- userId: '',
- screenName: '',
- avatarUrl: ''
- }
- },
- // 审核提交表单
- auditForm: {
- videoId: null,
- decision: null,
- reasonTags: [],
- remark: ''
- }
- }
- },
- watch: {
- $route() {
- this.$router.go()
- }
- },
- created() {
- const videoId = this.$route.params.id
- this.fetchNextVideo(videoId)
- },
- mounted() {
- // 全局监听键盘事件,开启极限盲审模式
- window.addEventListener('keydown', this.handleKeyboardShortcuts)
- },
- beforeDestroy() {
- window.removeEventListener('keydown', this.handleKeyboardShortcuts)
- },
- methods: {
- // 获取下一条待审核视频
- fetchNextVideo(videoId) {
- this.loading = true
- // 重置表单状态
- this.auditForm = { decision: null, reasonTags: [], remark: '' }
- this.currentTime = 0
- this.duration = 0
- const queryParams = {}
- queryParams.videoId = videoId
- getAuditVideo(queryParams).then(resp => {
- if (resp.code === 0) {
- this.currentVideo = resp.data
- this.auditForm.videoId = this.currentVideo.videoId
- } else {
- this.$message.warning(resp.msg)
- }
- }).finally(() => {
- this.loading = false
- })
- },
- // 提交审核决策
- submitAudit(status) {
- this.auditForm.decision = status
- // 校验逻辑:如果不通过,必须选择至少一个违规标签或填写备注
- if (status === this.auditResult.REVIEW_REJECTED && this.auditForm.reasonTags.length === 0 && !this.auditForm.remark.trim()) {
- this.$message.error('请选择违规标签或填写不通过的具体审核意见!')
- return
- }
- this.submitting = true
- submitAuditVideo(this.auditForm).then(resp => {
- if (resp.code === 0) {
- var nextVideoId = resp.data
- this.submitting = false
- this.$notify({
- title: status === this.auditResult.PUBLISHED ? '审核通过' : '已驳回',
- message: `视频 [${this.currentVideo.videoId}] 处理完毕,已自动加载下一条。`,
- type: status === this.auditResult.PUBLISHED ? 'success' : 'warning',
- duration: 2500
- })
- this.$router.push('/vod_audit/' + nextVideoId)
- }
- })
- },
- // 播放器时间更新
- onTimeUpdate(e) {
- this.currentTime = e.target.currentTime
- this.duration = e.target.duration || 0
- },
- // 时间轴快捷百分比跳跃
- jumpTo(percentage) {
- const player = this.$refs.videoPlayer
- if (player && this.duration) {
- player.currentTime = this.duration * percentage
- }
- },
- // 工具函数:格式化时间 (00:00)
- formatTime(seconds) {
- if (isNaN(seconds)) return '00:00'
- const min = Math.floor(seconds / 60).toString().padStart(2, '0')
- const sec = Math.floor(seconds % 60).toString().padStart(2, '0')
- return `${min}:${sec}`
- },
- // 键盘快捷键矩阵处理器
- handleKeyboardShortcuts(e) {
- // 如果光标正停留在文本输入框内,不触发快捷键,防止冲突
- if (e.target.tagName === 'INPUT' || e.target.tagName === 'TEXTAREA') return
- const player = this.$refs.videoPlayer
- switch (e.key) {
- case ' ': // 空格键:暂停/播放
- e.preventDefault()
- if (player) {
- player.paused ? player.play() : player.pause()
- }
- break
- case 'ArrowRight': // 方向右键:快进5秒
- if (player) player.currentTime += 5
- break
- case 'ArrowLeft': // 方向左键:快退5秒
- if (player) player.currentTime -= 5
- break
- case 'A': // Shift + A: 快捷通过
- if (e.shiftKey) this.submitAudit(2)
- break
- case 'R': // Shift + R: 快捷拒绝
- if (e.shiftKey) this.submitAudit(3)
- break
- }
- }
- }
- }
- </script>
- <style scoped>
- /* 全局页面沙箱 - 修正为直接吃满 100vh 浏览器视口 */
- .audit-container {
- height: 100vh;
- width: 100vw;
- background-color: #f4f7f9;
- gap: 20px;
- padding: 20px; /* 给四周留出高级的视窗边距 */
- box-sizing: border-box; /* 锁定边距不撑开大盘 */
- overflow: hidden; /* 严禁外层容器出现滚动条 */
- }
- /* ==================== 🎬 左侧内容区 ==================== */
- .audit-main {
- padding: 0;
- display: flex;
- flex-direction: column;
- gap: 20px;
- height: 100%; /* 锁定跟随父级高度 */
- overflow-y: auto; /* 仅允许内容区独立轴向滚动 */
- }
- .video-card {
- background: #ffffff;
- border-radius: 8px;
- padding: 20px;
- box-shadow: 0 2px 12px rgba(0, 0, 0, 0.04);
- }
- .video-header {
- display: flex;
- align-items: center;
- gap: 12px;
- margin-bottom: 16px;
- }
- .video-title {
- font-size: 18px;
- font-weight: 600;
- color: #1e293b;
- margin: 0;
- }
- /* 核心播放器容器 */
- .player-wrapper {
- background-color: #000000;
- border-radius: 6px;
- overflow: hidden;
- position: relative;
- width: 100%;
- aspect-ratio: 16 / 9; /* 保持16:9黄金比例,自适应各屏幕 */
- max-height: 500px;
- display: flex;
- justify-content: center;
- align-items: center;
- }
- .main-player {
- width: 100%;
- height: 100%;
- object-fit: contain; /* 防止视频拉伸变形 */
- }
- /* 抽审进度控制条 */
- .timeline-fast-tracks {
- margin-top: 14px;
- display: flex;
- align-items: center;
- gap: 12px;
- background: #f8fafc;
- padding: 8px 12px;
- border-radius: 6px;
- border: 1px dashed #e2e8f0;
- }
- .track-label {
- font-size: 13px;
- color: #64748b;
- font-weight: 500;
- }
- .current-time-tips {
- margin-left: auto;
- font-family: monospace;
- font-size: 13px;
- color: #334155;
- background: #cbd5e1;
- padding: 2px 8px;
- border-radius: 4px;
- }
- /* 视频元数据描述卡片 */
- .meta-card {
- background: #ffffff;
- border-radius: 8px;
- padding: 20px;
- box-shadow: 0 2px 12px rgba(0, 0, 0, 0.04);
- }
- .section-title {
- font-size: 14px;
- font-weight: 600;
- color: #475569;
- margin-top: 0;
- margin-bottom: 12px;
- border-left: 3px solid #1890ff;
- padding-left: 8px;
- }
- .description-text {
- font-size: 14px;
- color: #334155;
- line-height: 1.6;
- background: #f8fafc;
- padding: 12px;
- border-radius: 6px;
- margin-bottom: 16px;
- }
- .meta-grid {
- display: grid;
- grid-template-columns: repeat(3, 1fr);
- gap: 16px;
- }
- .meta-item {
- font-size: 13px;
- }
- .meta-item .label { color: #64748b; }
- .meta-item .value { color: #1e293b; font-weight: 500; }
- /* ==================== 🛠️ 右侧风控工作台 ==================== */
- .audit-aside {
- display: flex;
- flex-direction: column;
- gap: 20px;
- height: 100%; /* 锁定跟随父级高度 */
- overflow-y: auto; /* 防止右侧内容过多时溢出屏幕 */
- }
- .aside-card {
- background: #ffffff;
- border-radius: 8px;
- padding: 20px;
- box-shadow: 0 2px 12px rgba(0, 0, 0, 0.04);
- }
- /* 创作者信息 */
- .author-info {
- display: flex;
- align-items: center;
- gap: 12px;
- margin-bottom: 12px;
- }
- .author-name {
- font-size: 15px;
- font-weight: 600;
- color: #1e293b;
- }
- .author-id {
- font-size: 12px;
- color: #94a3b8;
- margin-top: 2px;
- }
- .author-badge-group {
- display: flex;
- gap: 6px;
- }
- /* 审核决策控制流 */
- ::v-deep .el-checkbox {
- margin-bottom: 8px;
- margin-right: 16px;
- width: 120px; /* 两列规整排列 */
- }
- .submit-buttons {
- display: flex;
- gap: 12px;
- margin-top: 20px;
- }
- .btn-reject, .btn-approve {
- flex: 1;
- height: 40px;
- font-weight: 600;
- border-radius: 6px;
- }
- /* 盲审快捷键指南卡片 */
- .shortcut-card {
- background: #0f172a; /* 深色极客风,与快捷键功能相呼应 */
- color: #94a3b8;
- }
- .shortcut-title {
- color: #f1f5f9;
- margin: 0 0 12px 0;
- font-size: 13px;
- font-weight: 500;
- }
- .shortcut-grid {
- display: grid;
- grid-template-columns: 1fr;
- gap: 8px;
- }
- .shortcut-item {
- display: flex;
- justify-content: space-between;
- align-items: center;
- font-size: 12px;
- }
- kbd {
- background-color: #334155;
- color: #ffffff;
- border-radius: 3px;
- border: 1px solid #475569;
- padding: 2px 6px;
- font-family: monospace;
- font-weight: bold;
- box-shadow: 0 1px 0 rgba(0,0,0,0.2);
- }
- .text-success { color: #10b981; }
- .text-danger { color: #ef4444; }
- </style>
|