U8XkyServerService.php 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843
  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\Record;
  7. use App\Model\Vendor;
  8. use Illuminate\Support\Facades\DB;
  9. class U8XkyServerService extends Service
  10. {
  11. //新增存货到本地-----------------------------------------------
  12. public function inventoryAdd($data, $user){
  13. list($status, $msg) = $this->inventoryRule($data, $user);
  14. if(! $status) return [false, $msg];
  15. try {
  16. DB::beginTransaction();
  17. $inventoryData = [
  18. 'order_number' => $this->generateBillNo([
  19. 'type' => Inventory::Order_type,
  20. 'login_type' => $user['login_type'],
  21. 'period' => date("Ym")
  22. ]),
  23. 'code' => $data['code'] ?? "",
  24. 'title' => $data['title'] ?? "",
  25. 'size' => $data['size'] ?? "",
  26. 'category_code' => $data['category_code'] ?? "",
  27. 'category_code_title' => $data['category_code_title'] ?? "",
  28. 'unit_group_type' => $data['unit_group_type'] ?? "",
  29. 'unit_group_code' => $data['unit_group_code'] ?? "",
  30. 'unit_group_code_title' => $data['unit_group_code_title'] ?? "",
  31. 'unit_code' => $data['unit_code'] ?? "",
  32. 'unit_code_title' => $data['unit_code_title'] ?? "",
  33. 'vendor_code' => $data['vendor_code'] ?? "",
  34. 'vendor_code_title' => $data['vendor_code_title'] ?? "",
  35. 'bSale' => $data['bSale'] ?? 0, //内销
  36. 'bExpSale' => $data['bExpSale'] ?? 0, //外销
  37. 'bPurchase' => $data['bPurchase'] ?? 0, // 采购
  38. 'bSelf' => $data['bSelf'] ?? 0, // 自制
  39. 'bComsume' => $data['bComsume'] ?? 0, // 生产耗材
  40. 'iImpTaxRate' => $data['iImpTaxRate'] ?? 0, // 进项税率
  41. 'iTaxRate' => $data['iTaxRate'] ?? 0, // 销项税率
  42. 'customs_change_rate' => $data['customs_change_rate'] ?? 0,
  43. 'login_type' => $user['login_type'],
  44. 'crt_id' => $user['userid'],
  45. 'crt_time' => time(),
  46. ];
  47. Inventory::insert($inventoryData);
  48. DB::commit();
  49. } catch (\Throwable $exception) {
  50. DB::rollBack();
  51. return [false, "创建存货失败: " . $exception->getMessage()];
  52. }
  53. return [true, ''];
  54. }
  55. public function inventoryEdit($data, $user){
  56. list($status, $msg) = $this->inventoryRule($data, $user);
  57. if(! $status) return [false, $msg];
  58. try {
  59. DB::beginTransaction();
  60. $inventoryData = [
  61. 'code' => $data['code'] ?? "",
  62. 'title' => $data['title'] ?? "",
  63. 'size' => $data['size'] ?? "",
  64. 'category_code' => $data['category_code'] ?? "",
  65. 'category_code_title' => $data['category_code_title'] ?? "",
  66. 'unit_group_type' => $data['unit_group_type'] ?? "",
  67. 'unit_group_code' => $data['unit_group_code'] ?? "",
  68. 'unit_group_code_title' => $data['unit_group_code_title'] ?? "",
  69. 'unit_code' => $data['unit_code'] ?? "",
  70. 'unit_code_title' => $data['unit_code_title'] ?? "",
  71. 'vendor_code' => $data['vendor_code'] ?? "",
  72. 'vendor_code_title' => $data['vendor_code_title'] ?? "",
  73. 'bSale' => $data['bSale'] ?? 0, //内销
  74. 'bExpSale' => $data['bExpSale'] ?? 0, //外销
  75. 'bPurchase' => $data['bPurchase'] ?? 0, // 采购
  76. 'bSelf' => $data['bSelf'] ?? 0, // 自制
  77. 'bComsume' => $data['bComsume'] ?? 0, // 生产耗材
  78. 'iImpTaxRate' => $data['iImpTaxRate'] ?? 0, // 进项税率
  79. 'iTaxRate' => $data['iTaxRate'] ?? 0, // 销项税率
  80. 'customs_change_rate' => $data['customs_change_rate'] ?? 0,
  81. 'upd_time' => time(),
  82. ];
  83. Inventory::where('id', $data['id'])->update($inventoryData);
  84. DB::commit();
  85. } catch (\Throwable $exception) {
  86. DB::rollBack();
  87. return [false, "编辑存货失败: " . $exception->getMessage()];
  88. }
  89. return [true, ''];
  90. }
  91. public function inventoryRule(&$data, $user, $is_add = true){
  92. if(! $is_add && empty($data['id'])) return [false, 'ID不能为空'];
  93. $id = $data['id'] ?? 0;
  94. $code = $data['code'] ?? "";
  95. if(empty($data['title'])) return [false, '存货名称不能为空'];
  96. $title = $data['title'];
  97. if(empty($data['category_code'])) return [false, '存货分类编码不能为空'];
  98. $category_code = $data['category_code'];
  99. if(empty($data['category_code_title'])) return [false, '存货分类名不能为空'];
  100. $category_code_title = $data['category_code_title'];
  101. if(! isset($data['unit_group_type'])) return [false, '计量单位组类型不存在'];
  102. $unit_group_type = $data['unit_group_type'] ?? "";
  103. if(empty($data['unit_group_code'])) return [false, '计量单位组编码不能为空'];
  104. $unit_group_code = $data['unit_group_code'] ?? "";
  105. if(empty($data['unit_group_code_title'])) return [false, '计量单位组名不能为空'];
  106. $unit_group_code_title = $data['unit_group_code_title'] ?? "";
  107. if(empty($data['unit_code'])) return [false, '主计量单位编码不能为空'];
  108. $unit_code = $data['unit_code'] ?? "";
  109. if(empty($data['unit_code_title'])) return [false, '主计量单位名称不能为空'];
  110. $unit_code_title = $data['unit_code_title'] ?? "";
  111. if(! empty($data['vendor_code_title']) && empty($data['vendor_code'])) return [false, '生产企业编码不能为空'];
  112. $vendor_code = $data['vendor_code'] ?? "";
  113. $vendor_code_title = $data['vendor_code_title'] ?? "";
  114. $bSale = $data['bSale'] ?? 0;
  115. $bExpSale = $data['bExpSale'] ?? 0;
  116. $bPurchase = $data['bPurchase'] ?? 0;
  117. $bSelf = $data['bSelf'] ?? 0;
  118. $bComsume = $data['bComsume'] ?? 0;
  119. $iImpTaxRate = $data['iImpTaxRate'] ?? 0;
  120. $iTaxRate = $data['iTaxRate'] ?? 0;
  121. $res = $this->checkNumber($iImpTaxRate,2,'positive');
  122. if(! $res['valid']) return [false,'进项税率:' . $res['error']];
  123. $res = $this->checkNumber($iTaxRate,2,'positive');
  124. if(! $res['valid']) return [false,'销项税率:' . $res['error']];
  125. $customs_change_rate = $data['customs_change_rate'] ?? 0;
  126. $res = $this->checkNumber($customs_change_rate,2,'positive');
  127. if(! $res['valid']) return [false,'海关换算率:' . $res['error']];
  128. if(! empty($code)){
  129. $bool = Inventory::where('del_time', 0)
  130. ->where('login_type', $user['login_type'])
  131. ->when(! empty($id), function ($query) use($id){
  132. return $query->where('id', '<>', $id);
  133. })
  134. ->where('code', $code)
  135. ->exists();
  136. if($bool) return [false, '存货编码已存在记录,新增失败'];
  137. }
  138. if(! $is_add){
  139. $bool = Inventory::where('del_time', 0)
  140. ->where('login_type', $user['login_type'])
  141. ->where('id', $id)
  142. ->where('status', '>', Inventory::STATE_ZERO)
  143. ->exists();
  144. if($bool) return [false, '存货记录状态已变更,编辑失败'];
  145. }
  146. return [true, ''];
  147. }
  148. public function inventoryCommon($data,$user, $field = []){
  149. $model = Inventory::where('del_time',0)
  150. // ->where('crt_id', $user['userid'])
  151. ->where('login_type', $user['login_type'])
  152. ->orderby('id', 'desc');
  153. if(! empty($data['id'])) $model->where('id', $data['id']);
  154. if(isset($data['status'])) $model->where('status', $data['status']);
  155. if(! empty($data['crt_time'][0]) && ! empty($data['crt_time'][1])) {
  156. $return = $this->changeDateToTimeStampAboutRange($data['crt_time']);
  157. $model->where('crt_time','>=',$return[0]);
  158. $model->where('crt_time','<=',$return[1]);
  159. }
  160. return $model;
  161. }
  162. public function inventoryList($data, $user){
  163. $model = $this->inventoryCommon($data, $user);
  164. $list = $this->limit($model,'',$data);
  165. $list = $this->fillInventoryData($list,$user);
  166. return [true, $list];
  167. }
  168. public function fillInventoryData($data, $user){
  169. if(empty($data['data'])) return $data;
  170. $map = Record::where('del_time',0)
  171. ->where('type', Record::type_four)
  172. ->where('login_type', $user['login_type'])
  173. ->whereIn('order_number',array_column($data['data'],'order_number'))
  174. ->pluck('id','order_number')
  175. ->toArray();
  176. $e_map = DDEmployee::whereIn('crt_id', array_unique(array_column($data['data'], 'crt_id')))
  177. ->where('login_type', $user['login_type'])
  178. ->pluck('name_id', 'id')
  179. ->toArray();
  180. foreach ($data['data'] as $key => $value){
  181. $data['data'][$key]['crt_name'] = $e_map[$value['crt_id']] ?? "";
  182. $data['data'][$key]['crt_time'] = $value['crt_time'] ? date('Y-m-d H:i:s',$value['crt_time']) : '';
  183. $data['data'][$key]['status_title'] = Inventory::$name[$value['status']] ?? '';
  184. $data['data'][$key]['bSale'] = $value['bSale'] ? '是' : '否';
  185. $data['data'][$key]['bExpSale'] = $value['bExpSale'] ? '是' : '否';
  186. $data['data'][$key]['bPurchase'] = $value['bPurchase'] ? '是' : '否';
  187. $data['data'][$key]['bSelf'] = $value['bSelf'] ? '是' : '否';
  188. $data['data'][$key]['bComsume'] = $value['bComsume'] ? '是' : '否';
  189. if(isset($map[$value['order_number']])) {
  190. $state = $map[$value['order_number']];
  191. $state_title = Record::state_name[$state];
  192. }else{
  193. $state = Record::state_minus_one;
  194. $state_title = Record::state_name[$state];
  195. }
  196. $data['data'][$key]['state'] = $state;
  197. $data['data'][$key]['state_title'] = $state_title;
  198. }
  199. return $data;
  200. }
  201. public function inventoryDel($data){
  202. if($this->isEmpty($data,'id')) return [false,'请选择数据!'];
  203. try {
  204. DB::beginTransaction();
  205. $time = time();
  206. $bool = Inventory::where('del_time', 0)
  207. ->whereIn('id', $data['id'])
  208. ->where('status', '>', Inventory::STATE_ZERO)
  209. ->exists();
  210. if($bool) return [false, '存货记录状态已变更,删除失败'];
  211. Inventory::where('del_time',0)
  212. ->whereIn('id',$data['id'])
  213. ->update(['del_time' => $time]);
  214. DB::commit();
  215. }catch (\Exception $exception){
  216. DB::rollBack();
  217. return [false,$exception->getMessage()];
  218. }
  219. return [true, ''];
  220. }
  221. //新增存货到本地-----------------------------------------------
  222. //新增供应商到本地-----------------------------------------------
  223. public function vendorAdd($data, $user){
  224. list($status, $msg) = $this->vendorRule($data, $user);
  225. if(! $status) return [false, $msg];
  226. try {
  227. DB::beginTransaction();
  228. $vendorData = [
  229. 'order_number' => $this->generateBillNo([
  230. 'type' => Vendor::Order_type,
  231. 'login_type' => $user['login_type'],
  232. 'period' => date("Ym")
  233. ]),
  234. 'code' => $data['code'] ?? "",
  235. 'title' => $data['title'] ?? "",
  236. 'easy_title' => $data['easy_title'] ?? "",
  237. 'category_code' => $data['category_code'] ?? "",
  238. 'category_code_title' => $data['category_code_title'] ?? "",
  239. 'login_type' => $user['login_type'] ?? "",
  240. 'crt_id' => $user['userid'],
  241. 'crt_time' => time(),
  242. ];
  243. Vendor::insert($vendorData);
  244. DB::commit();
  245. } catch (\Throwable $exception) {
  246. DB::rollBack();
  247. return [false, "创建供应商失败: " . $exception->getMessage()];
  248. }
  249. return [true, ''];
  250. }
  251. public function vendorEdit($data, $user){
  252. list($status, $msg) = $this->vendorRule($data, $user);
  253. if(! $status) return [false, $msg];
  254. try {
  255. DB::beginTransaction();
  256. $vendorData = [
  257. 'code' => $data['code'] ?? "",
  258. 'title' => $data['title'] ?? "",
  259. 'easy_title' => $data['easy_title'] ?? "",
  260. 'category_code' => $data['category_code'] ?? "",
  261. 'category_code_title' => $data['category_code_title'] ?? "",
  262. 'upd_time' => time(),
  263. ];
  264. Vendor::where('id', $data['id'])->update($vendorData);
  265. DB::commit();
  266. } catch (\Throwable $exception) {
  267. DB::rollBack();
  268. return [false, "编辑供应商失败: " . $exception->getMessage()];
  269. }
  270. return [true, ''];
  271. }
  272. public function vendorRule(&$data, $user, $is_add = true){
  273. if(! $is_add && empty($data['id'])) return [false, 'ID不能为空'];
  274. $id = $data['id'] ?? 0;
  275. $code = $data['code'] ?? "";
  276. if(empty($data['title'])) return [false, '供应商全称不能为空'];
  277. $title = $data['title'];
  278. if(empty($data['easy_title'])) return [false, '供应商简称不能为空'];
  279. $easy_title = $data['easy_title'];
  280. if(empty($data['category_code'])) return [false, '供应商分类编码不能为空'];
  281. $category_code = $data['category_code'];
  282. if(empty($data['category_code_title'])) return [false, '供应商分类名不能为空'];
  283. $category_code_title = $data['category_code_title'];
  284. if(! empty($code)){
  285. $bool = Vendor::where('del_time', 0)
  286. ->when(! empty($id), function ($query) use($id){
  287. return $query->where('id', '<>', $id);
  288. })
  289. ->where('code', $code)
  290. ->exists();
  291. if($bool) return [false, '供应商编码已存在记录,新增失败'];
  292. }
  293. if(! $is_add){
  294. $bool = Vendor::where('del_time', 0)
  295. ->where('id', $id)
  296. ->where('status', '>', Vendor::STATE_ZERO)
  297. ->exists();
  298. if($bool) return [false, '供应商记录状态已变更,编辑失败'];
  299. }
  300. return [true, ''];
  301. }
  302. public function vendorCommon($data,$user, $field = []){
  303. $model = Vendor::where('del_time',0)
  304. ->where('crt_id', $user['userid'])
  305. ->orderby('id', 'desc');
  306. if(! empty($data['id'])) $model->where('id', $data['id']);
  307. if(isset($data['status'])) $model->where('status', $data['status']);
  308. if(! empty($data['crt_time'][0]) && ! empty($data['crt_time'][1])) {
  309. $return = $this->changeDateToTimeStampAboutRange($data['crt_time']);
  310. $model->where('crt_time','>=',$return[0]);
  311. $model->where('crt_time','<=',$return[1]);
  312. }
  313. return $model;
  314. }
  315. public function vendorList($data, $user){
  316. $model = $this->vendorCommon($data, $user);
  317. $list = $this->limit($model,'',$data);
  318. $list = $this->fillVendorData($list,$user);
  319. return [true, $list];
  320. }
  321. public function fillVendorData($data, $user){
  322. if(empty($data['data'])) return $data;
  323. $map = Record::where('del_time',0)
  324. ->where('type', Record::type_five)
  325. ->where('login_type', $user['login_type'])
  326. ->whereIn('order_number',array_column($data['data'],'order_number'))
  327. ->pluck('id','order_number')
  328. ->toArray();
  329. $e_map = DDEmployee::whereIn('crt_id', array_unique(array_column($data['data'], 'crt_id')))
  330. ->where('login_type', $user['login_type'])
  331. ->pluck('name_id', 'id')
  332. ->toArray();
  333. foreach ($data['data'] as $key => $value){
  334. $data['data'][$key]['crt_name'] = $e_map[$value['crt_id']] ?? "";
  335. $data['data'][$key]['crt_time'] = $value['crt_time'] ? date('Y-m-d H:i:s',$value['crt_time']) : '';
  336. if(isset($map[$value['order_number']])) {
  337. $state = $map[$value['order_number']];
  338. $state_title = Record::state_name[$state];
  339. }else{
  340. $state = Record::state_minus_one;
  341. $state_title = Record::state_name[$state];
  342. }
  343. $data['data'][$key]['state'] = $state;
  344. $data['data'][$key]['state_title'] = $state_title;
  345. }
  346. return $data;
  347. }
  348. public function vendorDel($data){
  349. if($this->isEmpty($data,'id')) return [false,'请选择数据!'];
  350. try {
  351. DB::beginTransaction();
  352. $time = time();
  353. $bool = Vendor::where('del_time', 0)
  354. ->whereIn('id', $data['id'])
  355. ->where('status', '>', Inventory::STATE_ZERO)
  356. ->exists();
  357. if($bool) return [false, '供应商记录状态已变更,删除失败'];
  358. Vendor::where('del_time',0)
  359. ->whereIn('id',$data['id'])
  360. ->update(['del_time' => $time]);
  361. DB::commit();
  362. }catch (\Exception $exception){
  363. DB::rollBack();
  364. return [false,$exception->getMessage()];
  365. }
  366. return [true, ''];
  367. }
  368. public function filedCommon($data, $user, $field = []){
  369. $model = FieldData::select([
  370. 'userid',
  371. 'type',
  372. // 使用 GROUP_CONCAT 将 title 合并,并起别名为 titles
  373. DB::raw('GROUP_CONCAT(title SEPARATOR ",") as titles')
  374. ])
  375. ->where('login_type', $user['login_type'])
  376. ->groupBy('userid', 'type')
  377. ->orderBy('type', 'desc'); // 或者根据你的需求排序
  378. return $model;
  379. }
  380. public function fieldList($data, $user){
  381. $model = $this->filedCommon($data, $user);
  382. $list = $this->limit($model,'',$data);
  383. $list = $this->fillFieldData($list,$user);
  384. return [true, $list];
  385. }
  386. public function fillFieldData($data, $user){
  387. if(empty($data['data'])) return $data;
  388. $map = DDEmployee::whereIn('userid', array_unique(array_column($data['data'],'userid')))
  389. ->pluck('name', 'userid')
  390. ->toArray();
  391. foreach ($data['data'] as $key => $value){
  392. $data['data'][$key]['crt_time'] = $value['crt_time'] ? date('Y-m-d H:i:s',$value['crt_time']) : '';
  393. $data['data'][$key]['user_title'] = $map[$value['userid']] ?? '';
  394. $data['data'][$key]['type_title'] = FieldData::$name[$value['type']] ?? '';
  395. }
  396. return $data;
  397. }
  398. public function fieldDetail($data, $user){
  399. if(! isset($data['type'])) return [false, 'type不能为空'];
  400. if(! isset($data['userid'])) return [false, '人员id不能为空'];
  401. return [true, FieldData::where('type', $data['type'])
  402. ->where('login_type', $user['login_type'])
  403. ->where('userid', $data['userid'])
  404. ->get()->toArray()];
  405. }
  406. public function setField($data, $user){
  407. if(empty($user['qx'])) return [false, '权限不足,设置失败'];
  408. if(empty($data['userid'])) return [false, '人员ID不能为空'];
  409. if(! isset($data['field'])) return [false, '字段列不存在'];
  410. $insert = [];
  411. foreach ($data['field'] as $value){
  412. $insert[] = [
  413. 'userid' => $data['userid'],
  414. 'key' => $value['key'],
  415. 'title' => $value['title'],
  416. 'login_type' => $user['login_type']
  417. ];
  418. }
  419. try {
  420. FieldData::where('userid', $data['userid'])->where('login_type', $user['login_type'])->delete();
  421. if(! empty($insert)) FieldData::insert($insert);
  422. } catch (\Throwable $exception) {
  423. return [false, "异常: " . $exception->getMessage()];
  424. }
  425. return [true, ''];
  426. }
  427. public function ddEmployeeList($data, $user){
  428. $model = $this->ddEmployeeCommon($data, $user);
  429. $list = $this->limit($model,'',$data);
  430. $list = $this->fillDDEmployeeData($list,$user);
  431. return [true, $list];
  432. }
  433. public function ddEmployeeCommon($data,$user, $field = []){
  434. $model = DDEmployee::where('del_time',0)
  435. ->where('login_type', $user['login_type'])
  436. ->orderby('id', 'desc');
  437. if(! empty($data['id'])) $model->where('id', $data['id']);
  438. if(! empty($data['crt_time'][0]) && ! empty($data['crt_time'][1])) {
  439. $return = $this->changeDateToTimeStampAboutRange($data['crt_time']);
  440. $model->where('crt_time','>=',$return[0]);
  441. $model->where('crt_time','<=',$return[1]);
  442. }
  443. return $model;
  444. }
  445. public function fillDDEmployeeData($data, $user){
  446. if(empty($data['data'])) return $data;
  447. foreach ($data['data'] as $key => $value){
  448. $data['data'][$key]['crt_time'] = $value['crt_time'] ? date('Y-m-d H:i:s',$value['crt_time']) : '';
  449. }
  450. return $data;
  451. }
  452. //新增供应商到本地-----------------------------------------------
  453. //U8 存货库存列表
  454. public function stockList($data, $user)
  455. {
  456. try {
  457. $field_list = [];
  458. $employee = DDEmployee::where('userid', $user['userid'])->where('login_type', $user['login_type'])->first();
  459. $qx = $employee['qx'] ?? 0;
  460. if(empty($qx)) {
  461. $field_list = FieldData::where('userid', $user['userid'])
  462. ->where('login_type', $user['login_type'])
  463. ->where('type', FieldData::STATE_ZERO)
  464. ->pluck('key')
  465. ->all();
  466. }
  467. // 1. 构建基础查询:关联现存量表和存货档案表
  468. $query = DB::connection('sqlsrv')
  469. ->table('CurrentStock as S')
  470. ->select([
  471. 'S.cWhCode', // 仓库编码
  472. 'W.cWhName', // 仓库名称
  473. 'S.cInvCode', // 存货编码
  474. 'I.cInvName', // 存货名称
  475. 'I.cInvStd', // 规格型号
  476. 'I.cInvCCode', // 分类编码
  477. // --- 数量字段对齐你的结构 ---
  478. 'S.iQuantity', // 结存数量 (账面现存量)
  479. 'S.fAvaQuantity', // 可用数量
  480. 'S.fOutQuantity', // 待发货数量 (待出)
  481. 'S.fInQuantity', // 待入库数量 (待入)
  482. 'S.fStopQuantity', // 冻结数量
  483. // --- 批次与日期 ---
  484. 'S.cBatch', // 批号
  485. 'S.dMdate', // 生产日期
  486. 'S.dVDate', // 失效日期
  487. ])
  488. ->join('Inventory as I', 'S.cInvCode', '=', 'I.cInvCode')
  489. ->leftJoin('Warehouse as W', 'S.cWhCode', '=', 'W.cWhCode')
  490. ->leftJoin('InventoryClass as IC', 'I.cInvCCode', '=', 'IC.cInvCCode');
  491. // 2. 过滤条件:存货名称 (支持模糊查询)
  492. if (!empty($data['material_title'])) {
  493. $query->where('I.cInvName', 'like', '%' . $data['material_title'] . '%');
  494. }
  495. // 2. 过滤条件:存货编码 (支持模糊查询)
  496. if (!empty($data['material_code'])) {
  497. $query->where('S.cInvCode', 'like', '%' . $data['material_code'] . '%');
  498. }
  499. // 3. 过滤条件:存货分类 (支持左匹配,即选大类查出所有子类)
  500. if (!empty($data['category_code'])) {
  501. // U8 分类是级次结构,用 like '01%' 可以查出 01 开头的所有子类
  502. $query->where('I.cInvCCode', 'like', $data['category_code'] . '%');
  503. }
  504. // 6. 排序
  505. $query->orderBy('S.cInvCode', 'asc')->orderBy('S.cWhCode', 'asc');
  506. // 7. 调用你定义的分页方法
  507. $columns = ['*'];
  508. $result = $this->limit($query, $columns, $data);
  509. // 注意这里的 &$item,加了 & 符号才能直接修改原数组里的内容
  510. foreach ($result['data'] as &$item) {
  511. $numFields = ['iQuantity', 'fAvaQuantity', 'fOutQuantity', 'fInQuantity', 'fStopQuantity'];
  512. foreach ($numFields as $field) {
  513. if (isset($item->$field)) {
  514. $item->$field = (float)$item->$field;
  515. }
  516. }
  517. $item->dMdate = $item->dMdate ? date('Y-m-d', strtotime($item->dMdate)) : '';
  518. $item->dVDate = $item->dVDate ? date('Y-m-d', strtotime($item->dVDate)) : '';
  519. // 脱敏处理
  520. if (!empty($field_list)) {
  521. foreach ($field_list as $blackField) {
  522. if (isset($item->$blackField)) {
  523. $item->$blackField = '*****';
  524. }
  525. }
  526. }
  527. }
  528. unset($item); // 销毁引用
  529. return [true, $result];
  530. } catch (\Throwable $exception) {
  531. return [false, "查询库存失败: " . $exception->getMessage()];
  532. }
  533. }
  534. //U8 供应商列表
  535. public function vendorU8List($data, $user)
  536. {
  537. // 1. 构建基础查询
  538. $query = DB::connection('sqlsrv')
  539. ->table('Vendor as V')
  540. ->select([
  541. 'V.cVenCode', // 供应商编码
  542. 'V.cVenName', // 供应商名称
  543. 'V.cVenAbbName', // 供应商简称
  544. 'V.cVCCode', // 分类编码
  545. 'VC.cVCName', // 分类名称 (来自 VendorClass)
  546. 'V.cVenAddress', // 地址
  547. 'V.cVenPhone', // 电话
  548. 'V.dVenDevDate', // 发展日期
  549. 'V.cCreatePerson', // 创建人
  550. 'V.bVenTax', // 是否计税
  551. 'V.iId' // 内部ID
  552. ])
  553. // 关联供应商分类表获取分类名称
  554. ->leftJoin('VendorClass as VC', 'V.cVCCode', '=', 'VC.cVCCode');
  555. // 2. 增加搜索逻辑 (可选)
  556. if (!empty($data['keyword'])) {
  557. $keyword = $data['keyword'];
  558. $query->where(function($q) use ($keyword) {
  559. $q->where('V.cVenCode', 'like', "%{$keyword}%")
  560. ->orWhere('V.cVenName', 'like', "%{$keyword}%")
  561. ->orWhere('V.cVenAbbName', 'like', "%{$keyword}%");
  562. });
  563. }
  564. // 3. 排序 (默认按编码排序)
  565. $query->orderBy('V.cVenCode', 'ASC');
  566. // 4. 调用你定义的分页方法
  567. // 注意:limit 方法内部会执行 paginate 并将结果填充到 $data
  568. $columns = ['*']; // select 已经在上面定义过了,这里传 * 即可
  569. $result = $this->limit($query, $columns, $data);
  570. return [true, $result];
  571. }
  572. // U8 供应商分类树结构
  573. public function vendorClassTree($data, $user)
  574. {
  575. try {
  576. // 1. 获取所有供应商分类 (表名: VendorClass)
  577. $classes = DB::connection('sqlsrv')
  578. ->table('VendorClass')
  579. ->select('cVCCode', 'cVCName', 'iVCGrade', 'bVCEnd') // 编码、名称、级次
  580. ->orderBy('cVCCode', 'asc')
  581. ->get();
  582. if ($classes->isEmpty()) {
  583. return [true, []];
  584. }
  585. // 2. 格式化数据,以编码为 Key
  586. $classList = [];
  587. foreach ($classes as $item) {
  588. $classList[$item->cVCCode] = [
  589. 'label' => $item->cVCName,
  590. 'value' => $item->cVCCode, // 前端通常需要 value 字段
  591. 'code' => $item->cVCCode,
  592. 'grade' => $item->iVCGrade,
  593. 'is_end' => $item->bVCEnd,
  594. 'children' => []
  595. ];
  596. }
  597. // 3. 构建引用树
  598. $tree = [];
  599. foreach ($classList as $code => &$node) {
  600. // 获取父级编码
  601. $parentCode = $this->getParentCode($code);
  602. if ($parentCode === null || !isset($classList[$parentCode])) {
  603. // 顶级节点
  604. $tree[] = &$node;
  605. } else {
  606. // 挂载到父节点
  607. $classList[$parentCode]['children'][] = &$node;
  608. }
  609. }
  610. return [true, $tree];
  611. } catch (\Throwable $exception) {
  612. return [false, "获取供应商分类树失败: " . $exception->getMessage()];
  613. }
  614. }
  615. //U8 存货分类树结构
  616. public function inventoryClassTree($data, $user)
  617. {
  618. try {
  619. // 1. 从数据库获取所有存货分类
  620. $classes = DB::connection('sqlsrv')
  621. ->table('InventoryClass')
  622. ->select('cInvCCode', 'cInvCName', 'iInvCGrade', 'bInvCEnd')
  623. ->orderBy('cInvCCode', 'asc')
  624. ->get();
  625. if ($classes->isEmpty()) return [true, []];
  626. // 2. 将集合转换为数组并以编码作为 Key,方便查找
  627. $classList = [];
  628. foreach ($classes as $item) {
  629. $classList[$item->cInvCCode] = [
  630. 'label' => $item->cInvCName,
  631. 'code' => $item->cInvCCode,
  632. 'grade' => $item->iInvCGrade,
  633. 'is_end' => $item->bInvCEnd,
  634. 'children' => []
  635. ];
  636. }
  637. // 3. 构建树形结构
  638. $tree = [];
  639. foreach ($classList as $code => &$node) {
  640. // 获取当前分类的级次 (U8 逻辑通常根据编码长度判断父级)
  641. // 比如 0101 的父级是 01
  642. $parentCode = $this->getParentCode($code);
  643. if ($parentCode === null || !isset($classList[$parentCode])) {
  644. // 如果没有父级编码,或者父级编码不在列表里,说明是顶级分类
  645. $tree[] = &$node;
  646. } else {
  647. // 将当前节点引用到父节点的 children 数组中
  648. $classList[$parentCode]['children'][] = &$node;
  649. }
  650. }
  651. return [true, $tree];
  652. } catch (\Throwable $exception) {
  653. return [false, "获取分类树失败: " . $exception->getMessage()];
  654. }
  655. }
  656. /**
  657. * 辅助函数:根据 U8 编码规则获取父级编码
  658. * U8 的级次通常存储在 GradeDef 表,但通用逻辑是截取末尾
  659. */
  660. private function getParentCode($code)
  661. {
  662. $len = strlen($code);
  663. if ($len <= 2) return null; // 假设第一级是2位,小于等于2位则无父级
  664. // 这里假设级次是 2-2-2-2 (最常见配置)
  665. // 实际生产中,如果级次不固定,建议查询 GradeDef 表
  666. return substr($code, 0, $len - 2);
  667. }
  668. //U8 计量单位组(带默认主计量单位)
  669. public function getUnitGroups($data, $user)
  670. {
  671. $list = DB::connection('sqlsrv')
  672. ->select("
  673. SELECT
  674. G.cGroupCode,
  675. G.cGroupName,
  676. G.iGroupType,
  677. U.cComUnitCode,
  678. U.cComUnitName,
  679. U.iNumber
  680. FROM ComputationGroup AS G
  681. OUTER APPLY (
  682. SELECT TOP 1 cComUnitCode, cComUnitName, iNumber
  683. FROM ComputationUnit
  684. WHERE cGroupCode = G.cGroupCode
  685. ORDER BY
  686. bMainUnit DESC, -- 1. 优先主计量
  687. iNumber ASC, -- 2. 序号最小 (若 NULL 会排在最前)
  688. cComUnitCode ASC -- 3. 编码最小
  689. ) AS U
  690. ");
  691. return [true, $list];
  692. }
  693. //U8 计量单位档案
  694. public function getComputationUnitList($data, $user)
  695. {
  696. $list = DB::connection('sqlsrv')
  697. ->table('ComputationUnit as U')
  698. ->select(
  699. 'U.cComUnitCode', // 单位编码
  700. 'U.cComUnitName', // 单位名称
  701. 'U.cGroupCode', // 所属组编码
  702. 'U.bMainUnit', // 是否主单位
  703. 'U.iNumber' // 排序序号
  704. )
  705. ->orderBy('U.cGroupCode', 'ASC')
  706. ->orderBy('U.iNumber', 'ASC') // 按照你要求的 iNumber 排序
  707. ->get();
  708. return [true, $list];
  709. }
  710. }