Compare commits

79 Commits

Author SHA1 Message Date
46c9787216 修改套餐管理页面操作图标设置 2025-12-05 16:50:37 +08:00
e0a768de2e 新增套餐设置页面中的套餐管理页面并增加相应跳转逻辑 2025-12-05 16:37:34 +08:00
fc7f931728 新增套餐设置页面中的套餐管理页面并增加相应跳转逻辑 2025-12-05 16:34:12 +08:00
qk123
3ccb970a09 维护系统->检查方法、部位导出表格功能OK 2025-12-05 15:54:57 +08:00
ljj
8a9a622aeb 72.系统管理--》基础数据-》科室管理 2025-12-05 13:39:23 +08:00
qk123
29e7f0937b 维护系统->检查方法、部位条件搜索前后端实现 2025-12-04 14:47:39 +08:00
029d534b3c 新增检验项目设置中的检验类型页面并实现相关逻辑 2025-12-04 09:47:34 +08:00
qk123
213723b220 维护系统->检查方法后端接口数据校验修改 2025-12-04 09:33:44 +08:00
qk123
c636e77fd0 维护系统->检查部位前端接口缺少导入。 2025-12-04 09:21:35 +08:00
qk123
7407562ec5 维护系统->检查部位后端接口优化。 2025-12-03 17:15:35 +08:00
qk123
b3c27ec789 维护系统->检查方法前端页面优化、接口功能完善(搜索栏和导出表格未处理),后端接口优化。 2025-12-03 16:00:24 +08:00
qk123
601ae40666 维护系统->LIS分组前端页面开发,基础功能实现 2025-12-03 14:24:19 +08:00
3dda5ce72d 修改配置7 2025-12-03 13:48:25 +08:00
qk123
99d4c28b95 检查项目维护->Lis分组前端接口、后端实体优化。 2025-12-02 16:46:19 +08:00
qk123
0e70de6de7 Merge branch 'develop' of https://gitea.gentronhealth.com/Yajentine/his into develop 2025-12-02 16:44:57 +08:00
72eb1088df 修正前端PROD 2025-12-02 15:03:40 +08:00
a853c16854 修正prd 2025-12-02 14:43:35 +08:00
73a349f98a 冗余prod 2025-12-02 14:25:10 +08:00
5fb7a3f488 production 2025-12-02 14:12:06 +08:00
a8eb7d5697 prd->production 2025-12-02 14:08:09 +08:00
eb3b142ce4 修改配置 2025-12-02 14:03:49 +08:00
9f6a39ba30 修改配置文件 2025-12-02 14:02:25 +08:00
f7f2f98bbe 修改统一前端的PRD的标识 2025-12-02 13:52:18 +08:00
79e68ee14e 修改前端VUE的DEV环境配置 2025-12-02 12:53:01 +08:00
bc8987c463 配置3 2025-12-02 11:51:07 +08:00
3f7174fcd1 修复重复 2025-12-02 10:31:18 +08:00
080008f447 Merge branch 'develop' of https://gitea.gentronhealth.com/Yajentine/his into develop 2025-12-02 10:29:50 +08:00
5c3d935615 前端加上相关test的配置 2025-12-02 10:29:39 +08:00
a0845127c6 配置2 2025-12-02 10:28:02 +08:00
ba65ab303b 多环境配置 2025-12-02 09:25:35 +08:00
qk123
6fbdddf0ee 检查项目维护->Lis分组后端接口 2025-12-01 16:13:30 +08:00
py
173cbc7f87 门诊挂号-》完善新增患者字典信息 2025-12-01 14:45:27 +08:00
qk123
2e341a123e 检查项目维护检查方法、部位前端表格数据映射ok 2025-12-01 13:42:15 +08:00
qk123
8beff2ee14 检查方法、检查部位前端页面、完善后端接口逻辑 2025-11-28 17:03:08 +08:00
wzk
1b939ba5b7 刷新 2025-11-28 16:48:14 +08:00
wzk
577daabe1c 医嘱诊断和费用性质没有自动引入 2025-11-28 16:35:41 +08:00
wzk
327e88e6d4 检查项目设置-套餐设置-套餐管理界面放大或缩小数据显示不全 2025-11-28 15:24:52 +08:00
叶锦涛
548fabcffe 新建检验项目设置的检验类型页面 2025-11-28 13:56:27 +08:00
263e21e157 换卡bug 2025-11-28 10:41:14 +08:00
d7955aa628 配置 2025-11-28 09:58:52 +08:00
叶锦涛
d197f7555f 修复发票管理界面管理员和普通操作人员的筛选规则 2025-11-27 13:36:17 +08:00
wzk
a0e52da437 门诊号码管理维护界面前后端 2025-11-27 13:08:42 +08:00
qk123
88d79729fe 库房管理->采购管理->采购管理点击《详情》按钮跳转后无数据显示 2025-11-27 12:00:21 +08:00
6dc6480e64 Merge remote-tracking branch 'origin/develop' into develop 2025-11-27 10:40:12 +08:00
98c263a72b 患者档案和进货单bug 2025-11-27 10:39:56 +08:00
wzk
50a2ef9e00 门诊医生站->医嘱:新增独立的西药处方单补充 2025-11-27 09:41:59 +08:00
wzk
065f7052c6 检查项目设置->套餐设置->套餐管理补充 2025-11-27 09:24:26 +08:00
e23d42404d 患者档案和进货单bug 2025-11-27 08:56:19 +08:00
wzk
4120d4e001 检查项目设置->套餐设置->套餐管理 2025-11-27 08:44:32 +08:00
wzk
1bd2089047 检查项目设置-套餐设置 2025-11-26 16:18:47 +08:00
wzk
6d9ff7dc10 检查项目设置-套餐设置 2025-11-26 16:15:41 +08:00
wzk
ae7ca984f8 检查项目设置-套餐设置 2025-11-26 16:15:11 +08:00
wzk
69b7a4d865 检查项目设置-套餐设置界面 2025-11-26 14:39:57 +08:00
wzk
864bf55025 修复门诊医生站->【处方单】按钮,点击【处方单】按钮无响应问题 2025-11-26 13:34:41 +08:00
wzk
249ef5f87e 门诊挂号-》医生字段,对字段医生进行过滤,让医生和、就诊科室、挂号类型形成对照。 2025-11-26 13:15:42 +08:00
qk123
df3fb6c66c 检查方法、检查部位接口部分细节修改 2025-11-26 10:37:30 +08:00
qk123
10ec9f4c1b 检查方法、检查部位后端接口、实体、数据库表基本完成。 2025-11-26 10:31:30 +08:00
0b98763c05 门诊挂号查询优化 2025-11-25 16:14:03 +08:00
叶锦涛
3d3b21a775 修复操作员的工号的编辑功能 2025-11-25 15:58:17 +08:00
叶锦涛
4ba4f80946 修改发票管理页面 2025-11-25 15:39:29 +08:00
叶锦涛
ed7cb2dab5 修改发票管理页面 2025-11-25 15:30:33 +08:00
wzk
50ef9e6743 修复西药处方单bug 2025-11-25 13:49:38 +08:00
py
250d7dde34 门诊医生站-》开立处方医嘱:完善药品的字典信息 2025-11-25 11:27:58 +08:00
wzk
69f3e066db 门诊医生站-》医嘱:新增独立的西药处方单 2025-11-25 10:17:13 +08:00
叶锦涛
afb0c3933c 删除检查项目设置多余数据 2025-11-24 16:22:43 +08:00
叶锦涛
45ac07e57c 删除检查项目设置多余数据 2025-11-24 16:19:46 +08:00
qk123
262ea97824 同一患者保存病历后判断逻辑问题 2025-11-24 15:13:18 +08:00
ljj
dcfa13f239 系统管理-目录管理-》耗材目录。将该界面的字段标题设置可以手动拉宽每个字段的宽度,有的字段内容看不全,将界面的字段标题设置成可以调节每个字段的长度,使之能看全内容 2025-11-24 15:11:19 +08:00
叶锦涛
27c3c850d6 创建项目检查设置页面 2025-11-24 14:35:54 +08:00
wzk
122a15a73d 修复医嘱保存、签发不成功 2025-11-24 11:40:37 +08:00
wzk
61749aee4d 门诊号码管理修改关闭给予保存提示 2025-11-24 10:45:27 +08:00
97a29a31c5 门诊挂号查询优化 2025-11-24 09:39:23 +08:00
wzk
c3734f921c 用户管理中用户名字替换成用户姓名 2025-11-24 09:34:04 +08:00
c561586cfa 门诊换卡优化 2025-11-24 09:22:11 +08:00
qk123
60593233bc 中医添加医嘱后页面不显示BUG 2025-11-21 15:08:27 +08:00
wzk
992b03f9c2 解决退号记录的退号操作工作取值问题 2025-11-21 13:15:08 +08:00
wzk
36628342dc 用户管理将用户名称改成用户账号、用户昵称改成用户姓名。 2025-11-21 11:32:46 +08:00
qk123
3783e4a872 优化配方名称框、剂量单位框、脚注框 2025-11-21 09:59:09 +08:00
py
584f79294b 13 收费工作站-》门诊挂号1、将医生改成出诊医生。
2、将出诊医生字段选中‘内科医生1’,点击【确认】按钮,重新打开出诊医生字段的内容保存成功
2025-11-21 09:39:22 +08:00
128 changed files with 13426 additions and 982 deletions

76
.github/copilot-instructions.md vendored Normal file
View File

@@ -0,0 +1,76 @@
# OpenHIS — AI 编码助手 指南
目的:帮助自动化/AI 编码代理快速上手本仓库,包含架构要点、关键文件、常用构建/运行命令以及项目约定。请只按照仓库内真实可见的内容提出修改建议或补充说明。
- **代码组织**: 本项目是一个 Java 后端(多模块 Maven+ Vue3 前端Vite的大型应用。
- 后端主模块目录:`openhis-server-new/`(顶层为 `pom`,包含多个子模块)。关键子模块示例:`openhis-application`, `openhis-domain`, `openhis-common`, `core-*` 系列。
- 前端目录:`openhis-ui-vue3/`Vite + Vue 3使用 Pinia、Element Plus 等)。
- **大局观Big Picture**: 后端以 Spring BootJava 17实现使用多模块 Maven 管理公共库与业务模块;前端由单独仓库目录通过 Vite 构建并以环境变量(`VITE_APP_BASE_API`)与后端交互。后端扫描 `com.core``com.openhis` 包(见 `OpenHisApplication.java`),启动类位于:`openhis-server-new/openhis-application/src/main/java/com/openhis/OpenHisApplication.java`
- **运行/构建Windows PowerShell 示例)**:
- 构建后端(从仓库根执行):
```powershell
cd openhis-server-new
mvn clean package -DskipTests
```
- 仅运行后端模块(开发时常用):
```powershell
cd openhis-server-new/openhis-application
mvn spring-boot:run
# 或在 IDE 中运行 `com.openhis.OpenHisApplication` 的 main()
```
- 前端启动与构建(需要 Node.js v16.x仓库 README 建议 v16.15
```powershell
cd openhis-ui-vue3
npm install
npm run dev # 本地开发(热重载)
npm run build:prod # 生产构建
```
- **环境与配置**:
- 后端配置:`openhis-server-new/openhis-application/src/main/resources/application.yml`数据库、端口、profile 等。README 还提及 `application-druid.yml`(若存在请优先查看)。
- 前端配置:多个 `.env.*` 文件(例如 `.env.development`, `.env.staging`, `.env.production`),关键变量:`VITE_APP_BASE_API`(例如 `/dev-api`),前端通过 `import.meta.env.VITE_APP_BASE_API` 拼接后端 URL`src/utils/request.js`、多个视图与组件)。
- **重要约定 / 模式**:
- 后端采用 Java 17、Spring Boot 2.5.x 家族,父 POM在 `openhis-server-new/pom.xml` 定义。常用依赖版本在该 POM 的 `<properties>` 中集中维护。
- 模块间以 Maven 模块依赖与 `com.core` / `com.openhis` 包名分层(见 `pom.xml``<modules>``dependencyManagement`)。
- 前端通过 Vite 插件配置(`openhis-ui-vue3/vite/plugins`)管理 svg、自动导入等。UI 框架为 Element Plus状态管理为 Pinia。
- **集成点 & 外部依赖**:
- 数据库PostgreSQLREADME 建议 v16.2),仓库根含一个大型初始化 SQL`数据库初始话脚本请使用navicat16版本导入.sql`,用于初始化表与演示数据。
- 缓存/会话Redis需自行配置
- 其他Flowable工作流Druid连接池监控第三方服务通过特定配置类例如 `YbServiceConfig``OpenHisApplication` 中启用)。
- **调试与常见位置**:
- 启动类:`openhis-server-new/openhis-application/src/main/java/com/openhis/OpenHisApplication.java`
- 全局配置:`openhis-server-new/openhis-application/src/main/resources/``application.yml`、profile 文件等)。
- 前端入口:`openhis-ui-vue3/src/main.js`、路由在 `openhis-ui-vue3/src/router/index.js`
- API 文档与监控路径(通常由后端暴露并被前端访问):
- Swagger UI: `<VITE_APP_BASE_API>/swagger-ui/index.html`(前端视图在 `src/views/tool/swagger/index.vue`)。
- Druid: `<VITE_APP_BASE_API>/druid/login.html`(见前端相关视图引用)。
- **为 AI 代理的具体建议(如何安全、有效地修改代码)**:
- 修改后端时:优先在子模块(例如 `openhis-application`)本地运行 `mvn spring-boot:run` 验证启动与基础 API大量改动前先执行 `mvn -T1C -DskipTests clean package` 在 CI 环境上验证构建(本地机器也可用)。
- 修改前端时:检查/调整对应 `.env.*` 文件中的 `VITE_APP_BASE_API`,使用 `npm run dev` 本地联调后端接口(可通过代理或将 `VITE_APP_BASE_API` 指向后端地址)。
- 修改数据库结构或 seed请参考仓库根的 SQL 初始化脚本,任何 DDL/数据变更需同步该脚本并通知数据库管理员/运维。
- **举例(常见任务示例)**:
- 本地联调前端 + 后端PowerShell:
```powershell
# 启动后端
cd openhis-server-new/openhis-application
mvn spring-boot:run
# 启动前端(另开终端)
cd openhis-ui-vue3
npm run dev
```
如需我把这些内容合并为更短或更详细的版本,或把其中某部分(例如后端模块依赖关系图、关键 Java 包说明)展开,请告诉我要增强哪一节。

View File

@@ -0,0 +1,39 @@
import java.io.IOException;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.Scanner;
public class TestDeleteInspectionType {
public static void main(String[] args) {
try {
// 测试删除ID为1的检验类型
long inspectionTypeId = 1;
URL url = new URL("http://localhost:8080/system/inspection-type/" + inspectionTypeId);
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
connection.setRequestMethod("DELETE");
connection.setRequestProperty("Content-Type", "application/json");
connection.setRequestProperty("Accept", "application/json");
// 发送请求
int responseCode = connection.getResponseCode();
System.out.println("响应代码: " + responseCode);
// 读取响应
Scanner scanner;
if (responseCode >= 200 && responseCode < 300) {
scanner = new Scanner(connection.getInputStream());
} else {
scanner = new Scanner(connection.getErrorStream());
}
String response = scanner.useDelimiter("\\A").next();
System.out.println("响应内容: " + response);
scanner.close();
connection.disconnect();
} catch (IOException e) {
e.printStackTrace();
}
}
}

View File

@@ -0,0 +1,27 @@
-- 向adm_organization表添加缺失字段的SQL脚本
-- 添加register_flag字段对应registerFlag属性
ALTER TABLE "adm_organization"
ADD COLUMN "register_flag" int4;
-- 添加location字段
ALTER TABLE "adm_organization"
ADD COLUMN "location" varchar(255);
-- 添加intro字段
ALTER TABLE "adm_organization"
ADD COLUMN "intro" varchar(1000);
-- 添加remark字段
ALTER TABLE "adm_organization"
ADD COLUMN "remark" varchar(1000);
-- 添加字段注释
COMMENT ON COLUMN "adm_organization"."register_flag" IS '挂号标志';
COMMENT ON COLUMN "adm_organization"."location" IS '科室位置';
COMMENT ON COLUMN "adm_organization"."intro" IS '科室简介';
COMMENT ON COLUMN "adm_organization"."remark" IS '备注';
-- 设置register_flag默认值为0不允许挂号
ALTER TABLE "adm_organization"
ALTER COLUMN "register_flag" SET DEFAULT 0;

View File

@@ -0,0 +1,45 @@
package com.openhis.tool;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.Statement;
/**
* Database field adder tool
*/
public class DatabaseFieldAdder {
public static void main(String[] args) {
String url = "jdbc:postgresql://192.168.110.252:15432/postgresql?currentSchema=public";
String username = "postgresql";
String password = "Jchl1528";
try (Connection conn = DriverManager.getConnection(url, username, password);
Statement stmt = conn.createStatement()) {
// Check if field exists
String checkSql = "SELECT column_name FROM information_schema.columns " +
"WHERE table_name = 'adm_healthcare_service' AND column_name = 'practitioner_id'";
boolean fieldExists = stmt.executeQuery(checkSql).next();
if (!fieldExists) {
// Add field
String addSql = "ALTER TABLE \"public\".\"adm_healthcare_service\" " +
"ADD COLUMN \"practitioner_id\" int8";
stmt.execute(addSql);
// Add comment
String commentSql = "COMMENT ON COLUMN \"public\".\"adm_healthcare_service\".\"practitioner_id\" IS 'practitioner_id'";
stmt.execute(commentSql);
System.out.println("Successfully added practitioner_id field to adm_healthcare_service table");
} else {
System.out.println("practitioner_id field already exists");
}
} catch (Exception e) {
System.err.println("Error executing SQL: " + e.getMessage());
e.printStackTrace();
}
}
}

View File

