|
@@ -1,12 +1,37 @@
|
|
|
<script setup name="reportChapterDetail">
|
|
|
-import {ref} from 'vue'
|
|
|
+import {ref,onMounted,onUnmounted, reactive, nextTick} from 'vue'
|
|
|
import apiReport from '@/api/report'
|
|
|
import { useRoute } from 'vue-router'
|
|
|
-import AudioBox from '../components/AudioBox.vue'
|
|
|
+import EditChapterBaseInfo from './components/EditChapterBaseInfo.vue'
|
|
|
import moment from 'moment'
|
|
|
+import {useInitFroalaEditor} from '@/hooks/useFroalaEditor'
|
|
|
+import {useUserInfo} from '@/hooks/common'
|
|
|
+import { showToast } from 'vant'
|
|
|
+import {useUploadFileToOSS} from '@/hooks/useUploadFileToOSS'
|
|
|
+import MD5 from 'js-md5'
|
|
|
|
|
|
+const userInfo=useUserInfo()
|
|
|
+
|
|
|
+const {FroalaEditorIns,initFroalaEditor,frolaEditorContentChange}=useInitFroalaEditor()
|
|
|
const route=useRoute()
|
|
|
|
|
|
+let autoSaveTimer=null
|
|
|
+
|
|
|
+onMounted(() => {
|
|
|
+ getChapterDetail()
|
|
|
+ autoSaveTimer=setInterval(() => {
|
|
|
+ autoSaveReportContent()
|
|
|
+ }, 6000);
|
|
|
+})
|
|
|
+onUnmounted(()=>{
|
|
|
+ clearInterval(autoSaveTimer)
|
|
|
+})
|
|
|
+// 自动保存
|
|
|
+function autoSaveReportContent(){
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
let info=ref(null)
|
|
|
function getChapterDetail(){
|
|
|
apiReport.getChapterDetail({
|
|
@@ -14,96 +39,252 @@ function getChapterDetail(){
|
|
|
}).then(res=>{
|
|
|
if(res.Ret===200){
|
|
|
info.value=res.Data
|
|
|
- document.title=`${res.Data.ClassifyNameFirst}-${res.Data.Title}`
|
|
|
+ document.title=`${res.Data.ClassifyNameFirst}-${res.Data.TypeName}`
|
|
|
+
|
|
|
+ chapterBaseInfo.type=res.Data.TypeName
|
|
|
+ chapterBaseInfo.title=res.Data.Title
|
|
|
+ chapterBaseInfo.addType=res.Data.AddType
|
|
|
+ chapterBaseInfo.author=res.Data.Author||userInfo.RealName
|
|
|
+ chapterBaseInfo.createTime=moment(res.Data.CreateTime).format('YYYY-MM-DD')
|
|
|
+ if(res.Data.VideoKind==1){
|
|
|
+ chapterBaseInfo.audioUrl=res.Data.VideoUrl
|
|
|
+ chapterBaseInfo.audioDuration=res.Data.VideoPlaySeconds
|
|
|
+ chapterBaseInfo.audioSize=res.Data.VideoSize
|
|
|
+ }
|
|
|
+
|
|
|
+ nextTick(()=>{
|
|
|
+ const el=document.getElementById('editor')
|
|
|
+ initFroalaEditor('#editor',{height:el.offsetHeight-150})
|
|
|
+
|
|
|
+ FroalaEditorIns.value.html.set(res.Data.Content);
|
|
|
+ })
|
|
|
+
|
|
|
}
|
|
|
})
|
|
|
}
|
|
|
getChapterDetail()
|
|
|
|
|
|
+// 报告基本内容
|
|
|
+const showChapterBaseInfo=ref(false)
|
|
|
+const chapterBaseInfo=reactive({
|
|
|
+ type:'',
|
|
|
+ title:'',
|
|
|
+ addType:'',
|
|
|
+ ticket:[],
|
|
|
+ author:'',
|
|
|
+ createTime:'',
|
|
|
+ audioUrl:'',
|
|
|
+ audioDuration:0,
|
|
|
+ audioSize:0,
|
|
|
+})
|
|
|
+// 保存报告基本内容
|
|
|
+function handleChapterBaseInfoSave(e){
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+// 上传音频
|
|
|
+const showUploadAudio=ref(false)
|
|
|
+const temAudioData=reactive({
|
|
|
+ time:0,
|
|
|
+ size:0,
|
|
|
+ url:''
|
|
|
+})
|
|
|
+// 获取音频时长
|
|
|
+function handleGetAudioDuration(file){
|
|
|
+ return new Promise((resolve,reject)=>{
|
|
|
+ const fileUrl=URL.createObjectURL(file)
|
|
|
+ const audioEl=new Audio(fileUrl)
|
|
|
+ audioEl.addEventListener('loadedmetadata',(e)=>{
|
|
|
+ // console.log('e.path',e.path)
|
|
|
+ // console.log('e.composedPath',e.composedPath())
|
|
|
+ // console.log('获取音频时长',e.composedPath()[0].duration);
|
|
|
+ // console.log(audioEl.duration);
|
|
|
+ const t=e.composedPath()[0].duration
|
|
|
+ resolve(t)
|
|
|
+ })
|
|
|
+ })
|
|
|
+}
|
|
|
+function handleAudioUploadBeforeRead(e){
|
|
|
+ const temFile=e
|
|
|
+ if(temFile.name.indexOf('.mp3')==-1
|
|
|
+ && temFile.name.indexOf('.wav')==-1
|
|
|
+ && temFile.name.indexOf('.wma')==-1
|
|
|
+ &&temFile.name.indexOf('.m4a')==-1
|
|
|
+ ){
|
|
|
+ showToast('上传失败,上传音频格式不正确')
|
|
|
+ return false
|
|
|
+ }
|
|
|
+ return true
|
|
|
+}
|
|
|
+async function handleAudioUploadAfterRead(e){
|
|
|
+ // console.log(e);
|
|
|
+ const duration=await handleGetAudioDuration(e.file)
|
|
|
+ if(duration>60*15){
|
|
|
+ showToast('音频时长不得超过15分钟')
|
|
|
+ return
|
|
|
+ }
|
|
|
+ temAudioData.time=duration
|
|
|
+ temAudioData.size=e.file.size/1024/1024 //单位MB
|
|
|
+ // 生成文件名
|
|
|
+ const t=new Date().getTime().toString()
|
|
|
+ const temName=`static/yb/audio/${import.meta.env.MODE}/${MD5(t)}.${e.file.type.split('/')[1]}`
|
|
|
+ console.log(temName);
|
|
|
+ const url=await useUploadFileToOSS(e.file,temName)
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
</script>
|
|
|
|
|
|
<template>
|
|
|
- <div class="chapter-report-detail-page report-detail-wrap" v-if="info">
|
|
|
- <div class="top-stage-box">
|
|
|
- <span class="stage">第{{info.Stage}}期 / {{info.ClassifyNameFirst}}</span>
|
|
|
+ <div class="chapter-detail-edit-page">
|
|
|
+ <van-cell title="基础信息" is-link @click="showChapterBaseInfo=true"/>
|
|
|
+ <van-cell title="上传音频" is-link v-if="info?.ReportType==='week'" @click="showUploadAudio=true"/>
|
|
|
+ <div class="main-wrap">
|
|
|
+ <div class="editor-box" id="editor"></div>
|
|
|
</div>
|
|
|
- <h1 class="report-title">{{info.Title}}</h1>
|
|
|
- <div class="auth-box">
|
|
|
- <span>{{info.Author}}</span>
|
|
|
- <span>{{moment(info.CreateTime).format('YYYY-MM-DD HH:mm:ss')}}</span>
|
|
|
+ <!-- 底部操作 -->
|
|
|
+ <div class="bot-action-box">
|
|
|
+ <div class="left-box">
|
|
|
+ <div class="item" @click="handleRefreshAllChart">
|
|
|
+ <img src="@/assets/imgs/report/icon_refresh.png" alt="">
|
|
|
+ <span>刷新</span>
|
|
|
+ </div>
|
|
|
+ <div class="item" @click="handleReportOpt('yl')">
|
|
|
+ <img src="@/assets/imgs/report/icon_preview.png" alt="">
|
|
|
+ <span>预览</span>
|
|
|
+ </div>
|
|
|
+ <div class="item" @click="handleReportOpt('cg')">
|
|
|
+ <img src="@/assets/imgs/report/icon_save2.png" alt="">
|
|
|
+ <span>保存</span>
|
|
|
+ </div>
|
|
|
+ <div class="item" @click="handleReportOpt('fb')">
|
|
|
+ <img src="@/assets/imgs/report/icon_publish3.png" alt="">
|
|
|
+ <span>发布</span>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="right-btn" @click="showReportInsertPop=true">
|
|
|
+ <svg width="28" height="28" viewBox="0 0 28 28" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
|
+ <path d="M12.0499 15.9499V27.5H15.9499V15.9499H27.5V12.0499H15.9499V0.5H12.0499V12.0499H0.5V15.9499H12.0499Z" fill="white"/>
|
|
|
+ </svg>
|
|
|
+ </div>
|
|
|
</div>
|
|
|
- <!-- 音频 -->
|
|
|
- <AudioBox :url="info.VideoUrl" v-if="info.VideoUrl"/>
|
|
|
- <div class="report-abstract">{{info.Abstract}}</div>
|
|
|
- <div class="report-html-wrap" v-html="info.Content"></div>
|
|
|
</div>
|
|
|
+
|
|
|
+ <!-- 基础信息 -->
|
|
|
+ <van-popup
|
|
|
+ v-model:show="showChapterBaseInfo"
|
|
|
+ position="bottom"
|
|
|
+ :style="{height:'100%'}"
|
|
|
+ >
|
|
|
+ <EditChapterBaseInfo
|
|
|
+ v-if="showChapterBaseInfo"
|
|
|
+ :chapterInfo="info"
|
|
|
+ :defaultData="chapterBaseInfo"
|
|
|
+ @close="showChapterBaseInfo=false"
|
|
|
+ @confirm="handleChapterBaseInfoSave"
|
|
|
+ />
|
|
|
+ </van-popup>
|
|
|
+
|
|
|
+ <!-- 上传音频 -->
|
|
|
+ <van-popup
|
|
|
+ v-model:show="showUploadAudio"
|
|
|
+ position="bottom"
|
|
|
+ :style="{height:'100%'}"
|
|
|
+ >
|
|
|
+ <div class="upload-audio-wrap">
|
|
|
+
|
|
|
+ <div class="bot-btns">
|
|
|
+ <van-uploader
|
|
|
+ accept="*"
|
|
|
+ :after-read="handleAudioUploadAfterRead"
|
|
|
+ :before-read="handleAudioUploadBeforeRead"
|
|
|
+ >
|
|
|
+ <van-button class="bot-btn" type="default">重新上传</van-button>
|
|
|
+ </van-uploader>
|
|
|
+ <van-button class="bot-btn" type="primary">保存</van-button>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </van-popup>
|
|
|
</template>
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
-.top-stage-box{
|
|
|
- .stage{
|
|
|
- display: inline-block;
|
|
|
- background-color: #F2F3FF;
|
|
|
- border-radius: 8px;
|
|
|
- height: 72px;
|
|
|
- line-height: 72px;
|
|
|
- padding: 0 20px;
|
|
|
- font-size: 28px;
|
|
|
- }
|
|
|
+.chapter-detail-edit-page{
|
|
|
+ height: 100%;
|
|
|
+ display: flex;
|
|
|
+ flex-direction: column;
|
|
|
+ overflow: hidden;
|
|
|
}
|
|
|
-.report-detail-wrap{
|
|
|
- padding: 30px 34px;
|
|
|
- .report-title{
|
|
|
- margin: 30px 0;
|
|
|
- font-weight: 600;
|
|
|
- font-size: 42px;
|
|
|
- line-height: 56px;
|
|
|
+.van-cell{
|
|
|
+ flex-shrink: 0;
|
|
|
+}
|
|
|
+.main-wrap{
|
|
|
+ flex: 1;
|
|
|
+ width: calc(100% - 32PX);
|
|
|
+ margin: 0 auto;
|
|
|
+ margin-top: 30px;
|
|
|
+ .editor-box{
|
|
|
+ width: 100%;
|
|
|
+ height: 100%;
|
|
|
}
|
|
|
- .auth-box{
|
|
|
+}
|
|
|
+.bot-action-box{
|
|
|
+ padding: 20px 16PX;
|
|
|
+ display: flex;
|
|
|
+ align-items: center;
|
|
|
+ .left-box{
|
|
|
+ flex: 1;
|
|
|
+ background: #FFFFFF;
|
|
|
+ box-shadow: 0px 12px 60px 10px rgba(0, 0, 0, 0.05), 0px 32px 48px 4px rgba(0, 0, 0, 0.04), 0px 16px 20px -10px rgba(0, 0, 0, 0.08);
|
|
|
+ border-radius: 100px;
|
|
|
+ height: 112px;
|
|
|
display: flex;
|
|
|
- justify-content: space-between;
|
|
|
- font-size: $font-grey;
|
|
|
- font-size: 36px;
|
|
|
- padding-bottom: 40px;
|
|
|
- border-bottom: 1px solid $border-color;
|
|
|
- margin-bottom: 40px;
|
|
|
+ align-items: center;
|
|
|
+ margin-right: 20px;
|
|
|
+ padding: 0 20px;
|
|
|
+ .item{
|
|
|
+ flex: 1;
|
|
|
+ text-align: center;
|
|
|
+ font-size: 20px;
|
|
|
+ img{
|
|
|
+ width: 40px;
|
|
|
+ height: 40px;
|
|
|
+ display: block;
|
|
|
+ margin: 5px auto;
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
- .report-abstract{
|
|
|
- font-size: 34px;
|
|
|
- line-height: 54px;
|
|
|
- margin: 40px 0;
|
|
|
+ .right-btn{
|
|
|
+ flex-shrink: 0;
|
|
|
+ position: relative;
|
|
|
+ width: 96px;
|
|
|
+ height: 96px;
|
|
|
+ background-color: $theme-color;
|
|
|
+ border-radius: 50%;
|
|
|
+ box-shadow: 0px 6px 28px 4px rgba(0, 0, 0, 0.05), 0px 16px 20px 2px rgba(0, 0, 0, 0.06), 0px 10px 10px -6px rgba(0, 0, 0, 0.1);
|
|
|
+ svg{
|
|
|
+ width: 27px;
|
|
|
+ height: 27px;
|
|
|
+ position: absolute;
|
|
|
+ top: 50%;
|
|
|
+ left: 50%;
|
|
|
+ transform: translate(-50%,-50%);
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-@media screen and (min-width:$media-width){
|
|
|
- .top-stage-box{
|
|
|
- .stage{
|
|
|
- border-radius: 4px;
|
|
|
- height: 36px;
|
|
|
- line-height: 36px;
|
|
|
- padding: 0 10px;
|
|
|
- font-size: 14px;
|
|
|
- }
|
|
|
- }
|
|
|
- .report-detail-wrap{
|
|
|
- padding: 30px;
|
|
|
- max-width: 800px;
|
|
|
- margin: 0 auto;
|
|
|
- .report-title{
|
|
|
- margin: 15px 0;
|
|
|
- font-size: 21px;
|
|
|
- line-height: 28px;
|
|
|
- }
|
|
|
- .auth-box{
|
|
|
- font-size: 18px;
|
|
|
- padding-bottom: 20px;
|
|
|
- margin-bottom: 20px;
|
|
|
- }
|
|
|
- .report-abstract{
|
|
|
- font-size: 17px;
|
|
|
- line-height: 27px;
|
|
|
- margin: 20px 0;
|
|
|
+.upload-audio-wrap{
|
|
|
+ height: 100%;
|
|
|
+ position: relative;
|
|
|
+ .bot-btns{
|
|
|
+ width: 100%;
|
|
|
+ position: absolute;
|
|
|
+ bottom: 0;
|
|
|
+ padding: 20px 0;
|
|
|
+ text-align: center;
|
|
|
+ .bot-btn{
|
|
|
+ width: 315px;
|
|
|
+ margin: 0 10px;
|
|
|
}
|
|
|
}
|
|
|
-}
|
|
|
+}
|
|
|
</style>
|