Compare commits
No commits in common. "92f23a35f9ab5d29a16e3d2ac122fc6ea29c671e" and "64e91dffbe4c2f8705d51492bd8038bd6002d28f" have entirely different histories.
92f23a35f9
...
64e91dffbe
|
@ -15,5 +15,3 @@ VITE_APP_UPLOAD_API = 'https://file.ysaix.com:7868/prod-api'
|
||||||
VITE_APP_RES_FILE_PATH = 'https://file.ysaix.com:7868/src/assets/textbook/booktxt/'
|
VITE_APP_RES_FILE_PATH = 'https://file.ysaix.com:7868/src/assets/textbook/booktxt/'
|
||||||
|
|
||||||
VITE_APP_BUILD_BASE_PATH = 'https://file.ysaix.com:7868/'
|
VITE_APP_BUILD_BASE_PATH = 'https://file.ysaix.com:7868/'
|
||||||
|
|
||||||
VITE_SHOW_DEV_TOOLS = 'true'
|
|
||||||
|
|
|
@ -17,5 +17,3 @@ VITE_BUILD_COMPRESS = gzip
|
||||||
VITE_APP_RES_FILE_PATH = 'https://prev.ysaix.com:7868/src/assets/textbook/booktxt/'
|
VITE_APP_RES_FILE_PATH = 'https://prev.ysaix.com:7868/src/assets/textbook/booktxt/'
|
||||||
|
|
||||||
VITE_APP_BUILD_BASE_PATH = 'https://prev.ysaix.com:7868/'
|
VITE_APP_BUILD_BASE_PATH = 'https://prev.ysaix.com:7868/'
|
||||||
|
|
||||||
VITE_SHOW_DEV_TOOLS = 'true'
|
|
||||||
|
|
|
@ -73,7 +73,8 @@ function createLoginWindow() {
|
||||||
loginWindow.loadFile(join(__dirname, '../renderer/index.html'), { hash: 'login' })
|
loginWindow.loadFile(join(__dirname, '../renderer/index.html'), { hash: 'login' })
|
||||||
updateInit(loginWindow)
|
updateInit(loginWindow)
|
||||||
}
|
}
|
||||||
if (import.meta.env.VITE_SHOW_DEV_TOOLS === 'true') loginWindow.webContents.openDevTools()
|
|
||||||
|
loginWindow.webContents.openDevTools()
|
||||||
loginWindow.once('ready-to-show', () => {
|
loginWindow.once('ready-to-show', () => {
|
||||||
loginWindow.show()
|
loginWindow.show()
|
||||||
})
|
})
|
||||||
|
@ -95,6 +96,7 @@ function createMainWindow() {
|
||||||
frame: false, // 无边框
|
frame: false, // 无边框
|
||||||
autoHideMenuBar: true,
|
autoHideMenuBar: true,
|
||||||
maximizable: false,
|
maximizable: false,
|
||||||
|
fullscreen: true,
|
||||||
icon: join(__dirname, '../../resources/logo2.ico'),
|
icon: join(__dirname, '../../resources/logo2.ico'),
|
||||||
...(process.platform === 'linux' ? { icon } : {}),
|
...(process.platform === 'linux' ? { icon } : {}),
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
|
@ -127,7 +129,7 @@ function createMainWindow() {
|
||||||
shell.openExternal(details.url)
|
shell.openExternal(details.url)
|
||||||
return { action: 'deny' }
|
return { action: 'deny' }
|
||||||
})
|
})
|
||||||
if (import.meta.env.VITE_SHOW_DEV_TOOLS === 'true') mainWindow.webContents.openDevTools()
|
mainWindow.webContents.openDevTools()
|
||||||
|
|
||||||
if (is.dev && process.env['ELECTRON_RENDERER_URL']) {
|
if (is.dev && process.env['ELECTRON_RENDERER_URL']) {
|
||||||
mainWindow.loadURL(process.env['ELECTRON_RENDERER_URL'])
|
mainWindow.loadURL(process.env['ELECTRON_RENDERER_URL'])
|
||||||
|
@ -137,7 +139,6 @@ function createMainWindow() {
|
||||||
|
|
||||||
// mainWindow.setAlwaysOnTop(true, "screen-saver") // 将窗口设置为顶层窗口
|
// mainWindow.setAlwaysOnTop(true, "screen-saver") // 将窗口设置为顶层窗口
|
||||||
// mainWindow.setVisibleOnAllWorkspaces(true) // 如果窗口在所有工作区都可见
|
// mainWindow.setVisibleOnAllWorkspaces(true) // 如果窗口在所有工作区都可见
|
||||||
mainWindow.maximize();
|
|
||||||
// 第三步: 开启remote服务
|
// 第三步: 开启remote服务
|
||||||
remote.enable(mainWindow.webContents)
|
remote.enable(mainWindow.webContents)
|
||||||
}
|
}
|
||||||
|
|
File diff suppressed because one or more lines are too long
|
@ -138,7 +138,7 @@ export const createWindow = async (type, data) => {
|
||||||
wins_tool.setIgnoreMouseEvents(true, {forward: true}) // 忽略鼠标事件但是事件继续传递给窗口
|
wins_tool.setIgnoreMouseEvents(true, {forward: true}) // 忽略鼠标事件但是事件继续传递给窗口
|
||||||
wins_tool.setAlwaysOnTop(true,'screen-saver') // 将窗口设置为顶层窗口
|
wins_tool.setAlwaysOnTop(true,'screen-saver') // 将窗口设置为顶层窗口
|
||||||
wins_tool.setVisibleOnAllWorkspaces(true) // 如果窗口在所有工作区都可见
|
wins_tool.setVisibleOnAllWorkspaces(true) // 如果窗口在所有工作区都可见
|
||||||
if (import.meta.env.VITE_SHOW_DEV_TOOLS === 'true') wins_tool.webContents.openDevTools() // 打开调试工具
|
wins_tool.webContents.openDevTools() // 打开调试工具
|
||||||
eventHandles(type, wins_tool) // 事件监听处理
|
eventHandles(type, wins_tool) // 事件监听处理
|
||||||
return wins_tool
|
return wins_tool
|
||||||
}
|
}
|
||||||
|
@ -165,7 +165,7 @@ export const createWindow = async (type, data) => {
|
||||||
win.type = type // 唯一标识
|
win.type = type // 唯一标识
|
||||||
win.show()
|
win.show()
|
||||||
win.setFullScreen(true) // 设置窗口为全屏
|
win.setFullScreen(true) // 设置窗口为全屏
|
||||||
if (import.meta.env.VITE_SHOW_DEV_TOOLS === 'true') win.webContents.openDevTools() // 打开调试工具
|
win.webContents.openDevTools() // 打开调试工具
|
||||||
eventHandles(type, win) // 事件监听处理
|
eventHandles(type, win) // 事件监听处理
|
||||||
winPdf=win
|
winPdf=win
|
||||||
break
|
break
|
||||||
|
|
|
@ -880,20 +880,18 @@ watch(() => courseObj.node, (newVal,oldVal) => {
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<!--
|
|
||||||
<style>
|
<style>
|
||||||
.el-table .hidden-row {
|
.el-table .hidden-row {
|
||||||
display: none !important;
|
display: none !important;
|
||||||
/* color: #ccc !important; */
|
/* color: #ccc !important; */
|
||||||
}
|
}
|
||||||
.el-table .father-row {
|
.el-table .father-row {
|
||||||
--el-table-tr-bg-color: #fff;
|
--el-table-tr-bg-color: #fff;
|
||||||
}
|
}
|
||||||
.el-table .son-row {
|
.el-table .son-row {
|
||||||
--el-table-tr-bg-color: #f0f0f08a;
|
--el-table-tr-bg-color: #f0f0f08a;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
-->
|
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.page-classTaskAssign {
|
.page-classTaskAssign {
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="common-layout" style="width: 100%">
|
<div class="common-layout" style="width: 100%; height: 73vh;">
|
||||||
<el-container>
|
<el-container>
|
||||||
<el-container>
|
<el-container>
|
||||||
<el-header style="height: auto">
|
<el-header style="height: auto">
|
||||||
|
@ -7,7 +7,7 @@
|
||||||
<el-card>
|
<el-card>
|
||||||
<template #header>
|
<template #header>
|
||||||
<div style="font-size: 20px;font-weight: bold">
|
<div style="font-size: 20px;font-weight: bold">
|
||||||
等级分布
|
学情分布
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<Distribution></Distribution>
|
<Distribution></Distribution>
|
||||||
|
@ -18,7 +18,7 @@
|
||||||
<el-card>
|
<el-card>
|
||||||
<template #header>
|
<template #header>
|
||||||
<div style="font-size: 20px;font-weight: bold">
|
<div style="font-size: 20px;font-weight: bold">
|
||||||
用时分析
|
时长分析
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<TimeAnalyse></TimeAnalyse>
|
<TimeAnalyse></TimeAnalyse>
|
||||||
|
@ -29,7 +29,7 @@
|
||||||
<el-card>
|
<el-card>
|
||||||
<template #header>
|
<template #header>
|
||||||
<div style="font-size: 20px;font-weight: bold">
|
<div style="font-size: 20px;font-weight: bold">
|
||||||
价值透析
|
知识点概览
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<Konwledge></Konwledge>
|
<Konwledge></Konwledge>
|
||||||
|
@ -43,12 +43,12 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import {ref,watchEffect,provide } from 'vue'
|
import Distribution from '@/views/classTask/container/classOverview/distribution.vue'
|
||||||
import Distribution from './classOverview/distribution.vue'
|
import Konwledge from '@/views/classTask/container/classOverview/knowledge.vue'
|
||||||
import Konwledge from './classOverview/knowledge.vue'
|
import TimeAnalyse from '@/views/classTask/container/classOverview/timeAnalyse.vue'
|
||||||
import TimeAnalyse from './classOverview/timeAnalyse.vue'
|
|
||||||
import {defineProps,watch} from 'vue'
|
import {defineProps,watch} from 'vue'
|
||||||
import overviewStore from "@/store/modules/overview";
|
import overviewStore from "@/store/modules/overview";
|
||||||
|
// import {getBindlist} from "@/api/education/knowledgePoint";
|
||||||
const useOverview = overviewStore()
|
const useOverview = overviewStore()
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
tableList: {
|
tableList: {
|
||||||
|
@ -57,185 +57,12 @@ const props = defineProps({
|
||||||
return []
|
return []
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
evalId:{
|
// evalId:{
|
||||||
type: Number,
|
// type: Number,
|
||||||
default: 0
|
// default: 0
|
||||||
},
|
|
||||||
activeData: { // 数据
|
|
||||||
type: Object,
|
|
||||||
// required: true, // 必传
|
|
||||||
default: () => ({
|
|
||||||
quizlist: [], // 当前习题列表
|
|
||||||
studentList: [], // 当前课程-所有学生
|
|
||||||
workFeedList: [] // 当前课程-所有学生反馈数据
|
|
||||||
})
|
|
||||||
},
|
|
||||||
})
|
|
||||||
let studentList = ref([]) // 学生数据
|
|
||||||
const stuHasAnswers = ref([]) // 已经答过题的学生不管对错
|
|
||||||
|
|
||||||
// 初始-数据处理
|
|
||||||
const initData = () => {
|
|
||||||
console.log('xxx', props)
|
|
||||||
// window.test = activeCourse
|
|
||||||
studentList.value = props.activeData.studentList || []
|
|
||||||
const activeWorkFeedList = props.activeData.workFeedList || []
|
|
||||||
const quizlist = props.activeData.quizlist || []
|
|
||||||
// 习题特殊处理
|
|
||||||
let data = quizlist.map(o => {
|
|
||||||
// 解析题选项
|
|
||||||
const workdesc = o.workdesc || ''
|
|
||||||
let accSum = 0 // 该题总人数
|
|
||||||
let activeIds = [] // 已做答学生
|
|
||||||
let rightIds = [] // 正确学生
|
|
||||||
let hasAnswers= [] // 答过题的学生
|
|
||||||
let timeAnalyse = [] // 平均时长和编号
|
|
||||||
const quizFeedList = activeWorkFeedList.filter(f => f.entpcourseworkid == o.id) // 做该题的列表
|
|
||||||
let children = []
|
|
||||||
const allStudents = [];
|
|
||||||
if (o.worktype == '单选题') { // '单选题','多选题'
|
|
||||||
const list = workdesc.includes('#&') ? workdesc.split('#&') : isJson(workdesc)?JSON.parse(workdesc):[];
|
|
||||||
children = list.map((v,i) => {
|
|
||||||
const code = toCode(i) // 转换 A-Z
|
|
||||||
const isOk = (isJson(workdesc)?JSON.parse(o.workanswer):o.workanswer||'').includes(i+'') // 是否(包含)正确答案
|
|
||||||
// 改选项的学生id
|
|
||||||
const studentIds = quizFeedList.filter(f => f.feedcontent==v&&f.finishtimelength!='0').map(f => f.studentid)||[];
|
|
||||||
accSum += studentIds.length;
|
|
||||||
if (isOk) {
|
|
||||||
activeIds.push(...studentIds)
|
|
||||||
}
|
|
||||||
hasAnswers.push(...studentIds)
|
|
||||||
return { def: v, code, isOk, studentIds }
|
|
||||||
})
|
|
||||||
}
|
|
||||||
else if (o.worktype == '多选题') {
|
|
||||||
// 多选题的正确率单独处理
|
|
||||||
rightIds = quizFeedList.filter(f => {
|
|
||||||
const workanswer = (isJson(o.workanswer)) ? JSON.parse(o.workanswer) : o.workanswer || [];
|
|
||||||
const res = isSame((f.feedcontent||'').split(','), workanswer);
|
|
||||||
return f.entpcourseworkid == o.id && f.finishtimelength!='0' && res;
|
|
||||||
});
|
|
||||||
|
|
||||||
const list = workdesc.includes('#&') ? workdesc.split('#&') : isJson(workdesc)?JSON.parse(workdesc):[];
|
|
||||||
children = list.map((v,i) => {
|
|
||||||
const isOne = o.worktype == '单选题'
|
|
||||||
const code = toCode(i) // 转换 A-Z
|
|
||||||
// const isOk = isOne ? i == o.workanswer : o.workanswer.includes(i) // 是否(包含)正确答案
|
|
||||||
const isOk = (isJson(workdesc)?JSON.parse(o.workanswer):o.workanswer||'').includes(i+'') // 是否(包含)正确答案
|
|
||||||
// 改选项的学生id
|
|
||||||
const studentIds = quizFeedList.filter(f => f.feedcontent.includes(i)&&f.finishtimelength!='0').map(f => f.studentid)||[];
|
|
||||||
accSum += studentIds.length;
|
|
||||||
if (studentIds.length>0) {
|
|
||||||
allStudents.push(...studentIds);
|
|
||||||
}
|
|
||||||
if(isOk) {
|
|
||||||
activeIds=[...new Set(activeIds.concat(studentIds))] // 多选去重
|
|
||||||
}
|
|
||||||
hasAnswers=[...new Set(hasAnswers.concat(studentIds))]
|
|
||||||
return { def: v, code, isOk, studentIds }
|
|
||||||
})
|
|
||||||
}
|
|
||||||
else if (o.worktype == '填空题') { // 填空题
|
|
||||||
const regex = /<!--BA-->(.*?)<!--EA-->/g // 定义正则表达式,匹配 <!--BA-->xxx<!--EA--> 格式的内容
|
|
||||||
children = (o.title||'').match(regex).map((v,i) => {
|
|
||||||
const def = `填空项 ${i+1}`
|
|
||||||
//const code = '( )'
|
|
||||||
const code = '(略)', txt=v
|
|
||||||
// 改选项的学生id
|
|
||||||
const studentIds = quizFeedList.filter(f => !!(f.feedcontent||'').replace(/#$/,'').split('#')[i] && f.finishtimelength!='0').map(f => f.studentid)||[]
|
|
||||||
activeIds=[...new Set(activeIds.concat(studentIds))] // 多选去重
|
|
||||||
hasAnswers=[...new Set(hasAnswers.concat(studentIds))]
|
|
||||||
accSum = activeIds.length
|
|
||||||
return { def, code, txt, isOk:true, studentIds }
|
|
||||||
})
|
|
||||||
} else if (o.worktype == '判断题') { // 判断题
|
|
||||||
const list = ['正确', '错误'];
|
|
||||||
children = list.map((v,i) => {
|
|
||||||
const workanswer = o.workanswer
|
|
||||||
.replace('×', '0')
|
|
||||||
.replace('√', '1')
|
|
||||||
.replace('错误', '0')
|
|
||||||
.replace('正确', '1')
|
|
||||||
.replace('正确。', '1')
|
|
||||||
.replace('F', '0')
|
|
||||||
.replace('T', '1')
|
|
||||||
.replace('错', '0')
|
|
||||||
.replace('对', '1');
|
|
||||||
const workanswerFormat = isJson(workanswer) ? JSON.parse(workanswer) : workanswer||''
|
|
||||||
const code = v=='正确' ? '1' : '0'
|
|
||||||
let isOk = (workanswerFormat).includes(code)
|
|
||||||
// warn: 看是否需要转为回答正常的显示, 当前为学生实际回答字面的正确和错误
|
|
||||||
// if (workanswerFormat == '0') {
|
|
||||||
// isOk = !isOk;
|
|
||||||
// }
|
// }
|
||||||
// 改选项的学生id
|
|
||||||
const studentIds = quizFeedList.filter(f => {
|
|
||||||
const feedcontent = f.feedcontent
|
|
||||||
.replace('×', '0')
|
|
||||||
.replace('√', '1')
|
|
||||||
.replace('错误', '0')
|
|
||||||
.replace('正确', '1')
|
|
||||||
.replace('正确。', '1')
|
|
||||||
.replace('F', '0')
|
|
||||||
.replace('T', '1')
|
|
||||||
.replace('错', '0')
|
|
||||||
.replace('对', '1');
|
|
||||||
if(feedcontent == code&&f.finishtimelength!='0'){
|
|
||||||
return f
|
|
||||||
}
|
|
||||||
}).map(f => f.studentid)||[];
|
|
||||||
accSum += studentIds.length;
|
|
||||||
if(isOk) activeIds.push(...studentIds)
|
|
||||||
hasAnswers.push(...studentIds)
|
|
||||||
return { def: v, code: v, isOk, studentIds }
|
|
||||||
})
|
})
|
||||||
} else { // 论述题
|
|
||||||
// code = '( )'
|
|
||||||
const code = '(略)', def = '解答内容'
|
|
||||||
const studentIds = quizFeedList.filter(f => !!(f.feedcontent||'').replace(/#$/,'')&&f.finishtimelength!='0').map(f => f.studentid)||[]
|
|
||||||
activeIds=[...new Set(activeIds.concat(studentIds))] // 多选去重
|
|
||||||
hasAnswers=[...new Set(hasAnswers.concat(studentIds))]
|
|
||||||
accSum = activeIds.length
|
|
||||||
children = [{ def, code, isOk:true, studentIds }]
|
|
||||||
}
|
|
||||||
|
|
||||||
const studentSum = studentList.value.length || 0 // 当前推送答题人数
|
|
||||||
let points = percent((activeIds.length / (studentSum||1)).toFixed(2)) // 计算得分率
|
|
||||||
let rightSum = activeIds.length; // 回答正确人数
|
|
||||||
|
|
||||||
// 多选题单独处理
|
|
||||||
if (o.worktype == '多选题') {
|
|
||||||
// 单独重新处理人数问题
|
|
||||||
const uniqueTmpStuents = [...new Set(allStudents)];
|
|
||||||
accSum = uniqueTmpStuents.length;
|
|
||||||
// 单独处理得分率
|
|
||||||
points = percent((rightIds.length / (studentSum||1)).toFixed(2)) // 计算得分率
|
|
||||||
// 回答正确人数
|
|
||||||
rightSum = rightIds.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
// def: 原始题数据 type 类型 active: 选中 points: 得分率, accSum 题解答人数
|
|
||||||
return { def: o, id: o.id, type: o.worktype, active: [], points, accSum, rightSum, children,hasAnswers }
|
|
||||||
})
|
|
||||||
console.log('获取数据: ', data)
|
|
||||||
stuHasAnswers.value = [...data[0].hasAnswers]
|
|
||||||
provide('hasAnswer', stuHasAnswers.value)
|
|
||||||
}
|
|
||||||
// 百分比现在 0-100
|
|
||||||
const percent = v => v > 1 ? 1 : v < 0 ? 0 : Math.round(v * 100)
|
|
||||||
// Unicode 转 字符 差值65
|
|
||||||
const toCode = (v, b) => b ? v.charCodeAt() - 65 : String.fromCharCode(v + 65)
|
|
||||||
// 判断是否为json字符串
|
|
||||||
const isJson = str => {if(typeof str == 'string'){
|
|
||||||
try {
|
|
||||||
const res = JSON.parse(str)
|
|
||||||
if(typeof res == 'object' && res) return true
|
|
||||||
} catch (error) {}}return false
|
|
||||||
}
|
|
||||||
|
|
||||||
watch(() => props.tableList,() => {
|
watch(() => props.tableList,() => {
|
||||||
useOverview.getTableList(props.tableList)
|
useOverview.getTableList(props.tableList)
|
||||||
},{deep:true})
|
},{deep:true})
|
||||||
// === 监听器 ===
|
|
||||||
watchEffect(() => { initData() })
|
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -1,40 +1,32 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="chart-container">
|
<div className="chart-container">
|
||||||
<div ref="chartRef" class="chart"></div>
|
<div ref="chartRef" className="chart"></div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, nextTick, watch, inject,watchEffect } from 'vue';
|
import {ref,nextTick,watch} from 'vue';
|
||||||
import * as echarts from 'echarts';
|
import * as echarts from 'echarts';
|
||||||
import overviewStore from '@/store/modules/overview';
|
import overviewStore from '@/store/modules/overview'
|
||||||
const useOverview = overviewStore();
|
const useOverview = overviewStore()
|
||||||
|
|
||||||
// 获取图表容器的引用
|
// 获取图表容器的引用
|
||||||
const chartRef = ref(null);
|
const chartRef = ref(null);
|
||||||
|
|
||||||
// 数据源
|
// 数据源
|
||||||
const dataList = ref([
|
const dataList = ref([
|
||||||
{ name: '完美', value: 0, rating: 1, max: 100, min: 100 },
|
{name: '完美', value: 0,rating:1,max:100,min:100,},
|
||||||
{ name: '优秀', value: 0, rating: 2, max: 99, min: 80 },
|
{name: '优秀', value: 0,rating:2,max:99,min:80,},
|
||||||
{ name: '良好', value: 0, rating: 3, max: 79, min: 70 },
|
{name: '良好', value: 0,rating:3,max:79,min:70,},
|
||||||
{ name: '及格', value: 0, rating: 4, max: 69, min: 60 },
|
{name: '及格', value: 0,rating:4,max:69,min:60,},
|
||||||
{ name: '不及格', value: 0, rating: 5, max: 59, min: 0 },
|
{name: '不及格', value: 0,rating:5,max:59,min:0,},
|
||||||
]);
|
]);
|
||||||
// 答过题的学生才进行统计
|
|
||||||
const hasAnswersData = ref([]);
|
|
||||||
const hasStudents = ref([])
|
|
||||||
|
|
||||||
// 根据数据生成不同的颜色
|
// 根据数据生成不同的颜色
|
||||||
function getColor(name) {
|
function getColor(index) {
|
||||||
const colorMap = {
|
// 这里可以根据值生成不同的颜色
|
||||||
'完美': '#d14a61',
|
const colors = ['#d14a61','#675bba', '#e89110','#008c8c','#5793f3'];
|
||||||
'优秀': '#675bba',
|
return colors[index];
|
||||||
'良好': '#e89110',
|
|
||||||
'及格': '#008c8c',
|
|
||||||
'不及格': '#5793f3'
|
|
||||||
};
|
|
||||||
return colorMap[name];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 初始化图表
|
// 初始化图表
|
||||||
|
@ -42,81 +34,75 @@ function initChart() {
|
||||||
const myChart = echarts.init(chartRef.value);
|
const myChart = echarts.init(chartRef.value);
|
||||||
const total = dataList.value.reduce((acc, cur) => acc + cur.value, 0); // 计算总数
|
const total = dataList.value.reduce((acc, cur) => acc + cur.value, 0); // 计算总数
|
||||||
|
|
||||||
// 过滤掉值为0的数据项
|
|
||||||
const filteredData = dataList.value.filter(item => item.value > 0);
|
|
||||||
|
|
||||||
const options = {
|
const options = {
|
||||||
tooltip: {
|
tooltip: {
|
||||||
trigger: 'item',
|
trigger: 'axis',
|
||||||
formatter: params => {
|
axisPointer: {
|
||||||
const value = params.value;
|
type: 'shadow'
|
||||||
const percentage = value ? ((value / total) * 100).toFixed(2) : 0; // 计算百分比并保留两位小数
|
|
||||||
return `${params.name}: ${value}人 (${percentage}%)`; // 显示为百分比形式
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
grid: {
|
||||||
|
left: '3%',
|
||||||
|
right: '4%',
|
||||||
|
bottom: '3%',
|
||||||
|
containLabel: true
|
||||||
|
},
|
||||||
|
xAxis: {
|
||||||
|
type: 'category',
|
||||||
|
data: dataList.value.map(item => item.name),
|
||||||
|
axisTick: {
|
||||||
|
alignWithLabel: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
yAxis: {
|
||||||
|
type: 'value'
|
||||||
|
},
|
||||||
series: [{
|
series: [{
|
||||||
name: '数据',
|
name: '数据',
|
||||||
type: 'pie',
|
type: 'bar',
|
||||||
radius: '50%', // 设置饼图的半径为实心
|
barWidth: '30%',
|
||||||
data: filteredData.map(item => ({
|
data: dataList.value.map(item => item.value),
|
||||||
name: item.name,
|
|
||||||
value: item.value,
|
|
||||||
itemStyle: {
|
itemStyle: {
|
||||||
color: getColor(item.name)
|
color: function (params) {
|
||||||
|
// 确保这里返回正确的颜色
|
||||||
|
return getColor(params.dataIndex);
|
||||||
}
|
}
|
||||||
})),
|
},
|
||||||
|
// 显示柱子顶部的值
|
||||||
label: {
|
label: {
|
||||||
show: true,
|
show: true,
|
||||||
position: 'inside', // 将标签显示在饼图内部
|
position: 'top',
|
||||||
formatter: params => {
|
formatter: params => {
|
||||||
const value = params.value;
|
const value = dataList.value[params.dataIndex].value;
|
||||||
const percentage = value ? ((value / total) * 100).toFixed(2) : 0; // 计算百分比并保留两位小数
|
const percentage = value ? ((value / total) * 100).toFixed() : 0; // 计算百分比并保留两位小数
|
||||||
return `${params.name}\n${value}人 (${percentage}%)`; // 显示为百分比形式,换行显示
|
return `${value}人 ${percentage}%`; // 显示为百分比形式
|
||||||
},
|
},
|
||||||
color: '#333',
|
color: '#333',
|
||||||
fontSize: 12
|
fontSize: 12
|
||||||
},
|
|
||||||
labelLine: {
|
|
||||||
show: false // 不显示标签线
|
|
||||||
}
|
}
|
||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
|
|
||||||
myChart.setOption(options);
|
myChart.setOption(options);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 获取表的数据
|
// 获取表的数据
|
||||||
const showEcharts =() => {
|
const showEcharts =() => {
|
||||||
hasStudents.value.forEach((item, index) => {
|
useOverview.tableList.forEach((item,index) => {
|
||||||
if (item.rating === 0) {
|
if(item.rating === 0) return // 没批改不计数
|
||||||
dataList.value.forEach((item1,index1) => {
|
dataList.value.forEach((item1,index1) => {
|
||||||
if(item1.min <= Number(item.scoingRate) && Number(item.scoingRate) <= item1.max ){
|
if(item1.min <= Number(item.scoingRate) && Number(item.scoingRate) <= item1.max ){
|
||||||
item1.value++;
|
item1.value ++
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
} else {
|
})
|
||||||
dataList.value.forEach((item1, index1) => {
|
|
||||||
if (item1.rating === item.rating) {
|
|
||||||
item1.value++;
|
|
||||||
}
|
}
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
//执行
|
|
||||||
watch(() => useOverview.tableList,() => {
|
watch(() => useOverview.tableList,() => {
|
||||||
hasStudents.value = useOverview.tableList.filter(item => hasAnswersData.value.includes(item.studentid)).map(item => item);
|
showEcharts()
|
||||||
showEcharts();
|
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
initChart();
|
initChart();
|
||||||
});
|
})
|
||||||
},{deep: true})
|
})
|
||||||
//监听
|
|
||||||
watchEffect(() => {
|
|
||||||
const stus = inject('hasAnswer');
|
|
||||||
if (!stus) return;
|
|
||||||
hasAnswersData.value = [...stus]
|
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|
|
@ -16,14 +16,11 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import {nextTick, ref, watch,inject,watchEffect} from 'vue'
|
import {nextTick, ref, watch} from 'vue'
|
||||||
import overviewStore from '@/store/modules/overview'
|
import overviewStore from '@/store/modules/overview'
|
||||||
|
|
||||||
const useOverview = overviewStore()
|
const useOverview = overviewStore()
|
||||||
const tabPosition = ref('left')
|
const tabPosition = ref('left')
|
||||||
//答过题的学生才进行统计
|
|
||||||
const hasAnswersData = ref([])
|
|
||||||
const hasStudents = ref([])
|
|
||||||
const leftList = ref([
|
const leftList = ref([
|
||||||
{
|
{
|
||||||
label:'完美',
|
label:'完美',
|
||||||
|
@ -68,26 +65,16 @@ const handelChange = (item) => {
|
||||||
}
|
}
|
||||||
//取区域的学生
|
//取区域的学生
|
||||||
const showStudents = (index) => {
|
const showStudents = (index) => {
|
||||||
leftList.value[index].stuList = hasStudents.value.filter(item => {
|
leftList.value[index].stuList = useOverview.tableList.filter(item => {
|
||||||
if(item.rating === 0){
|
if(item.rating > 0){
|
||||||
if(leftList.value[index].min <= Number(item.scoingRate || 0) && Number(item.scoingRate || 0) <= leftList.value[index].max ){
|
if(leftList.value[index].min <= Number(item.scoingRate || 0) && Number(item.scoingRate || 0) <= leftList.value[index].max ){
|
||||||
return item
|
return item
|
||||||
}
|
}
|
||||||
}else{
|
|
||||||
if(item.rating == leftList.value[index].rating){
|
|
||||||
return item
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
watch(() => useOverview.tableList,() => {
|
watch(() => useOverview.tableList,() => {
|
||||||
hasStudents.value = useOverview.tableList.filter(item => hasAnswersData.value.includes(item.studentid)).map(item => item);
|
|
||||||
showStudents(0)
|
showStudents(0)
|
||||||
},{deep: true})
|
|
||||||
watchEffect(() => {
|
|
||||||
const stus = inject('hasAnswer')
|
|
||||||
if(!stus) return
|
|
||||||
hasAnswersData.value = [...stus]
|
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
|
@ -318,7 +318,6 @@
|
||||||
<div
|
<div
|
||||||
v-for="(score, index) in teacherRatingList"
|
v-for="(score, index) in teacherRatingList"
|
||||||
:key="index"
|
:key="index"
|
||||||
style="white-space: nowrap;"
|
|
||||||
:class="[
|
:class="[
|
||||||
'score-circle',
|
'score-circle',
|
||||||
{ active: classWorkFormScore.rating == score.ratingKey }
|
{ active: classWorkFormScore.rating == score.ratingKey }
|
||||||
|
@ -502,11 +501,11 @@ const classWorkFormScore = reactive({
|
||||||
teacherremark: '' //评分说明
|
teacherremark: '' //评分说明
|
||||||
})
|
})
|
||||||
const teacherRatingList = ref([
|
const teacherRatingList = ref([
|
||||||
{ ratingKey: '1', ratingValue: '完美' },
|
{ ratingKey: '1', ratingValue: '优' },
|
||||||
{ ratingKey: '2', ratingValue: '优秀' },
|
{ ratingKey: '2', ratingValue: '优-' },
|
||||||
{ ratingKey: '3', ratingValue: '良好' },
|
{ ratingKey: '3', ratingValue: '良' },
|
||||||
{ ratingKey: '4', ratingValue: '及格' },
|
{ ratingKey: '4', ratingValue: '良-' },
|
||||||
{ ratingKey: '5', ratingValue: '不及格' }
|
{ ratingKey: '5', ratingValue: '差' }
|
||||||
])
|
])
|
||||||
// 确定的线上图片数据
|
// 确定的线上图片数据
|
||||||
//#region 文件内容相关
|
//#region 文件内容相关
|
||||||
|
|
|
@ -94,19 +94,19 @@
|
||||||
</template>
|
</template>
|
||||||
<!-- 1-优 2-优减 3-良 4-良减 5-差 -->
|
<!-- 1-优 2-优减 3-良 4-良减 5-差 -->
|
||||||
<template v-if="scope.row.teacherRating == 1"
|
<template v-if="scope.row.teacherRating == 1"
|
||||||
><el-tag type="danger">完美</el-tag></template
|
><el-tag type="danger">优</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 2"
|
<template v-if="scope.row.teacherRating == 2"
|
||||||
><el-tag type="danger">优秀</el-tag></template
|
><el-tag type="danger">优-</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 3"
|
<template v-if="scope.row.teacherRating == 3"
|
||||||
><el-tag type="warning">良好</el-tag></template
|
><el-tag type="warning">良</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 4"
|
<template v-if="scope.row.teacherRating == 4"
|
||||||
><el-tag type="info">及格</el-tag></template
|
><el-tag type="info">良-</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 5"
|
<template v-if="scope.row.teacherRating == 5"
|
||||||
><el-tag type="info">不及格</el-tag></template
|
><el-tag type="info">差</el-tag></template
|
||||||
>
|
>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
|
@ -656,6 +656,8 @@ defineExpose({
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<style src="@/assets/styles/JYStyle.css"></style>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
/*:deep(.reserv-date-pick) {
|
/*:deep(.reserv-date-pick) {
|
||||||
width: 140px;
|
width: 140px;
|
||||||
|
@ -763,4 +765,3 @@ defineExpose({
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
<style src="@/assets/styles/JYStyle.css"></style>
|
|
||||||
|
|
|
@ -84,19 +84,19 @@
|
||||||
</template>
|
</template>
|
||||||
<!-- 1-优 2-优减 3-良 4-良减 5-差 -->
|
<!-- 1-优 2-优减 3-良 4-良减 5-差 -->
|
||||||
<template v-if="scope.row.teacherRating == 1"
|
<template v-if="scope.row.teacherRating == 1"
|
||||||
><el-tag type="danger">完美</el-tag></template
|
><el-tag type="danger">优</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 2"
|
<template v-if="scope.row.teacherRating == 2"
|
||||||
><el-tag type="danger">优秀</el-tag></template
|
><el-tag type="danger">优-</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 3"
|
<template v-if="scope.row.teacherRating == 3"
|
||||||
><el-tag type="warning">良好</el-tag></template
|
><el-tag type="warning">良</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 4"
|
<template v-if="scope.row.teacherRating == 4"
|
||||||
><el-tag type="info">及格</el-tag></template
|
><el-tag type="info">良-</el-tag></template
|
||||||
>
|
>
|
||||||
<template v-if="scope.row.teacherRating == 5"
|
<template v-if="scope.row.teacherRating == 5"
|
||||||
><el-tag type="info">不及格</el-tag></template
|
><el-tag type="info">差</el-tag></template
|
||||||
>
|
>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
|
@ -139,7 +139,7 @@
|
||||||
<!-- 训练报告-->
|
<!-- 训练报告-->
|
||||||
<div v-else-if="classWorkAnalysis.view == 'report'" style="width: 100%;overflow-y: scroll">
|
<div v-else-if="classWorkAnalysis.view == 'report'" style="width: 100%;overflow-y: scroll">
|
||||||
<!-- <ClassOverview :table-list="overviewData" :eval-id="courseObj.evalid"></ClassOverview> -->
|
<!-- <ClassOverview :table-list="overviewData" :eval-id="courseObj.evalid"></ClassOverview> -->
|
||||||
<ClassOverview :active-data="classWorkActiveData" :table-list="overviewData" style="width: 100%;"></ClassOverview>
|
<ClassOverview :table-list="overviewData" style="width: 100%;"></ClassOverview>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -658,6 +658,8 @@ onUnmounted(() => {
|
||||||
// })
|
// })
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<style src="@/assets/styles/JYStyle.css"></style>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
.teachClassTask{
|
.teachClassTask{
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
@ -703,6 +705,94 @@ onUnmounted(() => {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// .clwk_dialog {
|
||||||
|
|
||||||
|
// }
|
||||||
|
|
||||||
|
// .clwk_dialog {
|
||||||
|
// display: flex;
|
||||||
|
// justify-content: center;
|
||||||
|
// overflow: hidden;
|
||||||
|
// }
|
||||||
|
// .clwk_dialog .el-dialog {
|
||||||
|
// margin: 0 auto !important;
|
||||||
|
// height: 85%!important;
|
||||||
|
// overflow: hidden;
|
||||||
|
// }
|
||||||
|
// .clwk_dialog .el-dialog__header {
|
||||||
|
// /* position: absolute;
|
||||||
|
// top: 0;
|
||||||
|
// left: 0; */
|
||||||
|
// width: 100%!important;
|
||||||
|
// }
|
||||||
|
// .clwk_dialog .el-dialog__body {
|
||||||
|
// position: absolute;
|
||||||
|
// left: 0;
|
||||||
|
// top: 15px;
|
||||||
|
// bottom: 1px;
|
||||||
|
// right:0;
|
||||||
|
// padding:5px;
|
||||||
|
// z-index:1;
|
||||||
|
// display: flex;
|
||||||
|
// flex-direction: column;
|
||||||
|
// overflow: hidden;
|
||||||
|
// /* overflow:hidden;
|
||||||
|
// overflow-y: auto; */
|
||||||
|
// }
|
||||||
|
// .clwk_dialog .el-dialog__footer{
|
||||||
|
// position: absolute;
|
||||||
|
// bottom: 10px;
|
||||||
|
// right: 10px;
|
||||||
|
// }
|
||||||
|
// .clwk_dialog .classwork-score{
|
||||||
|
// overflow-y: auto;
|
||||||
|
// }
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style src="@/assets/styles/JYStyle.css"></style>
|
<style scoped>
|
||||||
|
/* .clwk_dialog {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.clwk_dialog .el-dialog {
|
||||||
|
margin: 0 auto !important;
|
||||||
|
height: 85%!important;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.clwk_dialog .el-dialog__header {
|
||||||
|
/* position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%!important;
|
||||||
|
}
|
||||||
|
.clwk_dialog .el-dialog__body {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
top: 15px;
|
||||||
|
bottom: 1px;
|
||||||
|
right:0;
|
||||||
|
padding:5px;
|
||||||
|
z-index:1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
overflow: hidden;
|
||||||
|
/* overflow:hidden;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
.clwk_dialog .el-dialog__footer{
|
||||||
|
position: absolute;
|
||||||
|
bottom: 10px;
|
||||||
|
right: 10px;
|
||||||
|
}
|
||||||
|
.clwk_dialog .classwork-score{
|
||||||
|
overflow-y: auto;
|
||||||
|
} */
|
||||||
|
</style>
|
||||||
|
|
|
@ -98,7 +98,7 @@ import FileImage from '@/components/file-image/index.vue'
|
||||||
import { asyncLocalFile } from '@/utils/talkFile'
|
import { asyncLocalFile } from '@/utils/talkFile'
|
||||||
import { toTimeText } from '@/utils/date'
|
import { toTimeText } from '@/utils/date'
|
||||||
import { ElMessage, ElMessageBox } from 'element-plus'
|
import { ElMessage, ElMessageBox } from 'element-plus'
|
||||||
import { deleteSmarttalk, updateSmarttalk, getPrepareById, getSmarttalkPage } from '@/api/file'
|
import { deleteSmarttalk, updateSmarttalk, getPrepareById } from '@/api/file'
|
||||||
import useUserStore from '@/store/modules/user'
|
import useUserStore from '@/store/modules/user'
|
||||||
import outLink from '@/utils/linkConfig'
|
import outLink from '@/utils/linkConfig'
|
||||||
import { sessionStore } from '@/utils/store'
|
import { sessionStore } from '@/utils/store'
|
||||||
|
@ -208,7 +208,6 @@ export default {
|
||||||
}
|
}
|
||||||
if (action === 'cancel'){
|
if (action === 'cancel'){
|
||||||
// 继续上课
|
// 继续上课
|
||||||
console.log(obj.bookImg)
|
|
||||||
if (obj.bookImg) {
|
if (obj.bookImg) {
|
||||||
//PPT
|
//PPT
|
||||||
listEntpcourse({
|
listEntpcourse({
|
||||||
|
@ -242,20 +241,9 @@ export default {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}else {
|
}else {
|
||||||
console.log(obj)
|
|
||||||
getSmarttalkPage({fileId: obj.entpcoursefileid}).then(res2=>{
|
|
||||||
console.log(res2)
|
|
||||||
//APT
|
//APT
|
||||||
if (res2&&res2.rows.length>0) {
|
this.$emit('on-start-class', item, obj)
|
||||||
this.$emit('on-start-class', res2.rows[0], obj)
|
|
||||||
}else {
|
|
||||||
ElMessage({
|
|
||||||
message: '该文件未关联PPT,无法打开!',
|
|
||||||
type: 'warning'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
done()
|
done()
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (action === 'close') {
|
if (action === 'close') {
|
||||||
|
|
|
@ -291,7 +291,6 @@ export default {
|
||||||
methods: {
|
methods: {
|
||||||
addAiPPT(item) {
|
addAiPPT(item) {
|
||||||
this.currentFileList.unshift(item.resData)
|
this.currentFileList.unshift(item.resData)
|
||||||
KjListItem.methods.openFileWin(item.resData);
|
|
||||||
this.pptDialog = false
|
this.pptDialog = false
|
||||||
},
|
},
|
||||||
// test() {
|
// test() {
|
||||||
|
|
Loading…
Reference in New Issue