|
@@ -1,20 +1,24 @@
|
|
|
<script setup name="reportChapterDetail">
|
|
|
import {ref,onMounted,onUnmounted, reactive, nextTick} from 'vue'
|
|
|
import apiReport from '@/api/report'
|
|
|
-import { useRoute } from 'vue-router'
|
|
|
-import EditChapterBaseInfo from './components/EditChapterBaseInfo.vue'
|
|
|
+import apiChart from '@/api/chart'
|
|
|
+import { useRoute, useRouter } from 'vue-router'
|
|
|
+import EditChapterBaseInfo from './components/EditChapterBaseInfo.vue'
|
|
|
+import AudioBox from '../components/AudioBox.vue'
|
|
|
+import ReportInsertContent from '../components/reportInsert/Index.vue'
|
|
|
import moment from 'moment'
|
|
|
import {useInitFroalaEditor} from '@/hooks/useFroalaEditor'
|
|
|
import {useUserInfo} from '@/hooks/common'
|
|
|
-import { showToast } from 'vant'
|
|
|
+import { showToast,showDialog } from 'vant'
|
|
|
import {useUploadFileToOSS} from '@/hooks/useUploadFileToOSS'
|
|
|
import MD5 from 'js-md5'
|
|
|
-import AudioBox from '../components/AudioBox.vue'
|
|
|
+
|
|
|
|
|
|
const userInfo=useUserInfo()
|
|
|
|
|
|
const {FroalaEditorIns,initFroalaEditor,frolaEditorContentChange}=useInitFroalaEditor()
|
|
|
const route=useRoute()
|
|
|
+const router=useRouter()
|
|
|
|
|
|
let autoSaveTimer=null
|
|
|
|
|
@@ -28,15 +32,49 @@ onUnmounted(()=>{
|
|
|
clearInterval(autoSaveTimer)
|
|
|
})
|
|
|
// 自动保存
|
|
|
-function autoSaveReportContent(){
|
|
|
-
|
|
|
+function autoSaveReportContent(e){
|
|
|
+ if(!e&&!frolaEditorContentChange.value) return
|
|
|
+ let arr=[]
|
|
|
+ ticketList.value.forEach(item=>{
|
|
|
+ let obj={
|
|
|
+ ticker:'',
|
|
|
+ sort:1,
|
|
|
+ lable:''
|
|
|
+ }
|
|
|
+ if(item.Selected){
|
|
|
+ obj.ticker=item.BaseColumnTicker
|
|
|
+ obj.sort=arr.length+1
|
|
|
+ obj.lable=item.BaseColumnName
|
|
|
+ arr.push(obj)
|
|
|
+ }
|
|
|
+ })
|
|
|
+ apiReport.chapterDetailSave({
|
|
|
+ ReportChapterId: Number(route.query.id),
|
|
|
+ Title:chapterBaseInfo.title,
|
|
|
+ AddType:chapterBaseInfo.addType,
|
|
|
+ Author:chapterBaseInfo.author,
|
|
|
+ CreateTime:chapterBaseInfo.createTime,
|
|
|
+ TickerList:arr,
|
|
|
+ Content: $('.fr-element').html(),
|
|
|
+ VideoUrl:chapterBaseInfo.audioUrl,
|
|
|
+ VideoName:`${chapterBaseInfo.title}(${moment().format('MMDD')})`,
|
|
|
+ VideoPlaySeconds:Number(chapterBaseInfo.audioDuration).toFixed(2).toString(),
|
|
|
+ VideoSize:Number(chapterBaseInfo.audioSize).toFixed(2).toString()
|
|
|
+ }).then(res=>{
|
|
|
+ if(res.Ret===200){
|
|
|
+ if(e==='cg'){
|
|
|
+ showToast('保存成功')
|
|
|
+ }
|
|
|
+ }
|
|
|
+ frolaEditorContentChange.value=false
|
|
|
+ })
|
|
|
}
|
|
|
|
|
|
|
|
|
let info=ref(null)
|
|
|
function getChapterDetail(){
|
|
|
apiReport.getChapterDetail({
|
|
|
- ReportChapterId:(Number(route.query.id))
|
|
|
+ ReportChapterId:Number(route.query.id)
|
|
|
}).then(res=>{
|
|
|
if(res.Ret===200){
|
|
|
info.value=res.Data
|
|
@@ -52,6 +90,11 @@ function getChapterDetail(){
|
|
|
chapterBaseInfo.audioDuration=res.Data.VideoPlaySeconds
|
|
|
chapterBaseInfo.audioSize=res.Data.VideoSize
|
|
|
}
|
|
|
+
|
|
|
+ // 晨报获取指标数据
|
|
|
+ if(res.Data.ReportType=='day'){
|
|
|
+ getDayTicketList()
|
|
|
+ }
|
|
|
|
|
|
// 由于周报有个上传音频区域 所以得在此处初始化富文本区域 不然高度有问题
|
|
|
nextTick(()=>{
|
|
@@ -65,7 +108,17 @@ function getChapterDetail(){
|
|
|
}
|
|
|
})
|
|
|
}
|
|
|
-getChapterDetail()
|
|
|
+
|
|
|
+// 获取晨报指标数据
|
|
|
+let ticketList=ref([])
|
|
|
+async function getDayTicketList(){
|
|
|
+ const res=await apiReport.chapterDayReportTicketList({
|
|
|
+ ReportChapterId:info.value.ReportChapterId
|
|
|
+ })
|
|
|
+ if(res.Ret===200){
|
|
|
+ ticketList.value=res.Data||[]
|
|
|
+ }
|
|
|
+}
|
|
|
|
|
|
// 报告基本内容
|
|
|
const showChapterBaseInfo=ref(false)
|
|
@@ -73,23 +126,47 @@ const chapterBaseInfo=reactive({
|
|
|
type:'',
|
|
|
title:'',
|
|
|
addType:'',
|
|
|
- ticket:[],
|
|
|
author:'',
|
|
|
createTime:'',
|
|
|
audioUrl:'',
|
|
|
audioDuration:0,
|
|
|
audioSize:0,
|
|
|
})
|
|
|
-// 保存报告基本内容
|
|
|
-function handleChapterBaseInfoSave(e){
|
|
|
+// 更新报告基本内容
|
|
|
+async function handleChapterBaseInfoSave(e){
|
|
|
console.log(e);
|
|
|
- // 继承报告
|
|
|
- if(e.addType==2){
|
|
|
+ chapterBaseInfo.type=e.type
|
|
|
+ chapterBaseInfo.title=e.title
|
|
|
+ chapterBaseInfo.author=e.author
|
|
|
+ chapterBaseInfo.createTime=e.createTime
|
|
|
+ ticketList.value=e.ticket
|
|
|
|
|
|
- }else{
|
|
|
-
|
|
|
+ // 如果修改报告的新增方式 则刷新数据
|
|
|
+ if(chapterBaseInfo.addType!=e.addType){
|
|
|
+ chapterBaseInfo.addType=e.addType
|
|
|
+ // 继承报告
|
|
|
+ if(e.addType==2){
|
|
|
+ const res=await apiReport.chapterReportPreContent({
|
|
|
+ TypeId:info.value.TypeId,
|
|
|
+ ReportType:info.value.ReportType
|
|
|
+ })
|
|
|
+ if(res.Ret===200){
|
|
|
+ chapterBaseInfo.type=res.Data.TypeName
|
|
|
+ chapterBaseInfo.title=res.Data.Title
|
|
|
+ chapterBaseInfo.author=res.Data.Author||userInfo.RealName
|
|
|
+ chapterBaseInfo.createTime=moment(res.Data.CreateTime).format('YYYY-MM-DD')
|
|
|
+ FroalaEditorIns.value.html.set(res.Data.Content);
|
|
|
+ // 晨报获取指标数据
|
|
|
+ if(res.data.ReportType=='day'){
|
|
|
+ getDayTicketList()
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ FroalaEditorIns.value.html.set('');
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
+
|
|
|
showChapterBaseInfo.value=false
|
|
|
}
|
|
|
|
|
@@ -156,6 +233,180 @@ function handleUpdateAudio(){
|
|
|
showUploadAudio.value=false
|
|
|
}
|
|
|
|
|
|
+// 报告插入数据弹窗
|
|
|
+const showReportInsertPop=ref(false)
|
|
|
+/**
|
|
|
+ * list:[UniqueCode] 图表code
|
|
|
+ * type:iframe/img 插入的为iframe或者图片
|
|
|
+ * chartType: chart-图表,sheet-表格
|
|
|
+ */
|
|
|
+function handleInsert({list,type,chartType}){
|
|
|
+ if(type==='iframe'){
|
|
|
+ let link;
|
|
|
+ if(chartType==='chart'){
|
|
|
+ link=import.meta.env.MODE==='production'?'https://chartlib.hzinsights.com/chartshow':'https://charttest.hzinsights.com/chartshow'
|
|
|
+ list.forEach(item => {
|
|
|
+ FroalaEditorIns.value.html.insert(`<p style='text-align:left; margin-top:10px;'>
|
|
|
+ <iframe src='${link}?code=${item}&fromPage=' width='100%' height='350' style='border-width:0px; min-height:350px;'></iframe>
|
|
|
+ </p>`,false)
|
|
|
+ });
|
|
|
+ }else if(chartType==='sheet'){
|
|
|
+ link=import.meta.env.MODE==='production'?'https://chartlib.hzinsights.com/sheetshow':'https://charttest.hzinsights.com/sheetshow'
|
|
|
+ list.forEach(item => {
|
|
|
+ FroalaEditorIns.value.html.insert(`<p style='text-align:left; margin-top:10px;'>
|
|
|
+ <iframe src='${link}?code=${item}' class='iframe${item}' width='100%' style='border-width:0px;'></iframe>
|
|
|
+ </p>`,false)
|
|
|
+ });
|
|
|
+ }
|
|
|
+ }else if(type==='img'){
|
|
|
+ list.forEach(item=>{
|
|
|
+ FroalaEditorIns.value.html.insert(`<img style='width:100%' src='${item}' />`,false)
|
|
|
+ })
|
|
|
+ }
|
|
|
+
|
|
|
+ showReportInsertPop.value=false
|
|
|
+}
|
|
|
+
|
|
|
+// 更新sheet表格高度
|
|
|
+function reInitSheetIframe(e){
|
|
|
+ const { height,code } = e.data;
|
|
|
+ let iframeDom = document.getElementsByClassName(`iframe${code}`)
|
|
|
+ Array.prototype.forEach.call(iframeDom, function (ele) {
|
|
|
+ ele.height = `${height+45}px`;
|
|
|
+ });
|
|
|
+}
|
|
|
+onMounted(()=>{
|
|
|
+ window.addEventListener('message',reInitSheetIframe)
|
|
|
+})
|
|
|
+onUnmounted(()=>{
|
|
|
+ window.removeEventListener('message',reInitSheetIframe)
|
|
|
+})
|
|
|
+
|
|
|
+// 刷新所有图表
|
|
|
+async function handleRefreshAllChart(){
|
|
|
+ let code_arr = [];
|
|
|
+ $('iframe').each((k,i) => {
|
|
|
+ try {
|
|
|
+ let href = $(i).attr('src');
|
|
|
+ code_arr.push(href.slice(href.indexOf('code=') + 5));
|
|
|
+ } catch (err) {
|
|
|
+ }
|
|
|
+ });
|
|
|
+ if(!code_arr.length) return showToast('请插入图表');
|
|
|
+ const res=await apiChart.refreshChartMultiple({ChartInfoCode:code_arr})
|
|
|
+ if(res.Ret===200){
|
|
|
+ $('iframe').each((k,i) => {
|
|
|
+ $(i).attr('src',$(i).attr('src'))
|
|
|
+ });
|
|
|
+ showToast('刷新成功')
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//周报校验是否上传了音频
|
|
|
+function handleWeekValidAudio(){
|
|
|
+ return new Promise((resolve,reject)=>{
|
|
|
+ showDialog({
|
|
|
+ title: '发布提示',
|
|
|
+ message: '您还未上传录音文件,确定发布报告吗?',
|
|
|
+ showCancelButton:true
|
|
|
+ }).then(()=>{
|
|
|
+ resolve(true)
|
|
|
+ }).catch(()=>{
|
|
|
+ resolve(false)
|
|
|
+ })
|
|
|
+ })
|
|
|
+}
|
|
|
+
|
|
|
+// 发布章节报告
|
|
|
+async function chapterReportPublish(PublishReport){
|
|
|
+ let arr=[]
|
|
|
+ ticketList.value.forEach(item=>{
|
|
|
+ let obj={
|
|
|
+ ticker:'',
|
|
|
+ sort:1,
|
|
|
+ lable:''
|
|
|
+ }
|
|
|
+ if(item.Selected){
|
|
|
+ obj.ticker=item.BaseColumnTicker
|
|
|
+ obj.sort=arr.length+1
|
|
|
+ obj.lable=item.BaseColumnName
|
|
|
+ arr.push(obj)
|
|
|
+ }
|
|
|
+ })
|
|
|
+ const res=await apiReport.chapterReportPublish({
|
|
|
+ ReportChapterId: Number(route.query.id),
|
|
|
+ Title:chapterBaseInfo.title,
|
|
|
+ AddType:chapterBaseInfo.addType,
|
|
|
+ Author:chapterBaseInfo.author,
|
|
|
+ CreateTime:chapterBaseInfo.createTime,
|
|
|
+ TickerList:arr,
|
|
|
+ Content: $('.fr-element').html(),
|
|
|
+ PublishReport:PublishReport,
|
|
|
+ VideoUrl:chapterBaseInfo.audioUrl,
|
|
|
+ VideoName:`${chapterBaseInfo.title}(${moment().format('MMDD')})`,
|
|
|
+ VideoPlaySeconds:Number(chapterBaseInfo.audioDuration).toFixed(2).toString(),
|
|
|
+ VideoSize:Number(chapterBaseInfo.audioSize).toFixed(2).toString()
|
|
|
+ })
|
|
|
+ if(res.Ret===200){
|
|
|
+ showToast('发布成功')
|
|
|
+ setTimeout(() => {
|
|
|
+ router.back()
|
|
|
+ }, 1000);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+// 发布报告-fb;保存-cg;预览-yl
|
|
|
+async function handleReportOpt(type){
|
|
|
+ if(type==='yl'){
|
|
|
+
|
|
|
+ const params={
|
|
|
+ Title:chapterBaseInfo.title,
|
|
|
+ Author:chapterBaseInfo.author,
|
|
|
+ PublishTime:chapterBaseInfo.createTime,
|
|
|
+ Abstract:'',
|
|
|
+ Content:$('.fr-element').html()
|
|
|
+ }
|
|
|
+ if(!params.Title){
|
|
|
+ showToast('请填写标题')
|
|
|
+ return
|
|
|
+ }
|
|
|
+ sessionStorage.setItem('reportPreData',JSON.stringify(params))
|
|
|
+ const routerEl=router.resolve({
|
|
|
+ path:'/report/preview',
|
|
|
+ query:{
|
|
|
+ id:-1
|
|
|
+ }
|
|
|
+ })
|
|
|
+ window.open(routerEl.href,'_blank')
|
|
|
+ return
|
|
|
+ }
|
|
|
+ if(type==='cg'){
|
|
|
+ autoSaveReportContent('cg')
|
|
|
+ }
|
|
|
+ if(type==='fb'){
|
|
|
+ if(info.value.ReportType==='week'&&!chapterBaseInfo.audioUrl){
|
|
|
+ const validRes=await handleWeekValidAudio()
|
|
|
+ if(!validRes) return
|
|
|
+ }
|
|
|
+ const res=await apiReport.chapterReportIsLast({ReportChapterId:info.value.ReportChapterId})
|
|
|
+ if(res.Ret===200){
|
|
|
+ if(res.Data){
|
|
|
+ showDialog({
|
|
|
+ title: '发布提示',
|
|
|
+ message: '本期报告品种已全部更新,点击发布将同时发布周报,确认同时发布吗?',
|
|
|
+ showCancelButton:true
|
|
|
+ }).then(()=>{
|
|
|
+ chapterReportPublish(1)
|
|
|
+ }).catch(()=>{
|
|
|
+ chapterReportPublish(0)
|
|
|
+ })
|
|
|
+ }else{
|
|
|
+ chapterReportPublish(0)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
</script>
|
|
|
|
|
|
<template>
|
|
@@ -201,6 +452,7 @@ function handleUpdateAudio(){
|
|
|
>
|
|
|
<EditChapterBaseInfo
|
|
|
v-if="showChapterBaseInfo"
|
|
|
+ :ticketData="ticketList"
|
|
|
:chapterInfo="info"
|
|
|
:defaultData="chapterBaseInfo"
|
|
|
@close="showChapterBaseInfo=false"
|
|
@@ -233,11 +485,20 @@ function handleUpdateAudio(){
|
|
|
</div>
|
|
|
</div>
|
|
|
</van-popup>
|
|
|
+
|
|
|
+ <!-- 报告插入数据模块 -->
|
|
|
+ <van-popup
|
|
|
+ v-model:show="showReportInsertPop"
|
|
|
+ position="bottom"
|
|
|
+ round
|
|
|
+ >
|
|
|
+ <report-insert-content v-if="showReportInsertPop" @insert="handleInsert"/>
|
|
|
+ </van-popup>
|
|
|
</template>
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
.chapter-detail-edit-page{
|
|
|
- height: 100%;
|
|
|
+ height: 100dvh;
|
|
|
display: flex;
|
|
|
flex-direction: column;
|
|
|
overflow: hidden;
|
|
@@ -310,9 +571,10 @@ function handleUpdateAudio(){
|
|
|
padding-bottom: 32px;
|
|
|
border-bottom: 1px solid $border-color;
|
|
|
margin-bottom: 32px;
|
|
|
+ word-wrap: break-word;
|
|
|
}
|
|
|
.bot-btns{
|
|
|
- width: 100%;
|
|
|
+ // width: 100%;
|
|
|
position: absolute;
|
|
|
bottom: 0;
|
|
|
padding: 20px 0;
|