U8XkyServerService.php 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140
  1. <?php
  2. namespace App\Service;
  3. use App\Model\DDEmployee;
  4. use App\Model\FieldData;
  5. use App\Model\Inventory;
  6. use App\Model\Vendor;
  7. use Illuminate\Support\Facades\DB;
  8. class U8XkyServerService extends Service
  9. {
  10. //新增存货到本地-----------------------------------------------
  11. public function inventoryAdd($data, $user){
  12. list($status, $msg) = $this->inventoryRule($data, $user);
  13. if(! $status) return [false, $msg];
  14. try {
  15. DB::beginTransaction();
  16. $inventoryData = [
  17. 'order_number' => $this->generateBillNo([
  18. 'type' => Inventory::Order_type,
  19. 'period' => date("Ym")
  20. ]),
  21. 'code' => $data['code'] ?? "",
  22. 'title' => $data['title'] ?? "",
  23. 'size' => $data['size'] ?? "",
  24. 'category_code' => $data['category_code'] ?? "",
  25. 'category_code_title' => $data['category_code_title'] ?? "",
  26. 'unit_group_type' => $data['unit_group_type'] ?? "",
  27. 'unit_group_code' => $data['unit_group_code'] ?? "",
  28. 'unit_group_code_title' => $data['unit_group_code_title'] ?? "",
  29. 'unit_code' => $data['unit_code'] ?? "",
  30. 'unit_code_title' => $data['unit_code_title'] ?? "",
  31. 'vendor_code' => $data['vendor_code'] ?? "",
  32. 'vendor_code_title' => $data['vendor_code_title'] ?? "",
  33. 'bSale' => $data['bSale'] ?? 0, //内销
  34. 'bExpSale' => $data['bExpSale'] ?? 0, //外销
  35. 'bPurchase' => $data['bPurchase'] ?? 0, // 采购
  36. 'bSelf' => $data['bSelf'] ?? 0, // 自制
  37. 'bComsume' => $data['bComsume'] ?? 0, // 生产耗材
  38. 'iImpTaxRate' => $data['iImpTaxRate'] ?? 0, // 进项税率
  39. 'iTaxRate' => $data['iTaxRate'] ?? 0, // 销项税率
  40. 'customs_change_rate' => $data['customs_change_rate'] ?? 0,
  41. 'crt_id' => $user['userid'],
  42. 'crt_time' => time(),
  43. ];
  44. Inventory::insert($inventoryData);
  45. DB::commit();
  46. } catch (\Throwable $exception) {
  47. DB::rollBack();
  48. return [false, "创建存货失败: " . $exception->getMessage()];
  49. }
  50. return [true, ''];
  51. }
  52. public function inventoryEdit($data, $user){
  53. list($status, $msg) = $this->inventoryRule($data, $user);
  54. if(! $status) return [false, $msg];
  55. try {
  56. DB::beginTransaction();
  57. $inventoryData = [
  58. 'code' => $data['code'] ?? "",
  59. 'title' => $data['title'] ?? "",
  60. 'size' => $data['size'] ?? "",
  61. 'category_code' => $data['category_code'] ?? "",
  62. 'category_code_title' => $data['category_code_title'] ?? "",
  63. 'unit_group_type' => $data['unit_group_type'] ?? "",
  64. 'unit_group_code' => $data['unit_group_code'] ?? "",
  65. 'unit_group_code_title' => $data['unit_group_code_title'] ?? "",
  66. 'unit_code' => $data['unit_code'] ?? "",
  67. 'unit_code_title' => $data['unit_code_title'] ?? "",
  68. 'vendor_code' => $data['vendor_code'] ?? "",
  69. 'vendor_code_title' => $data['vendor_code_title'] ?? "",
  70. 'bSale' => $data['bSale'] ?? 0, //内销
  71. 'bExpSale' => $data['bExpSale'] ?? 0, //外销
  72. 'bPurchase' => $data['bPurchase'] ?? 0, // 采购
  73. 'bSelf' => $data['bSelf'] ?? 0, // 自制
  74. 'bComsume' => $data['bComsume'] ?? 0, // 生产耗材
  75. 'iImpTaxRate' => $data['iImpTaxRate'] ?? 0, // 进项税率
  76. 'iTaxRate' => $data['iTaxRate'] ?? 0, // 销项税率
  77. 'customs_change_rate' => $data['customs_change_rate'] ?? 0,
  78. 'upd_time' => time(),
  79. ];
  80. Inventory::where('id', $data['id'])->update($inventoryData);
  81. DB::commit();
  82. } catch (\Throwable $exception) {
  83. DB::rollBack();
  84. return [false, "创建存货失败: " . $exception->getMessage()];
  85. }
  86. return [true, ''];
  87. }
  88. public function inventoryRule(&$data, $user, $is_add = true){
  89. if(! $is_add && empty($data['id'])) return [false, 'ID不能为空'];
  90. $id = $data['id'] ?? 0;
  91. $code = $data['code'] ?? "";
  92. if(empty($data['title'])) return [false, '存货名称不能为空'];
  93. $title = $data['title'];
  94. if(empty($data['category_code'])) return [false, '存货分类编码不能为空'];
  95. $category_code = $data['category_code'];
  96. if(empty($data['category_code_title'])) return [false, '存货分类名不能为空'];
  97. $category_code_title = $data['category_code_title'];
  98. if(! isset($data['unit_group_type'])) return [false, '计量单位组类型不存在'];
  99. $unit_group_type = $data['unit_group_type'] ?? "";
  100. if(empty($data['unit_group_code'])) return [false, '计量单位组编码不能为空'];
  101. $unit_group_code = $data['unit_group_code'] ?? "";
  102. if(empty($data['unit_group_code_title'])) return [false, '计量单位组名不能为空'];
  103. $unit_group_code_title = $data['unit_group_code_title'] ?? "";
  104. if(empty($data['unit_code'])) return [false, '主计量单位编码不能为空'];
  105. $unit_code = $data['unit_code'] ?? "";
  106. if(empty($data['unit_code_title'])) return [false, '主计量单位名称不能为空'];
  107. $unit_code_title = $data['unit_code_title'] ?? "";
  108. if(! empty($data['vendor_code_title']) && empty($data['vendor_code'])) return [false, '生产企业编码不能为空'];
  109. $vendor_code = $data['vendor_code'] ?? "";
  110. $vendor_code_title = $data['vendor_code_title'] ?? "";
  111. $bSale = $data['bSale'] ?? 0;
  112. $bExpSale = $data['bExpSale'] ?? 0;
  113. $bPurchase = $data['bPurchase'] ?? 0;
  114. $bSelf = $data['bSelf'] ?? 0;
  115. $bComsume = $data['bComsume'] ?? 0;
  116. $iImpTaxRate = $data['iImpTaxRate'] ?? 0;
  117. $iTaxRate = $data['iTaxRate'] ?? 0;
  118. $res = $this->checkNumber($iImpTaxRate,2,'positive');
  119. if(! $res['valid']) return [false,'进项税率:' . $res['error']];
  120. $res = $this->checkNumber($iTaxRate,2,'positive');
  121. if(! $res['valid']) return [false,'销项税率:' . $res['error']];
  122. $customs_change_rate = $data['customs_change_rate'] ?? 0;
  123. $res = $this->checkNumber($customs_change_rate,2,'positive');
  124. if(! $res['valid']) return [false,'海关换算率:' . $res['error']];
  125. if(! empty($code)){
  126. $bool = Inventory::where('del_time', 0)
  127. ->when(! empty($id), function ($query) use($id){
  128. return $query->where('id', '<>', $id);
  129. })
  130. ->where('code', $code)
  131. ->exists();
  132. if($bool) return [false, '存货编码已存在记录,新增失败'];
  133. }
  134. if(! $is_add){
  135. $bool = Inventory::where('del_time', 0)
  136. ->where('id', $id)
  137. ->where('status', '>', Inventory::STATE_ZERO)
  138. ->exists();
  139. if($bool) return [false, '存货记录状态已变更,编辑失败'];
  140. }
  141. return [true, ''];
  142. }
  143. public function inventoryCommon($data,$user, $field = []){
  144. $model = Inventory::where('del_time',0)
  145. ->where('crt_id', $user['userid'])
  146. ->orderby('id', 'desc');
  147. if(! empty($data['id'])) $model->where('id', $data['id']);
  148. if(isset($data['status'])) $model->where('status', $data['status']);
  149. if(! empty($data['crt_time'][0]) && ! empty($data['crt_time'][1])) {
  150. $return = $this->changeDateToTimeStampAboutRange($data['crt_time']);
  151. $model->where('crt_time','>=',$return[0]);
  152. $model->where('crt_time','<=',$return[1]);
  153. }
  154. return $model;
  155. }
  156. public function inventoryList($data, $user){
  157. $model = $this->inventoryCommon($data, $user);
  158. $list = $this->limit($model,'',$data);
  159. $list = $this->fillInventoryData($list,$user);
  160. return [true, $list];
  161. }
  162. public function fillInventoryData($data, $user){
  163. if(empty($data['data'])) return $data;
  164. foreach ($data['data'] as $key => $value){
  165. $data['data'][$key]['crt_time'] = $value['crt_time'] ? date('Y-m-d H:i:s',$value['crt_time']) : '';
  166. $data['data'][$key]['status_title'] = Inventory::$name[$value['status']] ?? '';
  167. $data['data'][$key]['bSale'] = $value['bSale'] ? '是' : '否';
  168. $data['data'][$key]['bExpSale'] = $value['bExpSale'] ? '是' : '否';
  169. $data['data'][$key]['bPurchase'] = $value['bPurchase'] ? '是' : '否';
  170. $data['data'][$key]['bSelf'] = $value['bSelf'] ? '是' : '否';
  171. $data['data'][$key]['bComsume'] = $value['bComsume'] ? '是' : '否';
  172. }
  173. return $data;
  174. }
  175. public function inventoryDel($data){
  176. if($this->isEmpty($data,'id')) return [false,'请选择数据!'];
  177. try {
  178. DB::beginTransaction();
  179. $time = time();
  180. $bool = Inventory::where('del_time', 0)
  181. ->whereIn('id', $data['id'])
  182. ->where('status', '>', Inventory::STATE_ZERO)
  183. ->exists();
  184. if($bool) return [false, '存货记录状态已变更,删除失败'];
  185. Inventory::where('del_time',0)
  186. ->whereIn('id',$data['id'])
  187. ->update(['del_time' => $time]);
  188. DB::commit();
  189. }catch (\Exception $exception){
  190. DB::rollBack();
  191. return [false,$exception->getMessage()];
  192. }
  193. return [true, ''];
  194. }
  195. //新增存货到本地-----------------------------------------------
  196. //新增供应商到本地-----------------------------------------------
  197. public function vendorAdd($data, $user){
  198. list($status, $msg) = $this->vendorRule($data, $user);
  199. if(! $status) return [false, $msg];
  200. try {
  201. DB::beginTransaction();
  202. $vendorData = [
  203. 'order_number' => $this->generateBillNo([
  204. 'type' => Vendor::Order_type,
  205. 'period' => date("Ym")
  206. ]),
  207. 'code' => $data['code'] ?? "",
  208. 'title' => $data['title'] ?? "",
  209. 'easy_title' => $data['easy_title'] ?? "",
  210. 'category_code' => $data['category_code'] ?? "",
  211. 'category_code_title' => $data['category_code_title'] ?? "",
  212. 'crt_id' => $user['userid'],
  213. 'crt_time' => time(),
  214. ];
  215. Vendor::insert($vendorData);
  216. DB::commit();
  217. } catch (\Throwable $exception) {
  218. DB::rollBack();
  219. return [false, "创建供应商失败: " . $exception->getMessage()];
  220. }
  221. return [true, ''];
  222. }
  223. public function vendorEdit($data, $user){
  224. list($status, $msg) = $this->vendorRule($data, $user);
  225. if(! $status) return [false, $msg];
  226. try {
  227. DB::beginTransaction();
  228. $vendorData = [
  229. 'code' => $data['code'] ?? "",
  230. 'title' => $data['title'] ?? "",
  231. 'easy_title' => $data['easy_title'] ?? "",
  232. 'category_code' => $data['category_code'] ?? "",
  233. 'category_code_title' => $data['category_code_title'] ?? "",
  234. 'upd_time' => time(),
  235. ];
  236. Vendor::where('id', $data['id'])->update($vendorData);
  237. DB::commit();
  238. } catch (\Throwable $exception) {
  239. DB::rollBack();
  240. return [false, "创建供应商失败: " . $exception->getMessage()];
  241. }
  242. return [true, ''];
  243. }
  244. public function vendorRule(&$data, $user, $is_add = true){
  245. if(! $is_add && empty($data['id'])) return [false, 'ID不能为空'];
  246. $id = $data['id'] ?? 0;
  247. $code = $data['code'] ?? "";
  248. if(empty($data['title'])) return [false, '供应商全称不能为空'];
  249. $title = $data['title'];
  250. if(empty($data['easy_title'])) return [false, '供应商简称不能为空'];
  251. $easy_title = $data['easy_title'];
  252. if(empty($data['category_code'])) return [false, '供应商分类编码不能为空'];
  253. $category_code = $data['category_code'];
  254. if(empty($data['category_code_title'])) return [false, '供应商分类名不能为空'];
  255. $category_code_title = $data['category_code_title'];
  256. if(! empty($code)){
  257. $bool = Vendor::where('del_time', 0)
  258. ->when(! empty($id), function ($query) use($id){
  259. return $query->where('id', '<>', $id);
  260. })
  261. ->where('code', $code)
  262. ->exists();
  263. if($bool) return [false, '供应商编码已存在记录,新增失败'];
  264. }
  265. if(! $is_add){
  266. $bool = Vendor::where('del_time', 0)
  267. ->where('id', $id)
  268. ->where('status', '>', Vendor::STATE_ZERO)
  269. ->exists();
  270. if($bool) return [false, '供应商记录状态已变更,编辑失败'];
  271. }
  272. return [true, ''];
  273. }
  274. public function vendorCommon($data,$user, $field = []){
  275. $model = Vendor::where('del_time',0)
  276. ->where('crt_id', $user['userid'])
  277. ->orderby('id', 'desc');
  278. if(! empty($data['id'])) $model->where('id', $data['id']);
  279. if(isset($data['status'])) $model->where('status', $data['status']);
  280. if(! empty($data['crt_time'][0]) && ! empty($data['crt_time'][1])) {
  281. $return = $this->changeDateToTimeStampAboutRange($data['crt_time']);
  282. $model->where('crt_time','>=',$return[0]);
  283. $model->where('crt_time','<=',$return[1]);
  284. }
  285. return $model;
  286. }
  287. public function vendorList($data, $user){
  288. $model = $this->vendorCommon($data, $user);
  289. $list = $this->limit($model,'',$data);
  290. $list = $this->fillVendorData($list,$user);
  291. return [true, $list];
  292. }
  293. public function fillVendorData($data, $user){
  294. if(empty($data['data'])) return $data;
  295. foreach ($data['data'] as $key => $value){
  296. $data['data'][$key]['crt_time'] = $value['crt_time'] ? date('Y-m-d H:i:s',$value['crt_time']) : '';
  297. $data['data'][$key]['status_title'] = Vendor::$name[$value['status']] ?? '';
  298. }
  299. return $data;
  300. }
  301. public function vendorDel($data){
  302. if($this->isEmpty($data,'id')) return [false,'请选择数据!'];
  303. try {
  304. DB::beginTransaction();
  305. $time = time();
  306. $bool = Vendor::where('del_time', 0)
  307. ->whereIn('id', $data['id'])
  308. ->where('status', '>', Inventory::STATE_ZERO)
  309. ->exists();
  310. if($bool) return [false, '供应商记录状态已变更,删除失败'];
  311. Vendor::where('del_time',0)
  312. ->whereIn('id',$data['id'])
  313. ->update(['del_time' => $time]);
  314. DB::commit();
  315. }catch (\Exception $exception){
  316. DB::rollBack();
  317. return [false,$exception->getMessage()];
  318. }
  319. return [true, ''];
  320. }
  321. public function filedCommon($data, $user, $field = []){
  322. $model = FieldData::select([
  323. 'userid',
  324. 'type',
  325. // 使用 GROUP_CONCAT 将 title 合并,并起别名为 titles
  326. DB::raw('GROUP_CONCAT(title SEPARATOR ",") as titles')
  327. ])
  328. ->groupBy('userid', 'type')
  329. ->orderBy('type', 'desc'); // 或者根据你的需求排序
  330. return $model;
  331. }
  332. public function fieldList($data, $user){
  333. $model = $this->filedCommon($data, $user);
  334. $list = $this->limit($model,'',$data);
  335. return [true, $list];
  336. }
  337. public function fieldDetail($data, $user){
  338. if(! isset($data['type'])) return [false, 'type不能为空'];
  339. if(! isset($data['userid'])) return [false, '人员id不能为空'];
  340. return [true, FieldData::where('type', $data['type'])
  341. ->where('userid', $data['userid'])
  342. ->get()->toArray()];
  343. }
  344. public function setField($data, $user){
  345. if(empty($user['qx'])) return [false, '权限不足,设置失败'];
  346. if(empty($data['userid'])) return [false, '人员ID不能为空'];
  347. if(! isset($data['field'])) return [false, '字段列不存在'];
  348. $insert = [];
  349. foreach ($data['field'] as $value){
  350. $insert[] = [
  351. 'userid' => $data['userid'],
  352. 'key' => $value['key'],
  353. 'title' => $value['title'],
  354. ];
  355. }
  356. try {
  357. FieldData::where('userid', $data['userid'])->delete();
  358. if(! empty($insert)) FieldData::insert($insert);
  359. } catch (\Throwable $exception) {
  360. return [false, "异常: " . $exception->getMessage()];
  361. }
  362. return [true, ''];
  363. }
  364. public function ddEmployeeList($data, $user){
  365. $model = $this->ddEmployeeCommon($data, $user);
  366. $list = $this->limit($model,'',$data);
  367. $list = $this->fillDDEmployeeData($list,$user);
  368. return [true, $list];
  369. }
  370. public function ddEmployeeCommon($data,$user, $field = []){
  371. $model = DDEmployee::where('del_time',0)
  372. ->orderby('id', 'desc');
  373. if(! empty($data['id'])) $model->where('id', $data['id']);
  374. if(! empty($data['crt_time'][0]) && ! empty($data['crt_time'][1])) {
  375. $return = $this->changeDateToTimeStampAboutRange($data['crt_time']);
  376. $model->where('crt_time','>=',$return[0]);
  377. $model->where('crt_time','<=',$return[1]);
  378. }
  379. return $model;
  380. }
  381. public function fillDDEmployeeData($data, $user){
  382. if(empty($data['data'])) return $data;
  383. foreach ($data['data'] as $key => $value){
  384. $data['data'][$key]['crt_time'] = $value['crt_time'] ? date('Y-m-d H:i:s',$value['crt_time']) : '';
  385. }
  386. return $data;
  387. }
  388. //新增供应商到本地-----------------------------------------------
  389. //U8 存货库存列表
  390. public function stockList($data, $user)
  391. {
  392. try {
  393. $field_list = [];
  394. $employee = DDEmployee::where('userid', $user['userid'])->first();
  395. $qx = $employee['qx'] ?? 0;
  396. if(empty($qx)) {
  397. $field_list = FieldData::where('userid', $user['userid'])
  398. ->where('type', FieldData::STATE_ZERO)
  399. ->pluck('key')
  400. ->all();
  401. }
  402. // 1. 构建基础查询:关联现存量表和存货档案表
  403. $query = DB::connection('sqlsrv')
  404. ->table('CurrentStock as S')
  405. ->select([
  406. 'S.cWhCode', // 仓库编码
  407. 'W.cWhName', // 仓库名称
  408. 'S.cInvCode', // 存货编码
  409. 'I.cInvName', // 存货名称
  410. 'I.cInvStd', // 规格型号
  411. 'I.cInvCCode', // 分类编码
  412. // --- 数量字段对齐你的结构 ---
  413. 'S.iQuantity', // 结存数量 (账面现存量)
  414. 'S.fAvaQuantity', // 可用数量
  415. 'S.fOutQuantity', // 待发货数量 (待出)
  416. 'S.fInQuantity', // 待入库数量 (待入)
  417. 'S.fStopQuantity', // 冻结数量
  418. // --- 批次与日期 ---
  419. 'S.cBatch', // 批号
  420. 'S.dMdate', // 生产日期
  421. 'S.dVDate', // 失效日期
  422. ])
  423. ->join('Inventory as I', 'S.cInvCode', '=', 'I.cInvCode')
  424. ->leftJoin('Warehouse as W', 'S.cWhCode', '=', 'W.cWhCode')
  425. ->leftJoin('InventoryClass as IC', 'I.cInvCCode', '=', 'IC.cInvCCode');
  426. // 2. 过滤条件:存货名称 (支持模糊查询)
  427. if (!empty($data['material_title'])) {
  428. $query->where('I.cInvName', 'like', '%' . $data['material_title'] . '%');
  429. }
  430. // 2. 过滤条件:存货编码 (支持模糊查询)
  431. if (!empty($data['material_code'])) {
  432. $query->where('S.cInvCode', 'like', '%' . $data['material_code'] . '%');
  433. }
  434. // 3. 过滤条件:存货分类 (支持左匹配,即选大类查出所有子类)
  435. if (!empty($data['category_code'])) {
  436. // U8 分类是级次结构,用 like '01%' 可以查出 01 开头的所有子类
  437. $query->where('I.cInvCCode', 'like', $data['category_code'] . '%');
  438. }
  439. // 6. 排序
  440. $query->orderBy('S.cInvCode', 'asc')->orderBy('S.cWhCode', 'asc');
  441. // 7. 调用你定义的分页方法
  442. $columns = ['*'];
  443. $result = $this->limit($query, $columns, $data);
  444. // 注意这里的 &$item,加了 & 符号才能直接修改原数组里的内容
  445. foreach ($result['data'] as &$item) {
  446. $numFields = ['iQuantity', 'fAvaQuantity', 'fOutQuantity', 'fInQuantity', 'fStopQuantity'];
  447. foreach ($numFields as $field) {
  448. if (isset($item->$field)) {
  449. $item->$field = (float)$item->$field;
  450. }
  451. }
  452. $item->dMdate = $item->dMdate ? date('Y-m-d', strtotime($item->dMdate)) : '';
  453. $item->dVDate = $item->dVDate ? date('Y-m-d', strtotime($item->dVDate)) : '';
  454. // 脱敏处理
  455. if (!empty($field_list)) {
  456. foreach ($field_list as $blackField) {
  457. if (isset($item->$blackField)) {
  458. $item->$blackField = '*****';
  459. }
  460. }
  461. }
  462. }
  463. unset($item); // 销毁引用
  464. return [true, $result];
  465. } catch (\Throwable $exception) {
  466. return [false, "查询库存失败: " . $exception->getMessage()];
  467. }
  468. }
  469. //U8 供应商列表
  470. public function vendorU8List($data, $user)
  471. {
  472. // 1. 构建基础查询
  473. $query = DB::connection('sqlsrv')
  474. ->table('Vendor as V')
  475. ->select([
  476. 'V.cVenCode', // 供应商编码
  477. 'V.cVenName', // 供应商名称
  478. 'V.cVenAbbName', // 供应商简称
  479. 'V.cVCCode', // 分类编码
  480. 'VC.cVCName', // 分类名称 (来自 VendorClass)
  481. 'V.cVenAddress', // 地址
  482. 'V.cVenPhone', // 电话
  483. 'V.dVenDevDate', // 发展日期
  484. 'V.cCreatePerson', // 创建人
  485. 'V.bVenTax', // 是否计税
  486. 'V.iId' // 内部ID
  487. ])
  488. // 关联供应商分类表获取分类名称
  489. ->leftJoin('VendorClass as VC', 'V.cVCCode', '=', 'VC.cVCCode');
  490. // 2. 增加搜索逻辑 (可选)
  491. if (!empty($data['keyword'])) {
  492. $keyword = $data['keyword'];
  493. $query->where(function($q) use ($keyword) {
  494. $q->where('V.cVenCode', 'like', "%{$keyword}%")
  495. ->orWhere('V.cVenName', 'like', "%{$keyword}%")
  496. ->orWhere('V.cVenAbbName', 'like', "%{$keyword}%");
  497. });
  498. }
  499. // 3. 排序 (默认按编码排序)
  500. $query->orderBy('V.cVenCode', 'ASC');
  501. // 4. 调用你定义的分页方法
  502. // 注意:limit 方法内部会执行 paginate 并将结果填充到 $data
  503. $columns = ['*']; // select 已经在上面定义过了,这里传 * 即可
  504. $result = $this->limit($query, $columns, $data);
  505. return [true, $result];
  506. }
  507. // U8 供应商分类树结构
  508. public function vendorClassTree($data, $user)
  509. {
  510. try {
  511. // 1. 获取所有供应商分类 (表名: VendorClass)
  512. $classes = DB::connection('sqlsrv')
  513. ->table('VendorClass')
  514. ->select('cVCCode', 'cVCName', 'iVCGrade', 'bVCEnd') // 编码、名称、级次
  515. ->orderBy('cVCCode', 'asc')
  516. ->get();
  517. if ($classes->isEmpty()) {
  518. return [true, []];
  519. }
  520. // 2. 格式化数据,以编码为 Key
  521. $classList = [];
  522. foreach ($classes as $item) {
  523. $classList[$item->cVCCode] = [
  524. 'label' => $item->cVCName,
  525. 'value' => $item->cVCCode, // 前端通常需要 value 字段
  526. 'code' => $item->cVCCode,
  527. 'grade' => $item->iVCGrade,
  528. 'is_end' => $item->bVCEnd,
  529. 'children' => []
  530. ];
  531. }
  532. // 3. 构建引用树
  533. $tree = [];
  534. foreach ($classList as $code => &$node) {
  535. // 获取父级编码
  536. $parentCode = $this->getParentCode($code);
  537. if ($parentCode === null || !isset($classList[$parentCode])) {
  538. // 顶级节点
  539. $tree[] = &$node;
  540. } else {
  541. // 挂载到父节点
  542. $classList[$parentCode]['children'][] = &$node;
  543. }
  544. }
  545. return [true, $tree];
  546. } catch (\Throwable $exception) {
  547. return [false, "获取供应商分类树失败: " . $exception->getMessage()];
  548. }
  549. }
  550. //U8 存货分类树结构
  551. public function inventoryClassTree($data, $user)
  552. {
  553. try {
  554. // 1. 从数据库获取所有存货分类
  555. $classes = DB::connection('sqlsrv')
  556. ->table('InventoryClass')
  557. ->select('cInvCCode', 'cInvCName', 'iInvCGrade', 'bInvCEnd')
  558. ->orderBy('cInvCCode', 'asc')
  559. ->get();
  560. if ($classes->isEmpty()) return [true, []];
  561. // 2. 将集合转换为数组并以编码作为 Key,方便查找
  562. $classList = [];
  563. foreach ($classes as $item) {
  564. $classList[$item->cInvCCode] = [
  565. 'label' => $item->cInvCName,
  566. 'code' => $item->cInvCCode,
  567. 'grade' => $item->iInvCGrade,
  568. 'is_end' => $item->bInvCEnd,
  569. 'children' => []
  570. ];
  571. }
  572. // 3. 构建树形结构
  573. $tree = [];
  574. foreach ($classList as $code => &$node) {
  575. // 获取当前分类的级次 (U8 逻辑通常根据编码长度判断父级)
  576. // 比如 0101 的父级是 01
  577. $parentCode = $this->getParentCode($code);
  578. if ($parentCode === null || !isset($classList[$parentCode])) {
  579. // 如果没有父级编码,或者父级编码不在列表里,说明是顶级分类
  580. $tree[] = &$node;
  581. } else {
  582. // 将当前节点引用到父节点的 children 数组中
  583. $classList[$parentCode]['children'][] = &$node;
  584. }
  585. }
  586. return [true, $tree];
  587. } catch (\Throwable $exception) {
  588. return [false, "获取分类树失败: " . $exception->getMessage()];
  589. }
  590. }
  591. /**
  592. * 辅助函数:根据 U8 编码规则获取父级编码
  593. * U8 的级次通常存储在 GradeDef 表,但通用逻辑是截取末尾
  594. */
  595. private function getParentCode($code)
  596. {
  597. $len = strlen($code);
  598. if ($len <= 2) return null; // 假设第一级是2位,小于等于2位则无父级
  599. // 这里假设级次是 2-2-2-2 (最常见配置)
  600. // 实际生产中,如果级次不固定,建议查询 GradeDef 表
  601. return substr($code, 0, $len - 2);
  602. }
  603. //U8 计量单位组(带默认主计量单位)
  604. public function getUnitGroups($data, $user)
  605. {
  606. $list = DB::connection('sqlsrv')
  607. ->select("
  608. SELECT
  609. G.cGroupCode,
  610. G.cGroupName,
  611. G.iGroupType,
  612. U.cComUnitCode,
  613. U.cComUnitName,
  614. U.iNumber
  615. FROM ComputationGroup AS G
  616. OUTER APPLY (
  617. SELECT TOP 1 cComUnitCode, cComUnitName, iNumber
  618. FROM ComputationUnit
  619. WHERE cGroupCode = G.cGroupCode
  620. ORDER BY
  621. bMainUnit DESC, -- 1. 优先主计量
  622. iNumber ASC, -- 2. 序号最小 (若 NULL 会排在最前)
  623. cComUnitCode ASC -- 3. 编码最小
  624. ) AS U
  625. ");
  626. return [true, $list];
  627. }
  628. //U8 计量单位档案
  629. public function getComputationUnitList($data, $user)
  630. {
  631. $list = DB::connection('sqlsrv')
  632. ->table('ComputationUnit as U')
  633. ->select(
  634. 'U.cComUnitCode', // 单位编码
  635. 'U.cComUnitName', // 单位名称
  636. 'U.cGroupCode', // 所属组编码
  637. 'U.bMainUnit', // 是否主单位
  638. 'U.iNumber' // 排序序号
  639. )
  640. ->orderBy('U.cGroupCode', 'ASC')
  641. ->orderBy('U.iNumber', 'ASC') // 按照你要求的 iNumber 排序
  642. ->get();
  643. return [true, $list];
  644. }
  645. //U8 存货新增到用友
  646. public function inventoryAdd1($data, $user){
  647. $inventory = Inventory::where('del_time', 0)
  648. ->where('id', $data['id'])
  649. ->first();
  650. if(empty($inventory)) return [false, '存货记录不存在或已被删除'];
  651. $inventory = $inventory->toArray();
  652. if($inventory['status'] != Inventory::STATE_TWO) return [false, '存货记录未审核通过'];
  653. $title = $inventory['title'];
  654. $category_code = $inventory['category_code'];
  655. if(! empty($data['code'])){
  656. //用户传入
  657. $no = $inventory['code'];
  658. $flag_title = "重填";
  659. }else{
  660. //生成编码
  661. list($status, $msg) = $this->generate('inventory','1005');
  662. if(! $status) return [false, $msg];
  663. $no = $msg;
  664. $flag_title = "重试";
  665. }
  666. $inventoryData = [
  667. 'cInvCode' => $no,
  668. 'cInvName' => $title,
  669. 'cInvCCode' => $category_code,
  670. 'cInvStd' => $inventory['size'] ?? null, // 规格型号
  671. 'bSale' => $inventory['bSale'], //内销
  672. 'bExpSale' => $inventory['bExpSale'], //外销
  673. 'bPurchase' => $inventory['bPurchase'], // 采购
  674. 'bSelf' => $inventory['bSelf'], // 自制
  675. 'bComsume' => $inventory['bComsume'], // 生产耗材
  676. 'iGroupType' => $inventory['unit_group_type'], // 计量单位组类别
  677. 'cGroupCode' => $inventory['unit_group_code'], //计量单位组
  678. 'cComUnitCode' => $inventory['unit_code'], // 主计量单位
  679. 'cShopUnit' => $inventory['unit_code'], // 零售计量单位
  680. 'iImpTaxRate' => $inventory['iImpTaxRate'], // 进项税率
  681. 'iTaxRate' => $inventory['iTaxRate'], // 销项税率
  682. 'dSDate' => date("Y-m-d 00:00:00.000"), // 启用日期
  683. 'cEnterprise' => $inventory['vendor_code_title'] ?? null,
  684. 'iSupplyType' => '0', // 供应类型
  685. 'fConvertRate' => '1.0', // 转换因子
  686. 'bInTotalCost' => '1', // 成本累计否
  687. 'cPlanMethod' => 'R',
  688. 'cSRPolicy' => 'PE',
  689. 'bBomMain' => '0', // 允许BOM母件
  690. 'bBomSub' => '0', // 允许BOM子件
  691. 'bProductBill' => '0', // 允许生产订单
  692. 'iPlanDefault' => '1',
  693. 'iAllocatePrintDgt' => '4',
  694. 'bService' => '0',
  695. 'bAccessary' => '0',
  696. 'iInvAdvance' => '0.0',
  697. 'bInvQuality' => '0',
  698. 'bInvBatch' => '0',
  699. 'bInvEntrust' => '0',
  700. 'bInvOverStock' => '0',
  701. 'bFree1' => '0',
  702. 'bFree2' => '0',
  703. 'bInvType' => '0',
  704. 'bFree3' => '0',
  705. 'bFree4' => '0',
  706. 'bFree5' => '0',
  707. 'bFree6' => '0',
  708. 'bFree7' => '0',
  709. 'bFree8' => '0',
  710. 'bFree9' => '0',
  711. 'bFree10' => '0',
  712. 'cCreatePerson' => 'demo',
  713. 'cModifyPerson' => 'demo', // 变更
  714. 'dModifyDate' => date("Y-m-d H:i:s.000"),
  715. 'bFixExch' => '0',
  716. 'bTrack' => '0',
  717. 'bSerial' => '0',
  718. 'bBarCode' => '0',
  719. 'bSolitude' => '0',
  720. 'bSpecialties' => '0',
  721. 'bPropertyCheck' => '0',
  722. 'iRecipeBatch' => '0',
  723. 'bPromotSales' => '0',
  724. 'bPlanInv' => '0',
  725. 'bProxyForeign' => '0',
  726. 'bATOModel' => '0',
  727. 'bCheckItem' => '0',
  728. 'bPTOModel' => '0',
  729. 'bEquipment' => '0',
  730. 'bMPS' => '0',
  731. 'bROP' => '0',
  732. 'bRePlan' => '0',
  733. 'bBillUnite' => '0',
  734. 'bCutMantissa' => '0',
  735. 'bConfigFree1' => '0',
  736. 'bConfigFree2' => '0',
  737. 'bConfigFree3' => '0',
  738. 'bConfigFree4' => '0',
  739. 'bConfigFree5' => '0',
  740. 'bConfigFree6' => '0',
  741. 'bConfigFree7' => '0',
  742. 'bConfigFree8' => '0',
  743. 'bConfigFree9' => '0',
  744. 'bConfigFree10' => '0',
  745. 'bPeriodDT' => '0',
  746. 'bOutInvDT' => '0',
  747. 'bBackInvDT' => '0',
  748. 'bDTWarnInv' => '0',
  749. 'bImportMedicine' => '0',
  750. 'bFirstBusiMedicine' => '0',
  751. 'bForeExpland' => '0',
  752. 'bInvModel' => '0',
  753. 'bKCCutMantissa' => '0',
  754. 'bReceiptByDT' => '0',
  755. 'bCheckBSATP' => '0',
  756. 'bCheckFree1' => '0',
  757. 'bCheckFree2' => '0',
  758. 'bCheckFree3' => '0',
  759. 'bCheckFree4' => '0',
  760. 'bCheckFree5' => '0',
  761. 'bCheckFree6' => '0',
  762. 'bCheckFree7' => '0',
  763. 'bCheckFree8' => '0',
  764. 'bCheckFree9' => '0',
  765. 'bCheckFree10' => '0',
  766. 'iCheckATP' => '0',
  767. 'bPiece' => '0',
  768. 'bSrvItem' => '0',
  769. 'bSrvFittings' => '0',
  770. 'bSpecialOrder' => '0',
  771. 'bTrackSaleBill' => '0',
  772. 'bCheckBatch' => '0',
  773. 'bMngOldpart' => '0',
  774. ];
  775. $inventorySubData = [
  776. 'cInvSubCode' => $no,
  777. 'iSurenessType' => '1',
  778. 'bIsAttachFile' => '0',
  779. 'bInByProCheck' => '1',
  780. 'iRequireTrackStyle' => '0',
  781. 'iExpiratDateCalcu' => '0',
  782. 'iBOMExpandUnitType' => '1',
  783. 'iDrawType' => $inventory['iDrawType'] ?? 0, // 领用方式
  784. 'fInvCIQExch' => $inventory['customs_change_rate'] ?? 1, // 海关换算率
  785. 'bInvKeyPart' => '1',
  786. 'iAcceptEarlyDays' => '999',
  787. 'dInvCreateDatetime' => date("Y-m-d H:i:s.000"),
  788. 'bPUQuota' => '0',
  789. 'bInvROHS' => '0',
  790. 'bPrjMat' => '0',
  791. 'bInvAsset' => '0',
  792. 'bSrvProduct' => '0',
  793. 'iAcceptDelayDays' => '0',
  794. 'bSCkeyProjections' => '0',
  795. 'iSupplyPeriodType' => '1',
  796. 'iAvailabilityDate' => '1',
  797. 'bImport' => '0',
  798. 'bCheckSubitemCost' => '1',
  799. 'fRoundFactor' => '0.0',
  800. 'bConsiderFreeStock' => '1',
  801. 'bSuitRetail' => '0',
  802. 'bFeatureMatch' => '0',
  803. 'bProduceByFeatureAllocate' => '0',
  804. 'bMaintenance' => '0',
  805. 'iMaintenanceCycleUnit' => '0',
  806. 'bCoupon' => '0',
  807. 'bStoreCard' => '0',
  808. 'bProcessProduct' => '0',
  809. 'bProcessMaterial' => '0',
  810. // 所有的价格自由项默认设为 '0'
  811. 'bPurPriceFree1' => '0', 'bPurPriceFree2' => '0', 'bPurPriceFree3' => '0', 'bPurPriceFree4' => '0', 'bPurPriceFree5' => '0',
  812. 'bPurPriceFree6' => '0', 'bPurPriceFree7' => '0', 'bPurPriceFree8' => '0', 'bPurPriceFree9' => '0', 'bPurPriceFree10' => '0',
  813. 'bOMPriceFree1' => '0', 'bOMPriceFree2' => '0', 'bOMPriceFree3' => '0', 'bOMPriceFree4' => '0', 'bOMPriceFree5' => '0',
  814. 'bOMPriceFree6' => '0', 'bOMPriceFree7' => '0', 'bOMPriceFree8' => '0', 'bOMPriceFree9' => '0', 'bOMPriceFree10' => '0',
  815. 'bSalePriceFree1' => '0', 'bSalePriceFree2' => '0', 'bSalePriceFree3' => '0', 'bSalePriceFree4' => '0', 'bSalePriceFree5' => '0',
  816. 'bSalePriceFree6' => '0', 'bSalePriceFree7' => '0', 'bSalePriceFree8' => '0', 'bSalePriceFree9' => '0', 'bSalePriceFree10' => '0',
  817. // 所有的控制自由项默认设为 '0'
  818. 'bControlFreeRange1' => '0', 'bControlFreeRange2' => '0', 'bControlFreeRange3' => '0', 'bControlFreeRange4' => '0', 'bControlFreeRange5' => '0',
  819. 'bControlFreeRange6' => '0', 'bControlFreeRange7' => '0', 'bControlFreeRange8' => '0', 'bControlFreeRange9' => '0', 'bControlFreeRange10' => '0',
  820. // 所有的批次属性默认设为 '0'
  821. 'bBatchProperty1' => '0', 'bBatchProperty2' => '0', 'bBatchProperty3' => '0', 'bBatchProperty4' => '0', 'bBatchProperty5' => '0',
  822. 'bBatchProperty6' => '0', 'bBatchProperty7' => '0', 'bBatchProperty8' => '0', 'bBatchProperty9' => '0', 'bBatchProperty10' => '0',
  823. 'bBondedInv' => '0',
  824. 'bBatchCreate' => '0',
  825. ];
  826. try {
  827. DB::connection('sqlsrv')->beginTransaction();
  828. $exists = DB::connection('sqlsrv')
  829. ->table('Inventory')
  830. ->where('cInvCode', $inventoryData['cInvCode'])
  831. ->lockForUpdate() // 锁定该行,防止并发冲突
  832. ->exists();
  833. if ($exists) return [false, '存货编码已存在,请' . $flag_title];
  834. DB::connection('sqlsrv')->table('Inventory')->insert($inventoryData);
  835. DB::connection('sqlsrv')->table('Inventory_sub')->insert($inventorySubData);
  836. DB::connection('sqlsrv')->commit();
  837. } catch (\Throwable $exception) {
  838. DB::connection('sqlsrv')->rollBack();
  839. return [false, "创建用友失败: " . $exception->getMessage()];
  840. }
  841. return [true, ''];
  842. }
  843. //U8 供应商新增
  844. public function vendorAdd1($data, $user){
  845. $vendor = Vendor::where('del_time', 0)
  846. ->where('id', $data['id'])
  847. ->first();
  848. if(empty($vendor)) return [false, '供应商记录不存在或已被删除'];
  849. $vendor = $vendor->toArray();
  850. if($vendor['status'] != Vendor::STATE_TWO) return [false, '供应商记录未审核通过'];
  851. if(! empty($vendor['code'])){
  852. //用户传入
  853. $no = $vendor['code'];
  854. $flag_title = "重填";
  855. }else{
  856. //生成编码
  857. list($status, $msg) = $this->generate('vendor');
  858. if(! $status) return [false, $msg];
  859. $no = $msg;
  860. $flag_title = "重试";
  861. }
  862. $vendorData = [
  863. 'cVenCode' => $no,
  864. 'cVenName' => $vendor['title'],
  865. 'cVenAbbName' => $vendor['easy_title'],
  866. 'cVCCode' => $vendor['category_code'],
  867. 'dVenDevDate' => date("Y-m-d 00:00:00.000"),
  868. 'iVenDisRate' => '0.0',
  869. 'iVenCreLine' => '0.0',
  870. 'iVenCreDate' => '0',
  871. 'cVenHeadCode' => $no,
  872. 'iAPMoney' => '0.0',
  873. 'iLastMoney' => '0.0',
  874. 'iLRMoney' => '0.0',
  875. 'iFrequency' => '0',
  876. 'bVenTax' => '1',
  877. 'cCreatePerson' => 'demo',
  878. 'cModifyPerson' => 'demo',
  879. 'dModifyDate' => date("Y-m-d H:i:s.000"),
  880. 'iGradeABC' => '-1',
  881. 'bLicenceDate' => '0',
  882. 'bBusinessDate' => '0',
  883. 'bProxyDate' => '0',
  884. 'bPassGMP' => '0',
  885. 'bVenCargo' => '1', // 采购
  886. 'bProxyForeign' => '0', // 委外
  887. 'bVenService' => '0', // 服务
  888. 'bVenOverseas' => '0', // 国外
  889. 'cVenExch_name' => '人民币',
  890. 'iVenGSPType' => '0',
  891. 'iVenGSPAuth' => '-1',
  892. 'bVenAccPeriodMng' => '0',
  893. 'bVenHomeBranch' => '0',
  894. 'dVenCreateDatetime' => date("Y-m-d H:i:s.000"),
  895. 'bIsVenAttachFile' => '0',
  896. 'bRetail' => '0',
  897. ];
  898. try {
  899. DB::connection('sqlsrv')->beginTransaction();
  900. $exists = DB::connection('sqlsrv')
  901. ->table('Vendor')
  902. ->where('cVenCode', $vendorData['cVenCode'])
  903. ->lockForUpdate() // 锁定该行,防止并发冲突
  904. ->exists();
  905. if ($exists) return [false, '供应商编码已存在,请' . $flag_title];
  906. DB::connection('sqlsrv')->table('Vendor')->insert($vendorData);
  907. DB::connection('sqlsrv')->commit();
  908. } catch (\Throwable $exception) {
  909. DB::connection('sqlsrv')->rollBack();
  910. return [false, "创建供应商失败: " . $exception->getMessage()];
  911. }
  912. return [true, ''];
  913. }
  914. // 生成编码
  915. public function generate($type, $classCode = "")
  916. {
  917. try {
  918. // 1. 确定对象映射
  919. $cardNumber = ($type === 'inventory') ? 'inventory' : 'Vendor';
  920. $cardNumber_name = ($type === 'inventory') ? '存货' : '供应商';
  921. $table = ($type === 'inventory') ? 'Inventory' : 'Vendor';
  922. $codeField = ($type === 'inventory') ? 'cInvCode' : 'cVenCode';
  923. // 2. 获取 U8 规则定义
  924. $rule = DB::connection('sqlsrv')->table('VoucherNumber')
  925. ->where('CardNumber', $cardNumber)
  926. ->first();
  927. if (!$rule) return [false, "未找到 {$cardNumber_name} 的规则定义"];
  928. /**
  929. * 3. 动态解析前缀 (Prefix1, Prefix2, Prefix3)
  930. * U8 的规则通常存放在 PrefixRule 或 Prefix 字段中
  931. */
  932. $prefix = "";
  933. for ($i = 1; $i <= 3; $i++) {
  934. $ruleField = "Prefix{$i}Rule";
  935. $valField = "Prefix{$i}";
  936. // 检查规则描述
  937. $ruleDesc = $rule->$ruleField ?? ''; // 例如 "存货分类编码" 或 "GYS"
  938. $staticVal = $rule->$valField ?? '';
  939. if (str_contains($ruleDesc, '分类') || str_contains($staticVal, '分类')) {
  940. // 如果规则提到“分类”,则填入传入的分类编码
  941. $prefix .= $classCode;
  942. } elseif (!empty($ruleDesc)) {
  943. // 如果规则是具体的字符(如 "GYS"),直接拼接
  944. $prefix .= $ruleDesc;
  945. } elseif (!empty($staticVal) && $staticVal !== '手工输入' && $staticVal !== '存货分类编码') {
  946. // 如果静态值不是提示语,则作为前缀
  947. $prefix .= $staticVal;
  948. }
  949. }
  950. // 4. 获取流水号配置 (Glide)
  951. $glideLen = $rule->GlideLen > 0 ? $rule->GlideLen : 4;
  952. $startNum = $rule->iStartNumber ?? 1;
  953. // 5. 查找当前最大编码
  954. $expectedLen = strlen($prefix) + $glideLen;
  955. $lastCode = DB::connection('sqlsrv')->table($table)
  956. ->where($codeField, 'like', $prefix . '%')
  957. ->whereRaw("LEN($codeField) = $expectedLen")
  958. ->orderBy($codeField, 'desc')
  959. ->value($codeField);
  960. // 6. 生成编码
  961. if (!$lastCode) {
  962. // 初始:前缀 + 起始值补零
  963. $finalCode = $prefix . str_pad($startNum, $glideLen, '0', STR_PAD_LEFT);
  964. } else {
  965. // 截取流水号自增
  966. $lastSerial = substr($lastCode, -$glideLen);
  967. $nextSerial = ++$lastSerial;
  968. // 溢出检查
  969. if (strlen($nextSerial) > $glideLen) {
  970. return [false, "流水号已溢出"];
  971. }
  972. $finalCode = $prefix . str_pad($nextSerial, $glideLen, '0', STR_PAD_LEFT);
  973. }
  974. return [true, $finalCode];
  975. } catch (\Throwable $e) {
  976. return [false, "生成失败: " . $e->getMessage()];
  977. }
  978. }
  979. }