U8XkyServerService.php 41 KB

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