feat(product): 新增产品管理模块与字典组件功能

- 新增产品管理相关路由和页面(dashboard、list、requirement、setting)
- 实现产品基础信息编辑弹窗组件(base-info-dialog.vue)
- 添加运行时字典功能(dict-select、dict-text、dict-tag组件)
- 集成字典管理store和API调用
- 规范ID类型定义为string避免精度丢失问题
- 完善国际化资源文件支持中英文对照
- 新增对象上下文业务域入口页导航实现说明
- 添加Vue DevTools浮动入口注释说明
- 统一权限控制支持全局和对象作用域区分
- 规范分页查询参数类型定义与使用方式
This commit is contained in:
2026-04-23 09:05:55 +08:00
parent c5911ea34b
commit 4122dfa50d
95 changed files with 9581 additions and 801 deletions

View File

@@ -0,0 +1,84 @@
const MAX_SAFE_INTEGER_BIGINT = BigInt(Number.MAX_SAFE_INTEGER);
function shouldStringifyUnsafeInteger(token: string) {
if (token.includes('.') || token.includes('e') || token.includes('E')) {
return false;
}
try {
const value = BigInt(token);
return value > MAX_SAFE_INTEGER_BIGINT || value < -MAX_SAFE_INTEGER_BIGINT;
} catch {
return false;
}
}
function replaceUnsafeIntegerTokens(raw: string) {
let result = '';
let index = 0;
let inString = false;
let isEscaping = false;
while (index < raw.length) {
const char = raw[index];
if (inString) {
result += char;
if (isEscaping) {
isEscaping = false;
} else if (char === '\\') {
isEscaping = true;
} else if (char === '"') {
inString = false;
}
index += 1;
} else if (char === '"') {
inString = true;
result += char;
index += 1;
} else {
const nextChar = raw[index + 1] ?? '';
const isNumberStart = char === '-' ? /\d/.test(nextChar) : /\d/.test(char);
if (!isNumberStart) {
result += char;
index += 1;
} else {
let end = index + 1;
while (end < raw.length && /[\d.+\-Ee]/.test(raw[end])) {
end += 1;
}
const token = raw.slice(index, end);
result += shouldStringifyUnsafeInteger(token) ? `"${token}"` : token;
index = end;
}
}
}
return result;
}
/**
* 保留超出 JS 安全整数范围的 Long 原始值,避免在 JSON.parse 阶段丢精度。
*/
export function safeJsonTransformResponse(data: unknown) {
if (typeof data !== 'string') {
return data;
}
const raw = data.trim();
if (!raw) {
return data;
}
try {
return JSON.parse(replaceUnsafeIntegerTokens(raw));
} catch {
return data;
}
}