Browse Source

Merge branch 'beta' of https://git.cocorobo.cn/CocoRoboLabs/pbl-teacher-table into beta

SanHQin 7 months ago
parent
commit
a354b54dfe

+ 1 - 1
dist/index.html

@@ -32,7 +32,7 @@
       width: 100%;
       background: #e6eaf0;
       font-family: '黑体';
-    }</style><link href=./static/css/app.3372c039b9d257c01095b00c91d19aa4.css rel=stylesheet></head><body><div id=app></div><script type=text/javascript src=./static/js/manifest.571c38d63f24b1ae9e16.js></script><script type=text/javascript src=./static/js/vendor.35dfb1d95a63059f336f.js></script><script type=text/javascript src=./static/js/app.86e7297a4fdb16b9a6ba.js></script></body></html><script>function stopSafari() {
+    }</style><link href=./static/css/app.950b02e0ab4f6fffc20037eab06db226.css rel=stylesheet></head><body><div id=app></div><script type=text/javascript src=./static/js/manifest.571c38d63f24b1ae9e16.js></script><script type=text/javascript src=./static/js/vendor.35dfb1d95a63059f336f.js></script><script type=text/javascript src=./static/js/app.4758f15f922a73b6724b.js></script></body></html><script>function stopSafari() {
     //阻止safari浏览器双击放大功能
     let lastTouchEnd = 0  //更新手指弹起的时间
     document.documentElement.addEventListener("touchstart", function (event) {

File diff suppressed because it is too large
+ 0 - 0
dist/static/css/app.3372c039b9d257c01095b00c91d19aa4.css


File diff suppressed because it is too large
+ 0 - 0
dist/static/css/app.3372c039b9d257c01095b00c91d19aa4.css.map


File diff suppressed because it is too large
+ 0 - 0
dist/static/css/app.950b02e0ab4f6fffc20037eab06db226.css


File diff suppressed because it is too large
+ 0 - 0
dist/static/css/app.950b02e0ab4f6fffc20037eab06db226.css.map


BIN
dist/static/img/hello.68de2a9.gif


File diff suppressed because it is too large
+ 0 - 0
dist/static/js/app.4758f15f922a73b6724b.js


File diff suppressed because it is too large
+ 0 - 0
dist/static/js/app.4758f15f922a73b6724b.js.map


File diff suppressed because it is too large
+ 0 - 0
dist/static/js/manifest.571c38d63f24b1ae9e16.js.map


+ 208 - 83
src/components/pages/aiAddCourse/addCourse.vue

@@ -403,7 +403,7 @@
                           )
                           " />
                       </button>
-                      <button class="c_pub_button_add" @click="clickGenTT">
+                      <button class="c_pub_button_add" @click="clickGenTT" v-if="templateid != '4480d65a-1e48-11ef-bee5-005056b86db5' && templateid != 'cf5722a4-401b-11ef-b873-005056b86dc3'">
                         一键智能总结生成
                       </button>
                     </div>
@@ -560,7 +560,7 @@
                 </div>
               </div>
             </div>
-            <div class="whiteBg" :style="{ minHeight: targetTextLoading ? '250px' : 'auto'}" style="background: #fff; margin: 0 0 10px;padding: 0 0 15px;" v-loading="targetTextLoading" element-loading-text="小可正在努力生成中,请稍等..." v-if="(!yiKeTemplateArray.includes(templateid)) && isuseT">
+            <div class="whiteBg" :style="{ minHeight: targetTextLoading ? '250px' : 'auto'}" style="background: #fff; margin: 0 0 10px;padding: 0 0 15px;" v-loading="targetTextLoading" element-loading-text="小可正在努力生成中,请稍等..." v-if="(!yiKeTemplateArray.includes(templateid)) && isuseT && templateid != '4480d65a-1e48-11ef-bee5-005056b86db5' && templateid != 'cf5722a4-401b-11ef-b873-005056b86dc3'">
               <div class="c_pub_button_confirm stopBtn" v-if="targetTextLoading && isTargetCancelToken1" @click="cancelAjax('target1')">停止</div>
               <div class="whiteBg" style="border-radius: 0; margin-top: 15px">
                 <div class="c_info_title">
@@ -897,7 +897,7 @@
                       @contextmenu.prevent="openAiDialog(1, 'aiDetail',1)"
                       @click="openAiDialog(2, 'aiDetail',1)">AI优化</button>
                   </el-tooltip>
-                  <button class="c_pub_button_confirm" @click="clickGenTT2" style="margin: 0 0 0 5px;">生成概况和目标</button>
+                  <button class="c_pub_button_confirm" @click="clickGenTT2" style="margin: 0 0 0 5px;" v-if="templateid != '4480d65a-1e48-11ef-bee5-005056b86db5' && templateid != 'cf5722a4-401b-11ef-b873-005056b86dc3'">生成概况和目标</button>
                 </div>
                 <div style="width: 100%; padding: 0px 20px; box-sizing: border-box">
                   <div style="width: calc(100%);" class='op_task_box'>
@@ -920,7 +920,7 @@
                 </div>
               </div>
             </div>
-            <div class="whiteBg" :style="{ minHeight: targetTextLoading2 ? '250px' : 'auto'}" style="background: #fff; margin: 0 0 10px;padding: 0 0 15px;" v-loading="targetTextLoading2" element-loading-text="小可正在努力生成中,请稍等..." v-if="(!isuseT && (!yiKeTemplateArray.includes(templateid))) || (isuseT && (!yiKeTemplateArray.includes(templateid)) && courseTextBool)">
+            <div class="whiteBg" :style="{ minHeight: targetTextLoading2 ? '250px' : 'auto'}" style="background: #fff; margin: 0 0 10px;padding: 0 0 15px;" v-loading="targetTextLoading2" element-loading-text="小可正在努力生成中,请稍等..." v-if="((!isuseT && (!yiKeTemplateArray.includes(templateid))) || (isuseT && (!yiKeTemplateArray.includes(templateid)) && courseTextBool)) && templateid != '4480d65a-1e48-11ef-bee5-005056b86db5' && templateid != 'cf5722a4-401b-11ef-b873-005056b86dc3'">
                 <div class="c_pub_button_confirm stopBtn" v-if="targetTextLoading2 && isTargetCancelToken2" @click="cancelAjax('target2')">停止</div>
                 <div class="whiteBg" style="border-radius: 0; margin-top: 15px">
                 <div class="c_info_title" style="flex-wrap: wrap; position:relative;margin:0 20px;">
@@ -1601,9 +1601,12 @@
               <button class="c_pub_button_return pub_btn_last_img" v-if="steps > 1 && steps != 5" @click="lastSteps">
                 {{ steps == 4 ? "返回课程" : "上一步" }}
               </button>
-              <button class="c_pub_button_confirm" v-if="isuseT && !isOutline && !courseTextBool && (!yiKeTemplateArray.includes(templateid))" @click="clickGenTT2">
+              <button class="c_pub_button_confirm" v-if="isuseT && !isOutline && !courseTextBool && (!yiKeTemplateArray.includes(templateid)) && templateid != '4480d65a-1e48-11ef-bee5-005056b86db5' && templateid != 'cf5722a4-401b-11ef-b873-005056b86dc3'" @click="clickGenTT2">
                 {{isuseT ? "重新生成概况和目标" : "生成概况和目标"}}
               </button>
+              <button class="c_pub_button_confirm" v-else-if="isuseT && !isOutline && !courseTextBool && (!yiKeTemplateArray.includes(templateid))" @click="clickGenTT2">
+                {{isuseT ? "重新生成概况" : "生成概况"}}
+              </button>
               <!-- ((isuseT || courseTextBool) && (!yiKeTemplateArray.includes(templateid))) &&  -->
               <el-tooltip effect="light" content="右键单击可配置提示词" placement="top"  v-if="steps < 4">
                 <button class="c_pub_button_confirm" @click="nextSteps(2)"
@@ -1623,12 +1626,12 @@
                   : "下一步" }}
                 </button>
               </el-tooltip>
-              <el-tooltip effect="light" content="右键单击可配置提示词" placement="top"  v-if="isOutline2 && cidType == 0">
+              <!-- <el-tooltip effect="light" content="右键单击可配置提示词" placement="top"  v-if="isOutline2 && cidType == 0">
                 <button class="c_pub_button_confirm" @click="nextSteps(2, 2)" @contextmenu.prevent="nextSteps(1, 2)" :class="{
                       pub_btn_next_img: steps != 3,
                       pub_btn_finish_img: steps == 3,
                     }">下一步2</button>
-              </el-tooltip>
+              </el-tooltip> -->
             </div>
           </div>
           <div class="rightBox" v-if="steps == 2" :style="{
@@ -2377,15 +2380,20 @@
                         <div class="taskTitle">
                           <div style="display: flex;width: 100%;justify-content: space-between;">
                             <span>任务{{ itemTaskIndex + 1 }}</span>
-                            <el-tooltip effect="light" content="右键单击可配置提示词" placement="bottom">
+                            <!-- <el-tooltip effect="light" content="右键单击可配置提示词" placement="bottom">
                               <button class="c_pub_button_confirm" style="margin: 0 0 0 auto;"
                                 @contextmenu.prevent="openAiDialog2(1, 'aiTask2',itemTaskIndex)"
                                 @click="openAiDialog2(2, 'aiTask2',itemTaskIndex)">重新生成任务</button>
-                            </el-tooltip>
+                            </el-tooltip> -->
                             <el-tooltip effect="light" content="右键单击可配置提示词" placement="bottom">
                               <button class="c_pub_button_confirm" style="margin: 0 0 0 20px;"
                                 @contextmenu.prevent="openAiDialog2(1, 'aiTask3',itemTaskIndex)"
-                                @click="openAiDialog2(2, 'aiTask3',itemTaskIndex)">重新生成任务2</button>
+                                @click="openAiDialog2(2, 'aiTask3',itemTaskIndex)">重新生成任务</button>
+                            </el-tooltip>
+                            <el-tooltip effect="light" content="右键单击可配置提示词" placement="bottom">
+                              <button class="c_pub_button_confirm" style="margin: 0 0 0 20px;"
+                                @contextmenu.prevent="openAiDialog2(1, 'aiTask3','all')"
+                                @click="openAiDialog2(2, 'aiTask3','all')">重新生成任务</button>
                             </el-tooltip>
                           </div>
                         </div>
@@ -5043,9 +5051,9 @@
                               <div v-if="itemTask.eList && itemTask.eList.length" class="elist_input">
                                 <div v-for="(eItem, eIndex) in itemTask.eList" :key="eIndex" class="elist_input_box">
                                   <div class="elist_inptu_text">
