U8XkyServerService.php 40 KB

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