irc_web/src/views/trials/trials-panel/reading/visit-review/components/ReadPage.vue

328 lines
8.9 KiB
Vue

<template>
<div v-loading="loading" class="read-page-container">
<!-- 检查列表 -->
<div class="left-panel">
<div class="task-container">
<div class="task-info">
<div
v-for="s in visitTaskList"
:key="s.VisitTaskId"
class="task-item"
:class="{'task-item-active': activeTaskVisitId==s.VisitTaskId}"
@click.prevent="toggleTask(s)"
>{{ s.TaskBlindName }}</div>
</div>
</div>
<div class="study-info">
<div
v-for="s in visitTaskList"
v-show="activeTaskVisitId === s.VisitTaskId"
:key="s.VisitTaskId"
style="height:100%;"
>
<study-list
v-if="selectArr.includes(s.VisitTaskId) && s.StudyList.length > 0"
:ref="s.VisitTaskId"
:visit-task-info="s"
@selectFile="selectFile"
/>
</div>
</div>
</div>
<!-- 图像 -->
<div class="middle-panel">
<image-viewer
ref="imageViewer"
:related-study-info="relatedStudyInfo"
@toggleTaskByViewer="toggleTaskByViewer"
@toggleTask="toggleTask"
@toggleImage="toggleImage"
/>
</div>
<!-- 表单 -->
<div class="right-panel">
<div
v-if="taskInfo && taskInfo.IsReadingShowSubjectInfo"
class="text-info"
>
<h3>
<span
v-if="currentVisitInfo && currentVisitInfo.SubjectCode"
>
{{ currentVisitInfo.SubjectCode }}
</span>
<span
v-if="currentVisitInfo && currentVisitInfo.TaskBlindName"
style="margin-left:5px;"
>
{{ currentVisitInfo.TaskBlindName }}
</span>
</h3>
</div>
<ecrf-list
v-if="currentVisitInfo"
:visit-task-info="currentVisitInfo"
/>
</div>
</div>
</template>
<script>
import { getRelatedVisitTask, getReadingImageFile, getNoneDicomMarkListOutDto } from '@/api/trials'
import StudyList from './StudyList'
import ImageViewer from './ImageViewer'
import EcrfList from './EcrfList'
import { mapGetters } from 'vuex'
export default {
name: 'ReadPage',
components: {
StudyList,
ImageViewer,
EcrfList
},
data() {
return {
loading: false,
// 当前任务信息
taskInfo: null,
// 当前激活的任务Id
activeTaskVisitId: '',
// 表单数据
formData: {
name: '',
description: ''
},
// 当前任务关联的任务信息
visitTaskList: [],
selectArr: [],
fileType: 'image',
currentStudyInfo: null,
currentVisitInfo: null,
relatedStudyInfo: null
}
},
computed: {
...mapGetters(['lastViewportTaskId'])
},
watch: {
lastViewportTaskId: {
immediate: true,
handler(id) {
if (!id) return
const idx = this.visitTaskList.findIndex(i => i.VisitTaskId === id)
if (idx === -1) return
this.currentVisitInfo = this.visitTaskList[idx]
// this.activeTaskVisitId = id
}
}
},
mounted() {
this.taskInfo = JSON.parse(localStorage.getItem('taskInfo'))
this.getRelatedTask()
},
methods: {
// 获取关联任务信息
async getRelatedTask() {
this.loading = true
try {
const params = {
visitTaskId: this.taskInfo.VisitTaskId
}
const res = await getRelatedVisitTask(params)
this.visitTaskList = res.Result.map((item) => ({
...item,
StudyList: [],
Annotations: []
}))
const idx = res.Result.findIndex(i => i.IsCurrentTask)
if (idx > -1) {
await this.setActiveTaskVisitId(res.Result[idx].VisitTaskId)
this.$nextTick(() => {
this.$refs[res.Result[idx].VisitTaskId][0].setInitActiveFile()
})
}
if (this.taskInfo.IsReadingTaskViewInOrder === 1 && res.Result.length > 1) {
const i = this.visitTaskList.findIndex(i => i.IsBaseLineTask)
if (i > -1) {
await this.getReadingImageFile(res.Result[i].VisitTaskId, i)
await this.getAnnotations(res.Result[i].VisitTaskId, i)
const studyList = this.visitTaskList[i].StudyList
if (studyList.length > 0) {
const fileInfo = studyList[0].NoneDicomStudyFileList[0]
this.relatedStudyInfo = { fileInfo, visitTaskInfo: this.visitTaskList[i], fileList: studyList[0].NoneDicomStudyFileList, fileIndex: 0, studyId: studyList[0].Id }
if (!this.selectArr.includes(res.Result[i].VisitTaskId)) {
this.selectArr.push(res.Result[i].VisitTaskId)
}
}
}
}
this.loading = false
} catch (e) {
console.log(e)
this.loading = false
}
},
// 获取任务关联的文件信息
getReadingImageFile(visitTaskId, visitTaskIdx) {
return new Promise(async(resolve, reject) => {
this.loading = true
try {
const params = {
subjectId: this.taskInfo.SubjectId,
trialId: this.$route.query.trialId,
visistTaskId: visitTaskId
}
const res = await getReadingImageFile(params)
this.$set(this.visitTaskList[visitTaskIdx], 'StudyList', res.Result)
this.loading = false
resolve()
} catch (e) {
console.log(e)
this.loading = false
reject(e)
}
})
},
// 获取任务关联的标注信息
getAnnotations(visitTaskId, visitTaskIdx) {
return new Promise(async(resolve, reject) => {
this.loading = true
try {
const params = {
visitTaskId: visitTaskId
}
const res = await getNoneDicomMarkListOutDto(params)
this.$set(this.visitTaskList[visitTaskIdx], 'Annotations', res.Result.NoneDicomMarkList)
this.loading = false
resolve()
} catch (e) {
console.log(e)
this.loading = false
reject(e)
}
})
},
// 切换任务
toggleTask(taskInfo) {
this.setActiveTaskVisitId(taskInfo.VisitTaskId)
},
// 视图触发任务切换
async toggleTaskByViewer(visitTaskNum) {
const i = this.visitTaskList.findIndex(v => v.VisitTaskNum === visitTaskNum)
if (i === -1) return
const visistTaskId = this.visitTaskList[i].VisitTaskId
this.setActiveTaskVisitId(visistTaskId, true)
},
// 设置激活的访视
async setActiveTaskVisitId(id, isInitActiveFile = false) {
if (!id) return
if (!this.selectArr.includes(id)) {
this.selectArr.push(id)
}
const idx = this.visitTaskList.findIndex(i => i.VisitTaskId === id)
if (idx === -1) return
if (this.visitTaskList[idx].StudyList.length === 0) {
await this.getReadingImageFile(id, idx)
await this.getAnnotations(id, idx)
}
this.activeTaskVisitId = id
this.$nextTick(() => {
if (isInitActiveFile) {
this.$refs[id][0].setInitActiveFile()
}
})
},
// 切换图片
toggleImage(obj) {
this.$refs[obj.taskId][0].activeImage(obj)
},
selectFile(obj) {
this.$refs['imageViewer'].setActiveCanvasImages(obj)
}
}
}
</script>
<style lang="scss" scoped>
.read-page-container {
box-sizing: border-box;
height: 100%;
display: flex;
::-webkit-scrollbar {
width: 5px;
height: 5px;
}
::-webkit-scrollbar-thumb {
border-radius: 10px;
background: #d0d0d0;
}
.left-panel {
display: flex;
width: 200px;
border: 1px solid #727272;
color: #fff;
::-webkit-scrollbar {
width: 3px;
height: 3px;
}
::-webkit-scrollbar-thumb {
border-radius: 10px;
background: #d0d0d0;
}
.task-container {
position: relative;
width: 25px;
overflow-y: auto;
}
.task-info {
position: absolute;
top: 5px;
right: 20px;
transform-origin: right top;
transform: rotate(-90deg);
display: flex;
.task-item {
margin-left: 10px;
white-space: nowrap;
padding: 0px 4px;
border: 1px solid #999999;
border-bottom:none ;
text-align: center;
background-color: #4e4e4e;
color: #d5d5d5;
cursor: pointer;
}
.task-item-active {
background-color: #607d8b;
border: 1px solid #607d8b;
}
}
.study-info {
width: 170px;
border-left: 1px solid #727272;
}
}
.middle-panel {
flex: 1;
border: 1px solid #727272;
margin: 0 5px;
}
.right-panel {
width: 400px;
border: 1px solid #727272;
padding: 0 10px;
.text-info {
display: flex;
align-items: center;
justify-content: space-between;
color: #ddd;
}
}
}
</style>