-                                    <!-- <span>目标:</span> -->
-                                    <span>核心素养:</span>
-                                    <input type="input" v-model="itemTask.eList[eIndex].target" placeholder="填写核心素养"  style="width: calc(100% - 280px);"/>
+                                    <span>目标:</span>
+                                    <!-- <span>核心素养:</span> -->
+                                    <input type="input" v-model="itemTask.eList[eIndex].target" placeholder="填写目标"  style="width: calc(100% - 280px);"/>
                                     <el-switch
                                       class="rateSwitch"
                                       style="margin-left: auto;"
@@ -5057,11 +5065,11 @@
                                       inactive-value="1">
                                     </el-switch>
                                   </div>
-                                  <div class="elist_inptu_text">
+                                  <div class="elist_inptu_text" v-show="false">
                                     <!-- <span>评价维度:</span> -->
-                                    <span>目标:</span>
+                                    <span>能力:</span>
                                     <!-- <el-tooltip effect="dark" :content="itemTask.eList[eIndex].value || '填写评价维度'" placement="top" popper-class="text_tooltip2"> -->
-                                    <input type="input" v-model="itemTask.eList[eIndex].value" placeholder="填写目标" />
+                                    <input type="input" v-model="itemTask.eList[eIndex].value" placeholder="填写能力" />
                                     <!-- </el-tooltip> -->
                                     <!-- <span>评星等级:</span>
                                     <el-rate v-model="itemTask.eList[eIndex].score" @change="setEListStar()"
@@ -5079,6 +5087,13 @@
                                     <span>评价标准:</span>
                                     <!-- <el-tooltip effect="dark" :content="itemTask.eList[eIndex].detail || '填写维度描述'" placement="top" popper-class="text_tooltip2"> -->
                                     <input type="input" v-model="itemTask.eList[eIndex].detail" placeholder="填写评价标准" />
+                                    <div class="remove" @click="
+                                      deletEList(
+                                        unitIndex,
+                                        itemTaskIndex,
+                                        eIndex
+                                      )
+                                      "></div>
                                     <!-- </el-tooltip> -->
                                   </div>
                                   <div class="elist_inptu_text" v-if="evalua">
@@ -5094,7 +5109,7 @@
                                       :props="{ checkStrictly: true }" :show-all-levels="false" clearable></el-cascader>
                                   </div>
                                   <div class="elist_inptu_text" style="align-items: flex-start;" v-loading="ruleLoading[itemTaskIndex] && ruleLoading[itemTaskIndex][eIndex]" element-loading-text="小可正在努力生成中,请稍等...">
-                                    <span>细则:</span>
+                                    <span>评价细则:</span>
                                     <div style="width: calc(100%);">
                                       <div @click="openRule(itemTaskIndex, eIndex)" class="ruleBtn" v-if="itemTask.eList[eIndex].rule">{{itemTask.eList[eIndex].isrule ? '收起细则' : '展开细则'}}</div>
                                       <el-tooltip effect="light" content="右键单击可配置提示词" placement="bottom" v-else>
