449f7391d38bff272e5d928bb945f5f27b668220..38c2fe23d77c9f27d52941dab77a100bb1839dd7
4 天以前 WXL (wul)
测试完成
38c2fe 对比 | 目录
4 天以前 WXL (wul)
测试完成
5f4d48 对比 | 目录
已添加1个文件
已修改10个文件
1968 ■■■■ 文件已修改
src/api/AiCentre/external.js 10 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/followvisit/again/index.vue 5 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/followvisit/discharge/index.vue 5 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/followvisit/discharge/outpatientService.vue 5 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/followvisit/record/detailpage/MergeAndModify.vue 449 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/followvisit/record/detailpage/index.vue 1127 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/followvisit/record/physical/index.vue 1 ●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/followvisit/zbAgain/index.vue 4 ●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/knowledge/questionnaire/compilequer/index.vue 43 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/views/patient/patient/outpatient.vue 317 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
vue.config.js 2 ●●● 补丁 | 查看 | 原始文档 | blame | 历史
src/api/AiCentre/external.js
@@ -92,7 +92,7 @@
    data: data
  });
}
// æ–°å¢žæœåŠ¡
// å†æ¬¡éšè®¿æœåŠ¡
export function addserviceSubtask(data) {
  return request({
    url: "/smartor/serviceSubtask/addSubTaskAgain",
@@ -100,4 +100,12 @@
    data: data
  });
}
// å†æ¬¡éšè®¿æœåŠ¡
export function addSubtask(data) {
  return request({
    url: "/smartor/serviceSubtask/addSubTask",
    method: "post",
    data: data
  });
}
src/views/followvisit/again/index.vue
@@ -1197,8 +1197,9 @@
      });
    },
    affiliation() {
      this.topqueryParams.drcode = store.getters.hisUserId;
      this.topqueryParams.nurseId = store.getters.hisUserId;
      this.topqueryParams.managementDoctor = store.getters.hisUserId;
      this.getList(1);
    },
    onthatday() {
src/views/followvisit/discharge/index.vue
@@ -1408,9 +1408,8 @@
      });
    },
    affiliation() {
      this.topqueryParams.drcode = store.getters.hisUserId;
      this.topqueryParams.nurseId = store.getters.hisUserId;
      this.topqueryParams.managementDoctor = store.getters.name;
      this.topqueryParams.managementDoctor = store.getters.hisUserId;
      this.getList(1);
    },
    onthatday() {
src/views/followvisit/discharge/outpatientService.vue
@@ -1181,8 +1181,9 @@
      });
    },
    affiliation() {
      this.topqueryParams.drcode = store.getters.hisUserId;
      this.topqueryParams.nurseId = store.getters.hisUserId;
      this.topqueryParams.managementDoctor = store.getters.hisUserId;
      this.getList(1);
    },
    onthatday() {
src/views/followvisit/record/detailpage/MergeAndModify.vue
¶Ô±ÈÐÂÎļþ
@@ -0,0 +1,449 @@
<template>
  <div class="merge-questionnaire-container">
    <div class="merge-header">
      <h3>合并编辑问卷 (共 {{ services.length }} ä¸ªæœªå®ŒæˆæœåŠ¡)</h3>
      <el-tag
        v-for="service in services"
        :key="service.id"
        type="info"
        style="margin-right: 10px"
      >
        {{ service.taskName }} ({{ service.sendname }})
      </el-tag>
    </div>
    <div class="merge-content">
      <div class="question-list">
        <div
          v-for="(question, index) in mergedQuestions"
          :key="question.uniqueKey"
          class="question-item"
        >
          <!-- é¢˜ç›®å±•示 - ä¸Žçˆ¶ç»„件保持一致 -->
          <div
            :class="
              question.isabnormal ? 'scriptTopic-isabnormal' : 'scriptTopic-dev'
            "
          >
            <div class="dev-text">
              {{ index + 1 }}、[{{
                getQuestionType(question.scriptType)
              }}]<span> {{ question.scriptContent }}</span>
            </div>
            <!-- å•选 -->
            <div class="dev-xx" v-if="question.scriptType == 1">
              <el-radio-group
                v-model="question.mergedResult"
                @change="
                  handleOptionChange(
                    $event,
                    index,
                    question.svyLibTemplateTargetoptions,
                    question
                  )
                "
                >11
                <el-radio
                  v-for="(
                    option, optIndex
                  ) in question.svyLibTemplateTargetoptions"
                  :key="optIndex"
                  :label="option.optioncontent"
                  :class="option.isabnormal ? 'red-star' : ''"
                >
                  {{ option.optioncontent }}
                </el-radio>
              </el-radio-group>
            </div>
            <!-- å¤šé€‰ -->
            <div class="dev-xx" v-if="question.scriptType == 2">
              <el-checkbox-group
                v-model="question.mergedResult"
                @change="updateScore($event, index, question.options, question)"
              >
                <el-checkbox
                  v-for="(option, optIndex) in question.options"
                  :key="optIndex"
                  :label="option.optioncontent"
                  :class="option.isabnormal ? 'red-star' : ''"
                >
                  {{ option.optioncontent }}
                </el-checkbox>
              </el-checkbox-group>
            </div>
            <!-- å¡«ç©º -->
            <div class="dev-xx" v-if="question.scriptType == 4">
              <el-input
                type="textarea"
                :rows="2"
                placeholder="请输入答案"
                v-model="question.mergedResult"
                clearable
              />
            </div>
            <!-- é™„加输入框 -->
            <div v-if="question.showAppendInput" class="append-input-container">
              <el-input
                type="textarea"
                :rows="2"
                placeholder="请输入具体信息"
                v-model="question.answerps"
                clearable
              />
            </div>
            <!-- æç¤ºä¿¡æ¯ -->
            <div v-show="question.prompt">
              <el-alert :title="question.prompt" type="warning" />
            </div>
          </div>
        </div>
      </div>
    </div>
    <div class="merge-footer">
      <el-button @click="handleCancel">取消</el-button>
      <el-button type="primary" @click="handleSave" :loading="isSaving"
        >保存到所有问卷</el-button
      >
    </div>
  </div>
