ERPTurbo_Client/packages/app-client/src/pages/supplier/create.tsx
shenyifei dfe9a89213 refactor(components): 优化采购模块空箱和费用组件实现
- 移除 PageList 组件中对全局 loading 状态的依赖
- 简化 EmptyBoxModule 组件逻辑,使用 PackageList 组件替代原有复杂实现
- 移除冗余的状态管理和弹窗渲染逻辑
- 优化 OrderCost 组件样式和费用项匹配逻辑
- 修复成本项 ID 匹配问题,确保数据正确关联
- 添加边框样式增强视觉效果
- 移除调试日志和无用代码
- 简化组件间数据传递方式
2025-12-11 12:42:01 +08:00

599 lines
18 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import { View } from "@tarojs/components";
import { Button, Input, Toast, Uploader, UploaderFileItem } from "@nutui/nutui-react-taro";
import { Icon } from "@/components";
import hocAuth from "@/hocs/auth";
import { CommonComponent } from "@/types/typings";
import { useEffect, useState } from "react";
import { business } from "@/services";
import Taro from "@tarojs/taro";
import { buildUrl, generateShortId, uploadFile } from "@/utils";
export default hocAuth(function Page(props: CommonComponent) {
const { setLoading, router } = props;
const supplierId = router.params
.supplierId as BusinessAPI.SupplierVO["supplierId"];
const [supplierVO, setSupplierVO] = useState<BusinessAPI.SupplierCreateCmd>({
supplierId: generateShortId(),
name: "",
idCard: "",
phone: "",
bankCard: "",
status: true,
});
const [btnLoading, setBtnLoading] = useState(false);
const [picList, setPicList] = useState<UploaderFileItem[]>([]);
// 错误状态
const [nameError, setNameError] = useState(false);
const [idCardError, setIdCardError] = useState(false);
const [bankCardError, setBankCardError] = useState(false);
const [phoneError, setPhoneError] = useState(false);
const [nameDuplicateError, setNameDuplicateError] = useState(false); // 姓名重复错误
// 初始化微信二维码图片列表
useEffect(() => {
if (supplierId) {
setLoading(true);
business.supplier
.showSupplier({
supplierShowQry: {
supplierId: supplierId,
},
})
.then(({ data: { data: supplierVO } }) => {
if (supplierVO) {
setSupplierVO({
...supplierVO,
});
if (supplierVO.wechatQr) {
setPicList([
{
url: supplierVO.wechatQr,
name: "wechat-qrcode",
status: "success",
},
]);
}
}
})
.finally(() => {
setLoading(false);
});
}
}, [supplierId]);
// 微信二维码变更处理函数
const handleWechatQrChange = (files: UploaderFileItem[]) => {
setPicList(files);
// 如果有文件且上传成功保存URL到supplierVO
if (files.length > 0 && files[0].url) {
setSupplierVO((prev) => ({
...prev,
wechatQr: files[0].url,
}));
} else {
// 如果没有文件清空URL
setSupplierVO((prev) => ({
...prev,
wechatQr: undefined,
}));
}
};
// 校验姓名函数
const validateName = (name: string) => {
if (!name) {
return false;
}
// 姓名至少2个字符
return name.length >= 2;
};
// 校验身份证号函数
const validateIdCard = (idCard: string) => {
if (!idCard) {
return false;
}
// 18位身份证号正则表达式
const idCardRegex =
/^[1-9]\d{5}(18|19|20)\d{2}((0[1-9])|(1[0-2]))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/;
return idCardRegex.test(idCard);
};
// 校验银行卡号函数
const validateBankCard = (bankCard: string) => {
if (!bankCard) {
return false;
}
// 银行卡号一般为16-19位数字
const bankCardRegex = /^\d{16,19}$/;
return bankCardRegex.test(bankCard);
};
// 校验手机号函数 (使用项目中已有的规则)
const validatePhone = (phone: string) => {
if (!phone) {
return false;
}
// 使用项目规范的手机号正则表达式
const phoneRegex = /^1[3456789]\d{9}$/;
return phoneRegex.test(phone);
};
// 处理姓名变化
const handleNameChange = (value: string) => {
setSupplierVO({
...supplierVO,
name: value,
});
// 校验并更新错误状态
const isValid = validateName(value);
setNameError(!isValid && value !== "");
if (isValid) {
setNameError(false);
}
// 清除姓名重复错误提示
setNameDuplicateError(false);
};
// 处理姓名失焦
const handleNameBlur = async (value: string) => {
if (value && !validateName(value)) {
setNameError(true);
return;
} else {
setNameError(false);
}
// 检查瓜农姓名是否在系统中存在
if (value) {
setLoading(true);
try {
const { data } = await business.supplier.checkSupplier({
supplierCheckQry: {
name: value,
},
});
if (data.success && data.data) {
// 系统中已存在该瓜农信息,设置姓名重复错误状态
setNameDuplicateError(true);
} else {
setNameDuplicateError(false);
}
} catch (error) {
console.error("检查瓜农信息失败", error);
setNameDuplicateError(false);
} finally {
setLoading(false);
}
}
};
// 处理身份证号变化
const handleIdCardChange = (value: string) => {
setSupplierVO({
...supplierVO,
idCard: value,
});
// 校验并更新错误状态
const isValid = validateIdCard(value);
setIdCardError(!isValid && value !== "");
if (isValid) {
setIdCardError(false);
}
};
// 处理身份证号失焦
const handleIdCardBlur = (value: string) => {
if (value && !validateIdCard(value)) {
setIdCardError(true);
} else {
setIdCardError(false);
}
};
// 处理银行卡号变化
const handleBankCardChange = (value: string) => {
setSupplierVO({
...supplierVO,
bankCard: value,
});
// 校验并更新错误状态
const isValid = validateBankCard(value);
setBankCardError(!isValid && value !== "");
if (isValid) {
setBankCardError(false);
}
};
// 处理银行卡号失焦
const handleBankCardBlur = (value: string) => {
if (value && !validateBankCard(value)) {
setBankCardError(true);
} else {
setBankCardError(false);
}
};
// 处理手机号变化
const handlePhoneChange = (value: string) => {
setSupplierVO({
...supplierVO,
phone: value,
});
// 校验并更新错误状态
const isValid = validatePhone(value);
setPhoneError(!isValid && value !== "");
if (isValid) {
setPhoneError(false);
}
};
// 处理手机号失焦
const handlePhoneBlur = (value: string) => {
if (value && !validatePhone(value)) {
setPhoneError(true);
} else {
setPhoneError(false);
}
};
const handleSave = async () => {
// 校验表单
const isNameValid = validateName(supplierVO.name || "");
const isIdCardValid = validateIdCard(supplierVO.idCard || "");
const isBankCardValid = validateBankCard(supplierVO.bankCard || "");
const isPhoneValid = validatePhone(supplierVO.phone || "");
setNameError(!isNameValid);
setIdCardError(!isIdCardValid);
setBankCardError(!isBankCardValid);
setPhoneError(!isPhoneValid);
// 检查是否已存在同名瓜农
let isDuplicate = false;
if (isNameValid) {
try {
setBtnLoading(true);
const { data: checkData } = await business.supplier.checkSupplier({
supplierCheckQry: {
name: supplierVO.name,
},
});
if (checkData.success && checkData.data) {
isDuplicate = true;
setNameDuplicateError(true);
}
} catch (error) {
console.error("检查瓜农信息失败", error);
} finally {
setBtnLoading(false);
}
}
const isValid =
isNameValid &&
isIdCardValid &&
isBankCardValid &&
isPhoneValid &&
!isDuplicate;
if (!isValid) {
if (isDuplicate) {
Toast.show("toast", {
icon: "fail",
title: "提示",
content: "该瓜农已存在,请勿重复创建",
});
} else {
Toast.show("toast", {
icon: "fail",
title: "提示",
content: "请完善瓜农信息后再进行保存操作",
});
}
return;
}
// 创建新瓜农
try {
setBtnLoading(true);
const { data } = await business.supplier.createSupplier({
supplierId: supplierVO.supplierId,
name: supplierVO.name,
idCard: supplierVO.idCard,
phone: supplierVO.phone,
bankCard: supplierVO.bankCard,
wechatQr: supplierVO.wechatQr,
status: supplierVO.status,
});
if (data.success) {
Toast.show("toast", {
icon: "success",
title: "成功",
content: "瓜农创建成功",
});
setTimeout(() => {
Taro.navigateBack();
}, 1500);
} else {
Toast.show("toast", {
icon: "fail",
title: "失败",
content: data.errMessage || "创建失败",
});
}
} catch (error) {
Toast.show("toast", {
icon: "fail",
title: "错误",
content: "创建过程中发生错误",
});
console.error("创建瓜农失败:", error);
} finally {
setBtnLoading(false);
}
};
return (
<>
<View className="flex-1">
<View
className="flex flex-1 flex-col gap-2.5 p-2.5"
style={{
//@ts-ignore
"--nutui-input-padding": 0,
}}
>
{/* 功能提醒 */}
<View className="flex items-center rounded-lg border border-blue-200 bg-blue-50 p-2.5">
<Icon
className={"mr-1 leading-4"}
name="circle-info"
color={"var(--color-blue-700)"}
size={18}
/>
<View className={"text-sm text-blue-700"}>
</View>
</View>
{/* 快捷工具 */}
<View className="flex gap-2">
<View className={"flex-1"}>
<Button
block
icon={<Icon name={"id-card"} size={28} color={"white"} />}
type={"primary"}
size={"xlarge"}
className="bg-primary flex flex-1 items-center justify-center text-white"
onClick={() => {
Taro.navigateTo({
url: buildUrl("/pages/public/camera/ocr", {
type: "idcard",
}),
complete: () => {
Taro.eventCenter.on("ocr", (res) => {
console.log("识别结果为:", res.result);
setSupplierVO({
...supplierVO,
name: res.result.name,
idCard: res.result.idCard,
});
Taro.eventCenter.off("ocr");
});
},
});
}}
>
<View></View>
</Button>
</View>
<View className={"flex-1"}>
<Button
block
icon={<Icon name={"credit-card"} size={28} color={"white"} />}
type={"primary"}
size={"xlarge"}
className="bg-primary flex flex-1 items-center justify-center text-white"
onClick={() => {
Taro.navigateTo({
url: buildUrl("/pages/public/camera/ocr", {
type: "bankcard",
}),
complete: () => {
Taro.eventCenter.on("ocr", (res) => {
console.log("识别结果为:", res.result);
setSupplierVO({
...supplierVO,
bankCard: res.result.number,
});
Taro.eventCenter.off("ocr");
});
},
});
}}
>
<View></View>
</Button>
</View>
</View>
{/* 瓜农信息 */}
<View className="rounded-lg bg-white p-2.5 shadow-sm">
<View className="mb-2.5">
<View className="flex items-center justify-between">
<View className={"text-primary text-sm font-bold"}>
{supplierVO.name || "瓜农"}
</View>
</View>
</View>
<View className="mb-2.5">
<View className="mb-1 block text-sm font-normal text-[#000000]">
</View>
<View
className={`flex h-10 w-full items-center rounded-md ${nameError || nameDuplicateError ? "border-4 border-red-500" : "border-4 border-gray-300"}`}
>
<Icon name="user" size={16} color="#999" className="mx-2" />
<Input
type="text"
placeholder="请输入姓名"
value={supplierVO.name}
onChange={(value) => handleNameChange(value)}
onBlur={() => handleNameBlur(supplierVO.name || "")}
className="flex-1"
/>
</View>
{nameError && (
<View className="mt-1 text-xs text-red-500">
{`姓名"${supplierVO.name}"至少2个字符`}
</View>
)}
{nameDuplicateError && (
<View className="mt-1 text-xs text-red-500">
</View>
)}
</View>
<View className="mb-2.5">
<View className="mb-1 block text-sm font-normal text-[#000000]">
</View>
<View
className={`flex h-10 w-full items-center rounded-md ${idCardError ? "border-4 border-red-500" : "border-4 border-gray-300"}`}
>
<Icon name="id-card" size={16} color="#999" className="mx-2" />
<Input
type="idcard"
placeholder="请输入身份证号"
value={supplierVO.idCard}
onChange={(value) => handleIdCardChange(value)}
onBlur={() => handleIdCardBlur(supplierVO.idCard || "")}
className="flex-1"
/>
</View>
{idCardError && (
<View className="mt-1 text-xs text-red-500">
</View>
)}
</View>
<View className="mb-2.5">
<View className="mb-1 block text-sm font-normal text-[#000000]">
</View>
<View
className={`flex h-10 w-full items-center rounded-md ${bankCardError ? "border-4 border-red-500" : "border-4 border-gray-300"}`}
>
<Icon
name="credit-card"
size={16}
color="#999"
className="mx-2"
/>
<Input
type="digit"
placeholder="请输入银行卡号"
value={supplierVO.bankCard}
onChange={(value) => handleBankCardChange(value)}
onBlur={() => handleBankCardBlur(supplierVO.bankCard || "")}
className="flex-1"
/>
</View>
{bankCardError && (
<View className="mt-1 text-xs text-red-500">
</View>
)}
</View>
<View className="mb-2.5">
<View className="mb-1 block text-sm font-normal text-[#000000]">
</View>
<View
className={`flex h-10 w-full items-center rounded-md ${phoneError ? "border-4 border-red-500" : "border-4 border-gray-300"}`}
>
<Icon name="phone" size={16} color="#999" className="mx-2" />
<Input
type="tel"
placeholder="请输入手机号码"
value={supplierVO.phone}
onChange={(value) => handlePhoneChange(value)}
onBlur={() => handlePhoneBlur(supplierVO.phone || "")}
className="flex-1"
/>
</View>
{phoneError && (
<View className="mt-1 text-xs text-red-500">
</View>
)}
</View>
</View>
{/* 若瓜农无法开发票,则可打款到微信 */}
<View className="rounded-lg bg-white p-2.5 shadow-sm">
<View className={`flex w-full border-gray-300`}>
<Uploader
className={"w-full"}
value={picList}
onChange={handleWechatQrChange}
sourceType={["album", "camera"]}
uploadIcon={<Icon name={"camera"} size={36} />}
uploadLabel={
<View className={"flex flex-col items-center"}>
<View className="text-sm"></View>
<View className="mt-1 text-xs text-gray-400">
</View>
<View className="mt-1 text-xs text-gray-400">
</View>
</View>
}
maxCount={1}
//@ts-ignore
upload={uploadFile}
multiple
/>
</View>
</View>
</View>
</View>
<View className="sticky bottom-0 z-10 bg-white">
<View className="flex justify-between gap-2 border-t border-gray-200 p-2.5">
<View className={"flex-1"}>
<Button
block
type="primary"
size={"xlarge"}
onClick={handleSave}
loading={btnLoading}
>
</Button>
</View>
</View>
</View>
</>
);
});