@@ -8235,11 +8250,11 @@ export default {
             this.loading = true
             functionA()
           } else {
-            if(ttype == 2){
-              this.openAiDialog(clickType, "aiTask3", 'all', functionA)
-            }else {
-              this.openAiDialog(clickType, "aiTask2", 'all', functionA)
-            }
+            // if(ttype == 2){
+            this.openAiDialog(clickType, "aiTask3", 'all', functionA)
+            // }else {
+            //   this.openAiDialog(clickType, "aiTask2", 'all', functionA)
+            // }
           }
           
         } else {
@@ -8799,23 +8814,23 @@ export default {
       // this.dialogVisibleAiCreateVideo = false;
     },
     async createTeacherAn(index){
-      this.lineCount = index;
+      // this.lineCount = index;
       this.loading = true
       let md = new MarkdownIt();
       if(this.courseState == 4){
-        if(index == 0 || index){
-          let _task = this.unitJson[0].chapterInfo[0].taskJson[index]
+        // if(index == 0 || index){
+        //   let _task = this.unitJson[0].chapterInfo[0].taskJson[index]
+        //   let name = _task.task+'-教案'
+        //   let _html =  _task.taskDetail3 ? md.render(_task.taskDetail3) : ''
+        //   await this.generate(name, _html, 0)
+        // }else {
+        for(var i = 0; i < this.unitJson[0].chapterInfo[0].taskJson.length; i++){
+          let _task = this.unitJson[0].chapterInfo[0].taskJson[i]
           let name = _task.task+'-教案'
           let _html =  _task.taskDetail3 ? md.render(_task.taskDetail3) : ''
           await this.generate(name, _html, 0)
-        }else {
-          for(var i = 0; i < this.unitJson[0].chapterInfo[0].taskJson.length; i++){
-            let _task = this.unitJson[0].chapterInfo[0].taskJson[i]
-            let name = _task.task+'-教案'
-            let _html =  _task.taskDetail3 ? md.render(_task.taskDetail3) : ''
-            await this.generate(name, _html, 0)
-          }
         }
+        // }
       }else if(this.courseState == 5){
         if(index == 0 || index){
           let _task = this.unitJson[0].chapterInfo[0].taskJson[index]
@@ -8921,11 +8936,11 @@ export default {
                       _this.$message.error("上传失败");
                   } else {
                     let _index = 0
-                    if(_this.lineCount == 0 || _this.lineCount){
-                      _index = _this.lineCount
-                    }else {
+                    // if(_this.lineCount == 0 || _this.lineCount){
+                    //   _index = _this.lineCount
+                    // }else {
                       _index = index
-                    }
+                    // }
                     _this.unitJson[0].chapterInfo[0].taskJson[_index].chapterData.unshift({
                         name: file.name,
                         url: data.Location,
@@ -15837,10 +15852,13 @@ export default {
         return;
       }
       this.openAiDialog(2, 'aiDetail',1)
-      setTimeout(() => {
-        // this.openAiDialog(2, 'aitargetTextDetail2',1)
-        this.openAiDialog(2, 'aitargetTextDetail3',1)
-      }, 500);
+      if(this.templateid != '4480d65a-1e48-11ef-bee5-005056b86db5' && this.templateid != 'cf5722a4-401b-11ef-b873-005056b86dc3'){
+        setTimeout(() => {
+          // this.openAiDialog(2, 'aitargetTextDetail2',1)
+          this.openAiDialog(2, 'aitargetTextDetail3',1)
+        }, 500);
+      }
+
       this.courseTextBool = true
     },
     openAiDialog2(clickType, type, callback, index, tindex){
@@ -16231,15 +16249,11 @@ export default {
 // ${this.getListClassC(this.checkboxList2) ?  '班级:' + this.getListClassC(this.checkboxList2) : "无"}
 // ${this.infoData.length ? '使用文件检索的方式完整的去分析文件内容,并请完全按照要求输出。' : ''}
 // let txt = (!this.yiKeTemplateArray.includes(this.templateid)) ? '任务设计、评价标准和教学目标(markdown格式)' : '任务设计和评价标准'
-let txt = (!this.yiKeTemplateArray.includes(this.templateid)) ? `子任务名称:名称应该总结了该任务是在做什么,
-子任务设计:任务设计。
-子教学目标:教学目标。
-学生过程性成果:学生过程性成果。
-评价标准:评价标准。` : `子任务名称:名称应该总结了该任务是在做什么,
-子任务设计:任务设计。
-子教学目标:教学目标。
-学生过程性成果:学生过程性成果。
-评价标准:评价标准。`
+let txt = (!this.yiKeTemplateArray.includes(this.templateid)) ? `  **子教学目标**:子教学目标是课程目标在该任务的分解目标,撰写格式同子教学目标  
+  **过程性成果**:过程性成果是最终成果的一部分或者是中间成果(例如背景设计-大纲-章节-童话)  
+  **子评价标准**:以学生能做到...开始,需要匹配子教学目标和教学任务和过程性成果  ` : `  **子教学目标**:子教学目标是课程目标在该任务的分解目标,撰写格式同子教学目标  
+  **过程性成果**:过程性成果是最终成果的一部分或者是中间成果(例如背景设计-大纲-章节-童话)  
+  **子评价标准**:以学生能做到...开始,需要匹配子教学目标和教学任务和过程性成果  `  
 
 let courseText = this.courseText ? this.courseText : this.teacherCourseText
 let target = this.targetCourseText2 ? this.targetCourseText2 : this.targetcoursetext
@@ -16256,15 +16270,17 @@ Instruction: Based on the context, follow "Format example", write content.
 ${msg} 以及##参考资料
 
 ## 参考资料
-课程名字:${this.courseName}
-${courseText ? '课程简要描述:' + courseText : ''}
-${sub.length ? '学科:' + sub.join(",") : ''}
-${mclass.length ? '面向年级:' + mclass.join(",") : ''}
+课程名字:${this.courseName} ${courseText ? '课程简要描述:' + courseText : ''} ${sub.length ? '学科:' + sub.join(",") : ''} ${mclass.length ? '面向年级:' + mclass.join(",") : ''}
 ${target ? '课程目标:' + target.replaceAll('#','').replaceAll('*','').replaceAll('-','').replaceAll('\n','') : ''}
-detail至少输出300tokens
+detail至少输出300tokens,markdown格式输出。
+detail包含子教学目标、过程性成果、子评价标准
+子教学目标:子教学目标是课程目标在该任务的分解目标,撰写格式同子教学目标  
+过程性成果:过程性成果是最终成果的一部分或者是中间成果(例如背景设计-大纲-章节-童话)  
+子评价标准:以学生能做到...开始,需要匹配子教学目标和教学任务和过程性成果  
+task2输出子任务名称但是不要输出序号或者前缀仅输出名称
 
 # Format example
-[{"task": "任务名称","detail": "${txt}"},{"task": "任务名称","detail": "${txt}"},{"task": "任务名称","detail": "${txt}"},{"task": "任务名称","detail": "${txt}"}]`
+[{"task": "任务名称","task2": "子任务名称","detail": "${txt}"},{"task": "任务名称","task2": "子任务名称","detail": "${txt}"},{"task": "任务名称","task2": "子任务名称","detail": "${txt}"},{"task": "任务名称","task2": "子任务名称","detail": "${txt}"}]`
 // detail输出不要输出成json,文本或者markdown都可以
         this.aiGet2(message, () => {
           this.loading = false
@@ -16425,12 +16441,13 @@ ${msg}
 let message2 = `NOTICE
 Role: 你是一名教师,你正在使用一个在线学习平台来教授学生,你已经完成了教案,你现在需要将教案变为学生上课用的任务单(work sheet)。该在线学习平台为每个任务提供了资源区(视频,ppt,阅读资料将放在这里),多种互动工具(学生点击工具即可上传他们的作业或者回答,这些互动工具类似kahoot一样允许教师轻松收集,统计,追踪学生作业),以及和任务绑定的目标和评价系统。
 Language: Please use the same language as the user requirement, if the user speaks Chinese, the specific text of your answer should also be in Chinese.
-ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example".
+ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example".作业提交工具只能出现一次。
 Instruction: Based on the context, follow "Format example", write content.
+注意:作业提交工具只能出现一次。
 
 # Context
 ## 要求
-${msg} 
+${msg2} 
  
 
 ## 参考资料
@@ -16450,8 +16467,12 @@ ${msg}
 工具名称:作业提交
 工具指引:同学们,当你们完成实验报告后,我们需要用PPT来汇报项目成果。PPT是一种强大的工具,能帮助我们将复杂的信息以直观、有趣的方式展示出来。首先,打开PPT工具,新建一个演示文稿,并设置好基本结构,如标题页、实验背景、实验方法、数据分析、结论与建议等。接着,可以通过选择合适的主题和背景,美化幻灯片,插入图片、图标和图表,让PPT更生动。记住要保持文字简洁、突出重点,并为每张幻灯片添加适当的动画效果。制作完成后,多次练习你的演讲,确保表达清晰自然。最后,保存并分享你的PPT,准备好精彩的展示吧!期待看到你们的优秀成果,有问题随时来问我哦!
 
+注意:作业提交工具只能出现一次。
+
 # Format example
-{"toolChoose":[{"tool":"工具名","detail":"工具描述"},{"tool":"工具名","detail":"工具描述"}]}`
+{"toolChoose":[{"tool":"工具名","detail":"工具描述"},{"tool":"工具名","detail":"工具描述"}]}
+
+注意:作业提交工具只能出现一次。`
 
 let message3 = `NOTICE
 Role: 你是一名教师,你正在使用一个在线学习平台来教授学生,你已经完成了教案,你现在需要将教案变为学生上课用的任务单(work sheet)。该在线学习平台为每个任务提供了资源区(视频,ppt,阅读资料将放在这里),多种互动工具(学生点击工具即可上传他们的作业或者回答,这些互动工具类似kahoot一样允许教师轻松收集,统计,追踪学生作业),以及和任务绑定的目标和评价系统。
@@ -16461,7 +16482,7 @@ Instruction: Based on the context, follow "Format example", write content.
 
 # Context
 ## 要求
-${msg} 
+${msg3} 
  
 
 ## 参考资料
@@ -16514,12 +16535,13 @@ ${msg}
 let message2 = `NOTICE
 Role: 你是一名教师,你正在使用一个在线学习平台来教授学生,你已经完成了教案,你现在需要将教案变为学生上课用的任务单(work sheet)。该在线学习平台为每个任务提供了资源区(视频,ppt,阅读资料将放在这里),多种互动工具(学生点击工具即可上传他们的作业或者回答,这些互动工具类似kahoot一样允许教师轻松收集,统计,追踪学生作业),以及和任务绑定的目标和评价系统。
 Language: Please use the same language as the user requirement, if the user speaks Chinese, the specific text of your answer should also be in Chinese.
-ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example".
+ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example".作业提交工具只能出现一次。
 Instruction: Based on the context, follow "Format example", write content.
+注意:作业提交工具只能出现一次。
 
 # Context
 ## 要求
-${msg} 
+${msg2} 
  
 
 ## 参考资料
@@ -16539,8 +16561,12 @@ ${msg}
 工具名称:作业提交
 工具指引:同学们,当你们完成实验报告后,我们需要用PPT来汇报项目成果。PPT是一种强大的工具,能帮助我们将复杂的信息以直观、有趣的方式展示出来。首先,打开PPT工具,新建一个演示文稿,并设置好基本结构,如标题页、实验背景、实验方法、数据分析、结论与建议等。接着,可以通过选择合适的主题和背景,美化幻灯片,插入图片、图标和图表,让PPT更生动。记住要保持文字简洁、突出重点,并为每张幻灯片添加适当的动画效果。制作完成后,多次练习你的演讲,确保表达清晰自然。最后,保存并分享你的PPT,准备好精彩的展示吧!期待看到你们的优秀成果,有问题随时来问我哦!
 
+注意:作业提交工具只能出现一次。
+
 # Format example
-{"toolChoose":[{"tool":"工具名","detail":"工具描述"},{"tool":"工具名","detail":"工具描述"}]}`
+{"toolChoose":[{"tool":"工具名","detail":"工具描述"},{"tool":"工具名","detail":"工具描述"}]}
+
+注意:作业提交工具只能出现一次。`
 
 let message3 = `NOTICE
 Role: 你是一名教师,你正在使用一个在线学习平台来教授学生,你已经完成了教案,你现在需要将教案变为学生上课用的任务单(work sheet)。该在线学习平台为每个任务提供了资源区(视频,ppt,阅读资料将放在这里),多种互动工具(学生点击工具即可上传他们的作业或者回答,这些互动工具类似kahoot一样允许教师轻松收集,统计,追踪学生作业),以及和任务绑定的目标和评价系统。
@@ -16550,7 +16576,7 @@ Instruction: Based on the context, follow "Format example", write content.
 
 # Context
 ## 要求
-${msg} 
+${msg3} 
  
 
 ## 参考资料
@@ -16935,6 +16961,16 @@ ${this.courseText && this.aiCallBack == 2 ? '注意,优化原有的<参考内
         }
         // this.$forceUpdate();
       };
+      this.isTextCancelSource.onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.textLoading = false
+        this.ttextLoading = false
+        this.isTextCancelToken = null
+        this.isTextCancelSource.close();
+        this.isTextCancelSource = null;
+      };
     },
     numberToChinese(number) {
       const chineseNum = ['零', '一', '二', '三', '四', '五', '六', '七', '八', '九'];
@@ -17093,7 +17129,7 @@ ${this.courseText && this.aiCallBack == 2 ? '注意,优化原有的<参考内
       this.isOutlineCancelToken = this.ajax.setCancelSource();
       let parm = {
         assistant_id: '6063369f-289a-11ef-8bf4-12e77c4cb76b',
-        message: [{"type":"text", "text":messages.replaceAll('\n', " ").replaceAll('*', "")}],
+        message: [{"type":"text", "text":messages}],
         session_name: uuidv4(),
         userId: this.userid,
         file_ids: fileid.length ? [...fileid] : '',
@@ -17128,7 +17164,7 @@ ${this.courseText && this.aiCallBack == 2 ? '注意,优化原有的<参考内
                     let regex = new RegExp("(?<=```json)([\\s\\S]*?)(?=```)");
                     let match = data.message.match(regex);
                     // eval("var aaa = " + match[0])
-                    dArray = JSON.parse(match[0].replace(/\n/g, '').replace(/\s{2,}/g, ' '));
+                    dArray = JSON.parse(match[0].replace(/\n/g, '      ').replace(/\s{2,}/g, '      '));
 
                 } catch (error) {
                   _this.$message.error('哎呀,网络波动了...小可正在为您重新生成中...')
@@ -17148,13 +17184,13 @@ ${this.courseText && this.aiCallBack == 2 ? '注意,优化原有的<参考内
               for (var i = 0; i < dArray.length; i++) {
                 let _task = dArray[i]
                 if(_this.unitJson[0].chapterInfo[0].taskJson[i]){
-                  _this.unitJson[0].chapterInfo[0].taskJson[i].task = _task.task.replaceAll('任务'+(i+1),'').replaceAll('任务'+_this.numberToChinese(i+1),'').replaceAll(':','').replaceAll(':','')
-                  _this.unitJson[0].chapterInfo[0].taskJson[i].taskDetail2 = _task.detail
+                  _this.unitJson[0].chapterInfo[0].taskJson[i].task = _task.task.replaceAll('任务'+(i+1),'').replaceAll('任务'+_this.numberToChinese(i+1),'').replaceAll(':','').replaceAll(':','')+(_task.task2 ? ":"+_task.task2 : "")
+                  _this.unitJson[0].chapterInfo[0].taskJson[i].taskDetail2 = _task.detail.replaceAll("      ", "  \n")
                 }else {
                   _this.unitJson[0].chapterInfo[0].taskJson.push({
-                    task: _task.task.replaceAll('任务'+(i+1),'').replaceAll('任务'+_this.numberToChinese(i+1),'').replaceAll(':','').replaceAll(':',''),
+                    task: _task.task.replaceAll('任务'+(i+1),'').replaceAll('任务'+_this.numberToChinese(i+1),'').replaceAll(':','').replaceAll(':','')+(_task.task2 ? ":"+_task.task2 : ""),
                     taskDetail: "",
-                    taskDetail2: _task.detail,
+                    taskDetail2: _task.detail.replaceAll("      ", "  \n"),
                     taskDetail3: "",
                     chapterData: [],
                     toolText: "",
@@ -17185,9 +17221,9 @@ ${this.courseText && this.aiCallBack == 2 ? '注意,优化原有的<参考内
               for (var i = 0; i < dArray.length; i++) {
                 let _task = dArray[i]
                 _this.unitJson[0].chapterInfo[0].taskJson.push({
-                  task: _task.task.replaceAll('任务'+(i+1),'').replaceAll('任务'+_this.numberToChinese(i+1),'').replaceAll(':','').replaceAll(':',''),
+                  task: _task.task.replaceAll('任务'+(i+1),'').replaceAll('任务'+_this.numberToChinese(i+1),'').replaceAll(':','').replaceAll(':','')+(_task.task2 ? ":"+_task.task2 : ""),
                   taskDetail: "",
-                  taskDetail2: _task.detail,
+                  taskDetail2: _task.detail.replaceAll("      ", "  \n"),
                   taskDetail3: "",
                   chapterData: [],
                   toolText: "",
@@ -19406,7 +19442,7 @@ ${msg}
       // })
       let params = {
         assistant_id: 'f8e1ebb2-2e0d-11ef-8bf4-12e77c4cb76b',
-        message: [{"type":"text", "text":messages.replaceAll('\n', " ").replaceAll('*', "")}],
+        message: [{"type":"text", "text":messages.replaceAll('\n', " ")}],
         session_name: _uuid,
         userId: _this.userid,
         uid: _uuid,
@@ -19494,9 +19530,18 @@ ${msg}
           this.taskDetailLoading.splice(this.taskDetailLoading.indexOf(loading), 1)
           // 处理流数据
         }
-        
         // this.$forceUpdate();
       };
+      this.isOutlineTaskCancelSource[_tindex2].onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.taskDetailLoading.splice(this.taskDetailLoading.indexOf(loading), 1)
+        this.ttaskDetailLoading.splice(this.ttaskDetailLoading.indexOf(loading), 1)
+        this.isOutlineTaskCancelToken[_tindex2] = null
+        this.isOutlineTaskCancelSource[_tindex2].close();
+        this.isOutlineTaskCancelSource[_tindex2] = null;
+      };
     },
     async aiOutlineDetail(msg, index) {
       let _this = this
@@ -19659,10 +19704,12 @@ ${target ? '课程目标:' + target.replaceAll('#','').replaceAll('*','').repl
       }
       let target = _this.targetCourseText2 ? _this.targetCourseText2 : _this.targetcoursetext
       let taskArray = []
+      let taskArrayD = []
       for (var k = 0; k < this.unitJson[0].chapterInfo[0].taskJson.length; k++) {
           // _text += `任务名称:${this.unitJson[0].chapterInfo[0].taskJson[i].task}`
           if(k != index){
             taskArray.push(this.unitJson[0].chapterInfo[0].taskJson[k].task)
+            taskArrayD.push(this.unitJson[0].chapterInfo[0].taskJson[k].taskDetail2.replaceAll('#','').replaceAll('*','').replaceAll('-','').replaceAll('\n',''))
           }
         }
       let messages = `NOTICE
@@ -19676,18 +19723,17 @@ Instruction: Based on the context, follow "Format example", write content.
 ${msg}
 
 ## 参考信息
-课程名字:${_this.courseName}
-任务名称:${_this.aiCallBack.task ? _this.aiCallBack.task : _this.unitJson[0].chapterInfo[0].taskJson[_tindex2].task}
+课程名字:${_this.courseName} 任务名称:${_this.aiCallBack.task ? _this.aiCallBack.task : _this.unitJson[0].chapterInfo[0].taskJson[_tindex2].task}
 ${_this.courseText ? '课程简要描述:' + _this.courseText : ''}
 ${target ? '课程目标:' + target.replaceAll('#','').replaceAll('*','').replaceAll('-','').replaceAll('\n','') : ''}
 ${taskArray.length ? '其他任务大纲名字:' + taskArray.join(',') : ''}
+${taskArray.length ? '其他任务大纲内容' + taskArray.join(',') : ''}
 
 # Format example
-子任务名称:名称应该总结了该任务是在做什么,
-子任务设计:任务设计。
-子教学目标:教学目标。
-学生过程性成果:学生过程性成果。
-评价标准:评价标准。
+**子任务标题**:环节+子任务名称
+**子教学目标**:子教学目标是课程目标在该任务的分解目标,撰写格式同子教学目标
+**过程性成果**:过程性成果是最终成果的一部分或者是中间成果(例如背景设计-大纲-章节-童话)
+**子评价标准**:以学生能做到...开始,需要匹配子教学目标和教学任务和过程性成果
 `
 // 教学目标:教学目标。
 // 任务设计:任务设计。
@@ -19717,7 +19763,7 @@ ${taskArray.length ? '其他任务大纲名字:' + taskArray.join(',') : ''}
       // })
       let params = {
         assistant_id: 'f8e1ebb2-2e0d-11ef-8bf4-12e77c4cb76b',
-        message: [{"type":"text", "text":messages.replaceAll('\n', " ").replaceAll('*', "")}],
+        message: [{"type":"text", "text":messages}],
         session_name: _uuid,
         userId: _this.userid,
         uid: _uuid,
@@ -19811,9 +19857,18 @@ ${taskArray.length ? '其他任务大纲名字:' + taskArray.join(',') : ''}
           }
           // 处理流数据
         }
-        
         // this.$forceUpdate();
       };
+      this.isOutlineTaskCancelSource[_tindex2].onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.taskDetailLoading.splice(this.taskDetailLoading.indexOf(loading), 1)
+        this.ttaskDetailLoading.splice(this.ttaskDetailLoading.indexOf(loading), 1)
+        this.isOutlineTaskCancelToken[_tindex2] = null
+        this.isOutlineTaskCancelSource[_tindex2].close();
+        this.isOutlineTaskCancelSource[_tindex2] = null;
+      };
     },
     async aiDetail2(msg, index) {
       let _this = this
@@ -19962,6 +20017,14 @@ ${_this.aiCallBack.taskDetail}
         }
         // this.$forceUpdate();
       };
+      _source.onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.taskDetailLoading2.splice(this.taskDetailLoading2.indexOf(loading), 1)
+        this.ttaskDetailLoading2.splice(this.ttaskDetailLoading2.indexOf(loading), 1)
+        _source.close();
+      };
     },
     async aiDetail3(msg, index, tindex) {
       let _this = this
@@ -20111,6 +20174,14 @@ ${msg} 输出格式和内容要求参考#格式与要求
         }
         // this.$forceUpdate();
       };
+      _source.onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.taskDetailLoading3.splice(this.taskDetailLoading3.indexOf(loading), 1)
+        this.ttaskDetailLoading3.splice(this.ttaskDetailLoading3.indexOf(loading), 1)
+        _source.close();
+      };
     },
     async aiDetail4(msg, index) {
       if (!msg.length) {
@@ -20449,6 +20520,14 @@ ${_text2}`
         }
         // this.$forceUpdate();
       };
+      _source.onerror = function(event) {
+          // 处理错误,可以尝试重新连接
+          console.error('EventSource error:', event);
+          this.$message.error("网络错误")
+          this.taskDetailLoading5.splice(this.taskDetailLoading5.indexOf(loading), 1)
+          this.ttaskDetailLoading5.splice(this.ttaskDetailLoading5.indexOf(loading), 1)
+          _source.close();
+      };
     },
     async aiDetail52(msg, index) {
       let _this = this
@@ -20618,6 +20697,14 @@ ${(_this.templateid == "4480d65a-1e48-11ef-bee5-005056b86db5" || _this.templatei
           // 处理流数据
         }
       };
+      _source.onerror = function(event) {
+          // 处理错误,可以尝试重新连接
+          console.error('EventSource error:', event);
+          this.$message.error("网络错误")
+          this.taskDetailLoading5.splice(this.taskDetailLoading5.indexOf(loading), 1)
+          this.ttaskDetailLoading5.splice(this.ttaskDetailLoading5.indexOf(loading), 1)
+          _source.close();
+      };
     },
     aiDetail52getAiContentText(_e, loading, _tindex2, _source){
       let _mdText = "";
@@ -20820,6 +20907,14 @@ ${this.aitype == 'aiCpote4' ? '## 目标层\n'+this.cpote['cpote3'] : ''}`
         }
         // this.$forceUpdate();
       };
+      _source.onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.cpoteLoading[loading] = false
+        this.cpotetLoading[loading] = false
+        _source.close();
+      };
     },
     async aiteacherTextDetail(msg) {
       let _this = this
@@ -20995,6 +21090,16 @@ ${this.teacherCourseText && this.aiCallBack == 2 ? '注意,优化原有的<参
         
         // this.$forceUpdate();
       };
+      this.isTeacherTextCancelSource.onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.teacherTextLoading = false
+        this.tteacherTextLoading = false
+        this.isTeacherTextCancelToken = null
+        this.isTeacherTextCancelSource.close();
+        this.isTeacherTextCancelSource = null;
+      };
     },
     async aitargetTextDetail(msg) {
       let _this = this
@@ -21171,6 +21276,16 @@ ${this.targetCourseText && this.aiCallBack == 2 ? '注意,优化原有的<参
         
         // this.$forceUpdate();
       };
+      this.isTargetCancelSource1.onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.targetTextLoading = false
+        this.ttargetTextLoading = false
+        this.isTargetCancelToken1 = null
+        this.isTargetCancelSource1.close();
+        this.isTargetCancelSource1 = null;
+      };
     },
     async aitargetTextDetail2(msg) {
       let _this = this
@@ -21510,6 +21625,16 @@ ${this.targetCourseText && this.aiCallBack == 2 ? '注意,优化原有的<参
         }
         // this.$forceUpdate();
       };
+      this.isTargetCancelSource2.onerror = function(event) {
+        // 处理错误,可以尝试重新连接
+        console.error('EventSource error:', event);
+        this.$message.error("网络错误")
+        this.targetTextLoading2 = false
+        this.ttargetTextLoading2 = false
+        this.isTargetCancelToken2 = null
+        this.isTargetCancelSource2.close();
+        this.isTargetCancelSource2 = null;
+      };
     },
     async aitargetTextDetail3(msg) {
       let _this = this
@@ -21654,7 +21779,7 @@ Instruction: Based on the context, follow "Format example", write content.
 ${msg}
 
 ##制定标准资料
-评价维度:${this.unitJson[0].chapterInfo[0].taskJson[i].eList[k].value}
+评价维度:${this.unitJson[0].chapterInfo[0].taskJson[i].eList[k].target}
 维度描述:${this.unitJson[0].chapterInfo[0].taskJson[i].eList[k].detail}
 
 # Format example

+ 15 - 13
src/components/pages/components/studentWorksDetail.vue

@@ -128,31 +128,32 @@
           :row-class-name="tableRowClassName"
           stripe
         >
-          <el-table-column
+          <!-- <el-table-column
             prop="sName"
             label="姓名"
             min-width="15"
             align="center"
-          ></el-table-column>
+          ></el-table-column> -->
           <el-table-column
-            prop="class"
-            label="班级"
+            prop="course"
+            label="课程"
             min-width="20"
             align="center"
             show-overflow-tooltip
           >
-            <template slot-scope="scope">
-              <div>{{ scope.row.class ? scope.row.class : "暂无班级" }}</div>
-            </template>
           </el-table-column>
           <el-table-column
-            prop="course"
-            label="课程"
+            prop="class"
+            label="班级"
             min-width="20"
             align="center"
             show-overflow-tooltip
           >
+            <template slot-scope="scope">
+              <div>{{ scope.row.class ? scope.row.class : "暂无班级" }}</div>
+            </template>
           </el-table-column>
+         
 
           <el-table-column
             prop="time"
@@ -160,7 +161,7 @@
             min-width="15"
             align="center"
           ></el-table-column>
-          <el-table-column label="操作" width="300px">
+          <el-table-column label="操作" align="center" width="300px">
             <template slot-scope="scope">
               <el-button
                 type="primary"
@@ -210,7 +211,7 @@
                   size="mini"
                   style="margin-left: 15px;"
                   @click="getPdf"
-                  >导出作业</el-button
+                  >导出作业</el-button
                 >
               </div>
 
@@ -1860,7 +1861,7 @@ export default {
         let num = 0;
 
         for (const key in val) {
-          if (val[key] || val[key] == 0) {
+          if (val[key] || key != 'content') {
             sco += val[key] * 1;
           }
           num += 1;
@@ -3989,6 +3990,7 @@ export default {
   /* margin: 10px 0 0 30px; */
   margin: 10px 0;
   position: relative;
+  margin-left: 19px;
 }
 .sd_score .score_box > span:nth-child(1) {
   margin-right: 20px;
@@ -5150,7 +5152,7 @@ ol {
   flex-direction: column;
   flex-wrap: nowrap;
   align-items: center;
-  width: 100%;
+  width: 80%;
   flex-wrap: nowrap;
   overflow: auto;
   /* height: 500px; */

+ 20 - 15
src/components/pages/components/worksDetail2.vue

@@ -107,14 +107,13 @@
                 :key="index+'a'"
                 :label="item.name"
                 :value="item.id"
-
               ></el-option>
             </el-select>
 
             
             <div class="typeCheck">
               <el-switch v-model="typeCheck"></el-switch
-              ><span>查看所有学生</span>
+              ><span style="font-size: 14px;">查看所有学生</span>
             </div>
           </div>
         </div>
@@ -233,9 +232,9 @@
                 type="primary"
                 size="small"
                 @click="
-                  lookWork2(scope.row.id, scope.row.userid, scope.row.courseid)
+                  lookWork2(scope.row.id, scope.row.userid, scope.row.courseid, dyList[scope.row.stage].name,dyList[scope.row.stage].taskList[scope.row.task].name)
                 "
-                >导出作业</el-button
+                >导出作业</el-button
               >
               <!-- <el-button
                 class="de_button"
@@ -1900,12 +1899,11 @@ export default {
       return function (val) {
         let sco = 0
         let num = 0
-
         for (const key in val) {
-          if (val[key]  || val[key] == 0) {
-            sco+= val[key] * 1
+          if (val[key] && key != 'content') {
+            sco += val[key] * 1
           }
-          num+=1
+          num += 1
         }
         return (sco / (num - 1)).toFixed(1)
       }
@@ -3340,7 +3338,7 @@ export default {
     },
     onPlayerPlay() {},
     // this.$store.commit("update", ["userInfo", userInfo]);
-    lookWork2(id, uid, cid) {
+    lookWork2(id, uid, cid,stage,task) {
       // this.scopeId = id;
       let params = {
         uid: uid,
@@ -3439,13 +3437,18 @@ export default {
               if (workEvaList[p].rate && eList && eList.length) {
                 _rate[p] = JSON.parse(workEvaList[p].rate);
               }
-              _rateList[p] = Object.keys(JSON.parse(workEvaList[p].rate));
+              _rateList[p] = Object.keys(JSON.parse(workEvaList[p].rate)).filter(e => {return e!='content'});
               for (var i = 0; i < _rateList[p].length; i++) {
                 var _c = Object.keys(workJson.rateList);
                 if (_c.indexOf(_rateList[p][i]) != -1) {
                   workJson.rateList[_rateList[i]] = _rate[_rateList[i]];
                 }
               }
+
+              // console.log('1',JSON.parse(JSON.stringify(_rateList[p])));
+              // console.log('2',JSON.parse(JSON.stringify(this.ooption[p])));
+
+              
               for (var i = 0; i < this.ooption[p].length; i++) {
                 if (_rateList[p].indexOf(this.ooption[p][i].name) != -1) {
                   this.ooption[p][i].value = _rate[p][this.ooption[p][i].name];
@@ -3931,7 +3934,7 @@ export default {
             _bg +
             _cocopi +
             _word;
-          this.generate(_html);
+          this.generate(_html,stage,task);
         })
         .catch((err) => {
           console.error(err);
@@ -3962,7 +3965,7 @@ export default {
         this.navId = a;
       }
     },
-    async generate(a) {
+    async generate(a,stage,task) {
       // 将html文件中需要用到的数据挂载到store上
       this.$store.commit("update", ["report", a]);
       const content = `<!DOCTYPE html>
@@ -3980,7 +3983,7 @@ export default {
       // debugger
       // 生成报告
       const link = document.createElement("a");
-      let dname = this.sInfo.course + "-" + this.sInfo.sName + ".html";
+      let dname = this.sInfo.course + "-"+stage+"-" + task + this.sInfo.sName + ".html";
       // link.download = "报告.html"; // 文件名
       link.download = dname; // 文件名
       link.style.display = "none";
@@ -4423,7 +4426,7 @@ export default {
   /* margin: 10px 0 0 30px; */
   margin: 10px 0;
   position: relative;
-
+  margin-left: 19px;
 }
 .sd_score .score_box > span:nth-child(1) {
   margin-right: 20px;
@@ -5256,7 +5259,7 @@ font-size: 14px !important;
   flex-direction: column;
   flex-wrap: nowrap;
   align-items: center;
-  width: 100%;
+  width: 80%;
   flex-wrap: nowrap;
   /* overflow: auto; */
   /* height: 500px; */
@@ -5773,12 +5776,14 @@ ol {
   margin-right: 15px;
   padding: 5px 0;
   cursor: pointer;
+  font-size: 14px;
 }
 .cutTab > .ctrlBtn2 {
   cursor: pointer;
   margin-right: 15px;
   border-bottom: 3px #409EFF solid;
   font-weight: 600;
+  font-size: 14px;
   padding: 5px 0;
   color: #409EFF;
 }

+ 6 - 1
src/components/pages/course.vue

@@ -58,7 +58,7 @@
               <div class="search">
                 <img src="../../assets/icon/search.png" alt="" />
               </div>
-              <input class="sInput" type="text" placeholder="请输入关键字" v-model="courseName" @input="getCourse()" />
+              <input class="sInput" type="text" placeholder="请输入关键字" v-model="courseName" />
             </div>
             <el-button type="primary" class="bgColor" style="margin-left: 10px;" @click="reset">重置</el-button>
           </div>
@@ -465,6 +465,11 @@ export default {
       oidArray:[]
     };
   },
+  watch: {
+    courseName(newValue, oldValue) {
+      this.searchCourse()
+    }
+  },
   methods: {
     getAIJ(){
       this.ajax

+ 1 - 1
src/components/pages/test/add/components/course2/courseIndex.vue

@@ -886,7 +886,7 @@
         this.isLoading = true;
         let params = {
           // type: this.groupA,
-					type: "4",
+					type: "0",
           uid: this.userid,
           oid: this.oid,
           org: this.org,

+ 330 - 2
src/components/pages/test/add/components/file/index.vue

@@ -10,7 +10,27 @@
                 </el-select>
             </div>
             <div class="set_type">
-                <span>题目分数:</span><el-input v-model="checkJson.score" class="input" placeholder="请输入分数" style="width: 120px"  @change="numberPan"></el-input>
+                <span>题目分数:</span><el-input v-model="checkJson.score" class="input" placeholder="请输入分数"
+                    style="width: 120px" @change="numberPan"></el-input>
+            </div>
+            <div class="set_file">
+                <div class="set_type">
+                    <span>模板上传:</span>
+                    <div class="btn" @click.stop="addImg($event)" v-loading="proVisible">
+                        <el-button type="primary" size="mini">上传模板</el-button>
+                        <input type="file" accept=".doc,.docx,.xls,.xlsx,.ppt,.pptx,.pdf" style="display: none" multiple="multiple"
+                            @change="beforeUpload($event)" />
+                    </div>
+                </div>
+                <div class="file_box" v-if="checkJson.mobanFile && checkJson.mobanFile.length" v-loading="loading">
+                    <div class="file_item" v-for="(item, index) in checkJson.mobanFile" :key="index">
+                        <div class="file_name">{{ item.name }}</div>
+                        <div class="file_btns">
+                            <div class="file_btn" @click="downloadFile(index)">下载</div>
+                            <div class="file_btn" @click="delFile(index)">删除</div>
+                        </div>
+                    </div>
+                </div>
             </div>
             <div class="set_title">
                 <span>表单问题:</span>
@@ -35,6 +55,43 @@
 </template>
 
 <script>
+
+const getFile = (url) => {
+  return new Promise((resolve, reject) => {
+    var credentials = {
+        accessKeyId: "AKIATLPEDU37QV5CHLMH",
+        secretAccessKey: "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",
+      }; //秘钥形式的登录上传
+      window.AWS.config.update(credentials);
+      window.AWS.config.region = "cn-northwest-1"; //设置区域
+      let url2 = url;
+      let _url2 = "";
+      if (
+        url2.indexOf("https://view.officeapps.live.com/op/view.aspx?src=") != -1
+      ) {
+        _url2 = url2.split(
+          "https://view.officeapps.live.com/op/view.aspx?src="
+        )[1];
+      } else {
+        _url2 = url2;
+      }
+      var s3 = new window.AWS.S3({ params: { Bucket: "ccrb" } });
+      let name = decodeURIComponent(_url2.split("https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/")[1])
+      var params = {
+        Bucket: "ccrb",
+        Key: name
+      };
+      s3.getObject(params, function (err, data) {
+        if (err) {
+          console.log(err, err.stack)
+          resolve({ data: 1 });
+        }else {
+          resolve({ data: data.Body });
+          console.log(data); 
+        }
+      });
+  });
+};
 // import EditorBar from "../../../../../tools/wangEnduit";
 export default {
     components: {
@@ -52,7 +109,9 @@ export default {
                 { type: 1, name: '附件' },
                 // { type: 2, name: '多选题' }
             ],
-            checkJson: {}
+            checkJson: {},
+            proVisible: false,
+            loading: false
         }
     },
     directives: {
@@ -96,7 +155,233 @@ export default {
             this.$forceUpdate();
             console.log(val);
         },
+        addImg(e) {
+            var el = e.currentTarget;
+            el.getElementsByTagName("input")[0].click();
+            e.target.value = "";
+        },
+        delFile(index){
+            this.checkJson.mobanFile.splice(index, 1);
+            this.$forceUpdate();
+            this.$emit("setJson", this.checkJson)
+        },
+        downloadFile(index){
+            this.loading = true;
+            let file = this.checkJson.mobanFile[index]
+            getFile(file.url).then((data) => {
+                this.loading = false;
+                if (data.data != 1) {
+                    // 下载文件, 并存成ArrayBuffer对象
+                    const file_name =file.name; // 获取文件名
+                    const file_data = data.data; // 获取文件数据
+                    let url = window.URL.createObjectURL(new Blob([file_data]));
+                    let a = document.createElement("a");
+                    a.name = file_name;
+                    a.href = url;
+                    a.download = file_name;
+                    a.click();
+                    console.log(data); 
+                    this.$message.success("下载成功");
+                }else {
+                    this.$message.error("下载失败");
+                }
+            });
+
+        },
+        beforeUpload(event) {
+            let file = "";
+            let cfindex2 = 0;
+            for (var cfindex = 0; cfindex < event.target.files.length; cfindex++) {
+                file = event.target.files[cfindex];
 
+                var credentials = {
+                    accessKeyId: "AKIATLPEDU37QV5CHLMH",
+                    secretAccessKey: "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"
+                }; //秘钥形式的登录上传
+                window.AWS.config.update(credentials);
+                window.AWS.config.region = "cn-northwest-1"; //设置区域
+
+                var bucket = new window.AWS.S3({ params: { Bucket: "ccrb" } }); //选择桶
+                var _this = this;
+
+                // _this.progress = 0;
+                _this.proVisible = true;
+                // _this.isFinishSize = 0;
+                // _this.isAllSize = (file.size / 1024 / 1024).toFixed(2);
+                let _name = file.name;
+                let size = file.size;
+                _this.$forceUpdate();
+                if (file) {
+                    var params = {
+                        Key:
+                            file.name.split(".")[0] +
+                            new Date().getTime() +
+                            "." +
+                            file.name.split(".")[file.name.split(".").length - 1],
+                        ContentType: file.type,
+                        Body: file,
+                        "Access-Control-Allow-Credentials": "*",
+                        ACL: "public-read"
+                    }; //key可以设置为桶的相抵路径,Body为文件, ACL最好要设置
+                    var options = {
+                        partSize: 2048 * 1024 * 1024,
+                        queueSize: 2,
+                        leavePartsOnError: true
+                    };
+                    bucket
+                        .upload(params, options)
+                        .on("httpUploadProgress", function (evt) {
+                            //这里可以写进度条
+                            // console.log("Uploaded : " + parseInt((evt.loaded * 80) / evt.total) + '%');
+                            // _this.progress = parseInt((evt.loaded / evt.total) * 100);
+                            // _this.isFinishSize = (evt.loaded / 1024 / 1024).toFixed(2);
+                            // _this.$forceUpdate();
+                        })
+                        .send(function (err, data) {
+                            cfindex2++;
+
+                            // _this.progress = 100;
+                            // _this.isFinishSize = _this.isAllSize;
+                            // _this.$forceUpdate();
+                            // setTimeout(() => {
+                            //     _this.proVisible = false;
+                            //     _this.$forceUpdate();
+                            // }, 1000);
+                            setTimeout(() => {
+                                if (
+                                    cfindex2 == event.target.files.length ||
+                                    cfindex2 > event.target.files.length
+                                ) {
+                                    _this.proVisible = false;
+                                }
+                            }, 1000);
+                            // loading.close();
+                            if (err) {
+                                _this.$message.error("上传失败");
+                            } else {
+                                let _type = 2;
+                                var imgA = [
+                                    "png",
+                                    "jpg",
+                                    "jpeg",
+                                    "bmp",
+                                    "gif",
+                                    "webp",
+                                    "psd",
+                                    "svg",
+                                    "tiff"
+                                ];
+                                var fileA = [
+                                    "DOC",
+                                    "DOCX",
+                                    "DOCM",
+                                    "DOTM",
+                                    "DOTX",
+                                    "PPTX",
+                                    "PPSX",
+                                    "PPT",
+                                    "PPS",
+                                    "PPTM",
+                                    "POTM",
+                                    "PPAM",
+                                    "POTX",
+                                    "PPSM",
+                                    "XLSX",
+                                    "XLS"
+                                ];
+                                var videoA = [
+                                    "AVI",
+                                    "NAVI",
+                                    "MPEG",
+                                    "ASF",
+                                    "MOV",
+                                    "WMV",
+                                    "3GP",
+                                    "RM",
+                                    "RMVB",
+                                    "FLV",
+                                    "F4V",
+                                    "H.264",
+                                    "H.265",
+                                    "REAL VIDEO",
+                                    "MKV",
+                                    "WebM",
+                                    "HDDVD",
+                                    "MP4",
+                                    "MPG",
+                                    "M4V",
+                                    "MGV",
+                                    "OGV",
+                                    "QTM",
+                                    "STR",
+                                    "AMC",
+                                    "DVX",
+                                    "EVO",
+                                    "DAT",
+                                    "OGG",
+                                    "OGM"
+                                ];
+                                if (
+                                    fileA.indexOf(
+                                        data.Location.split(".")[
+                                            data.Location.split(".").length - 1
+                                        ].toLocaleUpperCase()
+                                    ) != -1
+                                ) {
+                                    _type = 1; //word 文件
+                                } else if (
+                                    videoA.indexOf(
+                                        data.Location.split(".")[
+                                            data.Location.split(".").length - 1
+                                        ].toLocaleUpperCase()
+                                    ) != -1
+                                ) {
+                                    _type = 2; //视频
+                                } else if (
+                                    imgA.indexOf(
+                                        data.Location.split(".")[
+                                            data.Location.split(".").length - 1
+                                        ].toLocaleLowerCase()
+                                    ) != -1
+                                ) {
+                                    _type = 3; //图片
+                                } else if (
+                                    "pdf".indexOf(
+                                        data.Location.split(".")[
+                                            data.Location.split(".").length - 1
+                                        ].toLocaleLowerCase()
+                                    ) != -1
+                                ) {
+                                    _type = 4; //pdf
+                                } else {
+                                    _type = 5; //文件
+                                }
+                                if (_this.checkJson.mobanFile) {
+                                    _this.checkJson.mobanFile.push({
+                                        name: _name,
+                                        url: data.Location,
+                                        type: _type
+                                    });
+                                    _this.$forceUpdate();
+                                } else {
+                                    _this.checkJson.mobanFile = [];
+                                    _this.checkJson.mobanFile.push({
+                                        name: _name,
+                                        url: data.Location,
+                                        type: _type
+                                    });
+                                    _this.$forceUpdate();
+                                }
+                                _this.$forceUpdate();
+
+                                console.log(_this.checkJson);
+                                _this.$emit("setJson", _this.checkJson)
+                                console.log(data.Location);
+                            }
+                        });
+                }
+            }
+        },
     },
     mounted() {
         console.log(1);
@@ -205,4 +490,47 @@ export default {
     border-bottom: 10px solid #f6f6f6;
     border-top: 10px solid transparent;
 } */
+
+
+.file_box {
+    padding: 0 0 0 15px;
+    width: 100%;
+    box-sizing: border-box;
+    margin: 10px 0;
+}
+
+.file_item {
+    display: flex;
+    width: 100%;
+    padding: 5px;
+    box-sizing: border-box;
+    border-radius: 5px;
+}
+
+.file_item+.file_item {
+    margin-top: 10px;
+}
+
+.file_name {
+    color: #747474;
+}
+
+.file_item:hover{
+    background: #fff;
+}
+
+.file_btns {
+    margin-left: auto;
+    display: flex;
+    align-items: center;
+}
+
+.file_btn+.file_btn {
+    margin-left: 10px;
+}
+
+.file_btn {
+    color: #3681fc;
+    cursor: pointer;
+}
 </style>

+ 116 - 0
src/components/pages/test/add/edit/check/file.vue

@@ -27,6 +27,16 @@
         v-html="checkJson.detail"
         style="color: #00000099;margin-top: 5px;"
       ></div>
+      <div class="choices">
+        <div class="file_box" v-if="checkJson.mobanFile && checkJson.mobanFile.length" v-loading="loading">
+            <div class="file_item" v-for="(item, index) in checkJson.mobanFile" :key="index">
+                <div class="file_name">{{ item.name }}</div>
+                <div class="file_btns">
+                    <div class="file_btn" @click="downloadMobanFile(index)">下载</div>
+                </div>
+            </div>
+        </div>
+      </div>
       <div class="choices">
         <div class="uploadBtn" v-if="checktype == 1">
           <div class="btn" @click.stop="addImg($event)">
@@ -195,6 +205,45 @@ import checkfile from "../../../file/checkfile.vue";
 
 import uploadFile from "./uploadFile.vue";
 import { v4 as uuidv4 } from "uuid";
+
+const getFile = (url) => {
+  return new Promise((resolve, reject) => {
+    var credentials = {
+        accessKeyId: "AKIATLPEDU37QV5CHLMH",
+        secretAccessKey: "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",
+      }; //秘钥形式的登录上传
+      window.AWS.config.update(credentials);
+      window.AWS.config.region = "cn-northwest-1"; //设置区域
+      let url2 = url;
+      let _url2 = "";
+      if (
+        url2.indexOf("https://view.officeapps.live.com/op/view.aspx?src=") != -1
+      ) {
+        _url2 = url2.split(
+          "https://view.officeapps.live.com/op/view.aspx?src="
+        )[1];
+      } else {
+        _url2 = url2;
+      }
+      var s3 = new window.AWS.S3({ params: { Bucket: "ccrb" } });
+      let name = decodeURIComponent(_url2.split("https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/")[1])
+      var params = {
+        Bucket: "ccrb",
+        Key: name
+      };
+      s3.getObject(params, function (err, data) {
+        if (err) {
+          console.log(err, err.stack)
+          resolve({ data: 1 });
+        }else {
+          resolve({ data: data.Body });
+          console.log(data); 
+        }
+      });
+  });
+};
+
+
 export default {
   components: {
     wpdf,
@@ -240,6 +289,7 @@ export default {
       wurl: "",
       isTong: false,
 			uploadList:[],
+      loading: false
     };
   },
   watch: {
@@ -692,6 +742,29 @@ export default {
         this.wurl = item.url;
       }
     },
+    downloadMobanFile(index){
+        this.loading = true;
+        let file = this.checkJson.mobanFile[index]
+        getFile(file.url).then((data) => {
+            this.loading = false;
+            if (data.data != 1) {
+                // 下载文件, 并存成ArrayBuffer对象
+                const file_name =file.name; // 获取文件名
+                const file_data = data.data; // 获取文件数据
+                let url = window.URL.createObjectURL(new Blob([file_data]));
+                let a = document.createElement("a");
+                a.name = file_name;
+                a.href = url;
+                a.download = file_name;
+                a.click();
+                console.log(data); 
+                this.$message.success("下载成功");
+            }else {
+                this.$message.error("下载失败");
+            }
+        });
+
+    },
     downloadFile(f) {
       var credentials = {
         accessKeyId: "AKIATLPEDU37QV5CHLMH",
@@ -996,4 +1069,47 @@ export default {
 	font-weight: bold;
 	margin-bottom: 10px;
 }
+
+
+.file_box {
+    /* padding: 0 0 0 15px; */
+    width: 100%;
+    box-sizing: border-box;
+    margin: 10px 0;
+}
+
+.file_item {
+    display: flex;
+    width: 100%;
+    padding: 5px;
+    box-sizing: border-box;
+    border-radius: 5px;
+}
+
+.file_item+.file_item {
+    margin-top: 10px;
+}
+
+.file_name {
+    color: #747474;
+}
+
+.file_item:hover{
+    background: #eee;
+}
+
+.file_btns {
+    margin-left: auto;
+    display: flex;
+    align-items: center;
+}
+
+.file_btn+.file_btn {
+    margin-left: 10px;
+}
+
+.file_btn {
+    color: #3681fc;
+    cursor: pointer;
+}
 </style>

+ 1 - 1
src/components/pages/test/add/setInfo/index.vue

@@ -69,7 +69,7 @@
                                 <div class="typeBox" v-for="item in teaType" :key="item.id" style="margin: 0 0 10px">
                                     <span>{{ item.name + ":" }}</span>
                                     <el-select v-model="item.value" :placeholder="'请选择' + item.name"
-                                        @change="Ochange">
+                                        @change="Ochange" clearable>
                                         <el-option v-for="i in item.child" :key="i.id" :label="i.name" :value="i.id">
                                         </el-option>
                                     </el-select>

+ 167 - 12
src/components/pages/test/check/index.vue

@@ -144,7 +144,7 @@
               <div class="title">
                 <el-tooltip :content="selectType(item)" placement="top" effect="dark">
                   <span class="test_icon"
-                    :class="{ test_icon_check: item.type == 1 && item.atype == 2, test_icon_checkO: item.type == 1 && item.atype == 1, test_icon_gap: item.type == 3, test_icon_file: item.type == 5, test_course_file: item.type == 6, test_eva_file: item.type == 7,test_icon_time: item.type == 8 }"></span>
+                    :class="{ test_icon_check: item.type == 1 && item.atype == 2, test_icon_checkO: item.type == 1 && item.atype == 1, test_icon_gap: item.type == 3, test_icon_file: item.type == 5, test_course_file: item.type == 6 || item.type == 11, test_eva_file: item.type == 7,test_icon_time: item.type == 8 }"></span>
                 </el-tooltip>
                 <span style="line-height: 25px;">{{ item.title }}</span>
 								<el-button v-if="item.type == 5 && item.array.length>0" class="title_downBtn" type="primary" size="small" @click.stop="downloadFileType5(item.array,item.title,index)">批量下载附件</el-button>
@@ -313,6 +313,38 @@
                   </wordcloud>
                 </div>
               </div>
+              <div class="content4" v-if="item.type == 11">
+                <div class="out_box" v-for="(item, index) in item.courseArray" :key="index + '-' + index2">
+                  <div class="tup">
+                    <img :src="item.cover != null && item.cover != ''
+    ? JSON.parse(item.cover).length > 0
+      ? JSON.parse(item.cover)[0].url
+      : mr
+    : mr
+    " alt />
+                    <div class="bottom_box">
+                      <div>
+                        <el-tooltip :content="item.title" popper-class="text_tooltip" placement="top" effect="dark">
+                          <span>{{ item.title }}</span>
+                        </el-tooltip>
+                      </div>
+                      <div class="kc_t">
+                        <span>{{ item.username }}</span>
+                        <span>{{ item.state == 1 ? '阶段模式' : item.state == 2 ? '任务模式' : '极简模式' }}</span>
+                      </div>
+                      <div class="kc_time">
+                        <span style="color: #717C8D">创建日期:</span>{{ item.time }}
+                      </div>
+                      <div class="kc_time">
+                        <span style="color: #717C8D">修改日期:</span>{{ item.update_at }}
+                      </div>
+                    </div>
+                  </div>
+                </div>
+                <!-- <div class="courses" v-for="(courses, index2) in item.array" :key="index + '-' + index2">
+                  {{ courses }}
+                </div> -->
+              </div>
             </div>
           </div>
           <!-- 按题目手机端 !isDesktop-->
@@ -497,6 +529,38 @@
                   <div class="cutBtnCss" @click="item.cut=!item.cut">{{ item.cut?'查看详情':'查看视图' }}</div>
                 </div>
               </div>
+              <div class="content4" v-if="item.type == 11">
+                <div class="out_box" v-for="(item, index) in item.courseArray" :key="index + '-' + index2">
+                  <div class="tup">
+                    <img :src="item.cover != null && item.cover != ''
+    ? JSON.parse(item.cover).length > 0
+      ? JSON.parse(item.cover)[0].url
+      : mr
+    : mr
+    " alt />
+                    <div class="bottom_box">
+                      <div>
+                        <el-tooltip :content="item.title" popper-class="text_tooltip" placement="top" effect="dark">
+                          <span>{{ item.title }}</span>
+                        </el-tooltip>
+                      </div>
+                      <div class="kc_t">
+                        <span>{{ item.username }}</span>
+                        <span>{{ item.state == 1 ? '阶段模式' : item.state == 2 ? '任务模式' : '极简模式' }}</span>
+                      </div>
+                      <div class="kc_time">
+                        <span style="color: #717C8D">创建日期:</span>{{ item.time }}
+                      </div>
+                      <div class="kc_time">
+                        <span style="color: #717C8D">修改日期:</span>{{ item.update_at }}
+                      </div>
+                    </div>
+                  </div>
+                </div>
+                <!-- <div class="courses" v-for="(courses, index2) in item.array" :key="index + '-' + index2">
+                  {{ courses }}
+                </div> -->
+              </div>
             </div>
           </div>
 
@@ -518,9 +582,9 @@
 								</template>
               </el-table-column>
               <el-table-column v-for="(item, index) in chapters" :key="index" :label="item.json.title"
-                :min-width="item.type == 5 ? 200 : 150" align="left" :filters="item.type == 1 ? item.nameFilters : item.type ==  6? item.nameFilters : null"
-                :filter-method="item.type == 1 ? (value, row) => { return filterName(value, row, index) } : item.type == 6?(value, row) => { return filterCourse(value, row, index)}:null"
-                :filter-placement="item.type == 1 || item.type ==6 ? filterPlacement : null">
+                :min-width="item.type == 5 ? 200 : 150" align="left" :filters="item.type == 1 ? item.nameFilters : (item.type ==  6 || item.type ==  11)? item.nameFilters : null"
+                :filter-method="item.type == 1 ? (value, row) => { return filterName(value, row, index) } : (item.type == 6 || item.type == 11)?(value, row) => { return filterCourse(value, row, index)}:null"
+                :filter-placement="item.type == 1 || item.type ==6|| item.type ==11 ? filterPlacement : null">
 
                 <template slot-scope="scope">
                   <div v-if="scope.row.array[index].type == 1" style="display: flex; flex-wrap: wrap;">
@@ -554,6 +618,9 @@
                   <div v-if="scope.row.array[index].type == 7" style="display: flex; flex-wrap: wrap;">
                     {{ getScore(scope.row.array[index].json) }}
                   </div>
+                  <div v-if="scope.row.array[index].type == 11" style="display: flex; flex-wrap: wrap;">
+                    {{ scope.row.array[index].json.answer2}}
+                  </div>
                 </template>
               </el-table-column>
               <el-table-column label="操作" width="300px" fixed="right" v-if="isN!=1">
@@ -684,6 +751,22 @@
                         </div>
                       </div>
                     </div>
+                    <div v-if="k.type.type == 11">
+                      <div class="isDTitle2">
+                        <div style="display: flex;justify-content: space-between;">
+                          <div style="display: flex;">
+                            <span class="topicTitCss">第{{kin+1}}题:</span>
+                            <div class="isDTitle2JsonTit">
+                              {{ k.json.title }}
+                            </div>
+                          </div>
+                          <div style="color: rgba(0, 0, 0, 0.4);">[{{ selectType2(k.type,k.json.type) }}]</div>
+                        </div>
+                        <div class="IsDAnswer">
+                          {{ i.array[kin].json.answer2 }}
+                        </div>
+                      </div>
+                    </div>
                   </div>
 
                 </div>
@@ -979,7 +1062,7 @@ export default {
           return '问答题'
         } else if (item.type == 5) {
           return '附件'
-        } else if (item.type == 6) {
+        } else if (item.type == 6 || item.type == 11) {
           return '课程'
         } else if (item.type == 7) {
           return '评分'
@@ -1077,7 +1160,7 @@ export default {
     },
 		// 筛选课程
 		filterCourse(value, row, index) {
-			return row.array[index].json.answer2 == value;
+			return row.array[index].json.answer2.indexOf(value) != -1;
     },
     wordClickHandler(name, value) {
       //  this.$notify({
@@ -1203,7 +1286,7 @@ export default {
           this.pcount = res.data[3][0].count
           let chapters = this.setJSON(this.setJson2(JSON.parse(JSON.stringify(JSON.parse(res.data[0][0].chapters)))))
           this.chapters = this.setFilter(this.JSONSetting(JSON.parse(JSON.stringify(JSON.parse(res.data[0][0].chapters)))))
-
+          let courseCount11 = []
           let testArray = []
           let array = []
           for (let i = 0; i < this.works.length; i++) {
@@ -1215,7 +1298,25 @@ export default {
                 let _json2 = _json[ja].json
                 if (_json[ja].type == 6) {
                   let _cjson = await this.getCourse(_json2.answer2)
-                  _json2.answer2 =  _cjson ? _cjson.title : ''
+                  _json[ja].json.answer2 =  _cjson ? _cjson.title : ''
+                }
+                if(_json[ja].type == 11){
+                  let _answer = _json2.answer2
+                  _answer.length ? courseCount11 = courseCount11.concat(_answer) : ''
+                  _json[ja].json.courseId = _answer ? _answer : [];
+                  let params = {
+                    cid:_answer.length ? _answer.join(",") : ''
+                  }
+
+                  let data = await this.ajax.get(this.$store.state.api + 'getCourseInfoTestAll',params);
+                  let result = data.data[0]
+                  let _title = []
+                  // this.chapters.find(i=>i.type==6).nameFilters = result.map(r=>{return{text:r.title,value:r.courseId}})
+                  result.forEach(i=>{
+                    _title.push(i.title)
+                  })
+                  
+                  _json[ja].json.answer2 = _title.length ? _title.join(',') : '';
                 }
               }
               array.push({
@@ -1291,6 +1392,29 @@ export default {
                   count: 0
                 })
               }
+            } else if (topic.type == 11) {
+              const uniqueArray = courseCount11.filter((item, index) => courseCount11.indexOf(item) === index);
+              let _answer = uniqueArray
+              topic.array = _answer
+              topic.courseArray = [];
+							let params = {
+								cid:topic.array ? topic.array.join(",") : ''
+							}
+
+							let data = await this.ajax.get(this.$store.state.api + 'getCourseInfoTestAll',params);
+							let result = data.data[0]
+							// this.chapters.find(i=>i.type==6).nameFilters = result.map(r=>{return{text:r.title,value:r.courseId}})
+							result.forEach(i=>{
+								if(this.chapters.find(f=>f.type==11).nameFilters.find(c=>c.value==i.title)==undefined){
+									this.chapters.find(i=>i.type==11).nameFilters.push({text:i.title,value:i.title})
+                  console.log(this.chapters);
+                  
+								}
+							})
+
+							result.forEach(i=>i.update_at=new Date(i.update_at).toLocaleString('zh-CN', {year: 'numeric',month: '2-digit',day: '2-digit',hour: '2-digit',minute: '2-digit',second: '2-digit',hour12: false}).replace(/\//g,'-'))
+							topic.courseArray = result;
+
             }
             for (var j = 0; j < array.length; j++) {
               let el2 = array[j]
@@ -1662,6 +1786,9 @@ export default {
         } else if (item.type == 8 && item.json.answer2) {
           let _option = `<div style='margin:10px 0 0 40px;'>${item.json.answer2}</div>`
           _test += _option
+        } else if (item.type == 11 && item.json.answer2) {
+            let _option = `<div style='margin:10px 0 0 40px;'>打分课程:${item.json.answer2}</div>`
+            _test += _option
         }
       }
       _test += `</div>`
@@ -1708,7 +1835,10 @@ export default {
         } else if (item.type == 8 && item.json.answer2) {
           let _option = `<div style='margin:10px 0 0 40px;'>${item.json.answer2}</div>`
           _test += _option
-        } 
+        } else if (item.type == 11 && item.json.answer2) {
+            let _option = `<div style='margin:10px 0 0 40px;'>打分课程:${item.json.answer2}</div>`
+            _test += _option
+        }
       }
       _test += `</div>`
       let _html = _title + _content + _detail + _test;
@@ -1787,7 +1917,10 @@ export default {
         } else if (item.type == 8 && item.json.answer2) {
           let _option = `<div style='margin:10px 0 0 40px;'>${item.json.answer2}</div>`
           _test += _option
-        }
+        } else if (item.type == 11 && item.json.answer2) {
+            let _option = `<div style='margin:10px 0 0 40px;'>打分课程:${item.json.answer2}</div>`
+            _test += _option
+        } 
       }
       _test += `</div>`
       let _html = _title + _content + _detail + _test;
@@ -1834,6 +1967,24 @@ export default {
         } else if (item.type == 8 && item.json.answer2) {
           let _option = `<div style='margin:10px 0 0 40px;'>${item.json.answer2}</div>`
           _test += _option
+        } else if (item.type == 11 && item.json.answer2 && item.json.answer2.length) {
+            let _answer = item.json.answer2
+            _answer.length ? courseCount11 = courseCount11.concat(_answer) : ''
+            _json[ja].json.courseId = _answer ? _answer : [];
+            let params = {
+              cid:_answer.length ? _answer.join(",") : ''
+            }
+
+            let data = await this.ajax.get(this.$store.state.api + 'getCourseInfoTestAll',params);
+            let result = data.data[0]
+            let _title = []
+            // this.chapters.find(i=>i.type==6).nameFilters = result.map(r=>{return{text:r.title,value:r.courseId}})
+            result.forEach(i=>{
+              _title.push(i.title)
+            })
+            
+            let _option = `<div style='margin:10px 0 0 40px;'>打分课程:${ _title.length ? _title.join(',') : '' }</div>`
+            _test += _option
         }
       }
       _test += `</div>`
@@ -1971,6 +2122,8 @@ export default {
             _json[j + 1 + "." + chapters[j].json.title] = _score
           } else if (res[i].array[j].type == 8 && res[i].array[j].json.answer2) {
             _json[j + 1 + "." + chapters[j].json.title] = res[i].array[j].json.answer2
+          } else if (res[i].array[j].type == 11 && res[i].array[j].json.answer2) {
+            _json[j + 1 + "." + chapters[j].json.title] = res[i].array[j].json.answer2
           }
         }
         array.push(_json);
@@ -2958,8 +3111,10 @@ export default {
 
 .tup>img {
   width: 100%;
-  height: 100%;
-  object-fit: cover;
+  /* height: 100%; */
+  height: 150px;
+  /* object-fit: cover; */
+  object-fit: contain;
 }
 
 .kc_time {

+ 44 - 3
src/components/pages/test/examine/conpoments/personPage.vue

@@ -60,15 +60,25 @@
           :sortable="true"
           :sort-method="sortEvaMethod"
           align="center"
-          label="自评分"
+          label="自评分"
         >
         </el-table-column>
+        <el-table-column align="center" label="一级指标评分" >
+          <template slot-scope="scope">
+            <div class= "scoreBox" v-for="item, index in scope.row.json" :key="index">
+              <el-tooltip :content="item.name" placement="top" effect="dark">
+                <span>{{ item.name }}</span>
+              </el-tooltip>
+               :<span>{{ scoreChildren(item.children) }}</span>
+            </div>
+          </template>
+        </el-table-column>
         <el-table-column
           prop="cogSco"
           :sortable="true"
           :sort-method="sortCogMethod"
           align="center"
-          label="考核分数"
+          label="考核分"
         >
         </el-table-column>
         <el-table-column align="center" label="操作">
@@ -450,7 +460,19 @@ export default {
 
       // 立即调用函数并返回结果
       return accumulateSco2();
-    }
+    },
+
+    scoreChildren() {
+      return function(children){
+        let score = 0;
+        children.forEach(item => {
+          if (item.sco2 && parseFloat(item.sco2) > 0) {
+            score += parseFloat(item.sco2);
+          }
+        });
+        return score;
+      }
+    },
   },
   mounted() {
     this.getData();
@@ -1299,4 +1321,23 @@ input[type="number"] {
 .ColCon {
   flex: 1;
 }
+
+.scoreBox{
+  display: flex;
+  align-items: center;
+}
+
+.scoreBox > span:nth-child(1){
+  max-width: 100px;
+  width: 100px;
+  overflow: hidden;
+  white-space: nowrap;
+  text-overflow: ellipsis;
+  text-align: right;
+}
+
+.scoreBox > span:nth-child(2){
+  margin-left: 10px;
+}
+
 </style>

+ 1 - 1
src/components/pages/testPerson/test/test.vue

@@ -30,7 +30,7 @@
                 <div class="test_panel_title">
                     <div class="title" :style="{paddingLeft: (item.array.length > 0 || item.carray.length > 0) ? '0' : '30px'}" @click="openWork(index)">
                         <span class="open" :class="{active: item.open}" v-if="item.array.length > 0 || item.carray.length > 0"></span>
-                        <el-tooltip :content="item.username" placement="top" effect="dark">
+                        <el-tooltip :content="item.title" placement="top" effect="dark">
                             <span class="titleN">{{ item.title }}</span>
                         </el-tooltip>
                     </div>

+ 77 - 1
src/components/pages/testStudent/view/component/file.vue

@@ -42,6 +42,16 @@
         v-html="checkJson.detail"
         style="color: #00000099;margin-top: 5px;"
       ></div>
+      <div class="choices">
+        <div class="file_box" v-if="checkJson.mobanFile && checkJson.mobanFile.length" v-loading="loading">
+            <div class="file_item" v-for="(item, index) in checkJson.mobanFile" :key="index">
+                <div class="file_name">{{ item.name }}</div>
+                <div class="file_btns">
+                    <div class="file_btn" @click="downloadMobanFile(index)">下载</div>
+                </div>
+            </div>
+        </div>
+      </div>
       <div class="choices">
         <div class="uploadBtn" v-if="checktype == 1">
           <div class="btn" @click.stop="addImg($event)">
@@ -313,7 +323,8 @@ export default {
       wurl: "",
       isloading: false,
       isTong: false,
-      uploadList: []
+      uploadList: [],
+      loading: false
     };
   },
   watch: {
@@ -762,6 +773,29 @@ export default {
         return (bytes / 1073741824).toFixed(2) + "GB";
       }
     },
+    downloadMobanFile(index){
+        this.loading = true;
+        let file = this.checkJson.mobanFile[index]
+        getFile(file.url).then((data) => {
+            this.loading = false;
+            if (data.data != 1) {
+                // 下载文件, 并存成ArrayBuffer对象
+                const file_name =file.name; // 获取文件名
+                const file_data = data.data; // 获取文件数据
+                let url = window.URL.createObjectURL(new Blob([file_data]));
+                let a = document.createElement("a");
+                a.name = file_name;
+                a.href = url;
+                a.download = file_name;
+                a.click();
+                console.log(data); 
+                this.$message.success("下载成功");
+            }else {
+                this.$message.error("下载失败");
+            }
+        });
+
+    },
     downloadFile(f) {
       var credentials = {
         accessKeyId: "AKIATLPEDU37QV5CHLMH",
@@ -1185,4 +1219,46 @@ export default {
 	font-weight: bold;
 	margin-bottom: 10px;
 }
+
+.file_box {
+    /* padding: 0 0 0 15px; */
+    width: 100%;
+    box-sizing: border-box;
+    margin: 10px 0;
+}
+
+.file_item {
+    display: flex;
+    width: 100%;
+    padding: 5px;
+    box-sizing: border-box;
+    border-radius: 5px;
+}
+
+.file_item+.file_item {
+    margin-top: 10px;
+}
+
+.file_name {
+    color: #747474;
+}
+
+.file_item:hover{
+    background: #eee;
+}
+
+.file_btns {
+    margin-left: auto;
+    display: flex;
+    align-items: center;
+}
+
+.file_btn+.file_btn {
+    margin-left: 10px;
+}
+
+.file_btn {
+    color: #3681fc;
+    cursor: pointer;
+}
 </style>

+ 4 - 2
src/components/pages/works.vue

@@ -194,7 +194,8 @@
               min-width="20"
               align="center"
             ></el-table-column> -->
-            <el-table-column align="center" label="操作" width="200px">
+              
+              <el-table-column align="center" min-width="30" label="操作" >
               <template slot-scope="scope">
                 <!-- <el-button
                 type="primary"
@@ -256,7 +257,8 @@
                 }}
               </template>
             </el-table-column>
-            <el-table-column label="操作" min-width="30">
+             
+            <el-table-column label="操作"  align="center" min-width="30">
               <template slot-scope="scope">
                 <el-button
                   type="primary"

Some files were not shown because too many files changed in this diff