</template>
<script>
import {
  getsearchrResults,
  serviceSubtaskDetailedit,
  serviceSubtaskDetailadd,
  Editsingletaskson,
  getTaskservelist,
  alterpatient,
} from "@/api/AiCentre/index";
export default {
  props: {
    selectedServices: {
      type: Array,
      required: true,
    },
    patid: {
      type: [String, Number],
      required: true,
    },
  },
  data() {
    return {
      services: [],
      mergedQuestions: [], // åˆå¹¶åŽçš„问题数据
      isSaving: false,
    };
  },
  created() {
    this.loadServicesData();
  },
  methods: {
    getQuestionType(type) {
      const types = { 1: "单选", 2: "多选", 4: "问答" };
      return types[type] || "未知";
    },
    async loadServicesData() {
      this.services = this.selectedServices;
      const loading = this.$loading({
        lock: true,
        text: "正在加载未完成问卷数据...",
        spinner: "el-icon-loading",
        background: "rgba(0, 0, 0, 0.7)",
      });
      try {
        // åªåŠ è½½æœªå®Œæˆçš„æœåŠ¡é—®å·æ•°æ®
        const requests = this.services.map((service) =>
          getsearchrResults({
            taskid: service.taskid,
            patid: this.patid,
            subId: service.id,
            isFinish: false, // ç¡®ä¿åªèŽ·å–æœªå®Œæˆçš„é—®å·
          })
        );
        const responses = await Promise.all(requests);
        // åˆå¹¶æ‰€æœ‰é—®é¢˜ï¼ŒåŽ»é‡
        const allQuestions = [];
        const questionMap = {};
        responses.forEach((response, index) => {
          if (response.code == 200 && response.data.scriptResult) {
            response.data.scriptResult.forEach((q) => {
              const key = `${q.scriptContent}_${q.scriptType}`;
              if (!questionMap[key]) {
                questionMap[key] = {
                  ...q,
                  uniqueKey: key,
                  mergedResult: q.scriptType == 2 ? [] : null,
                  originalServices: [], // è®°å½•原始服务ID
                };
                allQuestions.push(questionMap[key]);
              }
              questionMap[key].originalServices.push(this.services[index].id);
            });
          }
        });
        // åˆå§‹åŒ–合并结果
        this.mergedQuestions = allQuestions.map((q) => ({
          ...q,
          mergedResult: q.scriptType == 2 ? [] : q.scriptResult || null,
        }));
        console.log(this.mergedQuestions);
      } catch (error) {
        this.$message.error("加载问卷数据失败: " + error.message);
      } finally {
        loading.close();
      }
    },
    // å¤„理选项变化 (与父组件保持一致)
    handleOptionChange(selectedOption, questionIndex, options, question) {
      const selectedOptionObj = options.find(
        (item) => item.optioncontent == selectedOption
      );
      // è®¾ç½®å¼‚常状态
      question.isabnormal = !!selectedOptionObj?.isabnormal;
      // å¤„理附加输入框显示
      question.showAppendInput = selectedOptionObj?.appendflag == 1;
      if (!question.showAppendInput) {
        question.answerps = "";
      }
      this.$forceUpdate();
    },
    // æ›´æ–°å¤šé€‰åˆ†æ•° (与父组件保持一致)
    updateScore(selectedValues, questionIndex, options, question) {
      const abnormalOptions = options.filter((opt) => opt.isabnormal);
      question.isabnormal = abnormalOptions.some((opt) =>
        selectedValues.includes(opt.optioncontent)
      );
      this.$forceUpdate();
    },
    handleCancel() {
      this.$emit("cancel");
    },
    async handleSave() {
      this.isSaving = true;
      try {
        const saveResults = [];
        const updateServicePromises = [];
        // 1. ä¿å­˜æ‰€æœ‰é—®å·é—®é¢˜
        for (const service of this.services) {
          const serviceId = service.id;
          const questionsToSave = this.mergedQuestions
            .filter((q) => q.originalServices.includes(serviceId))
            .map((question) => ({
              scriptid: question.id,
              scriptResultId: question.scriptResultId,
              scriptType: question.scriptType,
              questiontext: question.scriptContent,
              asrtext:
                question.scriptType == 2
                  ? question.mergedResult.join("&")
                  : question.mergedResult,
              answerps: question.answerps || null,
              isabnormal: question.isabnormal || false,
            }));
          // ä¿å­˜é—®é¢˜æ•°æ®
          for (const question of questionsToSave) {
            const saveData = {
              taskid: service.taskid,
              patid: this.patid,
              subId: serviceId,
              ...question,
            };
            if (saveData.isabnormal) {
              saveData.excep = 1;
            }
            try {
              const response = question.scriptResultId
                ? await serviceSubtaskDetailedit(saveData)
                : await serviceSubtaskDetailadd(saveData);
              saveResults.push({
                serviceId,
                success: response.code === 200,
                message:
                  response.message ||
                  (response.code === 200 ? "保存成功" : "保存失败"),
              });
            } catch (error) {
              saveResults.push({
                serviceId,
                success: false,
                message: error.message || "保存失败",
              });
            }
          }
          // 2. æ›´æ–°æœåŠ¡çŠ¶æ€ä¸ºå·²å®Œæˆ (sendstate = 6)
          updateServicePromises.push(this.updateServiceStatus(serviceId));
        }
        // ç­‰å¾…所有服务状态更新完成
        const updateResults = await Promise.all(updateServicePromises);
        updateResults.forEach((result) => {
          if (!result.success) {
            saveResults.push({
              serviceId: result.serviceId,
              success: false,
              message: result.message || "服务状态更新失败",
            });
          }
        });
        // ç»Ÿè®¡ç»“æžœ
        const successCount = saveResults.filter((r) => r.success).length;
        const totalCount = saveResults.length;
        // é€šçŸ¥çˆ¶ç»„ä»¶
        this.$emit("save", {
          successCount,
          totalCount,
          results: saveResults,
        });
        if (successCount === totalCount) {
          this.$message.success("所有问卷和服务状态更新成功");
        } else {
          this.$message.warning(
            `成功保存 ${successCount} é¡¹ï¼Œå¤±è´¥ ${totalCount - successCount} é¡¹`
          );
        }
      } catch (error) {
        this.$message.error("保存过程中发生错误: " + error.message);
      } finally {
        this.isSaving = false;
      }
    },
    // æ–°å¢žæ–¹æ³•:更新服务状态
    async updateServiceStatus(serviceId) {
      try {
        // èŽ·å–æœåŠ¡å½“å‰æ•°æ®
        const res = await getTaskservelist({
          patid: this.patid,
          subId: serviceId,
        });
        if (res.code === 200) {
          const serviceData = res.rows[0].serviceSubtaskList.find(
            (item) => item.id === serviceId
          );
          if (serviceData) {
            // æ›´æ–°æœåŠ¡çŠ¶æ€ä¸ºå·²å®Œæˆ (sendstate = 6)
            const updateRes = await Editsingletaskson({
              ...serviceData,
              sendstate: 6, // è®¾ç½®ä¸ºå·²å®ŒæˆçŠ¶æ€
              remark: "通过合并编辑完成", // å¯é€‰ï¼šæ·»åŠ å¤‡æ³¨
            });
            return {
              serviceId,
              success: updateRes.code === 200,
              message: updateRes.message || "服务状态更新成功",
            };
          }
        }
        return {
          serviceId,
          success: false,
          message: "获取服务数据失败",
        };
      } catch (error) {
        return {
          serviceId,
          success: false,
          message: error.message || "更新服务状态失败",
        };
      }
    },
  },
};
</script>
<style scoped>
.merge-questionnaire-container {
  display: flex;
  flex-direction: column;
  height: 100%;
  padding: 20px;
}
.merge-header {
  margin-bottom: 20px;
}
.merge-content {
  flex: 1;
  overflow-y: auto;
}
.question-list {
  padding: 10px;
}
/* ä¸Žçˆ¶ç»„件一致的题目样式 */
.scriptTopic-dev {
  margin-bottom: 20px;
  padding: 15px;
  border: 1px solid #ebeef5;
  border-radius: 4px;
}
.scriptTopic-isabnormal {
  margin-bottom: 20px;
  padding: 15px;
  border: 1px solid #f56c6c;
  border-radius: 4px;
  background-color: #fff6f6;
}
.dev-text {
  font-size: 16px;
  margin-bottom: 15px;
  color: #333;
}
.dev-xx {
  margin-left: 20px;
}
.append-input-container {
  margin-top: 15px;
}
.red-star {
  color: #f56c6c;
}
.merge-footer {
  margin-top: 20px;
  text-align: right;
  padding-top: 15px;
  border-top: 1px solid #ebeef5;
}
</style>
src/views/followvisit/record/detailpage/index.vue
@@ -27,6 +27,23 @@
                å‰å¾€CDSS查询
              </el-link>
            </div>
            <div class="merge-controls" v-if="Whetherall">
              <el-button
                type="primary"
                @click="toggleMergeMode"
                :disabled="selectedServices.length < 2"
              >
                {{ isMergeMode ? "取消合并" : "合并编辑问卷" }}
              </el-button>
              <el-button
                v-if="isMergeMode"
                type="success"
                @click="openMergeDialog"
                :disabled="selectedServices.length < 2"
              >
                å¼€å§‹åˆå¹¶ (已选 {{ selectedServices.length }} ä¸ªæœåŠ¡)
              </el-button>
            </div>
          </div>
          <!-- <el-button type="success">随访后短信</el-button> -->
        </div>
@@ -36,7 +53,14 @@
          :data="logsheetlist"
          :row-class-name="tableRowClassName"
          style="width: 100%"
          @selection-change="handleSelectionChange"
        >
          <el-table-column
            type="selection"
            width="55"
            :selectable="checkSelectable"
            v-if="Whetherall"
          ></el-table-column>
          <el-table-column
            prop="sendname"
            align="center"