@@ -252,12 +252,10 @@ public class GenController extends BaseController {
InputStream is = file.getInputStream();
Workbook wb = WorkbookFactory.create(is);
StringBuilder sb = new StringBuilder();
// 遍历每个sheet页每个表
for (int i = 0; i < wb.getNumberOfSheets(); i++) {
sb.append("-- ----------------------------------------------------------------------------------\n");
Sheet st = wb.getSheetAt(i);
// 从第一行读取表名表注释
Row row0 = st.getRow(0);// 表名
String tableName = row0.getCell(4).toString();// 表名

View File

@@ -100,7 +100,18 @@ public class OrganizationAppServiceImpl implements IOrganizationAppService {
@Override
public R<?> getOrgInfo(Long orgId) {
Organization organization = organizationService.getById(orgId);
return R.ok(organization, MessageUtils.createMessage(PromptMsgConstant.Common.M00004, new Object[] {"机构信息查询"}));
if (organization == null) {
return R.fail(MessageUtils.createMessage(PromptMsgConstant.Common.M00006, new Object[] {"机构信息"}));
}
// 转换为DTO对象确保数据格式一致
OrganizationDto organizationDto = new OrganizationDto();
BeanUtils.copyProperties(organization, organizationDto);
organizationDto.setTypeEnum_dictText(EnumUtils.getInfoByValue(OrganizationType.class, organizationDto.getTypeEnum()));
organizationDto.setClassEnum_dictText(EnumUtils.getInfoByValue(OrganizationClass.class, organizationDto.getClassEnum()));
organizationDto.setActiveFlag_dictText(EnumUtils.getInfoByValue(AccountStatus.class, organizationDto.getActiveFlag()));
return R.ok(organizationDto, MessageUtils.createMessage(PromptMsgConstant.Common.M00004, new Object[] {"机构信息查询"}));
}
/**

View File

@@ -60,4 +60,16 @@ public class OrganizationDto {
/** 子集合 */
private List<OrganizationDto> children = new ArrayList<>();
/** 挂号科室标记 */
private Integer registerFlag;
/** 科室位置 */
private String location;
/** 科室简介 */
private String intro;
/** 备注 */
private String remark;
}

View File

@@ -0,0 +1,219 @@
package com.openhis.web.basicmanage.controller;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.core.common.annotation.Log;
import com.core.common.core.domain.R;
import com.core.common.enums.BusinessType;
import com.core.common.utils.SecurityUtils;
import com.core.common.utils.StringUtils;
import com.openhis.administration.domain.OutpatientNoSegment;
import com.openhis.administration.service.IOutpatientNoSegmentService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
import javax.servlet.http.HttpServletRequest;
/**
* 门诊号码段管理控制器
*
* @author system
* @date 2025-01-XX
*/
@RestController
@RequestMapping("/business-rule/outpatient-no")
public class OutpatientNoSegmentController {
@Autowired
private IOutpatientNoSegmentService outpatientNoSegmentService;
/**
* 分页查询门诊号码段列表
*
* @param pageNo 页码
* @param pageSize 每页条数
* @param onlySelf 是否只查询自己的true=只查询自己的false=查询所有)
* @return 门诊号码段列表
*/
@GetMapping("/page")
public R<?> selectOutpatientNoSegmentPage(
@RequestParam(defaultValue = "1") Integer pageNo,
@RequestParam(defaultValue = "10") Integer pageSize,
@RequestParam(required = false) Boolean onlySelf) {
// 获取当前用户ID
Long userId = SecurityUtils.getUserId();
// 如果onlySelf为null默认只查询自己的
boolean onlySelfFlag = onlySelf != null ? onlySelf : true;
// 分页查询门诊号码段列表
Page<OutpatientNoSegment> page = new Page<>(pageNo, pageSize);
Page<OutpatientNoSegment> result = outpatientNoSegmentService.selectOutpatientNoSegmentPage(page, onlySelfFlag, userId);
return R.ok(result);
}
/**
* 新增门诊号码段
*
* @param outpatientNoSegment 门诊号码段信息
* @return 操作结果
*/
@Log(title = "门诊号码管理", businessType = BusinessType.INSERT)
@PostMapping
public R<?> addOutpatientNoSegment(@RequestBody OutpatientNoSegment outpatientNoSegment) {
// 校验必填字段
if (StringUtils.isEmpty(outpatientNoSegment.getStartNo()) ||
StringUtils.isEmpty(outpatientNoSegment.getEndNo()) ||
StringUtils.isEmpty(outpatientNoSegment.getUsedNo())) {
return R.fail("起始号码、终止号码和使用号码不能为空");
}
// 校验号码段是否重复
if (outpatientNoSegmentService.checkNumberSegmentOverlap(
outpatientNoSegment.getStartNo(),
outpatientNoSegment.getEndNo(),
null)) {
return R.fail("门诊号码设置重复");
}
// 设置创建人信息
outpatientNoSegment.setOperatorId(SecurityUtils.getUserId());
if (StringUtils.isEmpty(outpatientNoSegment.getOperatorName())) {
outpatientNoSegment.setOperatorName(SecurityUtils.getUsername());
}
outpatientNoSegment.setCreateBy(SecurityUtils.getUsername());
int result = outpatientNoSegmentService.insertOutpatientNoSegment(outpatientNoSegment);
return result > 0 ? R.ok("保存成功") : R.fail("保存失败");
}
/**
* 修改门诊号码段
*
* @param outpatientNoSegment 门诊号码段信息
* @return 操作结果
*/
@Log(title = "门诊号码管理", businessType = BusinessType.UPDATE)
@PutMapping
public R<?> updateOutpatientNoSegment(@RequestBody OutpatientNoSegment outpatientNoSegment) {
// 校验必填字段
if (StringUtils.isEmpty(outpatientNoSegment.getStartNo()) ||
StringUtils.isEmpty(outpatientNoSegment.getEndNo()) ||
StringUtils.isEmpty(outpatientNoSegment.getUsedNo())) {
return R.fail("起始号码、终止号码和使用号码不能为空");
}
// 校验号码段是否重复(排除自身)
if (outpatientNoSegmentService.checkNumberSegmentOverlap(
outpatientNoSegment.getStartNo(),
outpatientNoSegment.getEndNo(),
outpatientNoSegment.getId())) {
return R.fail("门诊号码设置重复");
}
// 设置更新人信息
outpatientNoSegment.setUpdateBy(SecurityUtils.getUsername());
int result = outpatientNoSegmentService.updateOutpatientNoSegment(outpatientNoSegment);
return result > 0 ? R.ok("保存成功") : R.fail("保存失败");
}
/**
* 删除门诊号码段
*
* @param request 包含ids数组的请求对象
* @return 操作结果
*/
@Log(title = "门诊号码管理", businessType = BusinessType.DELETE)
@DeleteMapping
public R<?> deleteOutpatientNoSegment(@RequestBody java.util.Map<String, Object> request) {
// 支持接收 Long[] 或 String[] 或混合类型处理大整数ID
Object idsObj = request.get("ids");
System.out.println("删除请求 - 接收到的ids原始数据: " + idsObj);
System.out.println("删除请求 - 接收到的ids类型: " + (idsObj != null ? idsObj.getClass().getName() : "null"));
if (idsObj == null) {
return R.fail("请选择要删除的数据");
}
// 转换为 Long[] 数组
Long[] ids = null;
if (idsObj instanceof java.util.List) {
java.util.List<?> idList = (java.util.List<?>) idsObj;
ids = new Long[idList.size()];
for (int i = 0; i < idList.size(); i++) {
Object idObj = idList.get(i);
if (idObj instanceof Long) {
ids[i] = (Long) idObj;
} else if (idObj instanceof Integer) {
ids[i] = ((Integer) idObj).longValue();
} else if (idObj instanceof String) {
try {
String idStr = (String) idObj;
System.out.println("删除请求 - 转换字符串ID: " + idStr);
ids[i] = Long.parseLong(idStr);
System.out.println("删除请求 - 转换后的Long ID: " + ids[i]);
// 验证转换是否正确
if (!String.valueOf(ids[i]).equals(idStr)) {
System.out.println("删除请求 - 警告ID转换后值不匹配原始: " + idStr + ", 转换后: " + ids[i]);
}
} catch (NumberFormatException e) {
System.out.println("删除请求 - ID转换失败: " + idObj + ", 错误: " + e.getMessage());
return R.fail("无效的ID格式: " + idObj);
}
} else if (idObj instanceof Number) {
ids[i] = ((Number) idObj).longValue();
} else {
return R.fail("无效的ID类型: " + (idObj != null ? idObj.getClass().getName() : "null"));
}
}
} else if (idsObj instanceof Long[]) {
ids = (Long[]) idsObj;
} else {
return R.fail("无效的ID数组格式");
}
System.out.println("删除请求 - 转换后的ids: " + java.util.Arrays.toString(ids));
if (ids == null || ids.length == 0) {
return R.fail("请选择要删除的数据");
}
// 获取当前用户ID
Long userId = SecurityUtils.getUserId();
System.out.println("删除请求 - 当前用户ID: " + userId);
// 校验删除权限和使用状态
for (Long id : ids) {
System.out.println("删除验证 - 检查ID: " + id);
OutpatientNoSegment segment = outpatientNoSegmentService.getById(id);
if (segment == null) {
// 记录日志以便调试
System.out.println("删除失败记录不存在ID=" + id + ",可能已被软删除或不存在");
return R.fail("数据不存在ID: " + id);
}
System.out.println("删除验证 - 找到记录: ID=" + segment.getId() + ", operatorId=" + segment.getOperatorId() + ", usedNo=" + segment.getUsedNo() + ", startNo=" + segment.getStartNo());
// 校验归属权
if (!segment.getOperatorId().equals(userId)) {
System.out.println("删除验证 - 权限检查失败: segment.operatorId=" + segment.getOperatorId() + ", userId=" + userId);
return R.fail("只能删除自己维护的门诊号码段");
}
// 校验使用状态(使用号码=起始号码表示未使用)
if (!segment.getUsedNo().equals(segment.getStartNo())) {
System.out.println("删除验证 - 使用状态检查失败: usedNo=" + segment.getUsedNo() + ", startNo=" + segment.getStartNo());
return R.fail("已有门诊号码段已有使用的门诊号码,请核对!");
}
}
System.out.println("删除验证 - 所有检查通过,开始执行删除");
int rows = outpatientNoSegmentService.deleteOutpatientNoSegmentByIds(ids);
System.out.println("删除执行 - 影响行数: " + rows);
return rows > 0 ? R.ok("删除成功") : R.fail("删除失败");
}
}

View File

@@ -90,6 +90,14 @@ public class HealthcareServiceDto {
private Integer appointmentRequiredFlag;
private String appointmentRequiredFlag_enumText;
/**
* 出诊医生ID
*/
@Dict(dictTable = "adm_practitioner", dictCode = "id", dictText = "name")
@JsonSerialize(using = ToStringSerializer.class)
private Long practitionerId;
private String practitionerId_dictText;
/**
* 费用定价ID
*/

View File

@@ -82,6 +82,11 @@ public class HealthcareServiceFormData {
@NotBlank(message = "预约要求不能为空")
private Integer appointmentRequiredFlag;
/**
* 出诊医生ID
*/
private Long practitionerId;
/** 医保编码 */
private String ybNo;

View File

@@ -12,6 +12,8 @@ import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import lombok.extern.slf4j.Slf4j;
import java.util.List;
/**
* 患者换卡服务实现类
*
@@ -43,8 +45,8 @@ public class PatientCardRenewalServiceImpl implements PatientCardRenewalService
// 2. 检查新卡号是否已被使用
LambdaQueryWrapper<PatientIdentifier> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.eq(PatientIdentifier::getIdentifierNo, request.getNewCardNo());
PatientIdentifier existingIdentifier = patientIdentifierService.getOne(queryWrapper);
if (existingIdentifier != null) {
List<PatientIdentifier> existingIdentifiers = patientIdentifierService.list(queryWrapper);
if (existingIdentifiers != null && !existingIdentifiers.isEmpty()) {
throw new IllegalArgumentException("新卡号已被其他患者使用,请更换新卡号");
}

View File

@@ -98,10 +98,20 @@ public class OutpatientRegistrationAppServiceImpl implements IOutpatientRegistra
queryWrapper.orderByDesc("update_time");
// 通过证件号匹配 patient
if (StringUtils.isNotEmpty(searchKey)) {
PatientIdentifier patientIdentifier = patientIdentifierService
.getOne(new LambdaQueryWrapper<PatientIdentifier>().eq(PatientIdentifier::getIdentifierNo, searchKey));
if (patientIdentifier != null) {
queryWrapper.or(q -> q.eq("id", patientIdentifier.getPatientId()));
List<PatientIdentifier> patientIdentifiers = patientIdentifierService
.list(new LambdaQueryWrapper<PatientIdentifier>().eq(PatientIdentifier::getIdentifierNo, searchKey));
if (patientIdentifiers != null && !patientIdentifiers.isEmpty()) {
// 如果有多个匹配结果,将它们全部添加到查询条件中
if (patientIdentifiers.size() == 1) {
// 单个结果时直接添加条件
queryWrapper.or(q -> q.eq("id", patientIdentifiers.get(0).getPatientId()));
} else {
// 多个结果时使用in条件
List<Long> patientIds = patientIdentifiers.stream()
.map(PatientIdentifier::getPatientId)
.collect(Collectors.toList());
queryWrapper.or(q -> q.in("id", patientIds));
}
}
}
// 患者信息
@@ -120,10 +130,11 @@ public class OutpatientRegistrationAppServiceImpl implements IOutpatientRegistra
e.setFirstEnum_enumText(patientIdList.contains(e.getId()) ? EncounterType.FOLLOW_UP.getInfo()
: EncounterType.INITIAL.getInfo());
// 患者标识
PatientIdentifier patientIdentifier = patientIdentifierService
.getOne(new LambdaQueryWrapper<PatientIdentifier>().eq(PatientIdentifier::getPatientId, e.getId()));
if (patientIdentifier != null) {
e.setIdentifierNo(patientIdentifier.getIdentifierNo());
List<PatientIdentifier> patientIdentifiers = patientIdentifierService
.list(new LambdaQueryWrapper<PatientIdentifier>().eq(PatientIdentifier::getPatientId, e.getId()));
if (patientIdentifiers != null && !patientIdentifiers.isEmpty()) {
// 取第一个标识号,如果需要可以根据业务需求选择其他逻辑
e.setIdentifierNo(patientIdentifiers.get(0).getIdentifierNo());
}
});

View File

@@ -147,10 +147,9 @@ public class CurrentDayEncounterDto {
private String operatorName;
/**
* 退号操作工号
* 退号操作工号(用户账号)
*/
@JsonSerialize(using = ToStringSerializer.class)
private Long operatorId;
private String operatorId;
/**
* 退款金额

View File

@@ -0,0 +1,29 @@
package com.openhis.web.check.appservice;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckMethod;
import io.swagger.models.auth.In;
import javax.servlet.http.HttpServletResponse;
/**
* 检查方法Service接口
*
* @author system
* @date 2025-07-22
*/
public interface ICheckMethodAppService{
R<?> getCheckMethodList();
R<?> addCheckMethod(CheckMethod checkMethod);
R<?> updateCheckMethod(CheckMethod checkPart);
R<?> removeCheckMethod(Integer checkMethodId);
R<?> searchCheckMethodList(Integer pageNo, Integer pageSize, String checkType, String name, String packageName);
R<?> exportCheckMethod(String checkType, String name, String packageName, HttpServletResponse response);
}

View File

@@ -0,0 +1,48 @@
package com.openhis.web.check.appservice;
import com.core.common.core.domain.R;
import com.openhis.web.check.dto.CheckPackageDto;
/**
* 检查套餐AppService接口
*
* @author system
* @date 2025-11-26
*/
public interface ICheckPackageAppService {
/**
* 获取检查套餐列表
* @return 检查套餐列表
*/
R<?> getCheckPackageList();
/**
* 根据ID获取检查套餐详情
* @param id 套餐ID
* @return 套餐详情
*/
R<?> getCheckPackageById(Long id);
/**
* 新增检查套餐
* @param checkPackageDto 套餐信息
* @return 新增结果
*/
R<?> addCheckPackage(CheckPackageDto checkPackageDto);
/**
* 更新检查套餐
* @param checkPackageDto 套餐信息
* @return 更新结果
*/
R<?> updateCheckPackage(CheckPackageDto checkPackageDto);
/**
* 删除检查套餐
* @param id 套餐ID
* @return 删除结果
*/
R<?> deleteCheckPackage(Long id);
}

View File

@@ -0,0 +1,20 @@
package com.openhis.web.check.appservice;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckPart;
import javax.servlet.http.HttpServletResponse;
public interface ICheckPartAppService {
R<?> getCheckPartList();
R<?> addCheckPart(CheckPart checkPart);
R<?> removeCheckPart(Integer checkPartId);
R<?> updateCheckPart(CheckPart checkPart);
R<?> searchCheckPartList(Integer pageNo, Integer pageSize, String checkType, String name, String packageName);
R<?> exportCheckPart(String checkType, String name, String packageName, HttpServletResponse response);
}

View File

@@ -0,0 +1,14 @@
package com.openhis.web.check.appservice;
import com.core.common.core.domain.R;
import com.openhis.check.domain.LisGroupInfo;
public interface ILisGroupInfoAppService {
R<?> getLisGroupInfoList();
R<?> add(LisGroupInfo lisGroupInfo);
R<?> update(LisGroupInfo lisGroupInfo);
R<?> delete(Integer lisGroupInfoId);
}

View File

@@ -0,0 +1,129 @@
package com.openhis.web.check.appservice.impl;
import cn.hutool.core.util.ObjectUtil;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckMethod;
import com.openhis.check.service.ICheckMethodService;
import com.openhis.web.check.appservice.ICheckMethodAppService;
import com.openhis.web.reportmanage.utils.ExcelFillerUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
@Service
@Slf4j
public class CheckMethodAppServiceImpl implements ICheckMethodAppService {
@Resource
private ICheckMethodService checkMethodService;
@Override
public R<?> getCheckMethodList() {
List<CheckMethod> list = checkMethodService.list();
return R.ok(list);
}
@Override
public R<?> searchCheckMethodList(Integer pageNo, Integer pageSize, String checkType, String name, String packageName) {
LambdaQueryWrapper<CheckMethod> wrapper = new LambdaQueryWrapper<>();
if (checkType != null && ObjectUtil.isNotEmpty(checkType)) {
wrapper.eq(CheckMethod::getCheckType, checkType);
}
if (name != null && ObjectUtil.isNotEmpty(name)) {
wrapper.like(CheckMethod::getName, name);
}
if (packageName != null && ObjectUtil.isNotEmpty(packageName)) {
wrapper.eq(CheckMethod::getPackageName, packageName);
}
List<CheckMethod> list = checkMethodService.list(wrapper);
return R.ok(list);
}
@Override
public R<?> addCheckMethod(CheckMethod checkMethod) {
//1.数据校验
if (ObjectUtil.isEmpty(checkMethod.getName())) {
return R.fail("检查方法名称不能为空!");
}
if (ObjectUtil.isEmpty(checkMethod.getCode())) {
return R.fail("检查方法代码不能为空!");
}
if (ObjectUtil.isEmpty(checkMethod.getCheckType())) {
return R.fail("检查方法的检查类型不能为空!");
}
//2.保存
boolean save = checkMethodService.save(checkMethod);
return R.ok(save);
}
@Override
public R<?> updateCheckMethod(CheckMethod checkMethod) {
//1.数据校验
if (ObjectUtil.isEmpty(checkMethod.getName())) {
return R.fail("检查方法名称不能为空!");
}
if (ObjectUtil.isEmpty(checkMethod.getCode())) {
return R.fail("检查方法代码不能为空!");
}
if (ObjectUtil.isEmpty(checkMethod.getCheckType())) {
return R.fail("检查方法的检查类型不能为空!");
}
//2.更新
boolean b = checkMethodService.updateById(checkMethod);
return R.ok(b);
}
@Override
public R<?> removeCheckMethod(Integer checkMethodId) {
boolean remove = checkMethodService.removeById(checkMethodId);
return R.ok(remove);
}
@Override
public R<?> exportCheckMethod(String checkType, String name, String packageName, HttpServletResponse response) {
LambdaQueryWrapper<CheckMethod> wrapper = new LambdaQueryWrapper<>();
if (checkType != null && ObjectUtil.isNotEmpty(checkType)) {
wrapper.eq(CheckMethod::getCheckType, checkType);
}
if (name != null && ObjectUtil.isNotEmpty(name)) {
wrapper.like(CheckMethod::getName, name);
}
if (packageName != null && ObjectUtil.isNotEmpty(packageName)) {
wrapper.eq(CheckMethod::getPackageName, packageName);
}
List<CheckMethod> list = checkMethodService.list(wrapper);
if (list.isEmpty()) {
return R.fail("导出Excel失败,无数据。");
}
try {
// 准备表头key对应实体的字段名
Map<String, String> headers = new LinkedHashMap<>();
headers.put("checkType", "检查类型");
headers.put("code", "方法代码");
headers.put("name", "方法名称");
headers.put("packageName", "套餐名称");
headers.put("exposureNum", "曝光次数");
headers.put("orderNum", "序号");
headers.put("remark", "备注");
// 文件名,只传文字部分
String excelName = "检查方法列表";
// 导出到Excel
ExcelFillerUtil.makeExcelFile(response, list, headers, excelName, null);
} catch (IOException | IllegalAccessException e) {
e.printStackTrace();
return R.fail("导出Excel失败" + e.getMessage());
}
return R.ok(null, "导出Excel成功");
}
}

View File

@@ -0,0 +1,206 @@
package com.openhis.web.check.appservice.impl;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckPackage;
import com.openhis.check.domain.CheckPackageDetail;
import com.openhis.check.service.ICheckPackageDetailService;
import com.openhis.check.service.ICheckPackageService;
import com.openhis.web.check.appservice.ICheckPackageAppService;
import com.openhis.web.check.dto.CheckPackageDetailDto;
import com.openhis.web.check.dto.CheckPackageDto;
import lombok.AllArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.BeanUtils;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;
/**
* 检查套餐AppService实现
*
* @author system
* @date 2025-11-26
*/
@Slf4j
@Service
@AllArgsConstructor
public class CheckPackageAppServiceImpl implements ICheckPackageAppService {
private final ICheckPackageService checkPackageService;
private final ICheckPackageDetailService checkPackageDetailService;
@Override
public R<?> getCheckPackageList() {
try {
List<CheckPackage> list = checkPackageService.list();
return R.ok(list);
} catch (Exception e) {
log.error("获取检查套餐列表失败", e);
return R.fail("获取检查套餐列表失败");
}
}
@Override
public R<?> getCheckPackageById(Long id) {
try {
CheckPackage checkPackage = checkPackageService.getById(id);
if (checkPackage == null) {
return R.fail("套餐不存在");
}
// 获取套餐明细
List<CheckPackageDetail> details = checkPackageDetailService.list(
new LambdaQueryWrapper<CheckPackageDetail>()
.eq(CheckPackageDetail::getPackageId, id)
.orderByAsc(CheckPackageDetail::getOrderNum)
);
// 转换为DTO
CheckPackageDto dto = new CheckPackageDto();
BeanUtils.copyProperties(checkPackage, dto);
List<CheckPackageDetailDto> detailDtos = details.stream().map(detail -> {
CheckPackageDetailDto detailDto = new CheckPackageDetailDto();
BeanUtils.copyProperties(detail, detailDto);
return detailDto;
}).collect(Collectors.toList());
dto.setItems(detailDtos);
return R.ok(dto);
} catch (Exception e) {
log.error("获取检查套餐详情失败", e);
return R.fail("获取检查套餐详情失败");
}
}
@Override
@Transactional(rollbackFor = Exception.class)
public R<?> addCheckPackage(CheckPackageDto checkPackageDto) {
try {
// 创建套餐主表数据
CheckPackage checkPackage = new CheckPackage();
BeanUtils.copyProperties(checkPackageDto, checkPackage);
// 设置套餐维护日期为当前系统日期
checkPackage.setMaintainDate(LocalDate.now());
checkPackage.setCreateTime(LocalDateTime.now());
checkPackage.setUpdateTime(LocalDateTime.now());
// 保存套餐主表
boolean saveResult = checkPackageService.save(checkPackage);
if (!saveResult) {
return R.fail("保存套餐失败");
}
// 保存套餐明细
if (checkPackageDto.getItems() != null && !checkPackageDto.getItems().isEmpty()) {
List<CheckPackageDetail> details = new ArrayList<>();
int orderNum = 1;
for (CheckPackageDetailDto detailDto : checkPackageDto.getItems()) {
CheckPackageDetail detail = new CheckPackageDetail();
BeanUtils.copyProperties(detailDto, detail);
detail.setPackageId(checkPackage.getId());
detail.setOrderNum(orderNum++);
detail.setCreateTime(LocalDateTime.now());
detail.setUpdateTime(LocalDateTime.now());
details.add(detail);
}
checkPackageDetailService.saveBatch(details);
}
return R.ok(checkPackage.getId(), "保存成功");
} catch (Exception e) {
log.error("新增检查套餐失败", e);
return R.fail("新增检查套餐失败: " + e.getMessage());
}
}
@Override
@Transactional(rollbackFor = Exception.class)
public R<?> updateCheckPackage(CheckPackageDto checkPackageDto) {
try {
// 检查套餐是否存在
CheckPackage existPackage = checkPackageService.getById(checkPackageDto.getId());
if (existPackage == null) {
return R.fail("套餐不存在");
}
// 更新套餐主表数据
CheckPackage checkPackage = new CheckPackage();
BeanUtils.copyProperties(checkPackageDto, checkPackage);
// 更新套餐维护日期为当前系统日期
checkPackage.setMaintainDate(LocalDate.now());
checkPackage.setUpdateTime(LocalDateTime.now());
boolean updateResult = checkPackageService.updateById(checkPackage);
if (!updateResult) {
return R.fail("更新套餐失败");
}
// 删除原有明细
checkPackageDetailService.remove(
new LambdaQueryWrapper<CheckPackageDetail>()
.eq(CheckPackageDetail::getPackageId, checkPackage.getId())
);
// 保存新的套餐明细
if (checkPackageDto.getItems() != null && !checkPackageDto.getItems().isEmpty()) {
List<CheckPackageDetail> details = new ArrayList<>();
int orderNum = 1;
for (CheckPackageDetailDto detailDto : checkPackageDto.getItems()) {
CheckPackageDetail detail = new CheckPackageDetail();
BeanUtils.copyProperties(detailDto, detail);
detail.setPackageId(checkPackage.getId());
detail.setOrderNum(orderNum++);
detail.setCreateTime(LocalDateTime.now());
detail.setUpdateTime(LocalDateTime.now());
details.add(detail);
}
checkPackageDetailService.saveBatch(details);
}
return R.ok("更新成功");
} catch (Exception e) {
log.error("更新检查套餐失败", e);
return R.fail("更新检查套餐失败: " + e.getMessage());
}
}
@Override
@Transactional(rollbackFor = Exception.class)
public R<?> deleteCheckPackage(Long id) {
try {
// 检查套餐是否存在
CheckPackage existPackage = checkPackageService.getById(id);
if (existPackage == null) {
return R.fail("套餐不存在");
}
// 删除套餐明细
checkPackageDetailService.remove(
new LambdaQueryWrapper<CheckPackageDetail>()
.eq(CheckPackageDetail::getPackageId, id)
);
// 删除套餐主表
boolean deleteResult = checkPackageService.removeById(id);
if (!deleteResult) {
return R.fail("删除套餐失败");
}
return R.ok("删除成功");
} catch (Exception e) {
log.error("删除检查套餐失败", e);
return R.fail("删除检查套餐失败: " + e.getMessage());
}
}
}

View File

@@ -0,0 +1,109 @@
package com.openhis.web.check.appservice.impl;
import cn.hutool.core.util.ObjectUtil;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckPart;
import com.openhis.check.service.ICheckPartService;
import com.openhis.web.check.appservice.ICheckPartAppService;
import com.openhis.web.reportmanage.utils.ExcelFillerUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
@Service
@Slf4j
public class CheckPartAppServiceImpl implements ICheckPartAppService {
@Resource
private ICheckPartService checkPartService;
@Override
public R<?> getCheckPartList() {
List<CheckPart> list = checkPartService.list();
return R.ok(list);
}
@Override
public R<?> searchCheckPartList(Integer pageNo, Integer pageSize, String checkType, String name, String packageName) {
LambdaQueryWrapper<CheckPart> wrapper = new LambdaQueryWrapper<>();
if (checkType != null && ObjectUtil.isNotEmpty(checkType)) {
wrapper.eq(CheckPart::getCheckType, checkType);
}
if (name != null && ObjectUtil.isNotEmpty(name)) {
wrapper.like(CheckPart::getName, name);
}
if (packageName != null && ObjectUtil.isNotEmpty(packageName)) {
wrapper.eq(CheckPart::getPackageName, packageName);
}
List<CheckPart> list = checkPartService.list(wrapper);
return R.ok(list);
}
@Override
public R<?> addCheckPart(CheckPart checkPart) {
//数据检验
//保存
boolean save = checkPartService.save(checkPart);
return R.ok(save);
}
@Override
public R<?> removeCheckPart(Integer checkPartId) {
boolean remove = checkPartService.removeById(checkPartId);
return R.ok(remove);
}
@Override
public R<?> updateCheckPart(CheckPart checkPart) {
boolean b = checkPartService.updateById(checkPart);
return R.ok(b);
}
@Override
public R<?> exportCheckPart(String checkType, String name, String packageName, HttpServletResponse response) {
LambdaQueryWrapper<CheckPart> wrapper = new LambdaQueryWrapper<>();
if (checkType != null && ObjectUtil.isNotEmpty(checkType)) {
wrapper.eq(CheckPart::getCheckType, checkType);
}
if (name != null && ObjectUtil.isNotEmpty(name)) {
wrapper.like(CheckPart::getName, name);
}
if (packageName != null && ObjectUtil.isNotEmpty(packageName)) {
wrapper.eq(CheckPart::getPackageName, packageName);
}
List<CheckPart> list = checkPartService.list(wrapper);
if (list.isEmpty()) {
return R.fail("导出Excel失败,无数据。");
}
try {
// 准备表头key对应实体的字段名
Map<String, String> headers = new LinkedHashMap<>();
headers.put("checkType", "检查类型");
headers.put("code", "部位代码");
headers.put("name", "部位名称");
headers.put("packageName", "套餐名称");
headers.put("exposureNum", "曝光次数");
headers.put("price", "金额");
headers.put("number", "序号");
headers.put("serviceScope", "服务范围");
headers.put("subType", "下级医技类型");
headers.put("remark", "备注");
// 文件名,只传文字部分
String excelName = "检查部位列表";
// 导出到Excel
ExcelFillerUtil.makeExcelFile(response, list, headers, excelName, null);
} catch (IOException | IllegalAccessException e) {
e.printStackTrace();
return R.fail("导出Excel失败" + e.getMessage());
}
return R.ok(null, "导出Excel成功");
}
}

View File

@@ -0,0 +1,48 @@
package com.openhis.web.check.appservice.impl;
import cn.hutool.core.util.ObjectUtil;
import com.core.common.core.domain.R;
import com.openhis.check.domain.LisGroupInfo;
import com.openhis.check.service.ILisGroupInfoService;
import com.openhis.web.check.appservice.ILisGroupInfoAppService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import java.util.List;
@Service
@Slf4j
public class LisGroupInfoAppServiceImpl implements ILisGroupInfoAppService {
@Resource
private ILisGroupInfoService lisGroupInfoService;
@Override
public R<?> getLisGroupInfoList() {
List<LisGroupInfo> list = lisGroupInfoService.list();
return R.ok(list);
}
@Override
public R<?> add(LisGroupInfo lisGroupInfo) {
if (ObjectUtil.isEmpty(lisGroupInfo)) {
return R.fail("信息不能为空");
}
boolean save = lisGroupInfoService.save(lisGroupInfo);
return R.ok(save);
}
@Override
public R<?> update(LisGroupInfo lisGroupInfo) {
if (ObjectUtil.isEmpty(lisGroupInfo)) {
return R.fail("信息不能为空");
}
boolean update = lisGroupInfoService.updateById(lisGroupInfo);
return R.ok( update);
}
@Override
public R<?> delete(Integer lisGroupInfoId) {
boolean b = lisGroupInfoService.removeById(lisGroupInfoId);
return R.ok(b);
}
}

View File

@@ -0,0 +1,82 @@
package com.openhis.web.check.controller;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckMethod;
import com.openhis.web.check.appservice.ICheckMethodAppService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.*;
import javax.annotation.Resource;
import javax.servlet.http.HttpServletResponse;
@RestController
@Slf4j
@RequestMapping("/check/method")
public class CheckMethodController {
@Resource
private ICheckMethodAppService checkMethodAppService;
/*
* 获取检查方法
* @Param 此处参数注释有问题,完全是按照需求给的图来注释的
*/
@GetMapping("/list")
public R<?> getCheckMethodList(){
return R.ok(checkMethodAppService.getCheckMethodList());
}
/*
* 条件查询检查方法
* @Para
* */
@GetMapping("/search")
public R<?> searchCheckMethodList(
@RequestParam(required = false) Integer pageNo,
@RequestParam(required = false) Integer pageSize,
@RequestParam(required = false) String checkType,
@RequestParam(required = false) String name,
@RequestParam(required = false) String packageName) {
return R.ok(checkMethodAppService.searchCheckMethodList(pageNo,pageSize,checkType,name,packageName));
}
/*
* 新增检查方法
* @Param
*/
@PostMapping("/add")
public R<?> addCheckMethod(@RequestBody CheckMethod checkMethod){
return R.ok(checkMethodAppService.addCheckMethod(checkMethod));
}
/*
* 删除检查方法
* @Param code代码
*/
@DeleteMapping("/delete/{checkMethodId}")
public R<?> deleteCheckMethod(@PathVariable Integer checkMethodId){
return R.ok(checkMethodAppService.removeCheckMethod(checkMethodId));
}
/*
* 更新检查方法
* @Param
*/
@PutMapping("/update")
public R<?> updateCheckMethod(@RequestBody CheckMethod checkMethod){
return R.ok(checkMethodAppService.updateCheckMethod(checkMethod));
}
/*
* 导出检查方法列表
* @Param
*/
@GetMapping("/export")
public void exportCheckMethod(
@RequestParam(required = false) String checkType,
@RequestParam(required = false) String name,
@RequestParam(required = false) String packageName,
HttpServletResponse response) {
checkMethodAppService.exportCheckMethod(checkType, name, packageName, response);
}
}

View File

@@ -0,0 +1,80 @@
package com.openhis.web.check.controller;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckPart;
import com.openhis.web.check.appservice.ICheckPartAppService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.*;
import javax.annotation.Resource;
import javax.servlet.http.HttpServletResponse;
@RestController
@Slf4j
@RequestMapping("/check/part")
public class CheckPartController {
@Resource
private ICheckPartAppService checkPartAppService;
/*
* 获取检查部位
* @Param检查方法 此处参数注释有问题,完全是按照需求给的图来注释的
*/
@GetMapping("/list")
public R<?> getCheckPartList(){
return R.ok(checkPartAppService.getCheckPartList());
}
/*
* 条件搜索检查部位
* @Param
* */
@GetMapping("/search")
public R<?> searchCheckPartList(@RequestParam(required = false) Integer pageNo,
@RequestParam(required = false) Integer pageSize,
@RequestParam(required = false) String checkType,
@RequestParam(required = false) String name,
@RequestParam(required = false) String packageName){
return R.ok(checkPartAppService.searchCheckPartList(pageNo,pageSize,checkType,name,packageName));
}
/*
* 新增检查部位
* @Param
*/
@PostMapping("/add")
public R<?> addCheckPart(@RequestBody CheckPart checkPart){
return R.ok(checkPartAppService.addCheckPart(checkPart));
}
/*
* 删除检查部位
* @Param code代码
*/
@DeleteMapping("/delete/{checkPartId}")
public R<?> deleteCheckPart(@PathVariable Integer checkPartId){
return R.ok(checkPartAppService.removeCheckPart(checkPartId));
}
/*
* 更新检查部位
* @Param
*/
@PutMapping("/update")
public R<?> updateCheckPart(@RequestBody CheckPart checkPart){
return R.ok(checkPartAppService.updateCheckPart(checkPart));
}
/*
* 导出检查部位列表
* @Param
*/
@GetMapping("/export")
public void exportCheckPart(
@RequestParam(required = false) String checkType,
@RequestParam(required = false) String name,
@RequestParam(required = false) String packageName,
HttpServletResponse response) {
checkPartAppService.exportCheckPart(checkType, name, packageName, response);
}
}

View File

@@ -0,0 +1,186 @@
package com.openhis.web.check.controller;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.core.common.core.controller.BaseController;
import com.core.common.core.domain.AjaxResult;
import com.core.common.core.domain.R;
import com.openhis.check.domain.CheckMethod;
import com.openhis.check.domain.CheckPackage;
import com.openhis.check.domain.CheckPart;
import com.openhis.check.domain.CheckType;
import com.openhis.check.service.ICheckMethodService;
import com.openhis.check.service.ICheckPackageService;
import com.openhis.check.service.ICheckPartService;
import com.openhis.check.service.ICheckTypeService;
import com.openhis.web.check.appservice.ICheckPackageAppService;
import com.openhis.web.check.dto.CheckPackageDto;
import lombok.AllArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.*;
import javax.validation.Valid;
import java.time.LocalDate;
import java.util.List;
/**
* 检查类型管理Controller
*
* @author system
* @date 2025-07-22
* @updated 2025-11-26 - 增加套餐设置相关接口
*/
@RestController
@RequestMapping({"/system/check-type", "/system"})
@Slf4j
@AllArgsConstructor
public class CheckTypeController extends BaseController {
private final ICheckTypeService checkTypeService;
private final ICheckMethodService checkMethodService;
private final ICheckPartService checkPartService;
private final ICheckPackageService checkPackageService;
private final ICheckPackageAppService checkPackageAppService;
/**
* 获取检查类型列表
*/
@GetMapping("/list")
public AjaxResult list() {
List<CheckType> list = checkTypeService.list();
return AjaxResult.success(list);
}
/**
* 获取检查方法列表
*/
@GetMapping({"/method/list", "/check-method/list"})
public AjaxResult methodList() {
List<CheckMethod> list = checkMethodService.list();
return AjaxResult.success(list);
}
/**
* 获取检查部位列表
*/
@GetMapping({"/part/list", "/check-part/list"})
public AjaxResult partList() {
List<CheckPart> list = checkPartService.list();
return AjaxResult.success(list);
}
/**
* 获取检查套餐列表(支持分页和筛选)
*/
@GetMapping({"/package/list", "/check-package/list"})
public AjaxResult packageList(
@RequestParam(required = false) Integer pageNo,
@RequestParam(required = false) Integer pageSize,
@RequestParam(required = false) String organization,
@RequestParam(required = false) String packageName,
@RequestParam(required = false) String packageLevel,
@RequestParam(required = false) String packageType,
@RequestParam(required = false) String department,
@RequestParam(required = false) String user,
@RequestParam(required = false) String startDate,
@RequestParam(required = false) String endDate) {
LambdaQueryWrapper<CheckPackage> wrapper = new LambdaQueryWrapper<>();
// 添加筛选条件
if (organization != null && !organization.isEmpty()) {
wrapper.eq(CheckPackage::getOrganization, organization);
}
if (packageName != null && !packageName.isEmpty()) {
wrapper.like(CheckPackage::getPackageName, packageName);
}
if (packageLevel != null && !packageLevel.isEmpty()) {
wrapper.eq(CheckPackage::getPackageLevel, packageLevel);
}
if (packageType != null && !packageType.isEmpty()) {
wrapper.eq(CheckPackage::getPackageType, packageType);
}
if (department != null && !department.isEmpty()) {
wrapper.like(CheckPackage::getDepartment, department);
}
if (user != null && !user.isEmpty()) {
wrapper.like(CheckPackage::getUser, user);
}
if (startDate != null && !startDate.isEmpty()) {
wrapper.ge(CheckPackage::getMaintainDate, LocalDate.parse(startDate));
}
if (endDate != null && !endDate.isEmpty()) {
wrapper.le(CheckPackage::getMaintainDate, LocalDate.parse(endDate));
}
// 按更新时间倒序排列
wrapper.orderByDesc(CheckPackage::getUpdateTime);
// 如果需要分页
if (pageNo != null && pageSize != null) {
com.baomidou.mybatisplus.extension.plugins.pagination.Page<CheckPackage> page =
new com.baomidou.mybatisplus.extension.plugins.pagination.Page<>(pageNo, pageSize);
com.baomidou.mybatisplus.extension.plugins.pagination.Page<CheckPackage> result =
checkPackageService.page(page, wrapper);
return AjaxResult.success(result);
} else {
List<CheckPackage> list = checkPackageService.list(wrapper);
return AjaxResult.success(list);
}
}
/**
* 新增检查类型
*/
@PostMapping
public AjaxResult add(@RequestBody CheckType checkType) {
return toAjax(checkTypeService.save(checkType));
}
/**
* 修改检查类型
*/
@PutMapping
public AjaxResult edit(@RequestBody CheckType checkType) {
return toAjax(checkTypeService.updateById(checkType));
}
/**
* 删除检查类型
*/
@DeleteMapping("/{checkTypeId}")
public AjaxResult remove(@PathVariable Long checkTypeId) {
return toAjax(checkTypeService.removeById(checkTypeId));
}
/**
* 根据ID获取检查套餐详情
*/
@GetMapping({"/package/{id}", "/check-package/{id}"})
public R<?> getCheckPackageById(@PathVariable Long id) {
return checkPackageAppService.getCheckPackageById(id);
}
/**
* 新增检查套餐
*/
@PostMapping({"/package", "/check-package"})
public R<?> addCheckPackage(@Valid @RequestBody CheckPackageDto checkPackageDto) {
return checkPackageAppService.addCheckPackage(checkPackageDto);
}
/**
* 更新检查套餐
*/
@PutMapping({"/package", "/check-package"})
public R<?> updateCheckPackage(@Valid @RequestBody CheckPackageDto checkPackageDto) {
return checkPackageAppService.updateCheckPackage(checkPackageDto);
}
/**
* 删除检查套餐
*/
@DeleteMapping({"/package/{id}", "/check-package/{id}"})
public R<?> deleteCheckPackage(@PathVariable Long id) {
return checkPackageAppService.deleteCheckPackage(id);
}
}

View File

@@ -0,0 +1,56 @@
package com.openhis.web.check.controller;
import com.core.common.core.domain.R;
import com.openhis.web.check.appservice.ILisGroupInfoAppService;
import org.springframework.web.bind.annotation.*;
import com.openhis.check.domain.LisGroupInfo;
import javax.annotation.Resource;
@RestController
@RequestMapping("/check/lisGroupInfo")
public class LisGroupInfoController {
@Resource
private ILisGroupInfoAppService lisGroupInfoAppService;
/*
*
* 获取Lis分组信息
*
* */
@GetMapping("/list")
public R<?> getLisGroupInfoList(){
return R.ok(lisGroupInfoAppService.getLisGroupInfoList());
}
/*
*
* 新增Lis分组信息
*
* */
@PostMapping("/add")
public R<?> addLisGroupInfo(@RequestBody LisGroupInfo lisGroupInfo){
return R.ok(lisGroupInfoAppService.add(lisGroupInfo));
}
/*
*
* 修改Lis分组信息
*
* */
@PutMapping("/update")
public R<?> updateLisGroupInfo(@RequestBody LisGroupInfo lisGroupInfo){
return R.ok(lisGroupInfoAppService.update(lisGroupInfo));
}
/*
*
* 删除Lis分组信息
*
* */
@DeleteMapping("/{lisGroupInfoId}")
public R<?> deleteLisGroupInfo(@PathVariable Integer lisGroupInfoId){
return R.ok(lisGroupInfoAppService.delete(lisGroupInfoId));
}
}

View File

@@ -0,0 +1,46 @@
package com.openhis.web.check.dto;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import lombok.Data;
import lombok.experimental.Accessors;
import java.time.LocalDateTime;
@Data
@Accessors(chain = true)
public class CheckMethodDto {
/**
* 检查方法ID
*/
@TableId(type = IdType.AUTO)
private Long id;
/* 检查类型 */
private String checkType;
/* 方法代码 */
private String code;
/* 方法名称 */
private String name;
/* 套餐名称 */
private String packageName;
/* 曝光次数 */
private Integer exposureNum;
/* 序号 */
private Integer orderNum;
/* 备注 */
private String remark;
/** 创建时间 */
private LocalDateTime createTime;
/** 更新时间 */
private LocalDateTime updateTime;
}

View File

@@ -0,0 +1,71 @@
package com.openhis.web.check.dto;
import lombok.Data;
import lombok.experimental.Accessors;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotNull;
import java.math.BigDecimal;
/**
* 检查套餐明细DTO
*
* @author system
* @date 2025-11-26
*/
@Data
@Accessors(chain = true)
public class CheckPackageDetailDto {
/** 套餐明细ID */
private Long id;
/** 套餐ID */
private Long packageId;
/** 项目编号 */
private String itemCode;
/** 项目名称/规格 */
@NotBlank(message = "项目名称不能为空")
private String itemName;
/** 检查项目ID(诊疗项目ID) */
private Long checkItemId;
/** 剂量 */
private String dose;
/** 途径 */
private String method;
/** 频次 */
private String frequency;
/** 天数 */
private String days;
/** 数量 */
@NotNull(message = "数量不能为空")
private Integer quantity;
/** 单价 */
@NotNull(message = "单价不能为空")
private BigDecimal unitPrice;
/** 金额 */
private BigDecimal amount;
/** 服务费 */
private BigDecimal serviceCharge;
/** 总金额 */
private BigDecimal total;
/** 产地 */
private String origin;
/** 序号 */
private Integer orderNum;
}

View File

@@ -0,0 +1,86 @@
package com.openhis.web.check.dto;
import lombok.Data;
import lombok.experimental.Accessors;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotNull;
import java.math.BigDecimal;
import java.time.LocalDate;
import java.util.List;
/**
* 检查套餐DTO
*
* @author system
* @date 2025-11-26
*/
@Data
@Accessors(chain = true)
public class CheckPackageDto {
/** 检查套餐ID */
private Long id;
/** 套餐名称 */
@NotBlank(message = "套餐名称不能为空")
private String packageName;
/** 套餐编码 */
private String code;
/** 套餐类别 */
@NotBlank(message = "套餐类别不能为空")
private String packageType;
/** 套餐级别 */
@NotBlank(message = "套餐级别不能为空")
private String packageLevel;
/** 适用科室 */
private String department;
/** 适用用户 */
private String user;
/** 卫生机构 */
private String organization;
/** 套餐金额 */
private BigDecimal packagePrice;
/** 折扣 */
private BigDecimal discount;
/** 制单人 */
private String creator;
/** 是否停用 */
private Integer isDisabled;
/** 显示套餐名 */
private Integer showPackageName;
/** 生成服务费 */
private Integer generateServiceFee;
/** 套餐价格启用状态 */
private Integer packagePriceEnabled;
/** 服务费 */
private BigDecimal serviceFee;
/** 备注 */
private String remark;
/** 描述 */
private String description;
/** 套餐维护日期 */
private LocalDate createDate;
/** 套餐明细列表 */
@NotNull(message = "套餐明细不能为空")
private List<CheckPackageDetailDto> items;
}

View File

@@ -0,0 +1,41 @@
package com.openhis.web.check.dto;
import lombok.Data;
import lombok.experimental.Accessors;
@Data
@Accessors(chain = true)
public class CheckPartDto {
/** 检查部位id */
private Long id;
/** 检查部位名称 */
private String name;
/** 检查部位编码 */
private String code;
/** 检查部位检查类型 */
private String checkType;
/** 曝光次数 */
private Integer exposureNum;
/** 费用套餐 */
private String packageName;
/** 金额 */
private Double price;
/** 序号 */
private Integer number;
/** 服务范围 */
private String serviceScope;
/** 下级医技类型 */
private String subType;
/** 备注 */
private String remark;
}

View File

@@ -0,0 +1,10 @@
package com.openhis.web.check.mapper;
import org.springframework.stereotype.Repository;
@Repository
public interface CheckMethodAppMapper{
}

View File

@@ -0,0 +1,7 @@
package com.openhis.web.check.mapper;
import org.springframework.stereotype.Repository;
@Repository
public interface CheckPartAppMapper {
}

View File

@@ -0,0 +1,7 @@
package com.openhis.web.check.mapper;
import org.springframework.stereotype.Repository;
@Repository
public interface LisGroupInfoAppMapper {
}

View File

@@ -0,0 +1,190 @@
package com.openhis.web.lab.controller;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.core.common.core.controller.BaseController;
import com.core.common.core.domain.AjaxResult;
import com.openhis.lab.domain.InspectionType;
import com.openhis.lab.service.IInspectionTypeService;
import lombok.AllArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.transaction.support.TransactionTemplate;
import org.springframework.web.bind.annotation.*;
import java.util.List;
/**
* 检验类型管理Controller
*
* @author system
* @date 2025-12-09
*/
@RestController
@RequestMapping("/system/inspection-type")
@Slf4j
@AllArgsConstructor
public class InspectionTypeController extends BaseController {
// 辅助方法:将字节数组转换为十六进制字符串
private static String bytesToHex(byte[] bytes) {
StringBuilder sb = new StringBuilder();
for (byte b : bytes) {
sb.append(String.format("%02X ", b));
}
return sb.toString().trim();
}
private final IInspectionTypeService inspectionTypeService;
private final TransactionTemplate transactionTemplate;
/**
* 获取检验类型列表
*/
@GetMapping("/list")
public AjaxResult list(InspectionType inspectionType) {
// 使用Wrapper构建查询条件确保order字段被正确处理
QueryWrapper<InspectionType> queryWrapper = new QueryWrapper<>(inspectionType);
List<InspectionType> list = inspectionTypeService.list(queryWrapper);
return AjaxResult.success(list);
}
/**
* 获取检验类型详细
*/
@GetMapping("/{inspectionTypeId}")
public AjaxResult getInfo(@PathVariable Long inspectionTypeId) {
return AjaxResult.success(inspectionTypeService.getById(inspectionTypeId));
}
/**
* 新增检验类型
*/
@PostMapping
public AjaxResult add(@RequestBody InspectionType inspectionType) {
// 确保新增时ID为null强制使用数据库自增序列
inspectionType.setId(null);
// 去除code字段的前后空格确保唯一性验证准确
if (inspectionType.getCode() != null) {
inspectionType.setCode(inspectionType.getCode().trim());
}
// 验证code字段是否唯一
QueryWrapper<InspectionType> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("code", inspectionType.getCode());
// 输出调试信息
System.out.println("检查编码唯一性code=" + inspectionType.getCode() + ", 长度=" + inspectionType.getCode().length());
System.out.println("code的十六进制表示" + bytesToHex(inspectionType.getCode().getBytes()));
// 直接查询具体记录,而不仅仅是计数
List<InspectionType> existingRecords = inspectionTypeService.list(queryWrapper);
System.out.println("数据库中存在的记录数:" + existingRecords.size());
for (InspectionType record : existingRecords) {
System.out.println("已存在记录id=" + record.getId() + ", code=" + record.getCode() + ", code长度=" + record.getCode().length() + ", code十六进制=" + bytesToHex(record.getCode().getBytes()));
}
if (!existingRecords.isEmpty()) {
return AjaxResult.error("检验类型编码已存在");
}
// 保存前再次验证
System.out.println("准备保存数据:" + inspectionType);
try {
// 使用事务确保一致性
return transactionTemplate.execute(status -> {
// 再次检查,防止并发问题
QueryWrapper<InspectionType> checkWrapper = new QueryWrapper<>();
checkWrapper.eq("code", inspectionType.getCode());
List<InspectionType> finalCheck = inspectionTypeService.list(checkWrapper);
if (!finalCheck.isEmpty()) {
return AjaxResult.error("检验类型编码已存在");
}
boolean result = inspectionTypeService.save(inspectionType);
System.out.println("保存结果:" + result);
return toAjax(result);
});
} catch (Exception e) {
System.out.println("保存失败,错误信息:" + e.getMessage());
// 捕获唯一性约束冲突异常
if (e.getMessage().contains("uk_inspection_type_code") ||
e.getMessage().contains("duplicate key value") ||
e.getMessage().contains("检验类型编码已存在")) {
return AjaxResult.error("检验类型编码已存在");
}
return AjaxResult.error("保存失败:" + e.getMessage());
}
}
/**
* 修改检验类型
*/
@PutMapping
public AjaxResult edit(@RequestBody InspectionType inspectionType) {
// 去除code字段的前后空格确保唯一性验证准确
if (inspectionType.getCode() != null) {
inspectionType.setCode(inspectionType.getCode().trim());
}
// 验证code字段是否唯一排除自身
QueryWrapper<InspectionType> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("code", inspectionType.getCode())
.ne("id", inspectionType.getId());
if (inspectionTypeService.count(queryWrapper) > 0) {
return AjaxResult.error("检验类型编码已存在");
}
try {
boolean result = inspectionTypeService.updateById(inspectionType);
return toAjax(result);
} catch (Exception e) {
// 捕获唯一性约束冲突异常
if (e.getMessage().contains("uk_inspection_type_code") ||
e.getMessage().contains("duplicate key value") ||
e.getMessage().contains("检验类型编码已存在")) {
return AjaxResult.error("检验类型编码已存在");
}
return AjaxResult.error("更新失败:" + e.getMessage());
}
}
/**
* 删除检验类型
*/
@DeleteMapping("/{inspectionTypeId}")
public AjaxResult remove(@PathVariable Long inspectionTypeId) {
log.info("删除检验类型ID: {}", inspectionTypeId);
try {
// 检查记录是否存在
InspectionType existing = inspectionTypeService.getById(inspectionTypeId);
log.info("删除前检查记录是否存在: {}", existing != null);
if (existing == null) {
log.warn("删除失败检验类型ID: {} 不存在", inspectionTypeId);
return AjaxResult.error("删除失败: 记录不存在");
}
// 使用MyBatis Plus的removeById方法执行逻辑删除
boolean result = inspectionTypeService.removeById(inspectionTypeId);
log.info("逻辑删除检验类型结果: {}", result);
AjaxResult response = toAjax(result);
log.info("删除响应: {}", response);
return response;
} catch (Exception e) {
log.error("删除检验类型失败ID: {}, 错误: {}", inspectionTypeId, e.getMessage(), e);
return AjaxResult.error("删除失败: " + e.getMessage());
}
}
// 测试删除接口,直接返回成功
@DeleteMapping("/test-delete/{id}")
public AjaxResult testDelete(@PathVariable Long id) {
log.info("测试删除接口ID: {}", id);
return AjaxResult.success("测试删除成功");
}
}

View File

@@ -6,7 +6,7 @@ spring:
druid:
# 主库数据源
master:
url: jdbc:postgresql://192.168.110.252:15432/postgresql?currentSchema=public&characterEncoding=UTF-8&client_encoding=UTF-8
url: jdbc:postgresql://192.168.110.252:15432/postgresql?currentSchema=hisdev&characterEncoding=UTF-8&client_encoding=UTF-8
username: postgresql
password: Jchl1528
# 从库数据源
@@ -59,6 +59,8 @@ spring:
wall:
config:
multi-statement-allow: true
# redis 配置
redis:
# 地址
@@ -85,3 +87,9 @@ spring:
messages:
basename: i18n/general_message/messages
encoding: utf-8
server:
# 服务器的HTTP端口默认为18080
port: 18080
servlet:
# 应用的访问路径
context-path: /openhis

View File

@@ -6,7 +6,7 @@ spring:
druid:
# 主库数据源
master:
url: jdbc:postgresql://192.168.110.252:15432/postgresql?currentSchema=public&characterEncoding=UTF-8&client_encoding=UTF-8
url: jdbc:postgresql://192.168.110.252:15432/postgresql?currentSchema=hisprd&characterEncoding=UTF-8&client_encoding=UTF-8
username: postgresql
password: Jchl1528
# 从库数据源
@@ -85,3 +85,9 @@ spring:
messages:
basename: i18n/general_message/messages
encoding: utf-8
server:
# 服务器的HTTP端口默认为18080
port: 18082
servlet:
# 应用的访问路径
context-path: /openhis

View File

@@ -0,0 +1,93 @@
# 数据源配置
spring:
datasource:
type: com.alibaba.druid.pool.DruidDataSource
driverClassName: org.postgresql.Driver
druid:
# 主库数据源
master:
url: jdbc:postgresql://192.168.110.252:15432/postgresql?currentSchema=histest&characterEncoding=UTF-8&client_encoding=UTF-8
username: postgresql
password: Jchl1528
# 从库数据源
slave:
# 从数据源开关/默认关闭
enabled:
url:
username:
password:
# 初始连接数
initialSize: 5
# 最小连接池数量
minIdle: 10
# 最大连接池数量
maxActive: 20
# 配置获取连接等待超时的时间
maxWait: 60000
# 配置连接超时时间
connectTimeout: 30000
# 配置网络超时时间
socketTimeout: 60000
# 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒
timeBetweenEvictionRunsMillis: 60000
# 配置一个连接在池中最小生存的时间,单位是毫秒
minEvictableIdleTimeMillis: 300000
# 配置一个连接在池中最大生存的时间,单位是毫秒
maxEvictableIdleTimeMillis: 900000
# 配置检测连接是否有效
validationQuery: SELECT 1 # FROM DUAL
testWhileIdle: true
testOnBorrow: false
testOnReturn: false
webStatFilter:
enabled: true
statViewServlet:
enabled: true
# 设置白名单,不填则允许所有访问
allow:
url-pattern: /druid/*
# 控制台管理用户名和密码
login-username: openhis
login-password: 123456
filter:
stat:
enabled: true
# 慢SQL记录
log-slow-sql: true
slow-sql-millis: 1000
merge-sql: true
wall:
config:
multi-statement-allow: true
# redis 配置
redis:
# 地址
host: 192.168.110.252
# 端口默认为6379
port: 6379
# 数据库索引
database: 1
# 密码
password: Jchl1528
# 连接超时时间
timeout: 10s
lettuce:
pool:
# 连接池中的最小空闲连接
min-idle: 0
# 连接池中的最大空闲连接
max-idle: 8
# 连接池的最大数据库连接数
max-active: 8
# #连接池最大阻塞等待时间(使用负值表示没有限制)
max-wait: -1ms
# 文言
messages:
basename: i18n/general_message/messages
encoding: utf-8
server:
# 服务器的HTTP端口默认为18080
port: 18081
servlet:
# 应用的访问路径
context-path: /openhis

View File

@@ -15,11 +15,6 @@ core:
# 开发环境配置
server:
# 服务器的HTTP端口默认为18080
port: 18080
servlet:
# 应用的访问路径
context-path: /openhis
tomcat:
# tomcat的URI编码
uri-encoding: UTF-8
@@ -54,7 +49,7 @@ spring:
# 国际化资源文件路径
basename: i18n/messages
profiles:
active: local #本地local 生产prod (农大)
active: dev
# 文件上传
servlet:
multipart:
@@ -85,6 +80,11 @@ mybatis-plus:
mapperLocations: classpath*:mapper/**/*Mapper.xml
# 加载全局的配置文件
configLocation: classpath:mybatis/mybatis-config.xml
global-config:
db-config:
logic-delete-field: validFlag # 全局逻辑删除的实体字段名
logic-delete-value: 0 # 逻辑已删除值(默认为 1)
logic-not-delete-value: 1 # 逻辑未删除值(默认为 0)
# PageHelper分页插件
pagehelper:

View File

@@ -18,6 +18,7 @@
T3.extra_details,
T3.contact,
T3.appointment_required_flag,
T3.practitioner_id,
T3.definition_id,
T3.charge_name,
T3.price,
@@ -36,6 +37,7 @@
T1.extra_details,
T1.contact,
T1.appointment_required_flag,
T1.practitioner_id,
T2.ID AS definition_id,
T2.charge_name,
T2.price,

View File

@@ -96,7 +96,7 @@
T14.bill_date AS return_date,
T14.refund_reason AS return_reason,
T15."name" AS operator_name,
T14.enterer_id AS operator_id,
T17.user_name AS operator_id,
ABS(T14.display_amount) AS refund_amount,
T6.contract_no AS contract_no,
T16.refund_method AS refund_method
@@ -134,6 +134,7 @@
AND T14.status_enum = 3
AND T14.payment_enum = 1
LEFT JOIN adm_practitioner AS T15 ON T15.ID = T14.enterer_id AND T15.delete_flag = '0'
LEFT JOIN sys_user AS T17 ON T17.user_id = T15.user_id AND T17.delete_flag = '0'
-- 关联退号支付详情,获取退款方式(聚合多个支付方式)
LEFT JOIN (
SELECT reconciliation_id,

View File

@@ -60,6 +60,9 @@ public class HealthcareService extends HisBaseEntity {
/** 预约要求 */
private Integer appointmentRequiredFlag;
/** 出诊医生ID */
private Long practitionerId;
/** 医保编码 */
private String ybNo;

View File

@@ -77,4 +77,16 @@ public class Organization extends HisBaseEntity {
/** 默认挂号医生 */
private Long defDoctorId;
/** 挂号科室标记 */
private Integer registerFlag;
/** 科室位置 */
private String location;
/** 科室简介 */
private String intro;
/** 备注 */
private String remark;
}

View File

@@ -0,0 +1,56 @@
package com.openhis.administration.domain;
import java.util.Date;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import com.core.common.core.domain.HisBaseEntity;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;
/**
* 门诊号码段管理Entity实体
*
* @author system
* @date 2025-01-XX
*/
@Data
@TableName("adm_outpatient_no_segment")
@Accessors(chain = true)
@EqualsAndHashCode(callSuper = false)
public class OutpatientNoSegment extends HisBaseEntity {
/** ID */
@TableId(type = IdType.ASSIGN_ID)
@JsonSerialize(using = ToStringSerializer.class)
private Long id;
/** 操作员ID */
private Long operatorId;
/** 操作员姓名 */
private String operatorName;
/** 员工工号 */
private String staffNo;
/** 领用日期 */
@JsonFormat(pattern = "yyyy-MM-dd")
private Date receiveDate;
/** 起始号码 */
private String startNo;
/** 终止号码 */
private String endNo;
/** 使用号码 */
private String usedNo;
}

View File

@@ -0,0 +1,18 @@
package com.openhis.administration.mapper;
import org.springframework.stereotype.Repository;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.administration.domain.OutpatientNoSegment;
/**
* 门诊号码段管理Mapper接口
*
* @author system
* @date 2025-01-XX
*/
@Repository
public interface OutpatientNoSegmentMapper extends BaseMapper<OutpatientNoSegment> {
}

View File

@@ -0,0 +1,66 @@
package com.openhis.administration.service;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.openhis.administration.domain.OutpatientNoSegment;
/**
* 门诊号码段管理Service接口
*
* @author system
* @date 2025-01-XX
*/
public interface IOutpatientNoSegmentService {
/**
* 分页查询门诊号码段列表
*
* @param page 分页对象
* @param onlySelf 是否只查询自己的true=只查询自己的false=查询所有)
* @param userId 用户ID
* @return 分页结果
*/
Page<OutpatientNoSegment> selectOutpatientNoSegmentPage(Page<OutpatientNoSegment> page, boolean onlySelf, Long userId);
/**
* 新增门诊号码段
*
* @param outpatientNoSegment 门诊号码段信息
* @return 结果
*/
int insertOutpatientNoSegment(OutpatientNoSegment outpatientNoSegment);
/**
* 修改门诊号码段
*
* @param outpatientNoSegment 门诊号码段信息
* @return 结果
*/
int updateOutpatientNoSegment(OutpatientNoSegment outpatientNoSegment);
/**
* 删除门诊号码段
*
* @param ids 门诊号码段ID列表
* @return 结果
*/
int deleteOutpatientNoSegmentByIds(Long[] ids);
/**
* 根据ID查询门诊号码段
*
* @param id 门诊号码段ID
* @return 门诊号码段信息
*/
OutpatientNoSegment getById(Long id);
/**
* 检查号码段是否重复(全系统范围)
*
* @param startNo 起始号码
* @param endNo 终止号码
* @param excludeId 排除的ID用于更新时排除自身
* @return true=重复false=不重复
*/
boolean checkNumberSegmentOverlap(String startNo, String endNo, Long excludeId);
}

View File

@@ -0,0 +1,241 @@
package com.openhis.administration.service.impl;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import com.openhis.administration.domain.OutpatientNoSegment;
import com.openhis.administration.mapper.OutpatientNoSegmentMapper;
import com.openhis.administration.service.IOutpatientNoSegmentService;
/**
* 门诊号码段管理Service实现
*
* @author system
* @date 2025-01-XX
*/
@Service
public class OutpatientNoSegmentServiceImpl implements IOutpatientNoSegmentService {
@Autowired
private OutpatientNoSegmentMapper outpatientNoSegmentMapper;
/**
* 分页查询门诊号码段列表
*/
@Override
public Page<OutpatientNoSegment> selectOutpatientNoSegmentPage(Page<OutpatientNoSegment> page, boolean onlySelf, Long userId) {
LambdaQueryWrapper<OutpatientNoSegment> queryWrapper = new LambdaQueryWrapper<>();
// 如果不是管理员,只查询自己的号码段
if (onlySelf && userId != null) {
queryWrapper.eq(OutpatientNoSegment::getOperatorId, userId);
}
// 按创建时间正序排列(新记录排在后面)
queryWrapper.orderByAsc(OutpatientNoSegment::getCreateTime);
return outpatientNoSegmentMapper.selectPage(page, queryWrapper);
}
/**
* 新增门诊号码段
*/
@Override
public int insertOutpatientNoSegment(OutpatientNoSegment outpatientNoSegment) {
return outpatientNoSegmentMapper.insert(outpatientNoSegment);
}
/**
* 修改门诊号码段
*/
@Override
public int updateOutpatientNoSegment(OutpatientNoSegment outpatientNoSegment) {
return outpatientNoSegmentMapper.updateById(outpatientNoSegment);
}
/**
* 删除门诊号码段
*/
@Override
public int deleteOutpatientNoSegmentByIds(Long[] ids) {
if (ids == null || ids.length == 0) {
return 0;
}
return outpatientNoSegmentMapper.deleteBatchIds(java.util.Arrays.asList(ids));
}
/**
* 根据ID查询门诊号码段
* 注意:由于 @TableLogic 注解selectById 会自动过滤已删除的记录delete_flag = '1'
* 这意味着如果记录已经被软删除,此方法会返回 null
*/
@Override
public OutpatientNoSegment getById(Long id) {
System.out.println("=== 查询门诊号码段开始 ===");
System.out.println("查询ID: " + id);
System.out.println("ID类型: " + (id != null ? id.getClass().getName() : "null"));
System.out.println("ID值字符串: " + String.valueOf(id));
// 使用 selectById 查询(会自动过滤 delete_flag = '1' 的记录)
OutpatientNoSegment segment = outpatientNoSegmentMapper.selectById(id);
if (segment != null) {
System.out.println("查询成功 - 找到记录:");
System.out.println(" - 记录ID: " + segment.getId());
System.out.println(" - 操作员ID: " + segment.getOperatorId());
System.out.println(" - 起始号码: " + segment.getStartNo());
System.out.println(" - 终止号码: " + segment.getEndNo());
System.out.println(" - 使用号码: " + segment.getUsedNo());
} else {
System.out.println("查询失败 - 未找到记录");
System.out.println("可能原因:");
System.out.println(" 1. 记录不存在");
System.out.println(" 2. 记录已被软删除delete_flag = '1'");
System.out.println(" 3. ID 不匹配");
// 尝试直接查询数据库(包括已删除的记录)来验证记录是否存在
try {
LambdaQueryWrapper<OutpatientNoSegment> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.eq(OutpatientNoSegment::getId, id);
// 注意:这里仍然会受到 @TableLogic 的影响,无法查询已删除的记录
// 如果需要查询已删除的记录,需要使用原生 SQL 或禁用逻辑删除
OutpatientNoSegment segmentWithDeleted = outpatientNoSegmentMapper.selectOne(queryWrapper);
if (segmentWithDeleted != null) {
System.out.println("使用 LambdaQueryWrapper 查询结果: 找到记录ID=" + segmentWithDeleted.getId());
} else {
System.out.println("使用 LambdaQueryWrapper 查询结果: 未找到记录");
}
} catch (Exception e) {
System.out.println("使用 LambdaQueryWrapper 查询时出错: " + e.getMessage());
}
// 尝试查询所有记录(不限制 delete_flag来验证 ID 是否存在
// 注意:由于 @TableLogic无法直接查询已删除的记录
// 但我们可以查询所有未删除的记录,看看是否有类似的 ID
try {
LambdaQueryWrapper<OutpatientNoSegment> allQuery = new LambdaQueryWrapper<>();
// 查询所有未删除的记录
List<OutpatientNoSegment> allSegments = outpatientNoSegmentMapper.selectList(allQuery);
System.out.println("数据库中所有未删除的记录数: " + allSegments.size());
System.out.println("所有记录的 ID 列表:");
for (OutpatientNoSegment seg : allSegments) {
System.out.println(" - ID: " + seg.getId() + " (类型: " + seg.getId().getClass().getName() + ")");
// 检查是否有接近的 ID可能是精度问题
if (seg.getId() != null) {
String segIdStr = String.valueOf(seg.getId());
String searchIdStr = String.valueOf(id);
if (segIdStr.length() == searchIdStr.length() &&
segIdStr.substring(0, Math.min(10, segIdStr.length())).equals(
searchIdStr.substring(0, Math.min(10, searchIdStr.length())))) {
System.out.println(" 警告:发现相似的 ID");
}
}
}
} catch (Exception e) {
System.out.println("查询所有记录时出错: " + e.getMessage());
}
}
System.out.println("=== 查询门诊号码段结束 ===");
return segment;
}
/**
* 检查号码段是否重复(全系统范围)
* 规则:从末位往前找到第一个字母作为前缀,比较前缀和数字范围
*/
@Override
public boolean checkNumberSegmentOverlap(String startNo, String endNo, Long excludeId) {
if (startNo == null || endNo == null) {
return false;
}
// 提取前缀(从末位往前找到第一个字母)
String prefix1 = extractPrefix(startNo);
String prefix2 = extractPrefix(endNo);
// 前缀必须一致
if (!prefix1.equals(prefix2)) {
return false;
}
// 提取尾部数字
Long num1 = extractTailNumber(startNo);
Long num2 = extractTailNumber(endNo);
if (num1 == null || num2 == null || num1 > num2) {
return false;
}
// 查询所有相同前缀的号码段
LambdaQueryWrapper<OutpatientNoSegment> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.likeRight(OutpatientNoSegment::getStartNo, prefix1);
if (excludeId != null) {
queryWrapper.ne(OutpatientNoSegment::getId, excludeId);
}
List<OutpatientNoSegment> segments = outpatientNoSegmentMapper.selectList(queryWrapper);
// 检查是否有重叠
for (OutpatientNoSegment segment : segments) {
String otherPrefix = extractPrefix(segment.getStartNo());
if (!prefix1.equals(otherPrefix)) {
continue;
}
Long otherNum1 = extractTailNumber(segment.getStartNo());
Long otherNum2 = extractTailNumber(segment.getEndNo());
if (otherNum1 != null && otherNum2 != null) {
// 检查范围是否重叠
if (Math.max(num1, otherNum1) <= Math.min(num2, otherNum2)) {
return true;
}
}
}
return false;
}
/**
* 从末位往前找到第一个字母,返回前缀(包含该字母)
*/
private String extractPrefix(String value) {
if (value == null || value.isEmpty()) {
return "";
}
char[] chars = value.toCharArray();
for (int i = chars.length - 1; i >= 0; i--) {
if (Character.isLetter(chars[i])) {
return value.substring(0, i + 1);
}
}
return "";
}
/**
* 提取尾部数字
*/
private Long extractTailNumber(String value) {
if (value == null || value.isEmpty()) {
return null;
}
Pattern pattern = Pattern.compile("(\\d+)$");
Matcher matcher = pattern.matcher(value);
if (matcher.find()) {
try {
return Long.parseLong(matcher.group(1));
} catch (NumberFormatException e) {
return null;
}
}
return null;
}
}

View File

@@ -0,0 +1,74 @@
package com.openhis.check.domain;
import com.baomidou.mybatisplus.annotation.TableName;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableField;
import lombok.Data;
import lombok.experimental.Accessors;
import java.time.LocalDateTime;
/**
* 检查方法
*
* @author system
* @date 2025-07-22
*/
@Data
@Accessors(chain = true)
@TableName(value = "check_method", autoResultMap = true)
public class CheckMethod {
private static final long serialVersionUID = 1L;
/**
* 检查方法ID
*/
@TableId(type = IdType.AUTO)
private Integer id;
/* 检查类型 */
private String checkType;
/* 方法代码 */
private String code;
/* 方法名称 */
private String name;
/* 套餐名称 */
private String packageName;
/* 曝光次数 */
private Integer exposureNum;
/* 序号 */
private Integer orderNum;
/* 备注 */
private String remark;
/** 创建时间 */
private LocalDateTime createTime;
/** 更新时间 */
private LocalDateTime updateTime;
/**
* 禁用逻辑删除因为数据库表中没有delete_flag字段
*/
@TableField(exist = false)
private String deleteFlag;
/**
* 以下字段数据库表中不存在用于禁用MyBatis Plus自动添加的字段
*/
@TableField(exist = false)
private String createBy;
@TableField(exist = false)
private String updateBy;
@TableField(exist = false)
private Integer tenantId;
}

View File

@@ -0,0 +1,127 @@
package com.openhis.check.domain;
import com.baomidou.mybatisplus.annotation.TableName;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableLogic;
import com.baomidou.mybatisplus.annotation.TableField;
import lombok.Data;
import lombok.experimental.Accessors;
import java.math.BigDecimal;
import java.time.LocalDate;
import java.time.LocalDateTime;
/**
* 检查套餐
*
* @author system
* @date 2025-07-22
* @updated 2025-11-26 - 扩展字段以支持套餐设置PRD需求
*/
@Data
@Accessors(chain = true)
@TableName(value = "check_package", autoResultMap = true)
public class CheckPackage {
private static final long serialVersionUID = 1L;
/** 检查套餐ID */
@TableId(type = IdType.AUTO)
private Long id;
/** 套餐名称 */
@TableField("package_name")
private String packageName;
/** 套餐编码 */
private String code;
/** 套餐类别 */
@TableField("package_type")
private String packageType;
/** 套餐级别 (1:全院套餐 2:科室套餐 3:个人套餐) */
@TableField("package_level")
private String packageLevel;
/** 适用科室 (当套餐级别为科室套餐时必填) */
@TableField("department")
private String department;
/** 适用用户 (当套餐级别为个人套餐时必填) */
@TableField("\"user\"")
private String user;
/** 卫生机构 */
@TableField("organization")
private String organization;
/** 套餐金额 */
@TableField("package_price")
private BigDecimal packagePrice;
/** 折扣 (百分比) */
@TableField("discount")
private BigDecimal discount;
/** 制单人 */
@TableField("creator")
private String creator;
/** 是否停用 (0:启用 1:停用) */
@TableField("is_disabled")
private Integer isDisabled;
/** 显示套餐名 (0:否 1:是) */
@TableField("show_package_name")
private Integer showPackageName;
/** 生成服务费 (0:否 1:是) */
@TableField("generate_service_fee")
private Integer generateServiceFee;
/** 套餐价格启用状态 (0:不启用 1:启用) */
@TableField("package_price_enabled")
private Integer packagePriceEnabled;
/** 服务费 */
@TableField("service_fee")
private BigDecimal serviceFee;
/** 备注 */
private String remark;
/** 描述 */
private String description;
/** 序号 */
private Integer number;
/** 套餐维护日期 (系统自动生成) */
@TableField("maintain_date")
private LocalDate maintainDate;
/** 创建时间 */
private LocalDateTime createTime;
/** 更新时间 */
private LocalDateTime updateTime;
/**
* 禁用逻辑删除因为数据库表中没有delete_flag字段
*/
@TableField(exist = false)
private String deleteFlag;
/**
* 以下字段数据库表中不存在用于禁用MyBatis Plus自动添加的字段
*/
@TableField(exist = false)
private String createBy;
@TableField(exist = false)
private String updateBy;
@TableField(exist = false)
private Integer tenantId;
}

View File

@@ -0,0 +1,95 @@
package com.openhis.check.domain;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import lombok.Data;
import lombok.experimental.Accessors;
import java.math.BigDecimal;
import java.time.LocalDateTime;
/**
* 检查套餐明细表
*
* @author system
* @date 2025-11-26
*/
@Data
@Accessors(chain = true)
@TableName(value = "check_package_detail", autoResultMap = true)
public class CheckPackageDetail {
private static final long serialVersionUID = 1L;
/** 套餐明细ID */
@TableId(type = IdType.AUTO)
private Long id;
/** 套餐ID */
private Long packageId;
/** 项目编号 */
private String itemCode;
/** 项目名称/规格 */
private String itemName;
/** 检查项目ID(诊疗项目ID) */
private Long checkItemId;
/** 剂量 */
private String dose;
/** 途径 */
private String method;
/** 频次 */
private String frequency;
/** 天数 */
private String days;
/** 数量 */
private Integer quantity;
/** 单价 */
private BigDecimal unitPrice;
/** 金额 */
private BigDecimal amount;
/** 服务费 */
private BigDecimal serviceCharge;
/** 总金额 */
private BigDecimal total;
/** 产地 */
private String origin;
/** 序号 */
private Integer orderNum;
/** 创建时间 */
private LocalDateTime createTime;
/** 更新时间 */
private LocalDateTime updateTime;
/**
* 禁用MyBatis Plus自动添加的字段
*/
@TableField(exist = false)
private String createBy;
@TableField(exist = false)
private String updateBy;
@TableField(exist = false)
private Integer tenantId;
@TableField(exist = false)
private String deleteFlag;
}

View File

@@ -0,0 +1,81 @@
package com.openhis.check.domain;
import com.baomidou.mybatisplus.annotation.TableName;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableField;
import lombok.Data;
import lombok.experimental.Accessors;
import java.time.LocalDateTime;
/**
* 检查部位
*
* @author system
* @date 2025-07-22
*/
@Data
@Accessors(chain = true)
@TableName(value = "check_part", autoResultMap = true)
public class CheckPart {
private static final long serialVersionUID = 1L;
/** 检查部位ID */
@TableId(type = IdType.AUTO)
private Integer id;
/** 检查部位名称 */
private String name;
/** 检查部位编码 */
private String code;
/** 检查部位检查类型 */
private String checkType;
/** 曝光次数 */
private Integer exposureNum;
/** 费用套餐 */
private String packageName;
/** 金额 */
private Double price;
/** 序号 */
private Integer number;
/** 服务范围 */
private String serviceScope;
/** 下级医技类型 */
private String subType;
/** 备注 */
private String remark;
/** 创建时间 */
private LocalDateTime createTime;
/** 更新时间 */
private LocalDateTime updateTime;
/**
* 禁用逻辑删除因为数据库表中没有delete_flag字段
*/
@TableField(exist = false)
private String deleteFlag;
/**
* 以下字段数据库表中不存在用于禁用MyBatis Plus自动添加的字段
*/
@TableField(exist = false)
private String createBy;
@TableField(exist = false)
private String updateBy;
@TableField(exist = false)
private Integer tenantId;
}

View File

@@ -0,0 +1,73 @@
package com.openhis.check.domain;
import com.baomidou.mybatisplus.annotation.TableName;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableLogic;
import com.baomidou.mybatisplus.annotation.TableField;
import lombok.Data;
import lombok.experimental.Accessors;
import java.time.LocalDateTime;
/**
* 检查类型
*
* @author system
* @date 2025-07-22
*/
@Data
@Accessors(chain = true)
@TableName(value = "check_type", autoResultMap = true)
public class CheckType {
private static final long serialVersionUID = 1L;
/** 检查类型ID */
@TableId(type = IdType.AUTO)
private Long id;
/** 检查类型名称 */
private String name;
/** 检查类型编码 */
private String code;
/** 检查类型 */
private String type;
/** 是否选中 */
private Boolean selected;
/** 科室 */
private String department;
/** 序号 */
private Integer number;
/** 备注 */
private String remark;
/** 创建时间 */
private LocalDateTime createTime;
/** 更新时间 */
private LocalDateTime updateTime;
/**
* 禁用逻辑删除因为数据库表中没有delete_flag字段
*/
@TableField(exist = false)
private String deleteFlag;
/**
* 以下字段数据库表中不存在用于禁用MyBatis Plus自动添加的字段
*/
@TableField(exist = false)
private String createBy;
@TableField(exist = false)
private String updateBy;
@TableField(exist = false)
private Integer tenantId;
}

View File

@@ -0,0 +1,40 @@
package com.openhis.check.domain;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import lombok.Data;
import lombok.experimental.Accessors;
import java.time.LocalDate;
import java.time.LocalDateTime;
@Data
@Accessors(chain = true)
@TableName(value = "lis_group_info",autoResultMap = true)
public class LisGroupInfo {
/** id */
@TableId(type = IdType.AUTO)
private Integer id;
/** 卫生机构 */
private String hospital;
/** 创建日期 */
private LocalDate date;
/** LIS分组名称 */
private String groupName;
/** 采血管类型 */
private String tube;
/** 备注 */
private String remark;
/** 创建时间 */
private LocalDateTime createTime;
/** 更新时间 */
private LocalDateTime updateTime;
}

View File

@@ -0,0 +1,33 @@
package com.openhis.check.domain;
import com.baomidou.mybatisplus.annotation.TableName;
import com.core.common.core.domain.HisBaseEntity;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;
/**
* 套餐项目关系表
*
* @author system
* @date 2025-07-22
*/
@Data
@EqualsAndHashCode(callSuper = true)
@Accessors(chain = true)
@TableName("package_item")
public class PackageItem extends HisBaseEntity {
private static final long serialVersionUID = 1L;
/** ID */
private Long id;
/** 套餐ID */
private Long packageId;
/** 检查项目ID */
private Long checkItemId;
/** 项目类型1检查类型 2检查方法 3检查部位 */
private Integer itemType;
}

View File

@@ -0,0 +1,16 @@
package com.openhis.check.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.check.domain.CheckMethod;
import org.springframework.stereotype.Repository;
/**
* 检查方法Mapper接口
*
* @author system
* @date 2025-07-22
*/
@Repository
public interface CheckMethodMapper extends BaseMapper<CheckMethod> {
}

View File

@@ -0,0 +1,16 @@
package com.openhis.check.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.check.domain.CheckPackageDetail;
import org.apache.ibatis.annotations.Mapper;
/**
* 检查套餐明细Mapper接口
*
* @author system
* @date 2025-11-26
*/
@Mapper
public interface CheckPackageDetailMapper extends BaseMapper<CheckPackageDetail> {
}

View File

@@ -0,0 +1,16 @@
package com.openhis.check.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.check.domain.CheckPackage;
import org.springframework.stereotype.Repository;
/**
* 检查套餐Mapper接口
*
* @author system
* @date 2025-07-22
*/
@Repository
public interface CheckPackageMapper extends BaseMapper<CheckPackage> {
}

View File

@@ -0,0 +1,16 @@
package com.openhis.check.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.check.domain.CheckPart;
import org.springframework.stereotype.Repository;
/**
* 检查部位Mapper接口
*
* @author system
* @date 2025-07-22
*/
@Repository
public interface CheckPartMapper extends BaseMapper<CheckPart> {
}

View File

@@ -0,0 +1,16 @@
package com.openhis.check.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.check.domain.CheckType;
import org.springframework.stereotype.Repository;
/**
* 检查类型Mapper接口
*
* @author system
* @date 2025-07-22
*/
@Repository
public interface CheckTypeMapper extends BaseMapper<CheckType> {
}

View File

@@ -0,0 +1,9 @@
package com.openhis.check.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.check.domain.LisGroupInfo;
import org.springframework.stereotype.Repository;
@Repository
public interface LisGroupInfoMapper extends BaseMapper<LisGroupInfo> {
}

View File

@@ -0,0 +1,16 @@
package com.openhis.check.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.check.domain.PackageItem;
import org.springframework.stereotype.Repository;
/**
* 套餐项目关系表Mapper接口
*
* @author system
* @date 2025-07-22
*/
@Repository
public interface PackageItemMapper extends BaseMapper<PackageItem> {
}

View File

@@ -0,0 +1,14 @@
package com.openhis.check.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.check.domain.CheckMethod;
/**
* 检查方法Service接口
*
* @author system
* @date 2025-07-22
*/
public interface ICheckMethodService extends IService<CheckMethod> {
}

View File

@@ -0,0 +1,14 @@
package com.openhis.check.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.check.domain.CheckPackageDetail;
/**
* 检查套餐明细Service接口
*
* @author system
* @date 2025-11-26
*/
public interface ICheckPackageDetailService extends IService<CheckPackageDetail> {
}

View File

@@ -0,0 +1,14 @@
package com.openhis.check.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.check.domain.CheckPackage;
/**
* 检查套餐Service接口
*
* @author system
* @date 2025-07-22
*/
public interface ICheckPackageService extends IService<CheckPackage> {
}

View File

@@ -0,0 +1,14 @@
package com.openhis.check.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.check.domain.CheckPart;
/**
* 检查部位Service接口
*
* @author system
* @date 2025-07-22
*/
public interface ICheckPartService extends IService<CheckPart> {
}

View File

@@ -0,0 +1,14 @@
package com.openhis.check.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.check.domain.CheckType;
/**
* 检查类型Service接口
*
* @author system
* @date 2025-07-22
*/
public interface ICheckTypeService extends IService<CheckType> {
}

View File

@@ -0,0 +1,6 @@
package com.openhis.check.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.check.domain.LisGroupInfo;
public interface ILisGroupInfoService extends IService<LisGroupInfo>{
}

View File

@@ -0,0 +1,14 @@
package com.openhis.check.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.check.domain.PackageItem;
/**
* 套餐项目关系表Service接口
*
* @author system
* @date 2025-07-22
*/
public interface IPackageItemService extends IService<PackageItem> {
}

View File

@@ -0,0 +1,18 @@
package com.openhis.check.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.check.domain.CheckMethod;
import com.openhis.check.mapper.CheckMethodMapper;
import com.openhis.check.service.ICheckMethodService;
import org.springframework.stereotype.Service;
/**
* 检查方法Service实现类
*
* @author system
* @date 2025-07-22
*/
@Service
public class CheckMethodServiceImpl extends ServiceImpl<CheckMethodMapper, CheckMethod> implements ICheckMethodService {
}

View File

@@ -0,0 +1,19 @@
package com.openhis.check.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.check.domain.CheckPackageDetail;
import com.openhis.check.mapper.CheckPackageDetailMapper;
import com.openhis.check.service.ICheckPackageDetailService;
import org.springframework.stereotype.Service;
/**
* 检查套餐明细Service实现
*
* @author system
* @date 2025-11-26
*/
@Service
public class CheckPackageDetailServiceImpl extends ServiceImpl<CheckPackageDetailMapper, CheckPackageDetail>
implements ICheckPackageDetailService {
}

View File

@@ -0,0 +1,18 @@
package com.openhis.check.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.check.domain.CheckPackage;
import com.openhis.check.mapper.CheckPackageMapper;
import com.openhis.check.service.ICheckPackageService;
import org.springframework.stereotype.Service;
/**
* 检查套餐Service实现类
*
* @author system
* @date 2025-07-22
*/
@Service
public class CheckPackageServiceImpl extends ServiceImpl<CheckPackageMapper, CheckPackage> implements ICheckPackageService {
}

View File

@@ -0,0 +1,18 @@
package com.openhis.check.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.check.domain.CheckPart;
import com.openhis.check.mapper.CheckPartMapper;
import com.openhis.check.service.ICheckPartService;
import org.springframework.stereotype.Service;
/**
* 检查部位Service实现类
*
* @author system
* @date 2025-07-22
*/
@Service
public class CheckPartServiceImpl extends ServiceImpl<CheckPartMapper, CheckPart> implements ICheckPartService {
}

View File

@@ -0,0 +1,18 @@
package com.openhis.check.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.check.domain.CheckType;
import com.openhis.check.mapper.CheckTypeMapper;
import com.openhis.check.service.ICheckTypeService;
import org.springframework.stereotype.Service;
/**
* 检查类型Service实现类
*
* @author system
* @date 2025-07-22
*/
@Service
public class CheckTypeServiceImpl extends ServiceImpl<CheckTypeMapper, CheckType> implements ICheckTypeService {
}

View File

@@ -0,0 +1,12 @@
package com.openhis.check.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.check.mapper.LisGroupInfoMapper;
import com.openhis.check.domain.LisGroupInfo;
import com.openhis.check.service.ILisGroupInfoService;
import org.springframework.stereotype.Service;
@Service
public class LisGroupInfoServiceImpl extends ServiceImpl<LisGroupInfoMapper,LisGroupInfo> implements ILisGroupInfoService{
}

View File

@@ -0,0 +1,18 @@
package com.openhis.check.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.check.domain.PackageItem;
import com.openhis.check.mapper.PackageItemMapper;
import com.openhis.check.service.IPackageItemService;
import org.springframework.stereotype.Service;
/**
* 套餐项目关系表Service实现类
*
* @author system
* @date 2025-07-22
*/
@Service
public class PackageItemServiceImpl extends ServiceImpl<PackageItemMapper, PackageItem> implements IPackageItemService {
}

View File

@@ -0,0 +1,72 @@
package com.openhis.lab.domain;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableLogic;
import com.baomidou.mybatisplus.annotation.TableName;
import lombok.Data;
import lombok.experimental.Accessors;
import java.time.LocalDateTime;
/**
* 检验类型
*
* @author system
* @date 2025-12-09
*/
@Data
@Accessors(chain = true)
@TableName(value = "inspection_type", autoResultMap = true)
public class InspectionType {
private static final long serialVersionUID = 1L;
/** 主键ID */
@TableId(type = IdType.AUTO)
private Long id;
/** 检验类型编码 */
private String code;
/** 检验类型名称 */
private String name;
/** 所属科室 */
private String department;
/** 排序 */
@TableField("\"order\"")
private Integer sortOrder;
/** 备注 */
private String remark;
/** 有效标志1有效0无效 */
@TableField("valid_flag")
@TableLogic(value = "1", delval = "0")
private Integer validFlag;
/** 创建时间 */
@TableField("created_at")
private LocalDateTime createdAt;
/** 更新时间 */
@TableField("updated_at")
private LocalDateTime updatedAt;
/** 版本号 */
private Integer version;
/**
* 以下字段数据库表中不存在用于禁用MyBatis Plus自动添加的字段
*/
@TableField(exist = false)
private String createBy;
@TableField(exist = false)
private String updateBy;
@TableField(exist = false)
private Integer tenantId;
}

View File

@@ -0,0 +1,16 @@
package com.openhis.lab.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.openhis.lab.domain.InspectionType;
import org.springframework.stereotype.Repository;
/**
* 检验类型Mapper接口
*
* @author system
* @date 2025-12-09
*/
@Repository
public interface InspectionTypeMapper extends BaseMapper<InspectionType> {
}

View File

@@ -0,0 +1,14 @@
package com.openhis.lab.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.openhis.lab.domain.InspectionType;
/**
* 检验类型Service接口
*
* @author system
* @date 2025-12-09
*/
public interface IInspectionTypeService extends IService<InspectionType> {
}

View File

@@ -0,0 +1,18 @@
package com.openhis.lab.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.lab.domain.InspectionType;
import com.openhis.lab.mapper.InspectionTypeMapper;
import com.openhis.lab.service.IInspectionTypeService;
import org.springframework.stereotype.Service;
/**
* 检验类型Service实现类
*
* @author system
* @date 2025-12-09
*/
@Service
public class InspectionTypeServiceImpl extends ServiceImpl<InspectionTypeMapper, InspectionType> implements IInspectionTypeService {
}

View File

@@ -4,4 +4,14 @@
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.openhis.administration.mapper.OrganizationMapper">
<!-- 自定义selectById方法确保包含所有字段包括新增字段 -->
<select id="selectById" parameterType="java.lang.Long" resultType="com.openhis.administration.domain.Organization">
SELECT id, bus_no, name, active_flag, type_enum, class_enum, py_str, wb_str,
yb_no, yb_name, caty, display_order, medins_id, medins_admdvs,
medins_type, medins_lv, def_doctor_id, create_by, create_time,
update_by, update_time, tenant_id, delete_flag,
register_flag, location, intro, remark
FROM adm_organization
WHERE id = #{id} AND delete_flag = '0'
</select>
</mapper>

View File

@@ -2,7 +2,10 @@
VITE_APP_TITLE = 医院信息管理系统
# 开发环境配置
VITE_APP_ENV = 'development'
VITE_APP_ENV = 'dev'
# OpenHIS管理系统/开发环境
VITE_APP_BASE_API = '/dev-api'
# 租户ID配置
VITE_APP_TENANT_ID = '1'

View File

@@ -2,10 +2,13 @@
VITE_APP_TITLE=医院信息管理系统
# 生产环境配置
VITE_APP_ENV=production
VITE_APP_ENV= 'prod'
# OpenHIS管理系统/生产环境
VITE_APP_BASE_API=/prod-api
VITE_APP_BASE_API= '/prd-api'
# 租户ID配置
VITE_APP_TENANT_ID= '1'
# 是否在打包时开启压缩,支持 gzip 和 brotli
VITE_BUILD_COMPRESS = gzip

View File

@@ -13,6 +13,9 @@ VITE_UPLOAD_TYPE=server
# OpenHIS管理系统/SPUG环境
VITE_APP_BASE_API = '/admin-api'
# 租户ID配置
VITE_APP_TENANT_ID=1
# 是否删除debugger
VITE_DROP_DEBUGGER=false

View File

@@ -7,5 +7,8 @@ VITE_APP_ENV = 'staging'
# OpenHIS管理系统/生产环境
VITE_APP_BASE_API = '/stage-api'
# 租户ID配置
VITE_APP_TENANT_ID=1
# 是否在打包时开启压缩,支持 gzip 和 brotli
VITE_BUILD_COMPRESS = gzip

12
openhis-ui-vue3/.env.test Normal file
View File

@@ -0,0 +1,12 @@
# 页面标题
VITE_APP_TITLE = 医院信息管理系统
# 测试环境配置
VITE_APP_ENV = 'test'
# OpenHIS管理系统/测试环境
VITE_APP_BASE_API = '/test-api'
# 租户ID配置
VITE_APP_TENANT_ID = '1'

View File

@@ -44,7 +44,7 @@
"sass": "1.69.5",
"unplugin-auto-import": "0.17.1",
"unplugin-vue-setup-extend-plus": "1.0.0",
"vite": "5.0.4",
"vite": "^5.0.4",
"vite-plugin-compression": "0.5.1",
"vite-plugin-svg-icons": "2.0.1"
}
@@ -7020,7 +7020,7 @@
},
"node_modules/vite": {
"version": "5.0.4",
"resolved": "https://registry.npmmirror.com/vite/-/vite-5.0.4.tgz",
"resolved": "https://registry.npmjs.org/vite/-/vite-5.0.4.tgz",
"integrity": "sha512-RzAr8LSvM8lmhB4tQ5OPcBhpjOZRZjuxv9zO5UcxeoY2bd3kP3Ticd40Qma9/BqZ8JS96Ll/jeBX9u+LJZrhVg==",
"dev": true,
"dependencies": {
@@ -12419,7 +12419,7 @@
},
"vite": {
"version": "5.0.4",
"resolved": "https://registry.npmmirror.com/vite/-/vite-5.0.4.tgz",
"resolved": "https://registry.npmjs.org/vite/-/vite-5.0.4.tgz",
"integrity": "sha512-RzAr8LSvM8lmhB4tQ5OPcBhpjOZRZjuxv9zO5UcxeoY2bd3kP3Ticd40Qma9/BqZ8JS96Ll/jeBX9u+LJZrhVg==",
"dev": true,
"requires": {

View File

@@ -6,9 +6,11 @@
"license": "MIT",
"type": "module",
"scripts": {
"dev": "vite",
"build:prod": "vite build --mode production",
"dev": "vite --mode dev",
"build:prod": "vite build --mode prod",
"build:stage": "vite build --mode staging",
"build:test": "vite build --mode test",
"build:dev": "vite build --mode dev",
"preview": "vite preview",
"build:spug": "vite build --mode spug"
},
@@ -52,7 +54,7 @@
"sass": "1.69.5",
"unplugin-auto-import": "0.17.1",
"unplugin-vue-setup-extend-plus": "1.0.0",
"vite": "5.0.4",
"vite": "^5.0.4",
"vite-plugin-compression": "0.5.1",
"vite-plugin-svg-icons": "2.0.1"
}

View File

@@ -0,0 +1,44 @@
import request from '@/utils/request'
// 查询检验类型列表
export function listInspectionType(query) {
return request({
url: '/system/inspection-type/list',
method: 'get',
params: query
})
}
// 查询检验类型详细
export function getInspectionType(inspectionTypeId) {
return request({
url: `/system/inspection-type/${inspectionTypeId}`,
method: 'get'
})
}
// 新增检验类型
export function addInspectionType(data) {
return request({
url: '/system/inspection-type',
method: 'post',
data: data
})
}
// 修改检验类型
export function updateInspectionType(data) {
return request({
url: '/system/inspection-type',
method: 'put',
data: data
})
}
// 删除检验类型
export function delInspectionType(inspectionTypeId) {
return request({
url: `/system/inspection-type/${inspectionTypeId}`,
method: 'delete'
})
}

View File

@@ -0,0 +1,239 @@
import request from '@/utils/request'
// 查询检查类型列表
export function listCheckType(query) {
return request({
url: '/system/check-type/list',
method: 'get',
params: query
})
}
// 查询检查类型详细
export function getCheckType(checkTypeId) {
return request({
url: '/system/check-type/' + checkTypeId,
method: 'get'
})
}
// 新增检查类型
export function addCheckType(data) {
return request({
url: '/system/check-type',
method: 'post',
data: data
})
}
// 修改检查类型
export function updateCheckType(data) {
return request({
url: '/system/check-type',
method: 'put',
data: data
})
}
// 删除检查类型
export function delCheckType(checkTypeId) {
return request({
url: '/system/check-type/' + checkTypeId,
method: 'delete'
})
}
// 查询检查方法列表
export function listCheckMethod(query) {
return request({
url: '/check/method/list',
method: 'get',
params: query
})
}
// 搜索检查方法列表
export function searchCheckMethod(query) {
return request({
url: '/check/method/search',
method: 'get',
params: query
})
}
// 新增检查方法
export function addCheckMethod(data) {
return request({
url: '/check/method/add',
method: 'post',
data: data
})
}
// 修改检查方法
export function updateCheckMethod(data) {
return request({
url: '/check/method/update',
method: 'put',
data: data
})
}
// 删除检查方法
export function delCheckMethod(checkMethodId) {
return request({
url: '/check/method/delete/' + checkMethodId,
method: 'delete'
})
}
// 导出检查方法
export function exportCheckMethod(query) {
return request({
url: '/check/method/export',
method: 'get',
params: query,
responseType: 'blob'
})
}
// 导出检查部位
export function exportCheckPart(query) {
return request({
url: '/check/part/export',
method: 'get',
params: query,
responseType: 'blob'
})
}
// 查询检查部位列表
export function listCheckPart(query) {
return request({
url: '/check/part/list',
method: 'get',
params: query
})
}
// 搜索检查部位列表
export function searchCheckPart(query) {
return request({
url: '/check/part/search',
method: 'get',
params: query
})
}
// 新增检查部位
export function addCheckPart(data) {
return request({
url: '/check/part/add',
method: 'post',
data: data
})
}
// 删除检查部位
export function delCheckPart(checkPartId) {
return request({
url: '/check/part/delete/' + checkPartId,
method: 'delete'
})
}
// 更新检查部位
export function updateCheckPart(data) {
return request({
url: '/check/part/update',
method: 'put',
data: data
})
}
// 查询检查套餐列表
export function listCheckPackage(query) {
return request({
url: '/system/check-package/list',
method: 'get',
params: query
})
}
// 根据ID查询检查套餐详情
export function getCheckPackage(id) {
return request({
url: `/system/check-package/${id}`,
method: 'get'
})
}
// 新增检查套餐
export function addCheckPackage(data) {
return request({
url: '/system/check-package',
method: 'post',
data: data
})
}
// 修改检查套餐
export function updateCheckPackage(data) {
return request({
url: '/system/check-package',
method: 'put',
data: data
})
}
// 删除检查套餐
export function delCheckPackage(id) {
return request({
url: `/system/check-package/${id}`,
method: 'delete'
})
}
// 查询LIS分组列表
export function listLisGroup(query) {
return request({
url: '/check/lisGroupInfo/list',
method: 'get',
params: query
})
}
// 根据ID查询LIS分组详情
export function getLisGroup(id) {
return request({
url: `/check/lisGroupInfo/${id}`,
method: 'get'
})
}
// 新增LIS分组
export function addLisGroup(data) {
return request({
url: '/check/lisGroupInfo/add',
method: 'post',
data: data
})
}
// 修改LIS分组
export function updateLisGroup(data) {
return request({
url: '/check/lisGroupInfo/update',
method: 'put',
data: data
})
}
// 删除LIS分组
export function delLisGroup(id) {
return request({
url: `/check/lisGroupInfo/${id}`,
method: 'delete'
})
}

View File

@@ -0,0 +1,44 @@
import request from '@/utils/request'
// 查询检验类型列表
export function listInspectionType(query) {
return request({
url: '/system/inspection-type/list',
method: 'get',
params: query
})
}
// 查询检验类型详细
export function getInspectionType(inspectionTypeId) {
return request({
url: '/system/inspection-type/' + inspectionTypeId,
method: 'get'
})
}
// 新增检验类型
export function addInspectionType(data) {
return request({
url: '/system/inspection-type',
method: 'post',
data: data
})
}
// 修改检验类型
export function updateInspectionType(data) {
return request({
url: '/system/inspection-type',
method: 'put',
data: data
})
}
// 删除检验类型
export function delInspectionType(inspectionTypeId) {
return request({
url: '/system/inspection-type/' + inspectionTypeId,
method: 'delete'
})
}

View File

@@ -47,11 +47,6 @@ export const constantRoutes = [
component: () => import('@/views/register'),
hidden: true
},
{
path: "/:pathMatch(.*)*",
component: () => import('@/views/error/404'),
hidden: true
},
{
path: '/401',
component: () => import('@/views/error/401'),
@@ -83,6 +78,20 @@ export const constantRoutes = [
meta: { title: '个人中心', icon: 'user' }
}
]
},
// 添加套餐管理相关路由到公共路由,确保始终可用
{
path: '/maintainSystem/Inspection/PackageManagement',
component: Layout,
hidden: true,
children: [
{
path: '',
component: () => import('@/views/maintainSystem/Inspection/PackageManagement.vue'),
name: 'DirectPackageManagement',
meta: { title: '套餐管理' }
}
]
}
]
@@ -134,6 +143,20 @@ export const constantRoutes = [
component: () => import('@/views/maintainSystem/chargeConfig/index.vue'),
name: 'ChargeConfig',
meta: { title: '挂号收费系统参数维护', icon: 'config', permissions: ['maintainSystem:chargeConfig:list'] }
},
{
path: 'Inspection',
component: () => import('@/views/maintainSystem/Inspection/index.vue'),
name: 'Inspection',
meta: { title: '检验管理', icon: 'inspection' },
children: [
{
path: 'PackageManagement',
component: () => import('@/views/maintainSystem/Inspection/PackageManagement.vue'),
name: 'PackageManagement',
meta: { title: '套餐管理' }
}
]
}
]
},
@@ -345,6 +368,13 @@ export const constantRoutes = [
// 合并常量路由和动态路由,确保所有路由都能被访问
const allRoutes = [...constantRoutes, ...dynamicRoutes];
// 添加404路由到所有路由的最后
allRoutes.push({
path: "/:pathMatch(.*)*",
component: () => import('@/views/error/404'),
hidden: true
});
const router = createRouter({
history: createWebHistory(),
routes: allRoutes,

View File

@@ -123,6 +123,9 @@ export function filterDynamicRoutes(routes) {
if (auth.hasRoleOr(route.roles)) {
res.push(route)
}
} else {
// 如果没有权限设置,默认允许访问
res.push(route)
}
})
return res

View File

@@ -1,6 +1,7 @@
import { login, logout, getInfo } from '@/api/login'
import { getToken, setToken, removeToken } from '@/utils/auth'
import defAva from '@/assets/images/user.png'
import { defineStore } from 'pinia'
const useUserStore = defineStore(
'user',
@@ -17,7 +18,8 @@ const useUserStore = defineStore(
fixmedinsCode: '', // 医疗机构编码
roles: [],
permissions: [],
tenantId: ''
tenantId: '',
status: '' // 用户状态0-启用(管理员), 1-禁用(普通人员)
}),
actions: {
// 登录
@@ -52,13 +54,14 @@ const useUserStore = defineStore(
this.roles = ['ROLE_DEFAULT']
}
this.id = user.userId
this.name = user.userName
this.name = user.userName // 用户账号对应数据库的user_name字段如'admin'
this.orgId = user.orgId
this.orgName = user.orgName
this.nickName = user.nickName
this.practitionerId = res.practitionerId
this.fixmedinsCode = res.optionJson.fixmedinsCode
this.avatar = avatar
this.status = user.status // 保存用户状态
resolve(res)
}).catch(error => {
reject(error)

View File

@@ -12,7 +12,8 @@ let downloadLoadingInstance;
export let isRelogin = { show: false };
axios.defaults.headers['Content-Type'] = 'application/json;charset=utf-8'
axios.defaults.headers['X-Tenant-ID'] = '1'
// 从环境变量读取租户ID如果没有则使用默认值'1'
axios.defaults.headers['X-Tenant-ID'] = import.meta.env.VITE_APP_TENANT_ID || '1'
axios.defaults.headers['Request-Method-Name'] = 'login'
// 创建axios实例
const service = axios.create({

View File

@@ -4,12 +4,6 @@
<!-- 页面标题和用户信息 -->
<div class="header-section">
<h2>发票管理</h2>
<div class="user-info">
<span class="welcome-text">欢迎, {{ currentUser.name }}</span>
<span class="role-badge" :class="currentUser.role">
{{ currentUser.role === 'admin' ? '管理员' : '操作员' }}
</span>
</div>
</div>
<!-- 操作按钮区域 -->
@@ -58,23 +52,8 @@
<td class="sequence-number">{{ index + 1 }}</td>
<td class="employee-info">
<div class="input-container">
<select
v-if="item.isActive"
v-model="item.operator"
class="form-control"
>
<option value="">请选择操作员</option>
<option
v-for="user in userList"
:key="user.employeeId"
:value="user.name"
:data-employee-id="user.employeeId"
@change="updateEmployeeId(item, user.employeeId)"
>
{{ user.name }}
</option>
</select>
<span v-else class="employee-name">{{ item.operator || '-' }}</span>
<!-- 操作员字段始终不可编辑 -->
<span class="employee-name">{{ item.operator || '-' }}</span>
</div>
</td>
<td class="employee-id-cell">
@@ -83,8 +62,7 @@
v-if="item.isActive"
v-model="item.employeeId"
class="form-control"
placeholder="请输入员工工号"
maxlength="10"
placeholder="员工工号"
/>
<span v-else>{{ item.employeeId || '-' }}</span>
</div>
@@ -191,6 +169,7 @@
<script>
import { listUser } from '@/api/system/user';
import request from '@/utils/request'; // 导入请求工具
import useUserStore from '@/store/modules/user'; // 导入用户store
export default {
name: 'InvoiceManagement',
@@ -198,9 +177,10 @@ export default {
return {
// 用户信息和权限
currentUser: {
name: 'admin',
employeeId: '1702',
role: 'admin' // operator: 普通操作员, admin: 管理员
name: '',
nickName: '',
employeeId: '',
status: '' // 0: 启用(管理员), 1: 禁用(普通人员)
},
// 用户列表用于操作员下拉选择将在created中从后端获取
userList: [],
@@ -217,11 +197,23 @@ export default {
computed: {
// 计算属性:判断是否为管理员
isAdmin() {
return this.currentUser.role === 'admin';
// 管理员是指在用户管理界面中状态为启用的用户
return this.currentUser.status === '0';
}
},
created() {
console.log('组件初始化,开始加载数据');
// 获取当前登录用户信息
const userStore = useUserStore();
this.currentUser = {
name: userStore.nickName || userStore.name,
nickName: userStore.nickName,
employeeId: userStore.id || userStore.practitionerId,
status: userStore.status || '' // 0: 启用(管理员), 1: 禁用(普通人员)
};
console.log('当前登录用户信息:', this.currentUser);
// 从后端获取用户列表后再加载发票数据,确保用户信息可用
this.getUserList().then(() => {
console.log('用户列表加载完成,开始加载发票数据');
@@ -247,12 +239,17 @@ export default {
methods: {
// 获取用户列表
getUserList() {
return listUser().then((res) => {
// 传递分页参数,获取所有用户数据
const queryParams = {
pageNum: 1,
pageSize: 1000 // 设置较大的pageSize以获取所有用户
};
return listUser(queryParams).then((res) => {
// 从响应中提取用户列表,并转换为需要的格式
this.userList = res.data.records.map(user => ({
name: user.nickName, // 使用用户昵称作为显示名称
name: user.nickName || user.username || user.name, // 尝试多种可能的名称字段
employeeId: user.userId, // 使用用户ID作为员工工号
role: user.practitionerRolesDtoList?.some(role => role.roleKey === 'admin') ? 'admin' : 'operator'
status: user.status // 0: 启用(管理员), 1: 禁用(普通人员)
}));
console.log('获取到的用户列表:', this.userList);
return Promise.resolve();
@@ -405,7 +402,7 @@ export default {
},
// 根据用户权限过滤数据
filterDataByPermission() {
console.log('开始过滤数据,当前用户角色:', this.currentUser.role);
console.log('开始过滤数据,当前用户状态:', this.currentUser.status);
console.log('过滤前数据总量:', this.invoiceData.length);
if (this.isAdmin) {
@@ -451,42 +448,33 @@ export default {
return user ? user.name : '';
},
// 检查是否有权限修改指定记录
canModifyRecord(record) {
// 管理员可以修改所有记录,普通用户只能修改自己的
return this.isAdmin || record.employeeId === this.currentUser.employeeId;
},
// 检查权限并执行操作
checkPermissionAndExecute(record, operation) {
if (this.canModifyRecord(record)) {
operation();
} else {
alert('您没有权限修改此记录!');
}
},
// 更新员工ID
updateEmployeeId(item, employeeId) {
item.employeeId = employeeId;
},
addNewRow() {
// 新增行时自动填充当前用户信息
// 使用负数作为临时ID避免与后端数据库ID冲突
const newId = -Math.max(...this.invoiceData.map(item => Math.abs(item.id)), 0) - 1;
this.invoiceData.push({
let maxId = 0;
if (this.invoiceData.length > 0) {
maxId = Math.max(...this.invoiceData.map(item => Math.abs(item.id)));
}
const newId = -(maxId + 1);
const newSegmentId = Date.now(); // 生成唯一的segmentId
const currentDate = new Date().toISOString().split('T')[0];
const newRecord = {
id: newId,
segmentId: newSegmentId, // 为新记录设置segmentId
operator: this.currentUser.name, // 自动使用当前用户名称
employeeId: this.currentUser.employeeId,
date: new Date().toISOString().split('T')[0],
date: currentDate, // 自动填充当日日期
startNum: '',
endNum: '',
currentNum: '',
status: '未使用',
isActive: true, // 新增行默认处于编辑状态
isNewRecord: true // 添加标记表示这是新记录
});
};
console.log('添加新行,自动填充领用日期为当日:', { newRecord });
this.invoiceData.push(newRecord);
},
deleteRow(record) {
@@ -495,9 +483,22 @@ export default {
return;
}
// 对于新添加的记录直接从前端移除不需要调用API
if (record.isNewRecord) {
if (confirm('确定要删除这条未保存的记录吗?')) {
const index = this.invoiceData.findIndex(item => item.keyId === record.keyId);
if (index > -1) {
this.invoiceData.splice(index, 1);
this.filterDataByPermission();
alert('删除成功');
}
}
return;
}
// 严格检查权限:只有管理员或记录所有者可以删除
if (this.currentUser.role !== 'admin' && record.employeeId !== this.currentUser.employeeId) {
alert('您没有权限删除此记录!操作员只能删除自己的记录。');
if (this.currentUser.status !== '0' && record.employeeId !== this.currentUser.employeeId) {
alert('您没有权限删除此记录!普通人员只能删除自己的记录。');
return;
}
@@ -815,17 +816,15 @@ export default {
},
// 从字符串末尾向前提取前缀,直到遇到第一个字母
// 规则:前缀定义为从号码最末位开始往前推直到出现字母为止前面字符若无字母则无前缀
// 规则:前缀定义为从号码最末位开始往前推直到出现字母为止,其前面字符全部称为前缀;若无字母则无前缀
extractPrefixFromEnd(str) {
if (!str) return '';
// 从末尾向前遍历字符串
for (let i = str.length - 1; i >= 0; i--) {
// 如果遇到字母,返回从开始到该字母之前的部分
// 例如:对于"ABC123",从末尾向前找到第一个字母'C',前缀就是'AB'
// 对于"123ABC456",从末尾向前找到第一个字母'C',前缀就是"123AB"
// 如果遇到字母,返回从开始到该字母的所有字符
if (/[A-Za-z]/.test(str[i])) {
return str.substring(0, i);
return str.substring(0, i + 1);
}
}
// 如果没有找到字母,则无前缀
@@ -924,6 +923,21 @@ export default {
},
// 检查号码范围是否与其他记录重叠
// 提取数字部分进行比较的辅助方法
extractNumber(str) {
if (!str) return 0;
const match = str.match(/\d+$/);
return match ? parseInt(match[0], 10) : 0;
},
// 提取完整前缀的辅助方法(包括字母和数字)
extractPrefix(str) {
if (!str) return '';
// 匹配开头的所有非数字字符(字母等)和随后的数字部分,直到遇到第一个非数字字符为止
const match = str.match(/^[A-Za-z0-9]+/);
return match ? match[0] : '';
},
checkRangeOverlap(record) {
// 确保当前记录有必要的数据
if (!record || !record.startNum || !record.endNum) {
@@ -934,25 +948,10 @@ export default {
// 使用keyId作为比较依据因为它是前端唯一标识符
const otherRecords = this.invoiceData.filter(item => item.keyId !== record.keyId);
// 提取数字部分进行比较的辅助函数
const extractNumber = function(str) {
if (!str) return 0;
const match = str.match(/\d+$/);
return match ? parseInt(match[0], 10) : 0;
};
// 提取完整前缀的辅助函数(包括字母和数字)
const extractPrefix = function(str) {
if (!str) return '';
// 匹配开头的所有非数字字符(字母等)和随后的数字部分,直到遇到第一个非数字字符为止
const match = str.match(/^[A-Za-z0-9]+/);
return match ? match[0] : '';
};
// 提取当前记录的前缀和数字范围
const currentPrefix = extractPrefix(record.startNum);
const currentStartNum = extractNumber(record.startNum);
const currentEndNum = extractNumber(record.endNum);
const currentPrefix = this.extractPrefix(record.startNum);
const currentStartNum = this.extractNumber(record.startNum);
const currentEndNum = this.extractNumber(record.endNum);
// 确保当前记录的起始号码小于等于终止号码
if (currentStartNum > currentEndNum) {
@@ -963,14 +962,14 @@ export default {
if (!item.startNum || !item.endNum) continue;
// 提取其他记录的前缀
const otherPrefix = extractPrefix(item.startNum);
const otherPrefix = this.extractPrefix(item.startNum);
// 检查前缀是否匹配(实现数字对应数字,字母对应字母的匹配规则)
if (currentPrefix !== otherPrefix) continue;
// 提取其他记录的数字范围
const otherStartNum = extractNumber(item.startNum);
const otherEndNum = extractNumber(item.endNum);
const otherStartNum = this.extractNumber(item.startNum);
const otherEndNum = this.extractNumber(item.endNum);
// 全面检查范围重叠
const hasOverlap =
@@ -996,12 +995,6 @@ export default {
// 如果当前号码为空,允许保存
if (!currentNum) return true;
// 数字部分比较(纯数字或字母前缀+数字)
const extractNumber = (str) => {
const match = str.match(/\d+$/);
return match ? parseInt(match[0], 10) : 0;
};
// 检查两个字符串的每一位是否类型匹配(数字对应数字,字母对应字母)
const checkCharacterTypesMatch = (str1, str2) => {
const maxLength = Math.max(str1.length, str2.length);
@@ -1023,18 +1016,13 @@ export default {
return false;
}
const currentNumValue = extractNumber(currentNum);
const startNumValue = extractNumber(startNum);
const endNumValue = extractNumber(endNum);
const currentNumValue = this.extractNumber(currentNum);
const startNumValue = this.extractNumber(startNum);
const endNumValue = this.extractNumber(endNum);
return currentNumValue >= startNumValue && currentNumValue <= endNumValue;
},
// 验证单个记录
// 测试用例说明:
// 1. 测试起始和终止号码前缀一致性:例如"AB123"和"AB456"通过,"AB123"和"AC456"失败
// 2. 测试当前号码与起始号码前缀一致性:例如"AB123"作为起始,"AB125"作为当前号码通过,"AC125"失败
// 3. 测试无字母情况:纯数字"123456"作为起始、终止和当前号码都通过验证
validateRecord(record, rowIndex) {
const errors = [];
const rowInfo = rowIndex ? `${rowIndex}` : '';
@@ -1145,24 +1133,23 @@ export default {
const savePromises = recordsToSave.map((record, index) => {
// 准备发送到后端的数据格式适配adm_invoice_segment表结构
const dataToSend = {
// 对于更新操作同时传递id和segmentId确保后端能正确识别
...(!record.isNewRecord && { id: record.keyId }),
// 确保segmentId不为null优先使用keyId然后是segmentId新记录时使用生成的临时ID
segmentId: record.keyId || record.segmentId || Date.now(),
// 员工和开票员信息
employeeId: record.employeeId,
employeeName: record.operator || this.getUserNameById(record.employeeId),
// 员工和开票员信息确保字段名称与后端API期望一致
invoicingStaffId: record.employeeId,
invoicingStaffName: record.operator || this.getUserNameById(record.employeeId),
// 日期信息
employeeId: record.employeeId, // 同时提供employeeId字段以保持一致性
// 日期信息确保同时设置billBusDate和createDate字段
billBusDate: record.date ? new Date(record.date) : null,
createDate: record.date ? new Date(record.date) : null,
// 号码信息,使用数据库表对应的字段名
// 号码信息,使用后端API期望的字段名
beginNumber: record.startNum,
startNum: record.startNum, // 同时提供startNum字段以保持一致性
endNumber: record.endNum,
endNum: record.endNum, // 同时提供endNum字段以保持一致性
currentNumber: record.currentNum,
currentNum: record.currentNum, // 同时提供currentNum字段以保持一致性
// 状态信息
status: record.status || '未使用',
statusEnum: { value: record.status || '未使用' }, // 同时提供statusEnum字段
// 备注信息
remark: record.currentNum ?
(record.remark && !record.remark.includes('当前使用号码:') ?
@@ -1174,6 +1161,13 @@ export default {
deleteFlag: '0'
};
// 对于更新记录设置id字段对于新记录不设置id让后端自动生成
if (!record.isNewRecord) {
dataToSend.id = record.id || record.keyId;
}
// 对于所有记录确保设置segmentId字段因为数据库表中segment_id是必填的
dataToSend.segmentId = record.segmentId || Date.now(); // 确保segmentId有值
// 添加调试信息,打印完整的记录对象
console.log(`准备${record.isNewRecord ? '新增' : '更新'}的原始记录对象:`, {
keyId: record.keyId,
@@ -1200,6 +1194,8 @@ export default {
return response;
}).catch(err => {
console.error(`${operation}记录 ${index + 1} 失败:`, err);
console.error(`${operation}记录 ${index + 1} 失败详情:`, JSON.stringify(err, null, 2));
console.error(`${operation}记录 ${index + 1} 请求数据:`, JSON.stringify(dataToSend, null, 2));
throw err; // 重新抛出错误以便上层catch捕获
});
});

View File

@@ -33,8 +33,9 @@ export function deleteOrganization(orgIds) {
export function getOrgDetail(id) {
return request({
url: '/base-data-manage/organization/organization?orgId=' + id,
url: '/base-data-manage/organization/organization-getById',
method: 'get',
params: { orgId: id }
})
}

View File

@@ -41,6 +41,14 @@
<el-table-column label="科室分类" align="center" prop="classEnum_dictText" />
<el-table-column label="医保码" align="center" prop="ybNo" />
<el-table-column label="医保名称" align="center" prop="ybName" />
<el-table-column label="挂号科室" align="center">
<template #default="scope">
{{ scope.row.registerFlag ? '是' : '否' }}
</template>
</el-table-column>
<el-table-column label="科室位置" align="center" prop="location" show-overflow-tooltip />
<el-table-column label="科室简介" align="center" prop="intro" show-overflow-tooltip />
<el-table-column label="备注" align="center" prop="remark" show-overflow-tooltip />
<el-table-column label="状态" align="center" prop="activeFlag_dictText" />
<el-table-column label="操作" align="center">
<template #default="scope">
@@ -129,6 +137,35 @@
/>
</el-form-item>
</el-col>
<el-form-item label="挂号科室" prop="registerFlag">
<el-radio-group v-model="form.registerFlag" size="large">
<el-radio :label="true"></el-radio>
<el-radio :label="false"></el-radio>
</el-radio-group>
</el-form-item>
<el-form-item label="科室位置" prop="location">
<el-input v-model="form.location" placeholder="请输入科室位置" maxlength="100" show-word-limit />
</el-form-item>
<el-form-item label="科室简介" prop="intro">
<el-input
v-model="form.intro"
type="textarea"
placeholder="请输入科室简介"
maxlength="500"
show-word-limit
:rows="4"
/>
</el-form-item>
<el-form-item label="备注" prop="remark">
<el-input
v-model="form.remark"
type="textarea"
placeholder="请输入备注信息"
maxlength="1000"
show-word-limit
:rows="3"
/>
</el-form-item>
</el-form>
<template #footer>
<div class="dialog-footer">
@@ -141,7 +178,7 @@
</template>
<script setup name="Organization">
import { getList, deleteOrganization, addOrganization, updateOrganization, disableOrg, initOrgTypeOption, enableOrg } from './components/api';
import { getList, deleteOrganization, addOrganization, updateOrganization, disableOrg, initOrgTypeOption, enableOrg, getOrgDetail } from './components/api';
const { proxy } = getCurrentInstance();
const loading = ref(true);
@@ -154,6 +191,10 @@ const form = ref({
name: undefined,
typeEnum: undefined,
busNoParent: undefined,
registerFlag: false, // 挂号科室标记
location: undefined, // 科室位置
intro: undefined, // 科室简介
remark: undefined, // 备注
});
const orgTableRef = ref();
const orgRef = ref();
@@ -169,6 +210,10 @@ const rules = ref({
{ min: 2, max: 20, message: '长度在 2 到 20 个字符', trigger: 'change' },
],
typeEnum: [{ required: true, message: '请选择科室类型', trigger: 'change' }],
// 新增字段验证规则
location: [{ required: false, message: '请输入科室位置', trigger: 'blur' }],
intro: [{ required: false, message: '请输入科室简介', trigger: 'blur' }],
remark: [{ required: false, message: '请输入备注信息', trigger: 'blur' }],
});
getPageList();
@@ -230,6 +275,10 @@ function initOption() {
function reset() {
form.value.id = undefined;
form.value.registerFlag = false;
form.value.location = undefined;
form.value.intro = undefined;
form.value.remark = undefined;
orgRef.value.resetFields();
}
@@ -248,7 +297,7 @@ function getDictLabel(value) {
}
function getPageList() {
loading.value = false;
loading.value = true;
getList(queryParams.value).then((res) => {
// 处理返回的科室数据,确保科室分类显示与系统标准字典一致
const processedData = res.data.records.map(item => {
@@ -269,6 +318,12 @@ function getPageList() {
organization.value = processedData;
total.value = res.data.total;
}).catch(error => {
console.error('获取科室列表失败:', error);
proxy.$modal.msgError('获取科室列表失败,请稍后重试');
organization.value = [];
total.value = 0;
}).finally(() => {
loading.value = false;
});
}
@@ -285,17 +340,29 @@ function handelEdit(row) {
title.value = '编辑科室';
open.value = true;
setTimeout(() => {
form.value.id = row.id;
form.value.busNo = row.busNo;
form.value.name = row.name;
form.value.ybNo = row.ybNo;
form.value.ybName = row.ybName;
form.value.typeEnum = row.typeEnum;
// 调用后端API获取完整的科室信息确保包含所有字段
getOrgDetail(row.id).then(res => {
if (res.code === 200) {
const orgInfo = res.data;
form.value.id = orgInfo.id;
form.value.busNo = orgInfo.busNo;
form.value.name = orgInfo.name;
form.value.ybNo = orgInfo.ybNo;
form.value.ybName = orgInfo.ybName;
form.value.typeEnum = orgInfo.typeEnum;
// 确保科室分类值的类型正确,使其能正确匹配下拉选项中的值
form.value.classEnum = row.classEnum !== undefined ? String(row.classEnum) : undefined;
form.value.busNoParent = row.busNo.split('.').length > 1 ? row.busNo.split('.')[0] : undefined;
}, 50);
form.value.classEnum = orgInfo.classEnum !== undefined ? String(orgInfo.classEnum) : undefined;
form.value.busNoParent = orgInfo.busNo.split('.').length > 1 ? orgInfo.busNo.split('.')[0] : undefined;
form.value.registerFlag = !!orgInfo.registerFlag;
form.value.location = orgInfo.location;
form.value.intro = orgInfo.intro;
form.value.remark = orgInfo.remark;
}
}).catch(error => {
console.error('获取科室信息失败:', error);
proxy.$modal.msgError('获取科室信息失败');
});
}
function cancel() {
@@ -308,20 +375,43 @@ function cancel() {
function submitForm() {
proxy.$refs['orgRef'].validate((valid) => {
if (valid) {
// 创建表单数据副本避免直接修改原始form对象
const formData = { ...form.value };
// 确保registerFlag从布尔值转换为整数true=1, false=0
formData.registerFlag = Number(formData.registerFlag ? 1 : 0);
// 确保classEnum字段有值数据库必填
// 如果未定义设置默认值1
if (formData.classEnum === undefined || formData.classEnum === null || formData.classEnum === '') {
formData.classEnum = 1;
}
// 确保classEnum为数字类型
formData.classEnum = Number(formData.classEnum);
// 验证提交数据
console.log('提交的数据:', formData);
if (form.value.id == undefined) {
if (form.value.busNoParent) {
form.value.busNo = form.value.busNoParent;
formData.busNo = form.value.busNoParent;
}
addOrganization(form.value).then((res) => {
addOrganization(formData).then((res) => {
proxy.$modal.msgSuccess('操作成功');
open.value = false;
getPageList();
}).catch(error => {
console.error('添加科室失败:', error);
proxy.$modal.msgError('添加科室失败,请稍后重试');
});
} else {
updateOrganization(form.value).then((res) => {
updateOrganization(formData).then((res) => {
proxy.$modal.msgSuccess('操作成功');
open.value = false;
getPageList();
}).catch(error => {
console.error('更新科室失败:', error);
proxy.$modal.msgError('更新科室失败,请稍后重试');
});
}
}
@@ -340,8 +430,12 @@ function handleDelete() {
loading.value = true;
deleteOrganization(selectRowIds.value.join(',')).then((res) => {
proxy.$modal.msgSuccess('操作成功');
loading.value = false;
getPageList();
}).catch(error => {
console.error('删除科室失败:', error);
proxy.$modal.msgError('删除科室失败,请稍后重试');
}).finally(() => {
loading.value = false;
});
}
// 停用
@@ -349,6 +443,9 @@ function handleDisabled(id) {
disableOrg(id).then((res) => {
proxy.$modal.msgSuccess('操作成功');
getPageList();
}).catch(error => {
console.error('停用科室失败:', error);
proxy.$modal.msgError('停用科室失败,请稍后重试');
});
}
@@ -357,6 +454,9 @@ function handelEnable(id) {
enableOrg(id).then((res) => {
proxy.$modal.msgSuccess('操作成功');
getPageList();
}).catch(error => {
console.error('启用科室失败:', error);
proxy.$modal.msgError('启用科室失败,请稍后重试');
});
}

View File

@@ -7,19 +7,12 @@ import request from '@/utils/request'
/**
* 分页查询门诊号码段列表
* 要求:普通用户只能查看自己的,管理员可以查看所有
* 注意由于后端接口不存在直接返回失败响应让调用方使用localStorage数据避免404错误
*/
export function listOutpatientNo(query) {
// return request({
// url: '/business-rule/outpatient-no/page',
// method: 'get',
// params: query,
// 由于后端接口不存在直接返回失败响应不发送实际请求避免控制台显示404错误
// 调用方会在判断 code !== 200 时使用 localStorage 数据
return Promise.resolve({
code: 404,
msg: '接口不存在,已使用本地数据',
data: null
return request({
url: '/business-rule/outpatient-no/page',
method: 'get',
params: query,
})
}
@@ -50,11 +43,11 @@ export function updateOutpatientNo(data) {
* 删除门诊号码段
*要求:双重校验(归属权+使用状态)
*/
export function deleteOutpatientNo(params) {
export function deleteOutpatientNo(data) {
return request({
url: '/business-rule/outpatient-no',
method: 'delete',
params,
data,
})
}

View File

@@ -1,6 +1,6 @@
<template>
<!-- Windows XP风格窗口布局600px固定宽度 -->
<div class="outpatient-no-management-wrapper">
<!-- Windows XP风格窗口布局全屏显示 -->
<div class="outpatient-no-management">
<!--标题栏32px高 -->
<div class="title-bar">
@@ -44,62 +44,25 @@
<el-table-column type="selection" width="50" align="center" />
<el-table-column label="序号" type="index" width="60" align="center" />
<el-table-column label="操作员" prop="operatorName" min-width="120" />
<el-table-column label="员工工号" prop="staffNo" min-width="120">
<template #default="{ row }">
<el-input
v-if="row._editing"
v-model.trim="row.staffNo"
/>
<span v-else>{{ row.staffNo }}</span>
</template>
</el-table-column>
<el-table-column label="员工工号" prop="staffNo" min-width="120" />
<el-table-column label="领用日期" prop="receiveDate" min-width="140">
<template #default="{ row }">
<el-date-picker
v-if="row._editing"
v-model="row.receiveDate"
type="date"
value-format="YYYY-MM-DD"
placeholder="选择日期"
style="width: 100%"
/>
<span v-else>{{ row.receiveDate }}</span>
{{ formatReceiveDate(row.receiveDate) }}
</template>
</el-table-column>
<el-table-column label="起始号码" prop="startNo" min-width="140">
<el-table-column label="起始号码" prop="startNo" min-width="140" />
<el-table-column label="终止号码" prop="endNo" min-width="140" />
<el-table-column label="使用号码" prop="usedNo" min-width="140" />
<el-table-column label="操作" width="120" align="center" fixed="right">
<template #default="{ row }">
<el-input
v-if="row._editing"
v-model.trim="row.startNo"
@input="() => onStartNoChange(row)"
@blur="() => validateNumField(row, 'startNo')"
/>
<span v-else>{{ row.startNo }}</span>
</template>
</el-table-column>
<el-table-column label="终止号码" prop="endNo" min-width="140">
<template #default="{ row }">
<el-input
v-if="row._editing"
v-model.trim="row.endNo"
@blur="() => validateNumField(row, 'endNo')"
/>
<span v-else>{{ row.endNo }}</span>
</template>
</el-table-column>
<el-table-column label="使用号码" prop="usedNo" min-width="140">
<template #default="{ row }">
<el-input
v-if="row._editing"
v-model.trim="row.usedNo"
@blur="() => validateNumField(row, 'usedNo')"
/>
<span v-else>{{ row.usedNo }}</span>
</template>
</el-table-column>
<el-table-column label="操作" width="100" fixed="right">
<template #default="scope">
<el-button type="primary" link icon="Edit" @click="() => openEdit(scope.row, scope.$index)">编辑</el-button>
<el-button
type="primary"
link
size="small"
@click="handleEdit(row)"
>
编辑
</el-button>
</template>
</el-table-column>
</el-table>
@@ -112,40 +75,65 @@
/>
</div>
</div>
</div>
<!-- 编辑弹窗 -->
<el-dialog v-model="dialogVisible" :title="dialogTitle" width="600px" append-to-body>
<el-form label-width="100px">
<el-form-item label="操作员">
<el-dialog
v-model="editDialogVisible"
title="编辑门诊号码段"
width="600px"
:close-on-click-modal="false"
>
<el-form
ref="editFormRef"
:model="editForm"
:rules="editFormRules"
label-width="120px"
>
<el-form-item label="操作员" prop="operatorName">
<el-input v-model="editForm.operatorName" disabled />
</el-form-item>
<el-form-item label="员工工号">
<el-form-item label="员工工号" prop="staffNo">
<el-input v-model.trim="editForm.staffNo" />
</el-form-item>
<el-form-item label="领用日期">
<el-date-picker v-model="editForm.receiveDate" type="date" value-format="YYYY-MM-DD" style="width: 100%" />
<el-form-item label="领用日期" prop="receiveDate">
<el-date-picker
v-model="editForm.receiveDate"
type="date"
value-format="YYYY.MM.DD"
placeholder="选择日期"
style="width: 100%"
/>
</el-form-item>
<el-form-item label="起始号码">
<el-input v-model.trim="editForm.startNo" @blur="() => validateNumField(editForm, 'startNo')" />
<el-form-item label="起始号码" prop="startNo">
<el-input
v-model.trim="editForm.startNo"
@input="onEditFormStartNoChange"
@blur="validateEditFormField('startNo')"
/>
</el-form-item>
<el-form-item label="终止号码">
<el-input v-model.trim="editForm.endNo" @blur="() => validateNumField(editForm, 'endNo')" />
<el-form-item label="终止号码" prop="endNo">
<el-input
v-model.trim="editForm.endNo"
@blur="validateEditFormField('endNo')"
/>
</el-form-item>
<el-form-item label="使用号码">
<el-input v-model.trim="editForm.usedNo" @blur="() => validateNumField(editForm, 'usedNo')" />
<el-form-item label="使用号码" prop="usedNo">
<el-input
v-model.trim="editForm.usedNo"
@blur="validateEditFormField('usedNo')"
/>
</el-form-item>
</el-form>
<template #footer>
<div class="dialog-footer">
<el-button @click="dialogVisible = false"> </el-button>
<el-button type="primary" @click="confirmEdit"> </el-button>
</div>
<el-button @click="editDialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSaveEditForm">保存</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup name="outpatientNoManagement">
import { ref, reactive, toRefs, onActivated, getCurrentInstance } from 'vue'
import useUserStore from '@/store/modules/user'
import { getConfigKey } from '@/api/system/config'
import { logQuery, logCreate, logUpdate, logDelete } from './components/operationLog'
@@ -160,6 +148,34 @@ const getUserInfo = () => ({
name: userStore.name || userStore.nickName
})
// 格式化领用日期为 YYYY.MM.DD 格式(用于显示)
function formatReceiveDate(dateStr) {
if (!dateStr) return ''
// 如果是 YYYY-MM-DD 格式,转换为 YYYY.MM.DD
if (dateStr.includes('-')) {
return dateStr.replace(/-/g, '.')
}
// 如果已经是 YYYY.MM.DD 格式,直接返回
if (dateStr.includes('.')) {
return dateStr
}
return dateStr
}
// 将日期格式从 YYYY.MM.DD 转换为 yyyy-MM-dd用于发送到后端
function convertDateForBackend(dateStr) {
if (!dateStr) return null
// 如果是 YYYY.MM.DD 格式,转换为 yyyy-MM-dd
if (dateStr.includes('.')) {
return dateStr.replace(/\./g, '-')
}
// 如果已经是 yyyy-MM-dd 格式,直接返回
if (dateStr.includes('-')) {
return dateStr
}
return dateStr
}
const loading = ref(false)
const tableData = ref([])
const total = ref(0)
@@ -167,18 +183,31 @@ const ids = ref([])
const multiple = ref(true)
const viewAll = ref(false)
const canToggleViewAll = ref(false)
const dialogVisible = ref(false)
const dialogTitle = ref('编辑门诊号码段')
const editIndex = ref(-1)
// 编辑弹窗相关
const editDialogVisible = ref(false)
const editFormRef = ref(null)
const editForm = reactive({
id: null,
operatorId: null,
operatorName: '',
staffNo: '',
receiveDate: '',
startNo: '',
endNo: '',
usedNo: '',
operatorName: '',
staffNo: '',
_originalData: null,
})
// 编辑表单验证规则
const editFormRules = {
staffNo: [{ required: true, message: '请输入员工工号', trigger: 'blur' }],
receiveDate: [{ required: true, message: '请选择领用日期', trigger: 'change' }],
startNo: [{ required: true, message: '请输入起始号码', trigger: 'blur' }],
endNo: [{ required: true, message: '请输入终止号码', trigger: 'blur' }],
usedNo: [{ required: true, message: '请输入使用号码', trigger: 'blur' }],
}
const data = reactive({
queryParams: {
pageNo: 1,
@@ -216,18 +245,146 @@ function onAdd() {
const yyyy = now.getFullYear()
const mm = String(now.getMonth() + 1).padStart(2, '0')
const dd = String(now.getDate()).padStart(2, '0')
tableData.value.push({
id: undefined,
operatorId: userStore.id,
operatorName: userStore.name || userStore.nickName,
staffNo: userStore.id,
receiveDate: `${yyyy}-${mm}-${dd}`,
startNo: '',
endNo: '',
usedNo: '',
_editing: true,
_error: false,
})
// 打开编辑弹窗,用于新增
editForm.id = undefined
editForm.operatorId = userStore.id
editForm.operatorName = userStore.name || userStore.nickName
editForm.staffNo = userStore.id
editForm.receiveDate = `${yyyy}.${mm}.${dd}`
editForm.startNo = ''
editForm.endNo = ''
editForm.usedNo = ''
editForm._originalData = null
editDialogVisible.value = true
}
// 编辑行 - 打开编辑弹窗
function handleEdit(row) {
// 复制行数据到编辑表单
editForm.id = row.id
editForm.operatorId = row.operatorId
editForm.operatorName = row.operatorName
editForm.staffNo = row.staffNo
editForm.receiveDate = formatReceiveDate(row.receiveDate)
editForm.startNo = row.startNo
editForm.endNo = row.endNo
editForm.usedNo = row.usedNo
editForm._originalData = { ...row }
editDialogVisible.value = true
}
// 编辑表单中起始号码变化时自动设置使用号码
function onEditFormStartNoChange() {
if (!editForm.id && editForm.startNo) {
editForm.usedNo = editForm.startNo
}
}
// 验证编辑表单字段
function validateEditFormField(field) {
if (!isTailDigitsValid(editForm[field])) {
const msg = `最大位数为12位且必须以数字结尾`
alertWarn(msg)
return false
}
return true
}
// 保存编辑表单
async function handleSaveEditForm() {
// 表单验证
if (!editFormRef.value) return
try {
const valid = await editFormRef.value.validate()
if (!valid) return
} catch (error) {
return
}
// 字段验证
if (!validateEditFormField('startNo') || !validateEditFormField('endNo') || !validateEditFormField('usedNo')) {
return
}
// 构建验证用的行对象
const validateRowData = {
startNo: editForm.startNo,
endNo: editForm.endNo,
usedNo: editForm.usedNo,
}
// 验证行数据(编辑时排除当前记录的 id避免误判为重复
const rowIndex = editForm.id ? tableData.value.findIndex(r => r.id === editForm.id) : -1
if (!validateRow(validateRowData, rowIndex, editForm.id)) {
return
}
// 准备保存的数据
const saveData = {
id: editForm.id,
operatorId: editForm.operatorId,
operatorName: editForm.operatorName,
staffNo: editForm.staffNo,
receiveDate: convertDateForBackend(editForm.receiveDate),
startNo: editForm.startNo,
endNo: editForm.endNo,
usedNo: editForm.usedNo,
}
try {
let res
if (saveData.id) {
// 更新
res = await updateOutpatientNo(saveData)
} else {
// 新增
res = await addOutpatientNo(saveData)
}
if (res.code === 200) {
proxy.$modal.msgSuccess('保存成功')
editDialogVisible.value = false
getList()
} else {
// 显示后端返回的错误信息
const errorMsg = res.msg || res.message || '保存失败'
proxy.$modal.msgError(errorMsg)
if (saveData.id) {
logUpdate(saveData, false, errorMsg, getUserInfo())
} else {
logCreate(saveData, false, errorMsg, getUserInfo())
}
}
} catch (error) {
console.error('保存失败:', error)
// 从错误对象中提取错误信息
let errorMsg = '保存失败'
// 优先从 response.data.msg 获取
if (error.response?.data?.msg) {
errorMsg = error.response.data.msg
}
// 其次从 error.message 获取request.js 会通过 new Error(msg) 抛出)
else if (error.message) {
// 如果 error.message 包含 "Error: " 前缀,去掉它
errorMsg = error.message.replace(/^Error:\s*/, '')
}
// 最后尝试从 error.msg 获取
else if (error.msg) {
errorMsg = error.msg
}
// 显示错误信息
proxy.$modal.msgError(errorMsg)
// 记录日志
if (saveData.id) {
logUpdate(saveData, false, errorMsg, getUserInfo())
} else {
logCreate(saveData, false, errorMsg, getUserInfo())
}
}
}
// 新增时,起始号码变化时自动设置使用号码为起始号码
@@ -238,36 +395,46 @@ function onStartNoChange(row) {
}
function onClose() {
// 检查是否有未保存的数据变动
const hasUnsavedChanges = tableData.value.some(row => row._dirty || row._editing)
if (hasUnsavedChanges) {
// 有未保存的数据,提示用户
const message = '窗口数据有变动是否进行保存操作?'
if (proxy.$modal?.confirm) {
proxy.$modal.confirm(message).then(() => {
// 用户选择保存
onSave()
// 等待保存完成后关闭页面
setTimeout(() => {
proxy.$tab.closePage()
}, 800)
}).catch(() => {
// 用户选择不保存,直接关闭
proxy.$tab.closePage()
})
} else {
// 降级方案:使用原生 confirm
if (confirm(message)) {
onSave()
setTimeout(() => {
proxy.$tab.closePage()
}, 800)
} else {
proxy.$tab.closePage()
}
}
} else {
// 没有未保存的数据,直接关闭
proxy.$tab.closePage()
}
}
function tableRowClassName({ row }) {
return row._error ? 'error-row' : ''
}
function openEdit(row, index) {
editIndex.value = index
dialogTitle.value = '编辑门诊号码段'
editForm.receiveDate = row.receiveDate
editForm.startNo = row.startNo
editForm.endNo = row.endNo
editForm.usedNo = row.usedNo
editForm.operatorName = row.operatorName
editForm.staffNo = row.staffNo
dialogVisible.value = true
}
function confirmEdit() {
const tmp = { ...tableData.value[editIndex.value], ...editForm }
if (!validateRow(tmp, editIndex.value)) return
tableData.value[editIndex.value] = {
...tableData.value[editIndex.value],
...editForm,
_dirty: true, // 标记为已修改,顶部保存时提交
}
dialogVisible.value = false
}
// 字母前缀识别规则 - 从末位往前找到第一个字母
function extractPrefix(value) {
if (!value) return ''
@@ -315,7 +482,7 @@ function isTailDigitsValid(value) {
function validateNumField(row, field, rowIndex) {
if (!isTailDigitsValid(row[field])) {
const idxInTable = typeof rowIndex === 'number' ? rowIndex : tableData.value.indexOf(row)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : (editIndex.value >= 0 ? editIndex.value + 1 : undefined)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : undefined
const msg = lineNo ? `第【${lineNo}】行数据中最大位数为12位且必须以数字结尾` : '最大位数为12位且必须以数字结尾'
alertWarn(msg)
row._error = true
@@ -341,11 +508,11 @@ function onNumberInput(row, field) {
}
}
function validateRow(row, rowIndex) {
function validateRow(row, rowIndex, excludeId = null) {
row._error = false
if (!lengthWithinLimit(row.startNo) || !lengthWithinLimit(row.endNo) || !lengthWithinLimit(row.usedNo)) {
const idxInTable = typeof rowIndex === 'number' ? rowIndex : tableData.value.indexOf(row)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : (editIndex.value >= 0 ? editIndex.value + 1 : undefined)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : undefined
const msg = lineNo ? `第【${lineNo}】行数据中最大位数为12位且必须以数字结尾` : '最大位数为12位且必须以数字结尾'
alertWarn(msg)
row._error = true
@@ -353,7 +520,7 @@ function validateRow(row, rowIndex) {
}
if ((row.startNo?.length || 0) !== (row.endNo?.length || 0)) {
const idxInTable = typeof rowIndex === 'number' ? rowIndex : tableData.value.indexOf(row)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : (editIndex.value >= 0 ? editIndex.value + 1 : undefined)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : undefined
const msg = lineNo ? `第【${lineNo}】行数据中,起始号码与终止号码长度必须一致,请修改!` : '起始号码与终止号码长度必须一致'
alertWarn(msg)
row._error = true
@@ -364,7 +531,7 @@ function validateRow(row, rowIndex) {
const p3 = extractPrefix(row.usedNo)
if (!(p1 === p2 && p2 === p3)) {
const idxInTable = typeof rowIndex === 'number' ? rowIndex : tableData.value.indexOf(row)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : (editIndex.value >= 0 ? editIndex.value + 1 : undefined)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : undefined
const msg = lineNo ? `第【${lineNo}】行数据中,门诊号码的字母前缀必须相同,请修改!` : '行数据中,门诊号码的字母前缀必须相同,请修改!'
alertWarn(msg)
row._error = true
@@ -374,25 +541,29 @@ function validateRow(row, rowIndex) {
const eNum = extractTailNumber(row.endNo)
if (Number.isNaN(sNum) || Number.isNaN(eNum) || sNum > eNum) {
const idxInTable = typeof rowIndex === 'number' ? rowIndex : tableData.value.indexOf(row)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : (editIndex.value >= 0 ? editIndex.value + 1 : undefined)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : undefined
const msg = lineNo ? `第【${lineNo}】行数据中,起始/终止号码不合法` : '起始/终止号码不合法'
alertWarn(msg)
row._error = true
return false
}
// 放宽:不再强制使用号码必须处于起始与终止范围内
// 放宽:不再强制"使用号码"必须处于起始与终止范围内
const prefix = p1
for (let i = 0; i < tableData.value.length; i++) {
const other = tableData.value[i]
// 跳过自身:当从弹窗校验时 row 为临时对象,需用下标判断
if ((typeof rowIndex === 'number' && i === rowIndex) || other === row || !other.startNo || !other.endNo) continue
// 跳过自身:通过 rowIndex、row 对象比较或 excludeId 排除
if ((typeof rowIndex === 'number' && i === rowIndex) ||
other === row ||
!other.startNo ||
!other.endNo ||
(excludeId && other.id === excludeId)) continue
if (extractPrefix(other.startNo) !== prefix) continue
const os = extractTailNumber(other.startNo)
const oe = extractTailNumber(other.endNo)
if (!Number.isNaN(os) && !Number.isNaN(oe)) {
if (rangesOverlap(sNum, eNum, os, oe)) {
const idxInTable = typeof rowIndex === 'number' ? rowIndex : tableData.value.indexOf(row)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : (editIndex.value >= 0 ? editIndex.value + 1 : undefined)
const lineNo = idxInTable >= 0 ? idxInTable + 1 : undefined
const msg = lineNo ? `第【${lineNo}】行数据中,门诊号码和【${i + 1}】行的门诊号码有冲突,请修改!` : '门诊号码设置重复!'
alertWarn(msg)
row._error = true
@@ -403,90 +574,233 @@ function validateRow(row, rowIndex) {
return true
}
function onSave(row) {
async function onSave(row) {
const rows = row ? [row] : tableData.value.filter(r => ids.value.includes(r.id) || r._dirty || r._editing)
if (!rows.length) return
for (const r of rows) {
const idx = tableData.value.indexOf(r)
if (!validateRow(r, idx)) return
if (!rows.length) {
proxy.$modal.msgWarning('没有可保存的数据')
return
}
// 准备保存的数据
// 前端校验
for (const r of rows) {
const idx = tableData.value.indexOf(r)
if (!validateRow(r, idx, r.id)) return
}
// 准备保存的数据(将日期格式转换为后端需要的格式)
const saveData = rows.map((r) => ({
id: r.id,
operatorId: r.operatorId,
operatorName: r.operatorName,
staffNo: r.staffNo,
receiveDate: r.receiveDate,
receiveDate: convertDateForBackend(r.receiveDate), // 转换为 yyyy-MM-dd 格式
startNo: r.startNo,
endNo: r.endNo,
usedNo: r.usedNo,
}))
const ok = lcUpsertMany(saveData)
if (!ok) {
// 记录失败的操作日志
for (const record of saveData) {
if (record.id) {
logUpdate(record, false, '门诊号码设置重复', getUserInfo())
} else {
logCreate(record, false, '门诊号码设置重复', getUserInfo())
}
}
return
}
// 批量保存
let successCount = 0
let failCount = 0
// 记录成功的操作日志
for (const record of saveData) {
try {
let res
if (record.id) {
// 更新
res = await updateOutpatientNo(record)
if (res.code === 200) {
logUpdate(record, true, null, getUserInfo())
successCount++
} else {
const errorMsg = res.msg || res.message || '保存失败'
proxy.$modal.msgError(errorMsg)
logUpdate(record, false, errorMsg, getUserInfo())
failCount++
}
} else {
// 新增
res = await addOutpatientNo(record)
if (res.code === 200) {
logCreate(record, true, null, getUserInfo())
successCount++
} else {
const errorMsg = res.msg || res.message || '保存失败'
proxy.$modal.msgError(errorMsg)
logCreate(record, false, errorMsg, getUserInfo())
failCount++
}
}
} catch (error) {
console.error('保存失败:', error)
// 从错误对象中提取错误信息
let errorMsg = '保存失败'
if (error.response?.data?.msg) {
errorMsg = error.response.data.msg
} else if (error.message) {
errorMsg = error.message
} else if (error.msg) {
errorMsg = error.msg
}
proxy.$modal.msgError(errorMsg)
if (record.id) {
logUpdate(record, false, errorMsg, getUserInfo())
} else {
logCreate(record, false, errorMsg, getUserInfo())
}
failCount++
}
}
if (proxy.$modal?.alertSuccess) {
proxy.$modal.alertSuccess('保存成功!')
} else {
proxy.$message.success('保存成功')
if (successCount > 0) {
proxy.$modal.msgSuccess(`保存成功${successCount}`)
// 保存成功后,退出编辑状态
rows.forEach(r => {
r._editing = false
r._dirty = false
r._originalData = null // 清除原始数据
})
}
if (failCount > 0) {
proxy.$modal.msgError(`保存失败${failCount}`)
}
getList()
}
function onDelete() {
const rows = tableData.value.filter((r) => ids.value.includes(r.id))
if (!rows.length) return
async function onDelete() {
console.log('删除操作 - 选中的ID列表:', ids.value)
console.log('删除操作 - 表格数据:', tableData.value.map(r => ({ id: r.id, operatorId: r.operatorId })))
// 双重校验(归属权+使用状态)
const rows = tableData.value.filter((r) => ids.value.includes(r.id))
console.log('删除操作 - 筛选后的行数据:', rows.map(r => ({ id: r.id, operatorId: r.operatorId })))
if (!rows.length) {
proxy.$modal.msgWarning('请选择要删除的数据')
return
}
// 前端预校验(归属权+使用状态)
for (const r of rows) {
console.log('检查删除权限 - 行数据:', {
id: r.id,
operatorId: r.operatorId,
operatorIdType: typeof r.operatorId,
userStoreId: userStore.id,
userStoreIdType: typeof userStore.id,
usedNo: r.usedNo,
startNo: r.startNo,
usedNoType: typeof r.usedNo,
startNoType: typeof r.startNo
})
const canDeleteSelf = String(r.operatorId) === String(userStore.id)
const neverUsed = r.usedNo === r.startNo
// 确保字符串比较,避免类型问题
const neverUsed = String(r.usedNo || '') === String(r.startNo || '')
if (!canDeleteSelf) {
// 权限不足提示
console.warn('删除权限检查失败:', {
operatorId: r.operatorId,
userStoreId: userStore.id,
canDeleteSelf
})
alertWarn('只能删除自己维护的门诊号码段')
logDelete(rows, false, '只能删除自己维护的门诊号码段', getUserInfo())
return
}
if (!neverUsed) {
// 已使用提示
console.warn('使用状态检查失败:', {
usedNo: r.usedNo,
startNo: r.startNo,
neverUsed
})
alertWarn('已有门诊号码段已有使用的门诊号码,请核对!')
logDelete(rows, false, '已有门诊号码段已有使用的门诊号码', getUserInfo())
return
}
}
const doRealDelete = () => {
lcDeleteByIds(rows.map((r) => r.id))
//记录成功的删除操作日志
logDelete(rows, true, null, getUserInfo())
if (proxy.$modal?.alertSuccess) {
proxy.$modal.alertSuccess('删除成功')
} else {
proxy.$message.success('删除成功')
const doRealDelete = async () => {
try {
// 处理大整数 ID保持为字符串或使用 BigInt避免精度丢失
// JavaScript 的 Number.MAX_SAFE_INTEGER = 9007199254740991
// 如果 ID 超过这个值,转换为数字会丢失精度
const idsToDelete = rows
.map((r) => r.id)
.filter(id => id != null && id !== undefined)
.map(id => {
// 如果 ID 是字符串,检查是否需要转换为数字
if (typeof id === 'string') {
// 尝试转换为数字,但如果超过安全整数范围,保持为字符串
const numId = parseInt(id, 10)
if (!isNaN(numId) && numId <= Number.MAX_SAFE_INTEGER) {
return numId
}
// 大整数保持为字符串,后端应该能处理
return id
}
// 如果已经是数字,检查是否超过安全范围
if (typeof id === 'number') {
if (id > Number.MAX_SAFE_INTEGER) {
// 超过安全范围,转换为字符串
return String(id)
}
return id
}
// 其他类型,尝试转换
return id
})
.filter(id => id != null && id !== undefined)
if (idsToDelete.length === 0) {
proxy.$modal.msgWarning('没有可删除的数据')
return
}
console.log('准备删除的ID列表:', idsToDelete)
console.log('删除请求数据:', JSON.stringify({ ids: idsToDelete }))
console.log('删除请求数据类型:', idsToDelete.map(id => ({ value: id, type: typeof id, stringValue: String(id) })))
const res = await deleteOutpatientNo({ ids: idsToDelete })
console.log('删除响应:', res)
console.log('删除响应code:', res?.code)
console.log('删除响应msg:', res?.msg)
if (res && (res.code === 200 || res.code === 0)) {
logDelete(rows, true, null, getUserInfo())
proxy.$modal.msgSuccess('删除成功')
getList()
} else {
const errorMsg = res?.msg || res?.message || '删除失败'
logDelete(rows, false, errorMsg, getUserInfo())
proxy.$modal.msgError(errorMsg)
}
} catch (error) {
console.error('删除失败:', error)
// 从错误对象中提取错误信息
let errorMsg = '删除失败'
// 优先从 response.data.msg 获取
if (error.response?.data?.msg) {
errorMsg = error.response.data.msg
}
// 其次从 error.message 获取request.js 会通过 new Error(msg) 抛出)
else if (error.message) {
// 如果 error.message 包含 "Error: " 前缀,去掉它
errorMsg = error.message.replace(/^Error:\s*/, '')
}
// 最后尝试从 error.msg 获取
else if (error.msg) {
errorMsg = error.msg
}
// 显示错误信息(参考 PackageManagement 的实现方式)
proxy.$modal.msgError('删除失败: ' + errorMsg)
// 记录日志
logDelete(rows, false, errorMsg, getUserInfo())
}
}
if (proxy.$modal?.confirm) {
@@ -502,115 +816,48 @@ function getList() {
loading.value = true
queryParams.value.onlySelf = !viewAll.value
// 先尝试调用后端API
// 调用后端API
listOutpatientNo(queryParams.value).then((res) => {
if (res.code === 200) {
tableData.value = (res.data?.records || res.data || []).map((it) => ({
tableData.value = (res.data?.records || res.data || []).map((it) => {
// 处理大整数 ID如果 ID 超过 JavaScript 安全整数范围,保持为字符串
let id = it.id
if (typeof id === 'number' && id > Number.MAX_SAFE_INTEGER) {
// 数字超过安全范围,转换为字符串以避免精度丢失
id = String(id)
} else if (typeof id === 'number') {
// 数字在安全范围内,保持为数字
id = id
} else {
// 已经是字符串或其他类型,保持原样
id = id
}
return {
...it,
id: id, // 确保 ID 正确处理
_editing: false,
_error: false,
_dirty: false,
}))
// 确保日期格式正确:后端返回 yyyy-MM-dd转换为 YYYY.MM.DD 用于显示
receiveDate: it.receiveDate ? formatReceiveDate(it.receiveDate) : ''
}
})
total.value = res.data?.total || res.data?.length || 0
// 记录查询操作日志
logQuery(total.value, getUserInfo())
} else {
// API返回错误回退到localStorage
console.warn('后端API返回错误使用localStorage数据')
loadFromLocalStorage()
proxy.$modal.msgError(res.msg || '查询失败')
}
loading.value = false
}).catch((error) => {
// API调用失败如404回退到localStorage
console.warn('后端API调用失败使用localStorage数据:', error)
loadFromLocalStorage()
console.error('查询门诊号码段失败:', error)
proxy.$modal.msgError('查询失败,请稍后重试')
loading.value = false
})
}
// 从localStorage加载数据
function loadFromLocalStorage() {
const res = lcList({ ...queryParams.value })
tableData.value = res.records.map((it) => ({
...it,
_editing: false,
_error: false,
_dirty: false,
}))
total.value = res.total
// 记录查询操作日志
logQuery(total.value, getUserInfo())
loading.value = false
}
// 纯前端本地持久化方法localStorage
const STORAGE_KEY = 'ohis_outpatient_no_segments'
function lcReadAll() {
try {
const raw = localStorage.getItem(STORAGE_KEY)
const arr = raw ? JSON.parse(raw) : []
return Array.isArray(arr) ? arr : []
} catch (e) {
return []
}
}
function lcWriteAll(list) {
localStorage.setItem(STORAGE_KEY, JSON.stringify(list || []))
}
function lcList({ pageNo = 1, pageSize = 10, onlySelf = true }) {
const all = lcReadAll()
const filtered = onlySelf ? all.filter((x) => String(x.operatorId) === String(userStore.id)) : all
const start = (pageNo - 1) * pageSize
const end = start + pageSize
return { records: filtered.slice(start, end), total: filtered.length, all }
}
function checkOverlapAll(row, all) {
const prefix = extractPrefix(row.startNo)
const sNum = extractTailNumber(row.startNo)
const eNum = extractTailNumber(row.endNo)
for (const it of all) {
if (row.id && it.id === row.id) continue
if (!it.startNo || !it.endNo) continue
if (extractPrefix(it.startNo) !== prefix) continue
const os = extractTailNumber(it.startNo)
const oe = extractTailNumber(it.endNo)
if (!Number.isNaN(os) && !Number.isNaN(oe)) {
if (rangesOverlap(sNum, eNum, os, oe)) return true
}
}
return false
}
function lcUpsertMany(rows) {
const all = lcReadAll()
for (const r of rows) {
if (checkOverlapAll(r, all)) {
alertWarn('门诊号码设置重复!')
return false
}
if (!r.id) {
r.id = `${Date.now()}_${Math.random().toString(36).slice(2, 8)}`
}
const idx = all.findIndex((x) => x.id === r.id)
if (idx >= 0) all[idx] = { ...all[idx], ...r }
else all.push({ ...r })
}
lcWriteAll(all)
return true
}
function lcDeleteByIds(idList) {
const all = lcReadAll()
const remain = all.filter((x) => !idList.includes(x.id))
lcWriteAll(remain)
}
</script>
<style scoped>
@@ -618,17 +865,14 @@ function lcDeleteByIds(idList) {
/* 外层容器 - 全屏显示 */
.outpatient-no-management-wrapper {
display: flex;
justify-content: flex-start;
align-items: flex-start;
width: 100%;
height: calc(100vh - 84px);
padding: 0;
background-color: #f0f0f0;
overflow: hidden;
padding: 0;
margin: 0;
}
/* 主容器 - 全屏宽度和高度 */
/* 主容器 - 全屏显示 */
.outpatient-no-management {
width: 100%;
height: 100%;
@@ -830,28 +1074,6 @@ function lcDeleteByIds(idList) {
background: linear-gradient(to bottom, #FFFEF8 0%, #F5F4EF 50%, #E5E4DF 100%);
}
/* 编辑弹窗样式 */
:deep(.el-dialog) {
border-radius: 0;
border: 2px solid #0055E5;
}
:deep(.el-dialog__header) {
background: linear-gradient(to bottom, #0055E5 0%, #0F3D8C 100%);
padding: 10px 15px;
margin: 0;
}
:deep(.el-dialog__title) {
color: #FFFFFF;
font-weight: 700;
font-size: 14px;
}
:deep(.el-dialog__close) {
color: #FFFFFF;
}
/* 表格容器样式调整 */
.table-content :deep(.el-table__body-wrapper) {
flex: 1;

View File

@@ -281,7 +281,7 @@
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="医生:" prop="practitionerId">
<el-form-item label="出诊医生:" prop="practitionerId">
<el-select
v-model="form.practitionerId"
placeholder="医生"
@@ -755,6 +755,7 @@ const transformFormData = (form) => {
ybType,
title,
comment,
practitionerId,
} = form.value;
return {
@@ -771,6 +772,7 @@ const transformFormData = (form) => {
appointmentRequiredFlag,
extraDetails,
comment,
practitionerId,
},
chargeItemDefinitionFormData: {
id,
@@ -807,6 +809,7 @@ const transformFormEditData = (form) => {
ybType,
title,
comment,
practitionerId,
} = form.value;
return {
@@ -823,6 +826,7 @@ const transformFormEditData = (form) => {
appointmentRequiredFlag,
extraDetails,
comment,
practitionerId,
},
};
};

View File

@@ -66,7 +66,7 @@
</el-col> -->
</el-row>
<el-table v-loading="loading" :data="deviceList" @selection-change="handleSelectionChange" width="90%">
<el-table v-loading="loading" :data="deviceList" @selection-change="handleSelectionChange" width="90%" border resizable-column>
<el-table-column type="selection" width="50" align="center" />
<el-table-column label="编码" align="center" key="busNo" prop="busNo" :show-overflow-tooltip="true" />
<el-table-column label="器材名称" align="center" key="name" prop="name" :show-overflow-tooltip="true" />

View File

@@ -1,16 +1,16 @@
<template>
<!-- <div class="app-container"> -->
<!-- 添加或修改对话框 -->
<el-dialog :title="title" v-model="visible" width="1020px" append-to-body>
<el-dialog :title="title" v-model="visible" width="1200px" append-to-body>
<el-form ref="patientRef" :model="form" :rules="rules" label-width="120px" label-position="left">
<!-- 第一行姓名民族性别 -->
<!-- 第一行姓名民族文化程度性别 -->
<el-row :gutter="10">
<el-col :span="8">
<el-col :span="6">
<el-form-item label="姓名" prop="name" label-width="80px">
<el-input v-model="form.name" clearable :disabled="isViewMode" />
</el-form-item>
</el-col>
<el-col :span="8">
<el-col :span="6">
<el-form-item label="民族" prop="nationalityCode" label-width="80px">
<el-select v-model="form.nationalityCode" clearable filterable :disabled="isViewMode">
<el-option
@@ -22,12 +22,28 @@
</el-select>
</el-form-item>
</el-col>
<el-col :span="8">
<el-col :span="6">
<el-form-item label="文化程度" prop="educationLevel" label-width="80px">
<el-select v-model="form.educationLevel" placeholder="请选择文化程度" clearable :disabled="isViewMode">
<el-option
v-for="item in educationLevelList"
:key="item.value"
:label="item.info"
:value="item.value"
/>
</el-select>
</el-form-item>
</el-col>
<el-col :span="6">
<el-form-item label="性别" prop="genderEnum" label-width="80px">
<el-radio-group v-model="form.genderEnum" :disabled="isViewMode">
<el-radio :label="0">男性</el-radio>
<el-radio :label="1">女性</el-radio>
</el-radio-group>
<el-select v-model="form.genderEnum" placeholder="请选择性别" clearable :disabled="isViewMode">
<el-option
v-for="item in administrativegenderList"
:key="item.value"
:label="item.info"
:value="item.value"
/>
</el-select>
</el-form-item>
</el-col>
</el-row>
@@ -70,11 +86,18 @@
</el-col>
</el-row>
<!-- 第三行家编码*联系方式工作单位 -->
<!-- 第三行*联系方式工作单位 -->
<el-row :gutter="10">
<el-col :span="8">
<el-form-item label="国家编码" prop="countryCode" label-width="80px">
<el-input v-model="form.countryCode" clearable :disabled="isViewMode" />
<el-form-item label="国" prop="countryCode" label-width="80px">
<el-select v-model="form.countryCode" placeholder="请选择国籍" clearable filterable :disabled="isViewMode">
<el-option
v-for="item in countryCodeList"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
</el-form-item>
</el-col>
<el-col :span="8">
@@ -97,7 +120,7 @@
</el-col>
</el-row>
<!-- 第三行就诊卡号国家编码出生日期 -->
<!-- 第三行就诊卡号职业邮政编码 -->
<el-row>
<el-col :span="8">
<el-form-item label="就诊卡号" prop="identifierNo">
@@ -116,16 +139,29 @@
</el-select>
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="邮政编码" prop="postalCode">
<el-input v-model="form.postalCode" clearable :disabled="isViewMode" placeholder="请输入邮政编码" />
</el-form-item>
</el-col>
</el-row>
<!-- 第四行工作单位 -->
<el-row>
<el-col :span="8">
<el-form-item label="工作单位" prop="workCompany">
<el-input v-model="form.workCompany" clearable :disabled="isViewMode" />
</el-form-item>
</el-col>
<el-col :span="16">
<el-form-item label="单位地址" prop="companyAddress">
<el-input v-model="form.companyAddress" clearable :disabled="isViewMode" />
</el-form-item>
</el-col>
</el-row>
<!-- 址选择详细地址 -->
<!-- 现住址选择详细地址 -->
<el-row>
<el-col :span="8">
<el-form-item label="地址选择" prop="addressSelect">
<el-form-item label="现住址" prop="addressSelect">
<el-cascader
:options="options"
:props="{ checkStrictly: true, value: 'code', label: 'name' }"
@@ -147,6 +183,31 @@
</el-col>
</el-row>
<!-- 户籍地址选择详细地址 -->
<el-row>
<el-col :span="8">
<el-form-item label="户籍地址" prop="hukouAddressSelect">
<el-cascader
:options="options"
:props="{ checkStrictly: true, value: 'code', label: 'name' }"
v-model="selectedHukouOptions"
@change="handleHukouChange"
:disabled="isViewMode"
>
<template #default="{ node, data }">
<span>{{ data.name }}</span>
<span v-if="!node.isLeaf"> ({{ data.children.length }}) </span>
</template>
</el-cascader>
</el-form-item>
</el-col>
<el-col :span="16">
<el-form-item label="详细地址" prop="hukouAddress">
<el-input v-model="form.hukouAddress" clearable :disabled="isViewMode" />
</el-form-item>
</el-col>
</el-row>
<!-- 第六行血型ABO血型RH -->
<el-row>
<el-col :span="8">
@@ -178,6 +239,18 @@
</el-select>
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="患者来源" prop="patientDerived">
<el-select v-model="form.patientDerived" placeholder="患者来源" clearable :disabled="isViewMode">
<el-option
v-for="item in patientDerivedList"
:key="item.value"
:label="item.info"
:value="item.value"
/>
</el-select>
</el-form-item>
</el-col>
</el-row>
<!-- 第七行婚姻状态死亡时间 -->
@@ -307,16 +380,146 @@ const {
);
const selectedOptions = ref([]); // v-model 绑定的选中值
const selectedHukouOptions = ref([]); // 户籍地址v-model 绑定的选中值
const maritalstatusList = ref([]); //婚姻
const occupationtypeList = ref([]); //职业
const administrativegenderList = ref([]); //性别
const bloodtypeaboList = ref([]); //血型abo
const bloodtypearhList = ref([]); //血型RH
const familyrelationshiptypeList = ref([]); //家庭关系
const patientDerivedList = ref([]); //患者来源
// 使用 ref 定义查询所得用户信息数据
const patientInfo = ref(undefined);
const addressCom = ref(''); //地址
const educationLevelList = ref([]); //文化程度
const countryCodeList = ref([]); //国家地区代码
// 从字典管理获取患者来源数据
const getPatientDerivedOptions = async () => {
try {
console.log('开始获取患者来源字典数据...');
// 从字典管理获取患者来源数据字典类型为patient_derived
const patientDerivedDict = await proxy.getDictDataByType('patient_derived');
console.log('获取到的患者来源原始数据:', patientDerivedDict);
// 确保数据是数组
if (!Array.isArray(patientDerivedDict)) {
console.error('患者来源数据格式错误,不是数组:', patientDerivedDict);
return;
}
// 按字典排序字段sort字段升序排序
const sortedPatientDerived = patientDerivedDict.sort((a, b) => {
// 尝试获取排序字段可能的字段名sort, orderNum, dictSort等
const sortA = a.sort || a.orderNum || a.dictSort || 0;
const sortB = b.sort || b.orderNum || b.dictSort || 0;
return sortA - sortB;
});
console.log('按排序字段排序后的患者来源数据:', sortedPatientDerived);
// 转换为组件需要的格式,确保使用正确的字段名称
patientDerivedList.value = sortedPatientDerived.map(item => ({
value: item.value || item.dictValue || item.code || '', // 使用字典键值
info: item.label || item.dictLabel || item.name || '' // 使用info字段作为显示文本与模板中的:label="item.info"匹配)
}));
console.log('处理后的患者来源选项:', patientDerivedList.value);
} catch (error) {
console.error('获取患者来源字典数据失败:', error);
// 改进的降级方案:使用默认的患者来源选项
patientDerivedList.value = [
{ value: '1', info: '熟人介绍' },
{ value: '2', info: '电视广告' },
{ value: '3', info: '公交站牌' },
{ value: '99', info: '其他' }
];
console.warn('使用默认患者来源数据作为降级方案');
}
};
// 从字典管理获取性别数据
const getGenderOptions = async () => {
try {
// 从字典管理获取性别数据
const genderDict = await proxy.getDictDataByType('性别');
// 按字典排序字段排序
const sortedGenders = genderDict.sort((a, b) => {
return (a.sort || 0) - (b.sort || 0);
});
// 转换为组件需要的格式
administrativegenderList.value = sortedGenders.map(item => ({
value: item.value, // 使用字典键值
info: item.label // 使用字典标签
}));
} catch (error) {
console.error('获取性别字典数据失败:', error);
// 降级方案:使用默认的性别选项
administrativegenderList.value = [
{ value: '1', info: '男' },
{ value: '2', info: '女' },
{ value: '9', info: '未说明性别' },
{ value: '0', info: '未知的性别' }
];
}
};
// 从字典管理获取文化程度数据
const getEducationLevelOptions = async () => {
try {
// 从字典管理获取文化程度数据
const educationDict = await proxy.getDictDataByType('文化程度');
console.log('获取到的文化程度数据:', educationDict);
// 确保数据是数组
if (!Array.isArray(educationDict)) {
console.error('文化程度数据格式错误,不是数组:', educationDict);
return;
}
// 按字典编码顺序升序排列(根据截图显示,排序字段为字典编码)
const sortedEducation = educationDict.sort((a, b) => {
// 获取字典编码可能的字段名code, dictCode, 或 value
const codeA = a.code || a.dictCode || a.value || '';
const codeB = b.code || b.dictCode || b.value || '';
// 尝试将编码转换为数字进行升序排序
const numA = parseInt(codeA);
const numB = parseInt(codeB);
// 如果都是有效数字,则按数字排序,否则按字符串排序
if (!isNaN(numA) && !isNaN(numB)) {
return numA - numB;
} else {
return codeA.localeCompare(codeB);
}
});
console.log('按字典编码排序后的文化程度数据:', sortedEducation);
// 转换为组件需要的格式
educationLevelList.value = sortedEducation.map(item => ({
value: item.value || item.dictValue || item.code || '', // 使用字典键值或编码
info: item.label || item.dictLabel || item.name || '' // 使用字典标签
}));
console.log('处理后的文化程度选项:', educationLevelList.value);
} catch (error) {
console.error('获取文化程度字典数据失败:', error);
// 降级方案:使用默认的文化程度选项,按编码顺序排列
educationLevelList.value = [
{ value: '3912', info: '大学本科' },
{ value: '3913', info: '硕士研究生' },
{ value: '3914', info: '博士研究生' },
{ value: '3915', info: '初中毕业' },
{ value: '3916', info: '大学毕业' },
{ value: '3917', info: '技工学校毕业' },
{ value: '3918', info: '职业高中毕业' },
{ value: '3919', info: '小学毕业' },
{ value: '3920', info: '普通高中毕业' },
{ value: '3921', info: '中等专科毕业' }
].sort((a, b) => parseInt(a.value) - parseInt(b.value)); // 确保默认选项也按编码排序
}
};
const options = ref(pcas); // 地区数据
const title = ref('新增患者');
@@ -406,11 +609,25 @@ const data = reactive({
typeCode: '08',
birthDate: undefined,
age: undefined,
genderEnum: '1', // 默认设置为'男'
hukouAddressSelect: undefined,
hukouAddress: undefined,
postalCode: undefined,
companyAddress: undefined,
patientDerived: undefined,
},
rules: {
name: [{ required: true, message: '姓名不能为空', trigger: 'change' },
{ validator: validateUniquePatient, trigger: 'blur' }
],
postalCode: [
{ required: false, message: '邮政编码非必填', trigger: 'change' },
{ pattern: /^\d{6}$/, message: '邮政编码格式应为6位数字', trigger: 'blur' }
],
hukouAddressSelect: [{ required: false, message: '户籍地址选择非必填', trigger: 'change' }],
hukouAddress: [{ required: false, message: '户籍详细地址非必填', trigger: 'change' }],
companyAddress: [{ required: false, message: '单位地址非必填', trigger: 'change' }],
patientDerived: [{ required: false, message: '患者来源非必填', trigger: 'change' }],
genderEnum: [{ required: true, message: '请选择性别', trigger: 'change' }],
age: [{ required: true, message: '年龄不能为空', trigger: 'change' }],
phone: [{ required: true, message: '联系方式不能为空', trigger: 'change' }],
@@ -516,7 +733,7 @@ function getList() {
patientlLists().then((response) => {
console.log(response);
occupationtypeList.value = response.data.occupationType;
administrativegenderList.value = response.data.sex;
// 移除直接从patientlLists设置性别的代码
bloodtypeaboList.value = response.data.bloodTypeABO;
bloodtypearhList.value = response.data.bloodTypeRH;
familyrelationshiptypeList.value = response.data.familyRelationshipType;
@@ -539,7 +756,7 @@ function getPatientInfo(idCard) {
}
});
}
//址选择
//现住址选择
const handleChange = () => {
const checkedNodes = selectedOptions.value.map((code) => {
const node = findNodeByCode(options.value, code);
@@ -552,6 +769,19 @@ const handleChange = () => {
form.value.address = '';
};
//户籍地址选择
const handleHukouChange = () => {
const checkedNodes = selectedHukouOptions.value.map((code) => {
const node = findNodeByCode(options.value, code);
return node ? node.name : null;
});
form.value.hukouAddressProvince = checkedNodes[0] || '';
form.value.hukouAddressCity = checkedNodes[1] || '';
form.value.hukouAddressDistrict = checkedNodes[2] || '';
form.value.hukouAddressStreet = checkedNodes[3] || '';
form.value.hukouAddress = '';
};
// 递归查找节点
const findNodeByCode = (data, code) => {
for (const item of data) {
@@ -563,10 +793,415 @@ const findNodeByCode = (data, code) => {
}
return null;
};
// 从字典管理获取国家地区代码数据
const getCountryCodeOptions = async () => {
try {
console.log('开始获取国家地区代码数据...');
// 确保使用正确的字典名称获取完整数据
let countryDict = [];
// 定义多个可能的字典类型名称优先使用nat_regn_code
const dictTypes = ['nat_regn_code', 'country_code', 'COUNTRY_CODE', '国家地区代码'];
let success = false;
// 尝试使用useDict方法获取数据遍历不同的字典类型
if (proxy.useDict && typeof proxy.useDict === 'function') {
for (const dictType of dictTypes) {
try {
const dictResult = await proxy.useDict(dictType);
if (dictResult && dictResult[dictType] && Array.isArray(dictResult[dictType]) && dictResult[dictType].length > 0) {
countryDict = dictResult[dictType];
console.log(`通过useDict(${dictType})获取到的国家地区代码数据数量:`, countryDict.length);
success = true;
break;
}
} catch (err) {
console.warn(`useDict(${dictType})调用失败,尝试下一种方式:`, err.message);
}
}
}
// 如果useDict失败尝试直接调用API获取字典数据
if (!success && proxy.request && typeof proxy.request === 'function') {
try {
console.log('尝试直接调用API获取国家地区代码数据...');
// 直接调用字典数据API优先使用nat_regn_code接口路径
const response = await proxy.request({
url: '/system/dict/data/type/nat_regn_code',
method: 'get'
});
if (response && response.code === 200 && Array.isArray(response.data)) {
countryDict = response.data;
console.log('通过API直接调用获取到的国家地区代码数据数量:', countryDict.length);
success = true;
} else {
// 如果nat_regn_code接口失败尝试使用country_code
const fallbackResponse = await proxy.request({
url: '/system/dict/data/type/country_code',
method: 'get'
});
if (fallbackResponse && fallbackResponse.code === 200 && Array.isArray(fallbackResponse.data)) {
countryDict = fallbackResponse.data;
console.log('通过备用API获取到的国家地区代码数据数量:', countryDict.length);
success = true;
}
}
} catch (apiError) {
console.error('API直接调用失败:', apiError);
}
}
// 如果仍然未获取到数据使用与nat_regn_code字典一致的模拟数据
if (!Array.isArray(countryDict) || countryDict.length === 0) {
console.warn('未获取到国家地区代码数据或数据格式错误使用与nat_regn_code字典一致的模拟数据');
// 创建与nat_regn_code字典一致的模拟数据基于数据库中的数据格式
const mockCountries = [
{ dictValue: 'CHN', dictLabel: '中国' },
{ dictValue: 'HKG', dictLabel: '香港' },
{ dictValue: 'MAC', dictLabel: '澳门' },
{ dictValue: 'TWN', dictLabel: '台湾' },
{ dictValue: 'AFG', dictLabel: '阿富汗' },
{ dictValue: 'ALB', dictLabel: '阿尔巴尼亚' },
{ dictValue: 'DZA', dictLabel: '阿尔及利亚' },
{ dictValue: 'ASM', dictLabel: '美属萨摩亚' },
{ dictValue: 'AND', dictLabel: '安道尔' },
{ dictValue: 'AGO', dictLabel: '安哥拉' },
{ dictValue: 'AIA', dictLabel: '安圭拉' },
{ dictValue: 'ATA', dictLabel: '南极洲' },
{ dictValue: 'ATG', dictLabel: '安提瓜和巴布达' },
{ dictValue: 'ARG', dictLabel: '阿根廷' },
{ dictValue: 'ARM', dictLabel: '亚美尼亚' },
{ dictValue: 'ABW', dictLabel: '阿鲁巴' },
{ dictValue: 'AUS', dictLabel: '澳大利亚' },
{ dictValue: 'AUT', dictLabel: '奥地利' },
{ dictValue: 'AZE', dictLabel: '阿塞拜疆' },
{ dictValue: 'BHS', dictLabel: '巴哈马' },
{ dictValue: 'BHR', dictLabel: '巴林' },
{ dictValue: 'BGD', dictLabel: '孟加拉国' },
{ dictValue: 'BRB', dictLabel: '巴巴多斯' },
{ dictValue: 'BLR', dictLabel: '白俄罗斯' },
{ dictValue: 'BEL', dictLabel: '比利时' },
{ dictValue: 'BLZ', dictLabel: '伯利兹' },
{ dictValue: 'BEN', dictLabel: '贝宁' },
{ dictValue: 'BMU', dictLabel: '百慕大' },
{ dictValue: 'BTN', dictLabel: '不丹' },
{ dictValue: 'BOL', dictLabel: '玻利维亚' },
{ dictValue: 'BIH', dictLabel: '波黑' },
{ dictValue: 'BWA', dictLabel: '博茨瓦纳' },
{ dictValue: 'BVT', dictLabel: '布维岛' },
{ dictValue: 'BRA', dictLabel: '巴西' },
{ dictValue: 'IOT', dictLabel: '英属印度洋领土' },
{ dictValue: 'BRN', dictLabel: '文莱' },
{ dictValue: 'BGR', dictLabel: '保加利亚' },
{ dictValue: 'BFA', dictLabel: '布基纳法索' },
{ dictValue: 'BDI', dictLabel: '布隆迪' },
{ dictValue: 'KHM', dictLabel: '柬埔寨' },
{ dictValue: 'CMR', dictLabel: '喀麦隆' },
{ dictValue: 'CAN', dictLabel: '加拿大' },
{ dictValue: 'CPV', dictLabel: '佛得角' },
{ dictValue: 'CYM', dictLabel: '开曼群岛' },
{ dictValue: 'CAF', dictLabel: '中非' },
{ dictValue: 'TCD', dictLabel: '乍得' },
{ dictValue: 'CHL', dictLabel: '智利' },
{ dictValue: 'CXR', dictLabel: '圣诞岛' },
{ dictValue: 'CCK', dictLabel: '科科斯(基林)群岛' },
{ dictValue: 'COL', dictLabel: '哥伦比亚' },
{ dictValue: 'COM', dictLabel: '科摩罗' },
{ dictValue: 'COG', dictLabel: '刚果(布)' },
{ dictValue: 'COD', dictLabel: '刚果(金)' },
{ dictValue: 'COK', dictLabel: '库克群岛' },
{ dictValue: 'CRI', dictLabel: '哥斯达黎加' },
{ dictValue: 'CIV', dictLabel: '科特迪瓦' },
{ dictValue: 'HRV', dictLabel: '克罗地亚' },
{ dictValue: 'CUB', dictLabel: '古巴' },
{ dictValue: 'CYP', dictLabel: '塞浦路斯' },
{ dictValue: 'CZE', dictLabel: '捷克' },
{ dictValue: 'DNK', dictLabel: '丹麦' },
{ dictValue: 'DJI', dictLabel: '吉布提' },
{ dictValue: 'DMA', dictLabel: '多米尼克' },
{ dictValue: 'DOM', dictLabel: '多米尼加' },
{ dictValue: 'TLS', dictLabel: '东帝汶' },
{ dictValue: 'ECU', dictLabel: '厄瓜多尔' },
{ dictValue: 'EGY', dictLabel: '埃及' },
{ dictValue: 'SLV', dictLabel: '萨尔瓦多' },
{ dictValue: 'GNQ', dictLabel: '赤道几内亚' },
{ dictValue: 'ERI', dictLabel: '厄立特里亚' },
{ dictValue: 'EST', dictLabel: '爱沙尼亚' },
{ dictValue: 'ETH', dictLabel: '埃塞俄比亚' },
{ dictValue: 'FLK', dictLabel: '福克兰群岛' },
{ dictValue: 'FRO', dictLabel: '法罗群岛' },
{ dictValue: 'FJI', dictLabel: '斐济' },
{ dictValue: 'FIN', dictLabel: '芬兰' },
{ dictValue: 'FRA', dictLabel: '法国' },
{ dictValue: 'GUF', dictLabel: '法属圭亚那' },
{ dictValue: 'PYF', dictLabel: '法属波利尼西亚' },
{ dictValue: 'ATF', dictLabel: '法属南部领地' },
{ dictValue: 'GAB', dictLabel: '加蓬' },
{ dictValue: 'GMB', dictLabel: '冈比亚' },
{ dictValue: 'GEO', dictLabel: '格鲁吉亚' },
{ dictValue: 'DEU', dictLabel: '德国' },
{ dictValue: 'GHA', dictLabel: '加纳' },
{ dictValue: 'GIB', dictLabel: '直布罗陀' },
{ dictValue: 'GRC', dictLabel: '希腊' },
{ dictValue: 'GRL', dictLabel: '格陵兰' },
{ dictValue: 'GRD', dictLabel: '格林纳达' },
{ dictValue: 'GLP', dictLabel: '瓜德罗普' },
{ dictValue: 'GUM', dictLabel: '关岛' },
{ dictValue: 'GTM', dictLabel: '危地马拉' },
{ dictValue: 'GGY', dictLabel: '根西岛' },
{ dictValue: 'GIN', dictLabel: '几内亚' },
{ dictValue: 'GNB', dictLabel: '几内亚比绍' },
{ dictValue: 'GUY', dictLabel: '圭亚那' },
{ dictValue: 'HTI', dictLabel: '海地' },
{ dictValue: 'HMD', dictLabel: '赫德岛和麦克唐纳群岛' },
{ dictValue: 'VAT', dictLabel: '梵蒂冈' },
{ dictValue: 'HND', dictLabel: '洪都拉斯' },
{ dictValue: 'HUN', dictLabel: '匈牙利' },
{ dictValue: 'ISL', dictLabel: '冰岛' },
{ dictValue: 'IND', dictLabel: '印度' },
{ dictValue: 'IDN', dictLabel: '印度尼西亚' },
{ dictValue: 'IRN', dictLabel: '伊朗' },
{ dictValue: 'IRQ', dictLabel: '伊拉克' },
{ dictValue: 'IRL', dictLabel: '爱尔兰' },
{ dictValue: 'IMN', dictLabel: '马恩岛' },
{ dictValue: 'ISR', dictLabel: '以色列' },
{ dictValue: 'ITA', dictLabel: '意大利' },
{ dictValue: 'JAM', dictLabel: '牙买加' },
{ dictValue: 'JPN', dictLabel: '日本' },
{ dictValue: 'JEY', dictLabel: '泽西岛' },
{ dictValue: 'JOR', dictLabel: '约旦' },
{ dictValue: 'KAZ', dictLabel: '哈萨克斯坦' },
{ dictValue: 'KEN', dictLabel: '肯尼亚' },
{ dictValue: 'KIR', dictLabel: '基里巴斯' },
{ dictValue: 'PRK', dictLabel: '朝鲜' },
{ dictValue: 'KOR', dictLabel: '韩国' },
{ dictValue: 'KWT', dictLabel: '科威特' },
{ dictValue: 'KGZ', dictLabel: '吉尔吉斯斯坦' },
{ dictValue: 'LAO', dictLabel: '老挝' },
{ dictValue: 'LVA', dictLabel: '拉脱维亚' },
{ dictValue: 'LBN', dictLabel: '黎巴嫩' },
{ dictValue: 'LSO', dictLabel: '莱索托' },
{ dictValue: 'LBR', dictLabel: '利比里亚' },
{ dictValue: 'LBY', dictLabel: '利比亚' },
{ dictValue: 'LIE', dictLabel: '列支敦士登' },
{ dictValue: 'LTU', dictLabel: '立陶宛' },
{ dictValue: 'LUX', dictLabel: '卢森堡' },
{ dictValue: 'MKD', dictLabel: '前南马其顿' },
{ dictValue: 'MDG', dictLabel: '马达加斯加' },
{ dictValue: 'MWI', dictLabel: '马拉维' },
{ dictValue: 'MYS', dictLabel: '马来西亚' },
{ dictValue: 'MDV', dictLabel: '马尔代夫' },
{ dictValue: 'MLI', dictLabel: '马里' },
{ dictValue: 'MLT', dictLabel: '马耳他' },
{ dictValue: 'MHL', dictLabel: '马绍尔群岛' },
{ dictValue: 'MTQ', dictLabel: '马提尼克' },
{ dictValue: 'MRT', dictLabel: '毛里塔尼亚' },
{ dictValue: 'MUS', dictLabel: '毛里求斯' },
{ dictValue: 'MYT', dictLabel: '马约特' },
{ dictValue: 'MEX', dictLabel: '墨西哥' },
{ dictValue: 'FSM', dictLabel: '密克罗尼西亚联邦' },
{ dictValue: 'MDA', dictLabel: '摩尔多瓦' },
{ dictValue: 'MCO', dictLabel: '摩纳哥' },
{ dictValue: 'MNG', dictLabel: '蒙古' },
{ dictValue: 'MSR', dictLabel: '蒙特塞拉特' },
{ dictValue: 'MAR', dictLabel: '摩洛哥' },
{ dictValue: 'MOZ', dictLabel: '莫桑比克' },
{ dictValue: 'MMR', dictLabel: '缅甸' },
{ dictValue: 'NAM', dictLabel: '纳米比亚' },
{ dictValue: 'NRU', dictLabel: '瑙鲁' },
{ dictValue: 'NPL', dictLabel: '尼泊尔' },
{ dictValue: 'NLD', dictLabel: '荷兰' },
{ dictValue: 'ANT', dictLabel: '荷属安的列斯' },
{ dictValue: 'NCL', dictLabel: '新喀里多尼亚' },
{ dictValue: 'NZL', dictLabel: '新西兰' },
{ dictValue: 'NIC', dictLabel: '尼加拉瓜' },
{ dictValue: 'NER', dictLabel: '尼日尔' },
{ dictValue: 'NGA', dictLabel: '尼日利亚' },
{ dictValue: 'NIU', dictLabel: '纽埃' },
{ dictValue: 'NFK', dictLabel: '诺福克岛' },
{ dictValue: 'MNP', dictLabel: '北马里亚纳' },
{ dictValue: 'NOR', dictLabel: '挪威' },
{ dictValue: 'OMN', dictLabel: '阿曼' },
{ dictValue: 'PAK', dictLabel: '巴基斯坦' },
{ dictValue: 'PLW', dictLabel: '帕劳' },
{ dictValue: 'PSE', dictLabel: '巴勒斯坦' },
{ dictValue: 'PAN', dictLabel: '巴拿马' },
{ dictValue: 'PNG', dictLabel: '巴布亚新几内亚' },
{ dictValue: 'PRY', dictLabel: '巴拉圭' },
{ dictValue: 'PER', dictLabel: '秘鲁' },
{ dictValue: 'PHL', dictLabel: '菲律宾' },
{ dictValue: 'PCN', dictLabel: '皮特凯恩群岛' },
{ dictValue: 'POL', dictLabel: '波兰' },
{ dictValue: 'PRT', dictLabel: '葡萄牙' },
{ dictValue: 'PRI', dictLabel: '波多黎各' },
{ dictValue: 'QAT', dictLabel: '卡塔尔' },
{ dictValue: 'REU', dictLabel: '留尼汪' },
{ dictValue: 'ROM', dictLabel: '罗马尼亚' },
{ dictValue: 'RUS', dictLabel: '俄罗斯联邦' },
{ dictValue: 'RWA', dictLabel: '卢旺达' },
{ dictValue: 'SHN', dictLabel: '圣赫勒拿' },
{ dictValue: 'KNA', dictLabel: '圣基茨和尼维斯' },
{ dictValue: 'LCA', dictLabel: '圣卢西亚' },
{ dictValue: 'SPM', dictLabel: '圣皮埃尔和密克隆' },
{ dictValue: 'VCT', dictLabel: '圣文森特和格林纳丁斯' },
{ dictValue: 'WSM', dictLabel: '萨摩亚' },
{ dictValue: 'SMR', dictLabel: '圣马力诺' },
{ dictValue: 'STP', dictLabel: '圣多美和普林西比' },
{ dictValue: 'SAU', dictLabel: '沙特阿拉伯' },
{ dictValue: 'SEN', dictLabel: '塞内加尔' },
{ dictValue: 'SYC', dictLabel: '塞舌尔' },
{ dictValue: 'SLE', dictLabel: '塞拉利昂' },
{ dictValue: 'SGP', dictLabel: '新加坡' },
{ dictValue: 'SVK', dictLabel: '斯洛伐克' },
{ dictValue: 'SVN', dictLabel: '斯洛文尼亚' },
{ dictValue: 'SLB', dictLabel: '所罗门群岛' },
{ dictValue: 'SOM', dictLabel: '索马里' },
{ dictValue: 'ZAF', dictLabel: '南非' },
{ dictValue: 'SGS', dictLabel: '南乔治亚岛和南桑威奇群岛' },
{ dictValue: 'ESP', dictLabel: '西班牙' },
{ dictValue: 'LKA', dictLabel: '斯里兰卡' },
{ dictValue: 'SDN', dictLabel: '苏丹' },
{ dictValue: 'SUR', dictLabel: '苏里南' },
{ dictValue: 'SJM', dictLabel: '斯瓦尔巴群岛和扬马延岛' },
{ dictValue: 'SWZ', dictLabel: '斯威士兰' },
{ dictValue: 'SWE', dictLabel: '瑞典' },
{ dictValue: 'CHE', dictLabel: '瑞士' },
{ dictValue: 'SYR', dictLabel: '叙利亚' },
{ dictValue: 'TWN', dictLabel: '台湾' },
{ dictValue: 'TJK', dictLabel: '塔吉克斯坦' },
{ dictValue: 'TZA', dictLabel: '坦桑尼亚' },
{ dictValue: 'THA', dictLabel: '泰国' },
{ dictValue: 'TGO', dictLabel: '多哥' },
{ dictValue: 'TKL', dictLabel: '托克劳' },
{ dictValue: 'TON', dictLabel: '汤加' },
{ dictValue: 'TTO', dictLabel: '特立尼达和多巴哥' },
{ dictValue: 'TUN', dictLabel: '突尼斯' },
{ dictValue: 'TUR', dictLabel: '土耳其' },
{ dictValue: 'TKM', dictLabel: '土库曼斯坦' },
{ dictValue: 'TCA', dictLabel: '特克斯和凯科斯群岛' },
{ dictValue: 'TUV', dictLabel: '图瓦卢' },
{ dictValue: 'UGA', dictLabel: '乌干达' },
{ dictValue: 'UKR', dictLabel: '乌克兰' },
{ dictValue: 'ARE', dictLabel: '阿联酋' },
{ dictValue: 'GBR', dictLabel: '英国' },
{ dictValue: 'USA', dictLabel: '美国' },
{ dictValue: 'UMI', dictLabel: '美国本土外小岛屿' },
{ dictValue: 'URY', dictLabel: '乌拉圭' },
{ dictValue: 'UZB', dictLabel: '乌兹别克斯坦' },
{ dictValue: 'VUT', dictLabel: '瓦努阿图' },
{ dictValue: 'VEN', dictLabel: '委内瑞拉' },
{ dictValue: 'VNM', dictLabel: '越南' },
{ dictValue: 'VGB', dictLabel: '英属维尔京群岛' },
{ dictValue: 'VIR', dictLabel: '美属维尔京群岛' },
{ dictValue: 'WLF', dictLabel: '瓦利斯和富图纳' },
{ dictValue: 'ESH', dictLabel: '西撒哈拉' },
{ dictValue: 'YEM', dictLabel: '也门' },
{ dictValue: 'ZMB', dictLabel: '赞比亚' },
{ dictValue: 'ZWE', dictLabel: '津巴布韦' }
];
countryDict = mockCountries;
console.log('使用与nat_regn_code字典一致的模拟数据共生成:', countryDict.length, '条国家地区代码数据');
// 提示管理员检查数据库中的国家地区代码数据
console.warn('提示: 请检查数据库中的sys_dict_type和sys_dict_data表确保存在nat_regn_code类型的完整国家地区代码数据');
}
// 确保有足够的数据
console.log('最终获取到的国家地区代码数据数量:', countryDict.length);
// 转换为统一格式
const formattedCountries = countryDict.map(item => ({
value: item.value || item.dictValue || item.code || '', // 使用字典键值或编码
label: item.label || item.dictLabel || item.name || '' // 使用字典标签
}));
// 数据去重根据value字段去重
const uniqueCountries = [];
const valueSet = new Set();
formattedCountries.forEach(item => {
if (item.value && !valueSet.has(item.value)) {
valueSet.add(item.value);
uniqueCountries.push(item);
}
});
console.log('去重后的数据数量:', uniqueCountries.length);
// 按字典编码升序排序
const sortedCountries = uniqueCountries.sort((a, b) => {
// 获取字典编码
const codeA = a.value || '';
const codeB = b.value || '';
// 按字典编码字符串升序排序
return codeA.localeCompare(codeB);
});
countryCodeList.value = sortedCountries;
// 确保中国是第一个选项(如果存在)
const chinaIndex = countryCodeList.value.findIndex(item => item.label === '中国');
if (chinaIndex > 0) {
// 将中国选项移到数组开头
const chinaOption = countryCodeList.value.splice(chinaIndex, 1)[0];
countryCodeList.value.unshift(chinaOption);
}
// 设置中国为默认值(查找标签为'中国'的选项)
const chinaOption = countryCodeList.value.find(item => item.label === '中国');
if (chinaOption && !form.value.countryCode) {
form.value.countryCode = chinaOption.value;
}
console.log('最终处理后的国家地区代码选项数量:', countryCodeList.value.length);
console.log('前5个国家地区代码选项:', countryCodeList.value.slice(0, 5));
} catch (error) {
console.error('获取国家地区代码字典数据失败:', error);
// 改进的降级方案:提供更完整的默认选项
countryCodeList.value = [
{ value: 'CHN', label: '中国' },
{ value: 'USA', label: '美国' },
{ value: 'JPN', label: '日本' },
{ value: 'KOR', label: '韩国' },
{ value: 'GBR', label: '英国' },
{ value: 'FRA', label: '法国' },
{ value: 'DEU', label: '德国' },
{ value: 'CAN', label: '加拿大' },
{ value: 'AUS', label: '澳大利亚' },
{ value: 'RUS', label: '俄罗斯' }
];
// 设置中国为默认值
if (!form.value.countryCode) {
form.value.countryCode = 'CHN';
}
// 显示错误提示,提醒用户检查系统配置
if (proxy && proxy.modal && proxy.modal.msgError) {
proxy.modal.msgError('获取国家地区代码数据失败,请联系系统管理员');
}
}
};
// 显示弹框
function show() {
// queryParams.roleId = props.roleId;
getList();
// 调用从字典管理获取性别数据的函数
getGenderOptions();
// 调用从字典管理获取文化程度数据的函数
getEducationLevelOptions();
// 调用从字典管理获取国家地区代码数据的函数
getCountryCodeOptions();
// 调用从字典管理获取患者来源数据的函数
getPatientDerivedOptions();
visible.value = true;
}

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