@@ -213,118 +237,427 @@
        </el-table>
      </div>
    </div>
    <div class="Followuserinfos">
      <div>
        <el-form
          ref="userform"
          :model="form"
          :rules="userrules"
          label-width="120px"
        >
    <!-- æ·»åŠ åˆå¹¶ç¼–è¾‘å¯¹è¯æ¡† -->
    <el-dialog
      title="合并编辑问卷"
      :visible.sync="mergeDialogVisible"
      width="80%"
      top="5vh"
      v-dialogDrag
    >
      <MergeAndModify
        v-if="mergeDialogVisible"
        :selected-services="selectedServices"
        :patid="patid"
        @save="handleMergeSave"
        @cancel="mergeDialogVisible = false"
      />
    </el-dialog>
    <div class="action-container">
      <div class="call-action">
        <div class="call-container">
          <!-- <div class="call-header">
            <h2>一键呼叫功能</h2>
          </div> -->
          <div class="headline">
            <div>人工处理</div>
            <el-row :gutter="20">
              <el-col :span="8"
                ><el-form-item label="联系电话">
                  <el-input
                    placeholder="联系电话缺失"
                    v-model="userform.telcode"
                  >
                    <el-button
                      slot="append"
                      icon="el-icon-phone"
                      @click="handleCall(userform.telcode, 'tel')"
                      :disabled="!isValidPhone(userform.telcode)"
                    ></el-button
                  ></el-input> </el-form-item
              ></el-col>
              <el-col :span="8"
                ><el-form-item label="联系人电话">
                  <el-input
                    placeholder="联系人电话缺失"
                    v-model="userform.relativetelcode"
                  >
                    <el-button
                      slot="append"
                      icon="el-icon-phone"
                      @click="handleCall(userform.relativetelcode, 'relative')"
                      :disabled="!isValidPhone(userform.relativetelcode)"
                    ></el-button
                  ></el-input> </el-form-item
              ></el-col>
              <el-col :span="8"
                ><el-form-item label="联系人关系">
                  <el-input
                    placeholder="联系人关系缺失"
                    v-model="userform.relation"
                  ></el-input> </el-form-item
              ></el-col>
            </el-row>
            <div style="margin-left: 30px">
              <el-button type="primary" plain @click="Editsingletasksonyic('')"
                >保存服务</el-button
              >
            </div>
            <div>随访内容</div>
          </div>
          <el-row :gutter="20" v-if="callStatus !== 'idle'">
            <el-col :span="24">
              <el-alert
                :title="callStatusText"
                :type="callStatusType"
                :closable="false"
                show-icon
              />
            </el-col>
          </el-row>
          <div>
            <el-tabs v-model="activeName" type="border-card">
              <el-tab-pane name="wj">
                <span class="mulsz" slot="label"
                  ><i class="el-icon-notebook-1"></i> é—®å·éšè®¿ç»“æžœ</span
                >
                <div class="CONTENT">
                  <div class="title">{{ taskname ? taskname : "问卷" }}</div>
          <!-- æŒ‚断按钮(仅在通话中显示) -->
          <el-row :gutter="20" v-if="callStatus === 'connected'">
            <el-col :span="24" style="text-align: center; margin-top: 10px">
              <el-button
                type="danger"
                icon="el-icon-phone"
                @click="endCurrentCall"
                :loading="isEndingCall"
              >
                æŒ‚断电话
              </el-button>
            </el-col>
          </el-row>
          <el-form-item label="随访记录">
            <el-input type="textarea" v-model="form.remark"></el-input>
          </el-form-item>
                  <div class="preview-left" v-if="!Voicetype">
                    <div
                      class="topic-dev"
                      v-for="(item, index) in tableDatatop"
                      :key="item.id"
                    >
                      <!-- å•选 -->
                      <div
                        :class="
                          item.isabnormal
                            ? 'scriptTopic-isabnormal'
                            : 'scriptTopic-dev'
                        "
                        :key="index"
                        v-if="item.scriptType == 1 && !item.astrict"
                      >
                        <div class="dev-text">
                          {{ index + 1 }}、[单选]<span>{{
                            item.scriptContent
                          }}</span>
                        </div>
                        <div class="dev-xx">
                          <el-radio-group
                            v-model="item.scriptResult"
                            @change="
                              handleOptionChange(
                                $event,
                                index,
                                item.svyLibTemplateTargetoptions,
                                item
                              )
                            "
                          >
                            <el-radio
                              v-for="(
                                items, indexs
                              ) in item.svyLibTemplateTargetoptions"
                              :class="items.isabnormal ? 'red-star' : ''"
                              :key="indexs"
                              :label="items.optioncontent"
                              >{{ items.optioncontent }}</el-radio
                            >
                          </el-radio-group>
                        </div>
                        <div
                          v-if="item.showAppendInput||item.answerps"
                          class="append-input-container"
                        >
                          <el-input
                            type="textarea"
                            :rows="2"
                            placeholder="请输入具体信息"
                            v-model="item.answerps"
                            clearable
                          ></el-input>
                        </div>
                        <div v-show="item.prompt">
                          <el-alert :title="item.prompt" type="warning">
                          </el-alert>
                        </div>
                      </div>
                      <!-- å¤šé€‰ -->
                      <div
                        :class="
                          item.isabnormal
                            ? 'scriptTopic-isabnormal'
                            : 'scriptTopic-dev'
                        "
                        :key="index"
                        v-if="item.scriptType == 2 && !item.astrict"
                      >
                        <div class="dev-text">
                          {{ index + 1 }}、[多选]<span>{{
                            item.scriptContent
                          }}</span>
                        </div>
                        <div class="dev-xx">
                          <el-checkbox-group
                            v-model="item.scriptResult"
                            @change="updateScore($event, index, item)"
                          >
                            <el-checkbox
                              :class="items.isabnormal ? 'red-star' : ''"
                              @change="$forceUpdate()"
                              v-for="(
                                items, indexs
                              ) in item.svyLibTemplateTargetoptions"
                              :key="indexs"
                              :label="items.optioncontent"
                            >
                              {{ items.optioncontent }}
                            </el-checkbox>
                          </el-checkbox-group>
                        </div>
                        <div v-show="item.prompt && item.scriptResult[0]">
                          <el-alert :title="item.prompt" type="warning">
                          </el-alert>
                        </div>
                      </div>
                      <!-- å¡«ç©º -->
                      <div
                        class="scriptTopic-dev"
                        :key="index"
                        v-if="item.scriptType == 4 && !item.astrict"
                      >
                        <div class="dev-text">
                          {{ index + 1 }}、[问答]<span>{{
                            item.scriptContent
                          }}</span>
                        </div>
                        <div class="dev-xx">
                          <el-input
                            type="textarea"
                            :rows="2"
                            placeholder="请输入答案"
                            v-model="item.scriptResult"
                            clearable
                          >
                          </el-input>
                        </div>
                      </div>
                    </div>
                  </div>
          <el-form-item label="处理意见">
            <div>
              <el-button plain type="warning" @click="Editsingletaskson('1')"
                >暂不处理</el-button
              >
              <el-button plain type="success" @click="Editsingletaskson('2')"
                >病情稳定</el-button
              >
              <el-button plain type="primary" @click="Editsingletaskson('3')"
                >通知就诊</el-button
              >
              <!-- <el-button type="danger" @click="Editsingletaskson('4')"
    >失访</el-button
  > -->
              <el-button plain type="info" @click="Editsingletaskson('5')"
                >中心随访</el-button
              >
              <el-button
                type="primary"
                round
                v-if="this.form.isVisitAgain != 2"
                @click="sendAgain()"
                >再次随访</el-button
              >
            </div>
          </el-form-item>
        </el-form>
        <el-collapse>
          <el-collapse-item title="查看当前患者信息" name="1">
                  <div class="preview-left" v-else>
                    <div
                      class="topic-dev"
                      v-for="(item, index) in tableDatatop"
                      :key="item.id"
                    >
                      <div v-if="item.targetvalue">
                        <div class="dev-text">
                          {{ index + 1 }}、[单选]<span>{{
                            item.questiontext
                          }}</span>
                        </div>
                        <div class="dev-xx">
                          <el-radio-group
                            v-model="item.matchedtext"
                            @change="
                              handleOptionChange(
                                $event,
                                index,
                                item.ivrTaskScriptTargetoptionList,
                                item
                              )
                            "
                          >
                            <el-radio
                              v-for="(items, index) in item.scriptResult"
                              :key="items"
                              :label="items"
                              >{{ items }}</el-radio
                            >
                          </el-radio-group>
                        </div>
                        <div v-show="item.prompt">
                          <el-alert :title="item.prompt" type="warning">
                          </el-alert>
                        </div>
                      </div>
                      <div class="scriptTopic-dev" :key="index" v-else>
                        <div class="dev-text">
                          {{ index + 1 }}、[问答]<span>{{
                            item.questiontext
                          }}</span>
                        </div>
                        <div class="dev-xx">
                          <el-input
                            type="textarea"
                            :rows="2"
                            placeholder="请输入答案"
                            v-model="item.matchedtext"
                            clearable
                          >
                          </el-input>
                        </div>
                      </div>
                    </div>
                  </div>
                  <el-button
                    v-if="Voicetype"
                    type="primary"
                    @click="yuyingetdetail"
                    >保存服务详情</el-button
                  >
                  <el-button v-else type="primary" @click="getdetail"
                    >保存服务详情</el-button
                  >
                </div>
              </el-tab-pane>
              <el-tab-pane name="yy">
                <span class="mulsz" slot="label"
                  ><i class="el-icon-headset"></i> è¯­éŸ³éšè®¿è¯¦æƒ…</span
                >
                <div class="borderdiv">
                  <div class="title">{{ taskname ? taskname : "问卷" }}</div>
                  <div
                    style="
                      display: flex;
                      text-align: center;
                      align-items: center;
                      color: #59a0f0;
                    "
                  >
                    å®Œæ•´è¯­éŸ³ï¼š
                    <mini-audio
                      :audio-source="
                        voice ? voice : '@assets/order/example.mp3'
                      "
                    ></mini-audio>
                  </div>
                  <div class="preview-left">
                    <div v-for="item in voiceDatatop">
                      <div class="leftside">
                        <i class="el-icon-phone-outline"></i
                        ><span>{{ item.questiontext }}</span>
                      </div>
                      <div class="offside">
                        <i class="el-icon-user"></i>
                        <div class="offside-value">
                          <el-input
                            type="textarea"
                            :autosize="{ minRows: 1 }"
                            v-model="item.asrtext"
                          ></el-input>
                          <div>
                            <mini-audio
                              :audio-source="
                                item.questionvoice
                                  ? item.questionvoice
                                  : '@assets/order/example.mp3'
                              "
                            ></mini-audio>
                          </div>
                        </div>
                      </div>
                    </div>
                  </div>
                  <el-button
                    v-if="Voicetype"
                    type="primary"
                    @click="yuyingetdetail"
                    >保存随访详情</el-button
                  >
                  <el-button v-else type="primary" @click="getdetail"
                    >保存随访详情</el-button
                  >
                </div>
              </el-tab-pane>
            </el-tabs>
          </div>
        </div>
      </div>
      <div class="manual-action">
        <div class="Followuserinfos">
          <div>
            <el-form
              ref="userform"
              :model="form"
              :rules="userrules"
              label-width="120px"
            >
              <div class="headline">
                <div>人工处理</div>
                <div style="margin: 0 30px">
                  <el-button
                    type="primary"
                    plain
                    @click="Editsingletasksonyic('')"
                    >保存基础信息</el-button
                  >
                </div>
                <div>
                  <el-button
                    type="primary"
                    round
                    v-if="this.form.isVisitAgain != 2"
                    @click="sendAgain()"
                    >再次随访</el-button
                  >
                </div>
              </div>
              <el-row>
                <el-col :span="14"
                  ><el-form-item label="联系电话">
                    <el-input
                      placeholder="联系电话缺失"
                      v-model="userform.telcode"
                    >
                      <el-button
                        slot="append"
                        icon="el-icon-phone"
                        @click="handleCall(userform.telcode, 'tel')"
                        :disabled="!isValidPhone(userform.telcode)"
                      ></el-button
                    ></el-input> </el-form-item
                ></el-col>
              </el-row>
              <el-row>
                <el-col :span="14"
                  ><el-form-item label="联系人电话">
                    <el-input
                      placeholder="联系人电话缺失"
                      v-model="userform.relativetelcode"
                    >
                      <el-button
                        slot="append"
                        icon="el-icon-phone"
                        @click="
                          handleCall(userform.relativetelcode, 'relative')
                        "
                        :disabled="!isValidPhone(userform.relativetelcode)"
                      ></el-button
                    ></el-input> </el-form-item
                ></el-col>
                <el-col :span="10"
                  ><el-form-item label="联系人关系">
                    <el-input
                      placeholder="联系人关系缺失"
                      v-model="userform.relation"
                    ></el-input> </el-form-item
                ></el-col>
              </el-row>
              <div class="call-controls">
                <CallButton
                  ref="callButton"
                  :phoneNumber="currentPhoneNumber"
                  style="display: none"
                />
                <div v-if="callStatus === 'connected'" class="hangup-btn">
                  <el-button
                    type="danger"
                    icon="el-icon-phone"
                    @click="endCurrentCall"
                    :loading="isEndingCall"
                  >
                    æŒ‚断电话
                  </el-button>
                </div>
                <div class="call-status" v-if="callStatus !== 'idle'">
                  <el-alert
                    :title="callStatusText"
                    :type="callStatusType"
                    :closable="false"
                    show-icon
                  />
                </div>
              </div>
              <el-form-item label="随访记录">
                <el-input type="textarea" v-model="form.remark"></el-input>
              </el-form-item>
              <el-form-item label="处理意见">
                <div>
                  <el-button
                    plain
                    type="warning"
                    @click="Editsingletaskson('1')"
                    >暂不处理</el-button
                  >
                  <el-button
                    plain
                    type="success"
                    @click="Editsingletaskson('2')"
                    >病情稳定</el-button
                  >
                  <el-button
                    plain
                    type="primary"
                    @click="Editsingletaskson('3')"
                    >通知就诊</el-button
                  >
                  <el-button plain type="info" @click="Editsingletaskson('5')"
                    >中心随访</el-button
                  >
                </div>
              </el-form-item>
            </el-form>
            <div class="detailed">
              <h3>患者档案信息</h3>
              <el-form ref="userform" :model="userform" label-width="100px">
                <el-row :gutter="20">
                  <el-col :span="12">
@@ -374,261 +707,13 @@
                      /> </el-form-item
                  ></el-col>
                </el-row>
                <!-- <el-row :gutter="20">
                  <el-col :span="24">
                    <el-form-item label="标签" prop="desc">
                      <div class="xinz-inf">
                        <el-tag
                          :key="tag.tagname"
                          type="success"
                          v-for="tag in dynamicTags"
                          :disable-transitions="false"
                        >
                          {{ tag.tagname }}
                        </el-tag>
                      </div>
                    </el-form-item>
                  </el-col>
                </el-row> -->
              </el-form>
            </div>
          </el-collapse-item>
        </el-collapse>
          </div>
        </div>
      </div>
    </div>
    <div>
      <h2>一键呼叫功能</h2>
      <CallButton
        ref="callButton"
        :phoneNumber="currentPhoneNumber"
        style="display: none"
      />
    </div>
    <div>
      <el-tabs v-model="activeName" type="border-card">
        <el-tab-pane name="wj">
          <span class="mulsz" slot="label"
            ><i class="el-icon-notebook-1"></i> é—®å·éšè®¿ç»“æžœ</span
          >
          <div class="CONTENT">
            <div class="title">{{ taskname ? taskname : "问卷" }}</div>
            <div class="preview-left" v-if="!Voicetype">
              <div
                class="topic-dev"
                v-for="(item, index) in tableDatatop"
                :key="item.id"
              >
                <!-- å•选 -->
                <div
                  :class="
                    item.isabnormal
                      ? 'scriptTopic-isabnormal'
                      : 'scriptTopic-dev'
                  "
                  :key="index"
                  v-if="item.scriptType == 1 && !item.astrict"
                >
                  <div class="dev-text">
                    {{ index + 1 }}、[单选]<span>{{ item.scriptContent }}</span>
                  </div>
                  <div class="dev-xx">
                    <el-radio-group
                      v-model="item.scriptResult"
                      @change="
                        handleOptionChange(
                          $event,
                          index,
                          item.svyLibTemplateTargetoptions
                        )
                      "
                    >
                      <el-radio
                        v-for="(
                          items, indexs
                        ) in item.svyLibTemplateTargetoptions"
                        :class="items.isabnormal ? 'red-star' : ''"
                        :key="indexs"
                        :label="items.optioncontent"
                        >{{ items.optioncontent }}</el-radio
                      >
                    </el-radio-group>
                  </div>
                  <div v-show="item.prompt">
                    <el-alert :title="item.prompt" type="warning"> </el-alert>
                  </div>
                </div>
                <!-- å¤šé€‰ -->
                <div
                  :class="
                    item.isabnormal
                      ? 'scriptTopic-isabnormal'
                      : 'scriptTopic-dev'
                  "
                  :key="index"
                  v-if="item.scriptType == 2 && !item.astrict"
                >
                  <div class="dev-text">
                    {{ index + 1 }}、[多选]<span>{{ item.scriptContent }}</span>
                  </div>
                  <div class="dev-xx">
                    <el-checkbox-group
                      v-model="item.scriptResult"
                      @change="updateScore($event, index, item)"
                    >
                      <el-checkbox
                        :class="items.isabnormal ? 'red-star' : ''"
                        @change="$forceUpdate()"
                        v-for="(
                          items, indexs
                        ) in item.svyLibTemplateTargetoptions"
                        :key="indexs"
                        :label="items.optioncontent"
                      >
                        {{ items.optioncontent }}
                      </el-checkbox>
                    </el-checkbox-group>
                  </div>
                  <div v-show="item.prompt && item.scriptResult[0]">
                    <el-alert :title="item.prompt" type="warning"> </el-alert>
                  </div>
                </div>
                <!-- å¡«ç©º -->
                <div
                  class="scriptTopic-dev"
                  :key="index"
                  v-if="item.scriptType == 4 && !item.astrict"
                >
                  <div class="dev-text">
                    {{ index + 1 }}、[问答]<span>{{ item.scriptContent }}</span>
                  </div>
                  <div class="dev-xx">
                    <el-input
                      type="textarea"
                      :rows="2"
                      placeholder="请输入答案"
                      v-model="item.scriptResult"
                      clearable
                    >
                    </el-input>
                  </div>
                </div>
              </div>
            </div>
            <div class="preview-left" v-else>
              <div
                class="topic-dev"
                v-for="(item, index) in tableDatatop"
                :key="item.id"
              >
                <div v-if="item.targetvalue">
                  <div class="dev-text">
                    {{ index + 1 }}、[单选]<span>{{ item.questiontext }}</span>
                  </div>
                  <div class="dev-xx">
                    <el-radio-group
                      v-model="item.matchedtext"
                      @change="handleOptionChange($event, index, item)"
                    >
                      <el-radio
                        v-for="(items, index) in item.scriptResult"
                        :key="items"
                        :label="items"
                        >{{ items }}</el-radio
                      >
                    </el-radio-group>
                  </div>
                  <div v-show="item.prompt">
                    <el-alert :title="item.prompt" type="warning"> </el-alert>
                  </div>
                </div>
                <div class="scriptTopic-dev" :key="index" v-else>
                  <div class="dev-text">
                    {{ index + 1 }}、[问答]<span>{{ item.questiontext }}</span>
                  </div>
                  <div class="dev-xx">
                    <el-input
                      type="textarea"
                      :rows="2"
                      placeholder="请输入答案"
                      v-model="item.matchedtext"
                      clearable
                    >
                    </el-input>
                  </div>
                </div>
              </div>
            </div>
            <el-button v-if="Voicetype" type="primary" @click="yuyingetdetail"
              >保存服务详情</el-button
            >
            <el-button v-else type="primary" @click="getdetail"
              >保存服务详情</el-button
            >
          </div>
        </el-tab-pane>
        <el-tab-pane name="yy">
          <span class="mulsz" slot="label"
            ><i class="el-icon-headset"></i> è¯­éŸ³éšè®¿è¯¦æƒ…</span
          >
          <div class="borderdiv">
            <div class="title">{{ taskname ? taskname : "问卷" }}</div>
            <div
              style="
                display: flex;
                text-align: center;
                align-items: center;
                color: #59a0f0;
              "
            >
              å®Œæ•´è¯­éŸ³ï¼š
              <mini-audio
                :audio-source="
                  voice ? voice : '@assets/order/example.mp3'
                "
              ></mini-audio>
            </div>
            <div class="preview-left">
              <div v-for="item in voiceDatatop">
                <div class="leftside">
                  <i class="el-icon-phone-outline"></i
                  ><span>{{ item.questiontext }}</span>
                </div>
                <div class="offside">
                  <i class="el-icon-user"></i>
                  <div class="offside-value">
                    <el-input
                      type="textarea"
                      :autosize="{ minRows: 1 }"
                      v-model="item.asrtext"
                    ></el-input>
                    <div>
                      <mini-audio
                        :audio-source="
                          item.questionvoice
                            ? item.questionvoice
                            : '@assets/order/example.mp3'
                        "
                      ></mini-audio>
                    </div>
                  </div>
                </div>
              </div>
            </div>
            <el-button v-if="Voicetype" type="primary" @click="yuyingetdetail"
              >保存随访详情</el-button
            >
            <el-button v-else type="primary" @click="getdetail"
              >保存随访详情</el-button
            >
          </div>
        </el-tab-pane>
      </el-tabs>
    </div>
    <el-dialog
      title="患者再次随访"
      v-dialogDrags
@@ -797,9 +882,11 @@
  listcontactinformation,
} from "@/api/patient/homepage";
import CallButton from "@/components/CallButton";
import MergeAndModify from "./MergeAndModify.vue";
export default {
  components: {
    CallButton,
    MergeAndModify,
  },
  dicts: ["sys_normal_disable", "sys_user_sex", "sys_yujing", "sys_suggest"],
@@ -836,6 +923,9 @@
      tableDatatop: [], //题目表
      voiceDatatop: [], //题目表
      dynamicTags: [],
      isMergeMode: false,
      mergeDialogVisible: false,
      selectedServices: [], // é€‰ä¸­çš„æœåŠ¡åˆ—è¡¨
      zcrules: {
        resource: [
          { required: true, message: "请选择随访方式", trigger: "change" },
@@ -1070,7 +1160,6 @@
      const queryParams = {
        pid: Number(this.patid),
        allhosp: "0",
        pageNum: 1,
      };
      // æ‚£è€…基础信息
      messagelistpatient(queryParams).then((response) => {
@@ -1193,6 +1282,7 @@
          scriptid: item.id,
          excep: excep,
          questiontext: item.scriptContent,
          answerps: item.answerps || null, // æ·»åŠ é™„åŠ ä¿¡æ¯
        };
        if (item.scriptType == 2 && item.scriptResult[0]) {
          obj.asrtext = item.scriptResult.join("&");
@@ -1239,7 +1329,7 @@
            })
            .catch(() => {
              if (this.form.serviceType == 13) {
                if (this.visitCount!=1) {
                if (this.visitCount != 1) {
                  this.$router.push({
                    path: "/logisticsservice/zbAgain",
                  });
@@ -1249,7 +1339,7 @@
                  });
                }
              } else if (this.form.serviceType == 2) {
                if (this.visitCount!=1) {
                if (this.visitCount != 1) {
                  this.$router.push({
                    path: "/logisticsservice/again",
                  });
@@ -1266,6 +1356,7 @@
          console.error("发生错误:", error);
        });
    },
    // ç”µè¯============================
    // éªŒè¯ç”µè¯å·ç æ ¼å¼å¹¶è¿”回错误信息
    validatePhoneNumber(phone) {
      if (!phone) {
@@ -1297,7 +1388,6 @@
        };
      }
    },
    // ä½¿ç”¨ç¤ºä¾‹
    isValidPhone(phone) {
      return this.validatePhoneNumber(phone).isValid;
@@ -1321,7 +1411,6 @@
        });
      });
    },
    // å¤„理通话状态变化
    handleCallStatusChange(status) {
      console.log(status, "status");
@@ -1343,7 +1432,6 @@
        this.$message.error(`呼叫失败: ${status.text}`);
      }
    },
    // ç»“束当前通话
    endCurrentCall() {
      if (!this.currentCall) return;
@@ -1549,7 +1637,7 @@
        })
        .catch(() => {});
    },
    handleOptionChange(a, b, c) {
    aahandleOptionChange(a, b, c) {
      const result = c.find((item) => item.optioncontent == a);
      if (result.nextQuestion == 0) {
        this.tableDatatop = this.tableDatatop.reduce((acc, item, i) => {
@@ -1576,6 +1664,98 @@
      } else {
        this.tableDatatop[b].isabnormal = false;
      }
      this.$forceUpdate();
    },
    // åœ¨methods部分,修改handleOptionChange方法:
    handleOptionChange(selectedOption, questionIndex, options, a) {
      if (document.activeElement) {
        document.activeElement.blur();
      }
      // æ‰¾åˆ°è¢«é€‰ä¸­çš„选项对象
      const selectedOptionObj = options.find(
        (item) => item.optioncontent == selectedOption
      );
      // å¤„理异常状态高亮
      this.tableDatatop[questionIndex].isabnormal =
        !!selectedOptionObj.isabnormal;
      // å¤„理附加输入框显示
      this.tableDatatop[questionIndex].showAppendInput =
        selectedOptionObj.appendflag == 1;
      console.log(this.tableDatatop);
      // if (!this.tableDatatop[questionIndex].showAppendInput) {
      //   this.tableDatatop[questionIndex].answerps = ""; // æ¸…除附加信息
      // }
      // ä¿å­˜å½“前题目之前已经隐藏的题目状态
      const previouslyHiddenBeforeCurrent = this.tableDatatop
        .slice(0, questionIndex)
        .map((item, index) => (item.astrict ? index : -1))
        .filter((index) => index !== -1);
      // ä¿å­˜ä¹‹å‰å› nextQuestion=0而隐藏的题目范围
      const previouslyHiddenByEnd = this.tableDatatop
        .map((item, index) => (item.hiddenByEnd ? index : -1))
        .filter((index) => index !== -1);
      // å¦‚æžœbranchFlag为1,处理题目跳转
      if (a.branchFlag == 1) {
        if (selectedOptionObj.nextQuestion == 0) {
          // ç»“束问答 - éšè—åŽé¢æ‰€æœ‰é¢˜ç›®å¹¶æ ‡è®°
          this.tableDatatop = this.tableDatatop.map((item, index) => ({
            ...item,
            astrict: index > questionIndex,
            hiddenByEnd: index > questionIndex, // æ ‡è®°è¿™äº›é¢˜ç›®æ˜¯è¢«ç»“束问答隐藏的
          }));
        } else {
          // æ­£å¸¸è·³è½¬é€»è¾‘
          const nextQuestionIndex = selectedOptionObj.nextQuestion - 1;
          this.tableDatatop = this.tableDatatop.map((item, index) => {
            // ä¿ç•™å½“前题目之前的隐藏状态
            if (index < questionIndex) {
              return {
                ...item,
                astrict: previouslyHiddenBeforeCurrent.includes(index),
                hiddenByEnd: false, // æ¸…除结束标记
              };
            }
            // å½“前题目总是可见
            if (index === questionIndex) {
              return { ...item, astrict: 0, hiddenByEnd: false };
            }
            // æ˜¾ç¤ºç›®æ ‡ä¸‹ä¸€é¢˜
            if (index === nextQuestionIndex) {
              return { ...item, astrict: 0, hiddenByEnd: false };
            }
            // å¦‚果是之前被结束问答隐藏的题目,现在应该恢复显示
            if (item.hiddenByEnd) {
              return { ...item, astrict: 0, hiddenByEnd: false };
            }
            // éšè—å½“前题和目标题之间的题目
            if (index > questionIndex && index < nextQuestionIndex) {
              return { ...item, astrict: 1, hiddenByEnd: false };
            }
            // å…¶ä»–情况保持原状
            return item;
          });
        }
      } else {
        // å¦‚果没有跳转,只需确保下一题可见
        this.tableDatatop = this.tableDatatop.map((item, index) => ({
          ...item,
          astrict: index === questionIndex + 1 ? 0 : item.astrict,
          hiddenByEnd: index === questionIndex + 1 ? false : item.hiddenByEnd,
        }));
      }
      this.$forceUpdate();
    },
    overdata() {
@@ -1655,69 +1835,205 @@
      return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
    },
    updateScore(a, b, c) {},
    // åˆå¹¶ä¿®æ”¹ç›¸å…³=============================
    toggleMergeMode() {
      this.isMergeMode = !this.isMergeMode;
      if (!this.isMergeMode) {
        this.selectedServices = [];
      }
    },
    handleSelectionChange(selection) {
      this.selectedServices = selection
        .filter(
          (item) => !item.preachformson || !item.preachformson.includes("3")
        )
        .map((item) => ({
          id: item.id,
          taskid: item.taskid,
          taskName: item.taskName,
          sendname: item.sendname,
        }));
    },
    checkSelectable(row, index) {
      // å½“ sendstate ä¸º 6 æ—¶ä¸å¯é€‰
      return row.sendstate !== 6;
    },
    openMergeDialog() {
      if (this.selectedServices.length < 2) {
        this.$message.warning("请至少选择2个问卷服务进行合并");
        return;
      }
      this.mergeDialogVisible = true;
    },
    handleMergeSave(mergedData) {
      // å¤„理合并保存逻辑
      this.mergeDialogVisible = false;
      this.isMergeMode = false;
      this.selectedServices = [];
      // æ˜¾ç¤ºä¿å­˜ç»“æžœ
      if (mergedData.successCount == mergedData.totalCount) {
        this.$message.success(`成功保存 ${mergedData.successCount} ä¸ªé—®å·`);
      } else if (mergedData.successCount > 0) {
        this.$message.warning(
          `成功保存 ${mergedData.successCount} ä¸ªé—®å·ï¼Œå¤±è´¥ ${
            mergedData.totalCount - mergedData.successCount
          } ä¸ª`
        );
      } else {
        this.$message.error("所有问卷保存失败");
      }
      // åˆ·æ–°æ•°æ®
      this.getTaskservelist();
    },
  },
};
</script>
<style lang="scss">
<style lang="scss" scoped>
.Followupdetailspage {
  margin: 10px;
  display: flex;
  flex-direction: column;
  gap: 20px;
}
.action-container {
  display: flex;
  gap: 20px;
  margin: 0 10px 20px 10px;
  .manual-action {
    flex: 1;
    min-width: 0;
    height: 100%; /* ç¡®ä¿é«˜åº¦ç»§æ‰¿ */
  }
  .call-action {
    width: 60%;
    min-width: 0;
    height: 100%; /* ç¡®ä¿é«˜åº¦ç»§æ‰¿ */
  }
}
.call-container {
  padding: 20px;
  background: #fff;
  border: 1px solid #dcdfe6;
  box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.12), 0 0 6px 0 rgba(0, 0, 0, 0.04);
  border-radius: 4px;
  height: 100%;
  .call-header {
    margin-bottom: 20px;
    h2 {
      font-size: 20px;
      color: #333;
      margin: 0;
      padding-bottom: 10px;
      border-bottom: 1px solid #eee;
    }
  }
  .call-status {
    margin-bottom: 20px;
  }
  .hangup-btn {
    text-align: center;
    margin-top: 20px;
  }
}
.merge-controls {
  background: #f5f7fa;
  border-radius: 4px;
  margin-left: 20px;
}
.Followuserinfo {
  margin: 20px 10px;
  margin: 10px 10px 0 10px;
  align-items: center;
  padding: 30px;
  background: #ffff;
  border: 1px solid #dcdfe6;
  -webkit-box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.12),
    0 0 6px 0 rgba(0, 0, 0, 0.04);
  box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.12), 0 0 6px 0 rgba(0, 0, 0, 0.04);
  .userinfo-text {
    font-size: 20px;
    margin-right: 20px;
    margin-bottom: 10px;
  }
  .userinfo-value {
    color: rgb(15, 139, 211);
    span {
      margin-right: 20px;
    }
  }
}
::v-deep.el-table .warning-row {
  background: #c4e2ee;
}
.Followuserinfos {
  margin: 20px 10px;
  align-items: center;
  padding: 30px;
  background: #ffff;
  border: 1px solid #dcdfe6;
  -webkit-box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.12),
    0 0 6px 0 rgba(0, 0, 0, 0.04);
  box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.12), 0 0 6px 0 rgba(0, 0, 0, 0.04);
  height: 100%; /* ç¡®ä¿é«˜åº¦ç»§æ‰¿ */
  min-height: 880px; /* æœ€å°é«˜åº¦ä¸Žéšè®¿å†…容一致 */
  display: flex;
  flex-direction: column;
  .userinfo-text {
    font-size: 20px;
    margin-right: 20px;
    margin-bottom: 10px;
  }
  .userinfo-value {
    color: rgb(15, 139, 211);
    span {
      margin-right: 20px;
    }
  }
  .el-form {
    flex: 1;
    overflow-y: auto; /* å†…容超过高度时显示滚动条 */
    max-height: calc(880px - 60px); /* å‡åŽ»padding */
    padding-right: 10px; /* é˜²æ­¢æ»šåŠ¨æ¡é®æŒ¡å†…å®¹ */
  }
}
.append-input-container {
  margin-top: 15px;
  padding: 10px;
  background-color: #f5f7fa;
  border-radius: 4px;
  border: 1px solid #dcdfe6;
}
.borderdiv {
  min-height: 60vh;
  font-size: 20px;
  padding: 30px;
  .title {
    font-size: 22px;
    font-weight: bold;
    margin-bottom: 20px;
    text-align: center;
  }
  .leftside {
    margin: 30px 0;
    span {
      width: 400px;
      margin-left: 20px;
@@ -1727,9 +2043,11 @@
      border-radius: 10px;
    }
  }
  .offside {
    display: flex;
    flex-direction: row-reverse;
    .offside-value {
      padding: 10px;
      background: rgb(217, 173, 253);
@@ -1739,8 +2057,15 @@
    }
  }
}
.topic-dev[inert] {
  opacity: 0.5;
  pointer-events: none;
}
.CONTENT {
  padding: 10px;
  height: 100%;
  min-height: 660px; /* è®¾ç½®æœ€å°é«˜åº¦ */
  .title {
    font-size: 22px;
    font-weight: bold;
@@ -1748,40 +2073,47 @@
    text-align: center;
  }
}
.preview-left {
  margin: 20px;
  //   margin: 20px;
  padding: 30px;
  // background: #ffff;
  border: 1px solid #dcdfe6;
  -webkit-box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.12),
    0 0 6px 0 rgba(0, 0, 0, 0.04);
  box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.12), 0 0 6px 0 rgba(0, 0, 0, 0.04);
  max-height: 580px; /* è®¾ç½®æœ€å¤§é«˜åº¦ */
  overflow-y: auto; /* å†…容超过高度时显示滚动条 */
  .topic-dev {
    margin-bottom: 25px;
    font-size: 20px !important;
    .dev-text {
      margin-bottom: 10px;
    }
  }
}
.scriptTopic-isabnormal {
  color: red;
}
.detailed {
  width: 88%;
  border-radius: 8px;
  padding: 30px;
  margin-bottom: 30px;
  background-color: #ddf0f8;
  .bg-purple {
    margin-bottom: 20px;
  }
  .spanvalue {
    display: inline-block;
    min-width: 200px;
    border-bottom: 1px solid rgb(172, 172, 172);
  }
}
.headline {
  font-size: 24px;
  height: 40px;
@@ -1789,79 +2121,94 @@
  padding-left: 5px;
  margin-bottom: 10px;
  display: flex;
  // justify-content: space-between;
  .Add-details {
    font-size: 18px;
    color: #02a7f0;
    cursor: pointer;
  }
}
.red-star {
  ::v-deep.el-radio__label {
    position: relative;
    padding-right: 10px; /* æ ¹æ®éœ€è¦è°ƒæ•´ */
    padding-right: 10px;
  }
  ::v-deep.el-radio__label::after {
    content: "*";
    color: red;
    position: absolute;
    right: -5px; /* æ ¹æ®éœ€è¦è°ƒæ•´ */
    right: -5px;
    top: 0;
  }
  ::v-deep.el-input-group__textarea {
    white-space: pre-wrap; /* ä¿æŒç©ºç™½ç¬¦åºåˆ—并正常换行 */
    word-break: break-all; /* åœ¨é•¿å•词或URL地址内部进行换行 */
    white-space: pre-wrap;
    word-break: break-all;
  }
  ::v-deep.el-checkbox__label {
    position: relative;
    padding-right: 10px; /* æ ¹æ®éœ€è¦è°ƒæ•´ */
    padding-right: 10px;
  }
  ::v-deep.el-checkbox__label::after {
    content: "*";
    color: red;
    position: absolute;
    right: -5px; /* æ ¹æ®éœ€è¦è°ƒæ•´ */
    right: -5px;
    top: 0;
  }
}
::v-deep.offside-value .el-radio__label {
  color: #fff;
}
::v-deep.el-link.el-link--default {
  color: #02a7f0 !important;
}
.el-message-box__btns button:nth-child(2) {
  margin-left: 10px;
  background-color: #f57676;
  border-color: #f57676;
}
.el-icon-phone {
  transition: all 0.3s;
}
.el-button[disabled] .el-icon-phone {
  color: #c0c4cc;
}
.el-button:not([disabled]) .el-icon-phone {
  color: #409eff;
}
.el-button:not([disabled]):hover .el-icon-phone {
  color: #66b1ff;
  transform: scale(1.1);
}
.mulsz {
  font-size: 25px;
  margin-top: 20px;
}
.el-input.is-disabled .el-input__inner {
  background-color: #fff; /* èƒŒæ™¯é¢œè‰² */
  border-color: #dcdfe6; /* è¾¹æ¡†é¢œè‰² */
  color: #080808 !important; /* æ–‡å­—颜色 */
  cursor: not-allowed; /* é¼ æ ‡æ ·å¼ */
  background-color: #fff;
  border-color: #dcdfe6;
  color: #080808 !important;
  cursor: not-allowed;
}
.el-textarea.is-disabled .el-textarea__inner {
  background-color: #fff; /* èƒŒæ™¯é¢œè‰² */
  border-color: #dcdfe6; /* è¾¹æ¡†é¢œè‰² */
  color: #080808 !important; /* æ–‡å­—颜色 */
  cursor: not-allowed; /* é¼ æ ‡æ ·å¼ */
  background-color: #fff;
  border-color: #dcdfe6;
  color: #080808 !important;
  cursor: not-allowed;
}
</style>
src/views/followvisit/record/physical/index.vue
@@ -711,7 +711,6 @@
      const queryParams = {
        pid: Number(this.patid),
        allhosp: "0",
        pageNum: 1,
      };
      // æ‚£è€…基础信息
      messagelistpatient(queryParams).then((response) => {
src/views/followvisit/zbAgain/index.vue
@@ -1187,8 +1187,8 @@
      });
    },
    affiliation() {
      this.topqueryParams.drcode = store.getters.hisUserId;
      this.topqueryParams.nurseId = store.getters.hisUserId;
      this.topqueryParams.managementDoctor = store.getters.hisUserId;
      this.getList(1);
    },
    onthatday() {
src/views/knowledge/questionnaire/compilequer/index.vue
@@ -262,7 +262,7 @@
                  <el-option
                    class="ruleFormaa"
                    v-for="item in flatArray"
                    :key="item.deptCode"
                    :key="item.id"
                    :label="item.label"
                    :value="item.deptCode"
                  >
@@ -285,7 +285,7 @@
                  <el-option
                    class="ruleFormaa"
                    v-for="item in flatArray"
                    :key="item.deptCode"
                    :key="item.id"
                    :label="item.label"
                    :value="item.deptCode"
                  >
@@ -566,10 +566,10 @@
                    </el-form-item></el-col
                  >
                  <el-col :span="12"
                    ><el-form-item label="是否可用">
                      <el-radio-group v-model="topicobj.isenable">
                    ><el-form-item label="是否隐藏">
                      <el-radio-group v-model="topicobj.ishide">
                        <el-radio
                          v-for="(item, index) in usable"
                          v-for="(item, index) in hides"
                          :label="item.value"
                          >{{ item.label }}</el-radio
                        >
@@ -582,8 +582,8 @@
                  v-if="topicobj.scriptType == 1"
                >
                  <el-radio-group v-model="topicobj.branchFlag">
                    <el-radio :label="1">是</el-radio>
                    <el-radio :label="0">否</el-radio>
                    <el-radio label="1">是</el-radio>
                    <el-radio label="0">否</el-radio>
                  </el-radio-group>
                </el-form-item>
                <el-form-item label="选中提示" v-if="topicobj.scriptType != 1">
@@ -656,6 +656,12 @@
                        <el-radio-group v-model="item.isabnormal">
                          <el-radio :label="1">是</el-radio>
                          <el-radio :label="0">否</el-radio>
                        </el-radio-group>
                      </el-form-item>
                      <el-form-item label="是否选中收集附加信息">
                        <el-radio-group v-model="item.appendflag">
                          <el-radio label="1">是</el-radio>
                          <el-radio label="0">否</el-radio>
                        </el-radio-group>
                      </el-form-item>
                    </el-row>
@@ -749,9 +755,7 @@
        >
        <el-button type="primary" @click="laststep()">上一步</el-button>
        <el-button type="info" @click="closeFm('ruleForm')">关闭</el-button>
        <el-button @click="toExamine('ruleForm')"
          >保存问题数据</el-button
        >
        <el-button @click="toExamine('ruleForm')">保存问题数据</el-button>
      </div>
      <!-- é—®å·é¢„览 -->
      <div v-if="Editprogress == 3">
@@ -1215,8 +1219,8 @@
                            style="width: 100px; height: 100px"
                            :src="item.picturePath"
                            :preview-src-list="
                                item.picturePath ? [item.picturePath] : []
                              "
                              item.picturePath ? [item.picturePath] : []
                            "
                          >
                          </el-image>
                        </div>
@@ -1299,6 +1303,7 @@
        svyTemplateLibScripts: [],
        tempDetpRelevances: [],
        svyLibTemplateTagList: [],
        suitway: [],
        scoreType: "4",
        isenable: "0",
        longTemp: "0",
@@ -1404,6 +1409,10 @@
      usable: [
        { value: "0", label: "可用" },
        { value: "1", label: "停用" },
      ],
      hides: [
        { value: 0, label: "正常" },
        { value: 1, label: "隐藏" },
      ],
      longtype: [
        { value: "0", label: "普通任务" },
@@ -1622,8 +1631,8 @@
          this.confirmillness();
          this.putbelongDepts();
          this.$modal.closeLoading();
          // this.$router.go(-1);
          window.location.reload();
          this.$router.go(-1);
          // window.location.reload();
        });
      } else {
        this.ruleForm.isoperation = 1;
@@ -1643,8 +1652,8 @@
    // é¢˜ç›®æ ¡éªŒ
    toExamine() {
      // éåŽ†é¢˜ç›®é›†åˆ
      for (let i = 0; i <  this.ruleForm.svyTemplateLibScripts.length; i++) {
        const question =  this.ruleForm.svyTemplateLibScripts[i];
      for (let i = 0; i < this.ruleForm.svyTemplateLibScripts.length; i++) {
        const question = this.ruleForm.svyTemplateLibScripts[i];
        // å¦‚æžœ scriptType ä¸º 4,则跳过当前题目
        if (question.scriptType === 4) {
@@ -1675,7 +1684,7 @@
      // å¦‚果所有题目都校验通过,返回 true
      console.log("所有题目校验通过,选项名称无重复");
       this.Departmenttreatment();
      this.Departmenttreatment();
    },
    // ç§‘室/院区处理
src/views/patient/patient/outpatient.vue
@@ -67,16 +67,16 @@
            ></el-cascader>
          </el-form-item>
          <el-form-item label="就诊日期">
              <el-date-picker
                v-model="dateRange"
                style="width: 240px"
                value-format="yyyy-MM-dd"
                type="daterange"
                range-separator="-"
                start-placeholder="开始日期"
                end-placeholder="结束日期"
              ></el-date-picker>
              <!-- <el-date-picker
            <el-date-picker
              v-model="dateRange"
              style="width: 240px"
              value-format="yyyy-MM-dd"
              type="daterange"
              range-separator="-"
              start-placeholder="开始日期"
              end-placeholder="结束日期"
            ></el-date-picker>
            <!-- <el-date-picker
                v-model="dateRange"
                type="datetimerange"
                value-format="yyyy-MM-dd HH:mm:ss"
@@ -85,7 +85,7 @@
                :default-time="['12:00:00']"
              >
              </el-date-picker> -->
            </el-form-item>
          </el-form-item>
          <el-row>
            <!-- <el-form-item label=" å°±è¯Šæ—¥æœŸ " prop="admitdate">
              <el-date-picker
@@ -150,6 +150,18 @@
              @click="handleDelete"
              v-hasPermi="['system:user:remove']"
              >删除</el-button
            >
          </el-col>
          <el-col :span="1.5">
            <el-button
              type="warning"
              plain
              icon="el-icon-plus"
              size="medium"
              :disabled="multiple"
              @click="handleBatchAddTask"
            >
              æ‰¹é‡æ·»åŠ ä»»åŠ¡</el-button
            >
          </el-col>
          <el-col :span="19">
@@ -228,7 +240,13 @@
              <el-button
                size="medium"
                type="text"
                @click="gettoken360(scope.row.idcardno,scope.row.drcode,scope.row.drname)"
                @click="
                  gettoken360(
                    scope.row.idcardno,
                    scope.row.drcode,
                    scope.row.drname
                  )
                "
                ><span class="button-textsc">{{
                  scope.row.patname
                }}</span></el-button
@@ -337,7 +355,113 @@
        />
      </el-col>
    </el-row>
    <el-dialog
      title="批量添加任务"
      :visible.sync="batchTaskVisible"
      width="90%"
      append-to-body
    >
      <el-row :gutter="20">
        <!-- å·¦ä¾§ï¼šé€‰ä¸­æ‚£è€…列表 -->
        <el-col :span="12">
          <div class="batch-patient-section">
            <h4>选中患者({{ selectedPatients.length }}人)</h4>
            <el-table
              :data="selectedPatients"
              border
              style="width: 100%"
              size="small"
            >
              <el-table-column prop="patname" label="姓名" width="100" />
              <el-table-column prop="sex" label="性别" width="80">
                <template slot-scope="scope">
                  {{ scope.row.sex === 1 ? "男" : "女" }}
                </template>
              </el-table-column>
              <el-table-column
                label="就诊时间"
                align="center"
                key="admitdate"
                prop="admitdate"
              >
                <template slot-scope="scope">
                  <span>{{ formatTime(scope.row.admitdate) }}</span>
                </template>
              </el-table-column>
              <el-table-column prop="outhospno" label="病案号" />
              <el-table-column prop="deptname" label="科室" />
            </el-table>
          </div>
        </el-col>
        <!-- å³ä¾§ï¼šä»»åŠ¡åˆ—è¡¨ -->
        <el-col :span="12">
          <div class="batch-task-section">
            <h4>任务列表(请选择1个任务)</h4>
            <el-table
              :data="taskList"
              border
              style="width: 100%"
              size="small"
              @current-change="handleTaskSelectionChange"
              highlight-current-row
            >
              <el-table-column
                label="任务名称"
                fixed
                align="center"
                key="taskName"
                prop="taskName"
                :show-overflow-tooltip="true"
              />
              <el-table-column
                label="服务项目"
                align="center"
                key="templatename"
                prop="templatename"
                :show-overflow-tooltip="true"
              />
              <el-table-column
                label="创建人"
                align="center"
                key="createBy"
                prop="createBy"
                :show-overflow-tooltip="true"
              />
              <el-table-column
                label="创建时间"
                sortable
                align="center"
                prop="createTime"
              >
                <template slot-scope="scope">
                  <span>{{ formatTime(scope.row.createTime) }}</span>
                </template>
              </el-table-column>
              <template #empty>
                <div class="empty-message">
                  <i class="el-icon-warning"></i>
                  <span>患者科室无匹配服务</span>
                </div>
              </template>
            </el-table>
          </div>
        </el-col>
      </el-row>
      <!-- åº•部按钮 -->
      <div slot="footer" class="dialog-footer">
        <el-button @click="batchTaskVisible = false">取 æ¶ˆ</el-button>
        <el-button
          type="primary"
          :loading="batchLoading"
          @click="submitBatchTask"
          >创建任务</el-button
        >
      </div>
    </el-dialog>
    <!-- ç”¨æˆ·å¯¼å…¥å¯¹è¯æ¡† -->
    <el-dialog
      :title="upload.title"
@@ -383,6 +507,7 @@
          </div>
        </el-upload>
      </div>
      <!-- å¯¼å…¥æ£€æŸ¥ -->
      <div class="uploading" v-else-if="dractive == 2">
        <el-table :data="uploadingData" style="width: 100%">
@@ -446,7 +571,7 @@
import { getToken } from "@/utils/auth";
import Treeselect from "@riophae/vue-treeselect";
import "@riophae/vue-treeselect/dist/vue-treeselect.css";
import { query360PatInfo } from "@/api/AiCentre/index";
import { query360PatInfo, getTasklist, addSubtask } from "@/api/AiCentre/index";
import store from "@/store";
@@ -509,6 +634,12 @@
      Labelchange: false, //修改新增弹窗
      propss: { multiple: true },
      optionstag: [], //标签列表
      batchTaskVisible: false, // å¼¹çª—可见性
      selectedPatients: [], // é€‰ä¸­çš„æ‚£è€…列表
      taskList: [], // ä»»åŠ¡åˆ—è¡¨
      selectedTask: null, // é€‰ä¸­çš„任务列表
      batchLoading: false, // æ‰¹é‡æäº¤åŠ è½½çŠ¶æ€
      deptcode: "",
      Patientrange: [
        {
          name: "全部",
@@ -680,9 +811,9 @@
      if (this.dateRange) {
        this.queryParams.beginTime = this.dateRange[0];
        this.queryParams.endTime = this.dateRange[1];
      }else{
        this.queryParams.beginTime = '';
        this.queryParams.endTime = '';
      } else {
        this.queryParams.beginTime = "";
        this.queryParams.endTime = "";
      }
      listPatouthosp(this.queryParams).then((response) => {
        this.userList = response.rows;
@@ -733,11 +864,11 @@
      });
    },
    //患者360跳转
    gettoken360(sfzh,drcode,drname) {
    gettoken360(sfzh, drcode, drname) {
      this.postData.YeWuXX.BingRenXX.ZhengJianHM = sfzh;
      if (this.postData.XiaoXiTou.ZuHuMC=='丽水市中医院') {
        this.postData.YeWuXX.YongHuXX.YongHuID = '1400398571877961728';
        this.postData.YeWuXX.YongHuXX.YongHuXM = 'LSZYY';
      if (this.postData.XiaoXiTou.ZuHuMC == "丽水市中医院") {
        this.postData.YeWuXX.YongHuXX.YongHuID = "1400398571877961728";
        this.postData.YeWuXX.YongHuXX.YongHuXM = "LSZYY";
      }
      query360PatInfo(this.postData).then((res) => {
@@ -796,7 +927,7 @@
        leaveldeptcodes: [],
        leavehospitaldistrictcodes: [],
      };
        this.handleQuery();
      this.handleQuery();
    },
    // å¤šé€‰æ¡†é€‰ä¸­æ•°æ®
    handleSelectionChange(selection) {
@@ -913,6 +1044,113 @@
      this.upload.open = false;
      this.dractive = 1;
    },
    /** æ‰¹é‡æ·»åŠ ä»»åŠ¡æŒ‰é’®ç‚¹å‡» */
    handleBatchAddTask() {
      // æ ¡éªŒæ˜¯å¦é€‰ä¸­æ‚£è€…
      if (this.ids.length === 0) {
        this.$modal.msgWarning("请至少选中1名患者");
        return;
      }
      // èŽ·å–é€‰ä¸­æ‚£è€…çš„ç§‘å®¤ä¿¡æ¯ï¼ˆç”¨äºŽæ ¡éªŒåŒä¸€ç§‘å®¤ï¼‰
      const patientDepts = new Set();
      let deptcode = "";
      this.ids.forEach((patId) => {
        const patient = this.userList.find((item) => item.patid === patId);
        if (patient) {
          patientDepts.add(patient.deptname);
          deptcode = patient.deptcode;
        }
      });
      // æ ¡éªŒæ˜¯å¦åŒä¸€ç§‘室
      if (patientDepts.size > 1) {
        this.$modal.msgError("选中的患者不属于同一科室,无法批量添加任务");
        return;
      }
      // èŽ·å–é€‰ä¸­æ‚£è€…åˆ—è¡¨
      this.selectedPatients = this.userList.filter((item) =>
        this.ids.includes(item.patid)
      );
      // æ˜¾ç¤ºå¼¹çª—
      this.batchTaskVisible = true;
      // èŽ·å–ä»»åŠ¡åˆ—è¡¨
      this.loadTaskList(deptcode);
    },
    /** åŠ è½½ä»»åŠ¡åˆ—è¡¨ */
    loadTaskList(deptcode) {
      this.batchLoading = true;
      let topqueryParams = {
        pageNum: 1,
        pageSize: 10,
        serviceType: 3,
        type: 2,
        deptcode: deptcode,
      };
      getTasklist(topqueryParams).then((response) => {
        this.taskList = response.rows;
        this.batchLoading = false;
      });
    },
    /** å¤„理任务选择变化 */
    handleTaskSelectionChange(currentRow) {
      this.selectedTask = currentRow;
    },
    // é‡å†™æ‰¹é‡æäº¤æ–¹æ³•
    async submitBatchTask() {
      // æ ¡éªŒæ˜¯å¦é€‰ä¸­ä»»åŠ¡
      if (!this.selectedTask) {
        this.$modal.msgWarning("请选择1个任务");
        return;
      }
      this.batchLoading = true;
      const successPatients = [];
      const failedPatients = [];
      try {
        // éåŽ†é€‰ä¸­çš„æ‚£è€…ï¼Œé€ä¸ªè°ƒç”¨æŽ¥å£
        for (const patient of this.selectedPatients) {
          const params = {
            taskId: this.selectedTask.taskId,
            taskName: this.selectedTask.taskName,
            serviceType: this.selectedTask.serviceType,
            ...patient,
            age: "",
          };
          try {
            await addSubtask(params);
            successPatients.push(patient.patname);
          } catch (error) {
            failedPatients.push(patient.patname);
          }
        }
        // æ˜¾ç¤ºå¤„理结果
        let message = `成功为 ${successPatients.length} åæ‚£è€…添加任务`;
        if (failedPatients.length > 0) {
          message += `,${failedPatients.length} åæ‚£è€…添加失败`;
        }
        this.$modal.msgSuccess(message);
        this.batchTaskVisible = false;
        this.getList(); // åˆ·æ–°æ‚£è€…列表
      } catch (error) {
        this.$modal.msgError("批量添加任务过程中出错");
      } finally {
        this.batchLoading = false;
      }
    },
    /** æ‰¹é‡æ·»åŠ æ‚£è€…ä»»åŠ¡æŽ¥å£ */
    batchAddPatientTask(params) {},
  },
};
</script>
@@ -963,4 +1201,41 @@
.button-textsc {
  color: #28cfe6;
}
.batch-patient-section,
.batch-task-section {
  padding: 15px;
  border: 1px solid #ebeef5;
  border-radius: 4px;
  margin-bottom: 15px;
}
.el-table--enable-row-hover .el-table__body tr:hover > td {
  background-color: #f5f7fa;
}
.el-table__body tr.current-row > td {
  background-color: #e6f7ff;
}
/* ç¦ç”¨çŠ¶æ€ä¸‹æŒ‰é’®æ ·å¼ */
.el-button.is-disabled {
  cursor: not-allowed;
}
.batch-patient-section h4,
.batch-task-section h4 {
  margin-top: 0;
  margin-bottom: 15px;
  color: #303133;
  font-size: 16px;
}
/* è¡¨æ ¼æ ·å¼ä¼˜åŒ– */
.el-table {
  margin-bottom: 10px;
}
/* å¼¹çª—底部按钮间距 */
.dialog-footer {
  text-align: right;
  padding: 10px 20px 20px;
}
</style>
vue.config.js
@@ -37,7 +37,7 @@
      [process.env.VUE_APP_BASE_API]: {
        // target: `https://www.health-y.cn/lssf`,
        // target: `http://10.202.20.185:8095`,
        // target: `http://9.208.2.190:8095`,
        // target: `http://192.168.100.195:8095`,
        target:`http://localhost:8095`,
        // target: `http://192.168.100.193:8095`,
        // target: `http://192.168.101.166:8093`,