GoodsService.php 67 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636
  1. <?php
  2. namespace app\admin\service\goods;
  3. use app\model\Goods;
  4. use app\model\GoodsComponent;
  5. use app\model\GoodsDetail;
  6. use app\model\GoodsLabel;
  7. use app\model\GoodsRunning;
  8. use app\model\GoodsSku;
  9. use app\model\SysCategory;
  10. use app\model\SysSerial;
  11. use support\Db;
  12. use support\exception\BusinessException;
  13. use support\Redis;
  14. use support\Request;
  15. use support\Response;
  16. class GoodsService
  17. {
  18. public static function selectAll($goodsIds)
  19. {
  20. $goods = Goods::where('goods_status', 'ON')
  21. ->when($goodsIds != '', function ($query) use ($goodsIds) {
  22. $query->whereIn('goods_id', $goodsIds);
  23. })
  24. ->select('goods_id', 'goods_name')
  25. ->get();
  26. return json_success('', $goods);
  27. }
  28. public static function selectAllByCategoryForRuleAddComponent($category = "GOODS")
  29. {
  30. $categoryIds = [];
  31. if ($category == 'GOODS') {
  32. $categoryIds = [6, 7, 8, 10, 11, 12];
  33. } elseif ($category == 'SERVICE') {
  34. $categoryIds = [33, 34, 35, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 36, 37, 38, 39, 40, 41];
  35. }
  36. $goods = Goods::with('sku')
  37. // ->where('goods_classify', $category)
  38. ->when(!empty($categoryIds), function ($query) use ($categoryIds) {
  39. $query->whereIn('join_goods_category_id', $categoryIds);
  40. })->select('goods_id', 'goods_name', 'join_goods_category_id')
  41. ->get()
  42. ->toArray();
  43. foreach ($goods as &$good) {
  44. if (!empty($good['sku'])) {
  45. foreach ($good['sku'] as $key => $sku) {
  46. if (!empty($sku['goods_sku_specs_json'])) {
  47. $specsJson = json_decode($sku['goods_sku_specs_json'], true);
  48. $skuTitle = '';
  49. foreach ($specsJson as $item) {
  50. if (is_array($item)) {
  51. $item = implode(',', $item);
  52. }
  53. $skuTitle .= $item . '-';
  54. }
  55. $good['sku'][$key]['goods_sku_id'] = strval($good['sku'][$key]['goods_sku_id']);
  56. $good['sku'][$key]['goods_sku_title'] = rtrim($skuTitle, '-');
  57. }
  58. unset($good['sku'][$key]['goods_sku_specs_json'], $good['sku'][$key]['join_sku_goods_id']);
  59. }
  60. } else {
  61. $good['sku'] = [];
  62. }
  63. }
  64. return json_success('', $goods);
  65. }
  66. public static function select(Request $request, $classify = "GOODS")
  67. {
  68. $page = $request->get('page');
  69. $pageSize = $request->get('pageSize');
  70. $goodsName = $request->get('goods_name', '');
  71. $categoryId = $request->get('join_goods_category_id', null);
  72. $goodsCategory = $request->get('goods_category', null);
  73. if ($categoryId != null) {
  74. $categoryPath = SysCategory::where('category_id', $categoryId)->value('category_super_path');
  75. $categoryPath .= '#' . $categoryId . '#';
  76. $categoryIds = SysCategory::where('category_super_path', 'like', $categoryPath . '%')->pluck('category_id')->toArray();
  77. if (!empty($categoryIds)) {
  78. $categoryId = $categoryIds;
  79. } else {
  80. $categoryId = [$categoryId];
  81. }
  82. }
  83. $rows = Goods::with([
  84. 'category' => function ($query) {
  85. $query->select('category_id', 'category_name');
  86. },
  87. 'running' => function ($query) {
  88. $query->select('join_running_goods_id', 'goods_running_storage');
  89. },
  90. 'supplier' => function ($query) {
  91. $query->select('supplier_id', 'supplier_name');
  92. }
  93. ])->select('goods_id', 'join_goods_category_id', 'join_goods_supplier_id', 'goods_status', 'goods_sales_price', 'goods_category', 'goods_name', 'goods_title', 'goods_cover', 'goods_sort', 'goods_addtimes')
  94. ->when($goodsName != '', function ($query) use ($goodsName) {
  95. $query->where(function ($q) use ($goodsName) {
  96. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  97. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  98. });
  99. })->when($categoryId != null, function ($query) use ($categoryId) {
  100. $query->whereIn('join_goods_category_id', $categoryId);
  101. })->when($goodsCategory != null, function ($query) use ($goodsCategory) {
  102. $query->where('goods_category', $goodsCategory);
  103. })
  104. ->when($classify != '', function ($query) use ($classify) {
  105. if ($classify == 'GOODS') {
  106. $query->whereIn('join_goods_category_id', ['6', '7', '8', '10', '11', '12']);
  107. } else {
  108. $query->where('goods_classify', $classify);
  109. }
  110. })
  111. ->orderBy('goods_addtimes', 'DESC')
  112. ->forPage($page, $pageSize)
  113. ->get()
  114. ->toArray();
  115. $total = Goods::when($goodsName != '', function ($query) use ($goodsName) {
  116. $query->where(function ($q) use ($goodsName) {
  117. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  118. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  119. });
  120. })->when($categoryId != null, function ($query) use ($categoryId) {
  121. $query->whereIn('join_goods_category_id', $categoryId);
  122. })->when($goodsCategory != null, function ($query) use ($goodsCategory) {
  123. $query->where('goods_category', $goodsCategory);
  124. })->when($classify != '', function ($query) use ($classify) {
  125. if ($classify == 'GOODS') {
  126. $query->whereIn('join_goods_category_id', ['6', '7', '8', '9', '10', '11', '12', '30']);
  127. } else {
  128. $query->where('goods_classify', $classify);
  129. }
  130. })
  131. ->count();
  132. foreach ($rows as &$row) {
  133. $row['goods_cover'] = getenv('STORAGE_DOMAIN') . $row['goods_cover'];
  134. if (isset($row['running'])) {
  135. $row['running']['goods_running_storage'] = intval($row['running']['goods_running_storage']);
  136. }
  137. }
  138. return json_success('', compact('rows', 'page', 'pageSize', 'total'));
  139. }
  140. public static function selectSpecial(Request $request)
  141. {
  142. $page = $request->get('page');
  143. $pageSize = $request->get('pageSize');
  144. $goodsName = $request->get('goods_name', '');
  145. $categoryId = $request->get('join_goods_category_id', null);
  146. if ($categoryId == null) {
  147. $categoryId = [65, 43];
  148. } elseif (is_string($categoryId)) {
  149. $categoryId = [$categoryId];
  150. }
  151. $rows = Goods::with([
  152. 'category' => function ($query) {
  153. $query->select('category_id', 'category_name');
  154. },
  155. 'running' => function ($query) {
  156. $query->select('join_running_goods_id', 'goods_running_storage');
  157. },
  158. 'supplier' => function ($query) {
  159. $query->select('supplier_id', 'supplier_name');
  160. }
  161. ])->select('goods_id', 'join_goods_category_id', 'join_goods_supplier_id', 'goods_status', 'goods_sales_price', 'goods_category', 'goods_name', 'goods_title', 'goods_cover', 'goods_sort', 'goods_addtimes')
  162. ->when($goodsName != '', function ($query) use ($goodsName) {
  163. $query->where(function ($q) use ($goodsName) {
  164. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  165. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  166. });
  167. })->whereIn('join_goods_category_id', $categoryId)
  168. ->orderBy('goods_addtimes', 'DESC')
  169. ->forPage($page, $pageSize)
  170. ->get()
  171. ->toArray();
  172. $total = Goods::when($goodsName != '', function ($query) use ($goodsName) {
  173. $query->where(function ($q) use ($goodsName) {
  174. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  175. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  176. });
  177. })->whereIn('join_goods_category_id', $categoryId)->count();
  178. foreach ($rows as &$row) {
  179. $row['goods_cover'] = getenv('STORAGE_DOMAIN') . $row['goods_cover'];
  180. if (isset($row['running'])) {
  181. $row['running']['goods_running_storage'] = intval($row['running']['goods_running_storage']);
  182. }
  183. }
  184. return json_success('', compact('rows', 'page', 'pageSize', 'total'));
  185. }
  186. public static function selectPicking(Request $request)
  187. {
  188. $page = $request->get('page');
  189. $pageSize = $request->get('pageSize');
  190. $goodsName = $request->get('goods_name', '');
  191. $rows = Goods::with([
  192. 'category' => function ($query) {
  193. $query->select('category_id', 'category_name');
  194. },
  195. 'running' => function ($query) {
  196. $query->select('join_running_goods_id', 'goods_running_storage');
  197. },
  198. 'supplier' => function ($query) {
  199. $query->select('supplier_id', 'supplier_name');
  200. }
  201. ])->select('goods_id', 'join_goods_category_id', 'join_goods_supplier_id', 'goods_status', 'goods_sales_price', 'goods_category', 'goods_name', 'goods_title', 'goods_cover', 'goods_sort', 'goods_addtimes')
  202. ->when($goodsName != '', function ($query) use ($goodsName) {
  203. $query->where(function ($q) use ($goodsName) {
  204. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  205. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  206. });
  207. })->where('goods_classify', 'SERVICE')
  208. ->where('goods_category', 'FARM')
  209. ->orderBy('goods_addtimes', 'DESC')
  210. ->forPage($page, $pageSize)
  211. ->get()
  212. ->toArray();
  213. $total = Goods::when($goodsName != '', function ($query) use ($goodsName) {
  214. $query->where(function ($q) use ($goodsName) {
  215. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  216. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  217. });
  218. })->where('goods_classify', 'SERVICE')
  219. ->where('goods_category', 'FARM')
  220. ->count();
  221. foreach ($rows as &$row) {
  222. $row['goods_cover'] = getenv('STORAGE_DOMAIN') . $row['goods_cover'];
  223. if (isset($row['running'])) {
  224. $row['running']['goods_running_storage'] = intval($row['running']['goods_running_storage']);
  225. }
  226. }
  227. return json_success('', compact('rows', 'page', 'pageSize', 'total'));
  228. }
  229. public static function selectPackage(Request $request)
  230. {
  231. $page = $request->get('page');
  232. $pageSize = $request->get('pageSize');
  233. $goodsName = $request->get('goods_name', '');
  234. $categoryId = $request->get('join_goods_category_id', null);
  235. $rows = Goods::with([
  236. 'category' => function ($query) {
  237. $query->select('category_id', 'category_name');
  238. },
  239. 'running' => function ($query) {
  240. $query->select('join_running_goods_id', 'goods_running_storage');
  241. },
  242. 'supplier' => function ($query) {
  243. $query->select('supplier_id', 'supplier_name');
  244. },
  245. ])->select('goods_id', 'join_goods_category_id', 'join_goods_supplier_id', 'goods_status', 'goods_sales_price', 'goods_category', 'goods_name', 'goods_title', 'goods_cover', 'goods_sort', 'goods_addtimes')
  246. ->when($goodsName != '', function ($query) use ($goodsName) {
  247. $query->where(function ($q) use ($goodsName) {
  248. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  249. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  250. });
  251. })->when($categoryId != null, function ($query) use ($categoryId) {
  252. $query->where('join_goods_category_id', $categoryId);
  253. })
  254. ->where('goods_classify', 'PACKAGE')
  255. ->orderBy('goods_addtimes', 'DESC')
  256. ->forPage($page, $pageSize)
  257. ->get()
  258. ->toArray();
  259. $total = Goods::when($goodsName != '', function ($query) use ($goodsName) {
  260. $query->where(function ($q) use ($goodsName) {
  261. $q->where('goods_name', 'like', '%' . $goodsName . '%')
  262. ->OrWhere('goods_title', 'like', '%' . $goodsName . '%');
  263. });
  264. })->when($categoryId != null, function ($query) use ($categoryId) {
  265. $query->where('join_goods_category_id', $categoryId);
  266. })->where('goods_classify', 'PACKAGE')->count();
  267. foreach ($rows as &$row) {
  268. $row['goods_cover'] = getenv('STORAGE_DOMAIN') . $row['goods_cover'];
  269. if (isset($row['running'])) {
  270. $row['running']['goods_running_storage'] = intval($row['running']['goods_running_storage']);
  271. }
  272. // if (!empty($row['component'])) {
  273. // $ids = [];
  274. // $contentList = [];
  275. // foreach ($row['component'] as $component) {
  276. // $ids[] = $component['join_component_goods_id'];
  277. // $configJson = json_decode($component['goods_component_config_json'], true);
  278. // $contentList[] = [
  279. // 'goods_name' => $configJson['goods_name'] ?? '',
  280. // 'goods_sales_price' => $component['goods_component_price'],
  281. // 'nbr' => $configJson['nbr'] ?? 0
  282. // ];
  283. // }
  284. //
  285. // $row['join_component_goods_id'] = $ids;
  286. // $row['goodsContentList'] = $contentList;
  287. // }
  288. }
  289. return json_success('', compact('rows', 'page', 'pageSize', 'total'));
  290. }
  291. /**
  292. * @Desc 下拉选择服务商品
  293. * @Author Gorden
  294. * @Date 2024/4/24 13:32
  295. *
  296. * @param Request $request
  297. * @return Response
  298. */
  299. public static function selectList(Request $request, $goodsClassify = "SERVICE")
  300. {
  301. $keywords = $request->get('keywords', '');
  302. // if (!$keywords){
  303. // return json_success('暂无数据');
  304. // }
  305. // $categoryIds = SysCategory::whereIn('category_super_id', [5, 31, 32, 42, 66, 70, 72])->pluck('category_id');
  306. $goods = Goods::with('sku')
  307. // ->whereIn('join_goods_category_id', $categoryIds)
  308. ->when($keywords != '', function ($query) use ($keywords) {
  309. $query->where('goods_name', 'like', "%" . $keywords . "%");
  310. })
  311. ->select('goods_id', 'goods_name', 'goods_sales_price', 'join_goods_category_id')
  312. ->get()
  313. ->toArray();
  314. foreach ($goods as &$good) {
  315. if (!empty($good['sku'])) {
  316. foreach ($good['sku'] as $key => $sku) {
  317. if (!empty($sku['goods_sku_specs_json'])) {
  318. $good['sku'][$key]['goods_sku_specs_json'] = json_decode($sku['goods_sku_specs_json']);
  319. $skuName = '';
  320. foreach ($good['sku'][$key]['goods_sku_specs_json'] as $specsKey => $skuSpecs) {
  321. if (is_array($skuSpecs)) {
  322. $skuName = $skuName . ' ' . $specsKey . ' ' . implode(' ', $skuSpecs);
  323. } else {
  324. $skuName = $skuName . ' ' . $specsKey . ' ' . $skuSpecs;
  325. }
  326. }
  327. $good['sku'][$key]['sku_name'] = $skuName;
  328. }
  329. }
  330. } else {
  331. $good['sku'] = [];
  332. }
  333. }
  334. return json_success('', $goods);
  335. }
  336. /**
  337. * @Desc 商品详情
  338. * @Author Gorden
  339. * @Date 2024/3/28 10:25
  340. *
  341. * @param $goodsId
  342. * @return Response
  343. */
  344. public static function info($goodsId)
  345. {
  346. try {
  347. // 商品主表
  348. $main = Goods::where('goods_id', $goodsId)->first();
  349. if (!empty($main)) {
  350. $main = $main->toArray();
  351. $main['goods_sku_json'] = json_decode($main['goods_sku_json'], true);
  352. $main['specList'] = [];
  353. foreach ($main['goods_sku_json'] as $key => $sku) {
  354. $main['specList'][] = [
  355. 'label' => $key,
  356. 'tags' => $sku
  357. ];
  358. }
  359. } else {
  360. $main = [];
  361. }
  362. // 详情表
  363. $detail = GoodsDetail::where('join_detail_goods_id', $goodsId)->first();
  364. if (!empty($detail)) {
  365. $detail = $detail->toArray();
  366. } else {
  367. $detail = [];
  368. }
  369. // 标签表
  370. $label = GoodsLabel::where('join_label_goods_id', $goodsId)->first();
  371. if (!empty($label)) {
  372. $label = $label->toArray();
  373. } else {
  374. $label = [];
  375. }
  376. // Running表
  377. $running = GoodsRunning::where('join_running_goods_id', $goodsId)->first();
  378. if (!empty($running)) {
  379. $running = $running->toArray();
  380. if (!empty($running['goods_running_off_json']) && is_json($running['goods_running_off_json'])) {
  381. $goodsRunningOffJson = json_decode($running['goods_running_off_json'], true);
  382. $running['goods_off_addtimes'] = isset($goodsRunningOffJson['time']) ? date("Y-m-d H:i", $goodsRunningOffJson['time']) : '';
  383. }
  384. } else {
  385. $running = [];
  386. }
  387. // Sku表
  388. $skus = GoodsSku::where('join_sku_goods_id', $goodsId)->get();
  389. if (!empty($skus)) {
  390. $skus = $skus->toArray();
  391. $submitList = [];
  392. foreach ($skus as $key => $sku) {
  393. $skuSpecsJson = json_decode($sku['goods_sku_specs_json'], true);
  394. $skuSpecs = '';
  395. $skuNameValue = [];
  396. $i = 1;
  397. foreach ($skuSpecsJson as $k => $item) {
  398. if (is_array($item)) {
  399. $item = implode(',', $item);
  400. }
  401. $skuSpecs = $skuSpecs . $item . ',';
  402. $skuNameKey = 'skuName' . $i;
  403. $skuValueKey = 'skuValue' . $i;
  404. $skuNameValue[$skuNameKey] = $k;
  405. $skuNameValue[$skuValueKey] = $item;
  406. $skuNameValue[$k] = $item;
  407. $i++;
  408. }
  409. $storage = json_decode($sku['goods_sku_storage_json'], true);
  410. $priceStorage = [
  411. 'sku_id' => $sku['goods_sku_id'],
  412. 'sku' => rtrim($skuSpecs, ',') ?? '',
  413. 'stock' => $storage['storage'] ?? '',
  414. 'price' => $sku['goods_sku_sales_price'] ?? '',
  415. ];
  416. $submitList[] = array_merge($skuNameValue, $priceStorage);
  417. }
  418. $skus['submitList'] = $submitList;
  419. } else {
  420. $skus = [];
  421. }
  422. // 合并数据
  423. $data = array_merge($main, $detail, $label, $running, ['sku' => $skus]);
  424. $data['goods_sku_json_label'] = [];
  425. $data['goods_label'] = !empty($data['goods_label']) ? explode(',', $data['goods_label']) : [];
  426. $data['goods_json'] = $data['goods_json'] ? json_decode($data['goods_json'], true) : [];
  427. $data['goods_cover'] = getenv('STORAGE_DOMAIN') . $data['goods_cover'];
  428. if (!empty($data['goods_detail_slider_json'])) {
  429. $data['goods_detail_slider_json'] = json_decode($data['goods_detail_slider_json'], true);
  430. // ……
  431. if (isset($data['goods_detail_slider_json']['slider'])) {
  432. $data['goods_detail_slider_json'] = explode(',', $data['goods_detail_slider_json']['slider']);
  433. }
  434. $slider = '';
  435. foreach ($data['goods_detail_slider_json'] as $item) {
  436. $slider .= getenv('STORAGE_DOMAIN') . $item . ',';
  437. }
  438. $data['goods_detail_slider_json'] = rtrim($slider, ',');
  439. }
  440. $extendJson = [];
  441. if (!empty($data['goods_attribute_json'])) {
  442. $extendJson = json_decode($data['goods_attribute_json'], true);
  443. $data['goods_attribute_json'] = $extendJson;
  444. }
  445. $data['appointment_times'] = [];
  446. if ($data['is_support_appointment'] == 'Y' && isset($extendJson['dates'])) {
  447. $data['dates'] = $extendJson['dates'] ?? [];
  448. if (isset($extendJson['times'])) {
  449. $times = [];
  450. foreach ($extendJson['times'] as $time) {
  451. if (!empty($time['duration'])) {
  452. $startEndTime = explode('-', $time['duration']);
  453. $times[] = [
  454. 'person' => $time['person'],
  455. 'appointmentTimeStart' => $startEndTime[0],
  456. 'appointmentTimeEnd' => $startEndTime[1],
  457. ];
  458. }
  459. }
  460. $data['appointment_times'] = $times;
  461. }
  462. if (isset($extendJson['address'])) {
  463. $data['address'] = $extendJson['address'];
  464. }
  465. if (isset($extendJson['position'])) {
  466. $data['position'] = $extendJson['position'];
  467. }
  468. if (isset($extendJson['label'])) {
  469. $data['appointment_label'] = $extendJson['label'];
  470. }
  471. }
  472. if (!empty($data['goods_json']) && $data['join_goods_category_id'] == 65) {
  473. foreach ($data['goods_json'] as $key => $datum) {
  474. $data['goods_json'][$key]['color'] = rgbToHex($datum['color']);
  475. }
  476. } elseif (!empty($data['goods_json']) && $data['join_goods_category_id'] == 43) {
  477. $goodsJsonNew = [];
  478. foreach ($data['goods_json'] as $key1 => $item1) {
  479. $itemsNew = [];
  480. if (isset($item1['items'])) {
  481. foreach ($item1['items'] as $key2 => $item2) {
  482. $itemsNew[] = [
  483. 'key' => $key2,
  484. 'params' => $item2
  485. ];
  486. }
  487. }
  488. $goodsJsonNew[] = [
  489. 'title' => $key1,
  490. 'service' => $item1['service'] ?? '',
  491. 'items' => $itemsNew
  492. ];
  493. }
  494. $data['goods_json'] = $goodsJsonNew;
  495. }
  496. $data['goods_on_addtimes'] = date('Y-m-d\TH:i:s.u\Z', $data['goods_on_addtimes'] - 60 * 60 * 8);
  497. return json_success('', $data);
  498. } catch (\Exception $e) {
  499. dump($e->getMessage());
  500. return json_fail("查询错误~");
  501. }
  502. }
  503. public static function infoPackage($goodsId)
  504. {
  505. try {
  506. // 商品主表
  507. $main = Goods::where('goods_id', $goodsId)->first();
  508. if (!empty($main)) {
  509. $main = $main->toArray();
  510. $main['goods_sku_json'] = json_decode($main['goods_sku_json'], true);
  511. $main['specList'] = [];
  512. foreach ($main['goods_sku_json'] as $key => $sku) {
  513. $main['specList'][] = [
  514. 'label' => $key,
  515. 'tags' => $sku
  516. ];
  517. }
  518. } else {
  519. $main = [];
  520. }
  521. // 详情表
  522. $detail = GoodsDetail::where('join_detail_goods_id', $goodsId)->first();
  523. if (!empty($detail)) {
  524. $detail = $detail->toArray();
  525. } else {
  526. $detail = [];
  527. }
  528. // 标签表
  529. $label = GoodsLabel::where('join_label_goods_id', $goodsId)->first();
  530. if (!empty($label)) {
  531. $label = $label->toArray();
  532. } else {
  533. $label = [];
  534. }
  535. // Running表
  536. $running = GoodsRunning::where('join_running_goods_id', $goodsId)->first();
  537. if (!empty($running)) {
  538. $running = $running->toArray();
  539. if (!empty($running['goods_running_off_json']) && is_json($running['goods_running_off_json'])) {
  540. $goodsRunningOffJson = json_decode($running['goods_running_off_json'], true);
  541. $running['goods_off_addtimes'] = isset($goodsRunningOffJson['time']) ? date("Y-m-d H:i", $goodsRunningOffJson['time']) : '';
  542. }
  543. } else {
  544. $running = [];
  545. }
  546. // 组件表
  547. $component = GoodsComponent::where('join_component_master_goods_id', $goodsId)->get()->toArray();
  548. $componentArr['join_component_goods_id'] = [];
  549. $componentArr['goodsContentList'] = [];
  550. if ($component) {
  551. $ids = [];
  552. $contentList = [];
  553. foreach ($component as $item) {
  554. if (empty($item['join_component_goods_id'])) {
  555. continue;
  556. }
  557. $skus = GoodsSku::where('join_sku_goods_id', $item['join_component_goods_id'])
  558. ->select('goods_sku_id', 'join_sku_goods_id', 'goods_sku_specs_json', 'goods_sku_sales_price')
  559. ->get()
  560. ->toArray();
  561. foreach ($skus as $key => $sku2) {
  562. if (!empty($sku2['goods_sku_specs_json'])) {
  563. $skus[$key]['goods_sku_specs_json'] = json_decode($sku2['goods_sku_specs_json'], true);
  564. $skuName = '';
  565. foreach ($skus[$key]['goods_sku_specs_json'] as $specsKey => $skuSpecs) {
  566. if (is_array($skuSpecs)) {
  567. $skuName = $skuName . ' ' . $specsKey . ' ' . implode(' ', $skuSpecs);
  568. } else {
  569. $skuName = $skuName . ' ' . $specsKey . ' ' . $skuSpecs;
  570. }
  571. }
  572. $skus[$key]['sku_name'] = $skuName;
  573. }
  574. }
  575. $ids[] = $item['join_component_goods_id'];
  576. $configJson = json_decode($item['goods_component_json'], true);
  577. $contentList[] = [
  578. 'goods_id' => $item['join_component_goods_id'],
  579. 'goods_name' => $configJson['goods_name'] ?? '',
  580. 'goods_sales_price' => $item['goods_component_price'],
  581. 'nbr' => $configJson['nbr'] ?? 0,
  582. 'sku_id' => $configJson['sku_id'] ?? '',
  583. 'skus' => $skus
  584. ];
  585. }
  586. $componentArr['join_component_goods_id'] = $ids;
  587. $componentArr['goodsContentList'] = $contentList;
  588. }
  589. // 合并数据
  590. $data = array_merge($main, $detail, $label, $running, $componentArr);
  591. $data['goods_sku_json_label'] = [];
  592. $data['goods_label'] = !empty($data['goods_label']) ? explode(',', $data['goods_label']) : [];
  593. $data['goods_cover'] = getenv('STORAGE_DOMAIN') . $data['goods_cover'];
  594. if (!empty($data['goods_detail_slider_json'])) {
  595. $data['goods_detail_slider_json'] = json_decode($data['goods_detail_slider_json'], true);
  596. $slider = '';
  597. foreach ($data['goods_detail_slider_json'] as $item) {
  598. $slider .= getenv('STORAGE_DOMAIN') . $item . ',';
  599. }
  600. $data['goods_detail_slider_json'] = rtrim($slider, ',');
  601. }
  602. $data['goods_on_addtimes'] = date('Y-m-d\TH:i:s.u\Z', $data['goods_on_addtimes'] - 60 * 60 * 8);
  603. return json_success('', $data);
  604. } catch (\Exception $e) {
  605. dump($e->getTrace());
  606. return json_fail("查询错误~");
  607. }
  608. }
  609. /**
  610. * @Desc 添加商品
  611. * @Author Gorden
  612. * @Date 2024/3/11 10:20
  613. *
  614. * @param $params
  615. * @return Response
  616. */
  617. public static function insert($params): Response
  618. {
  619. Db::beginTransaction();
  620. try {
  621. $params['goods_id'] = "GD" . sprintf('%016d', SysSerial::getSerial()) . random_string(6, 'up');
  622. // 主表
  623. self::mainInsert($params);
  624. // 商品详情表
  625. self::detailInsert($params);
  626. // 商品标签表
  627. self::labelInsert($params);
  628. // 产品运行控制信息表
  629. self::goodsRunningInsert($params);
  630. // sku表
  631. self::goodsSkuSet($params, 'insert');
  632. // 待上架状态,上架时间大于当前时间
  633. if ($params['goods_status'] == 'PENDING' && strtotime($params['goods_on_addtimes']) > time()) {
  634. $redis = Redis::connection();
  635. $key = date('YmdHi', strtotime($params['goods_on_addtimes']));
  636. $redis->sAdd(Goods::LISTING_KEY_PREFIX . $key, $params['goods_id']);
  637. }
  638. // 自动下架
  639. if (!empty($params['goods_running_off_type']) && $params['goods_running_off_type'] == 'T' && !empty($params['goods_off_addtimes'])) {
  640. $redis = Redis::connection();
  641. $key = Goods::LISTING_OFF_KEY_PREFIX . date('YmdHi', strtotime($params['goods_off_addtimes']));
  642. $redis->sAdd($key, $params['goods_id']);
  643. }
  644. Db::commit();
  645. } catch (\PDOException $e) {
  646. Db::rollBack();
  647. dump($e->getMessage());
  648. return json_fail('数据写入失败~');
  649. } catch (BusinessException $e) {
  650. Db::rollBack();
  651. dump($e->getMessage());
  652. return json_fail($e->getMessage());
  653. } catch (\Exception $e) {
  654. Db::rollBack();
  655. dump($e->getTrace());
  656. return json_fail('数据写入失败~');
  657. }
  658. _syslog("添加商品", "商品名【" . $params['goods_name'] . "】");
  659. return json_success('success');
  660. }
  661. public static function insertPackage($params): Response
  662. {
  663. Db::beginTransaction();
  664. try {
  665. $params['goods_id'] = "GD" . sprintf('%016d', SysSerial::getSerial()) . random_string(6, 'up');
  666. // 主表
  667. self::mainInsert($params);
  668. // 商品详情表
  669. self::detailInsert($params);
  670. // 套包组件表
  671. self::componentUpdate($params, 'insert');
  672. // 商品标签表
  673. self::labelInsert($params);
  674. // 产品运行控制信息表
  675. self::goodsRunningInsert($params);
  676. // sku表
  677. self::goodsSkuSet($params, 'insert');
  678. // 待上架状态,上架时间大于当前时间
  679. if ($params['goods_status'] == 'PENDING' && strtotime($params['goods_on_addtimes']) > time()) {
  680. $redis = Redis::connection();
  681. $key = date('YmdHi', strtotime($params['goods_on_addtimes']));
  682. $redis->sAdd(Goods::LISTING_KEY_PREFIX . $key, $params['goods_id']);
  683. }
  684. Db::commit();
  685. } catch (\PDOException $e) {
  686. Db::rollBack();
  687. dump($e->getMessage());
  688. return json_fail('数据写入失败~');
  689. } catch (BusinessException $e) {
  690. Db::rollBack();
  691. dump($e->getMessage());
  692. return json_fail($e->getMessage());
  693. } catch (\Exception $e) {
  694. Db::rollBack();
  695. dump($e->getTrace());
  696. return json_fail('数据写入失败~');
  697. }
  698. _syslog("添加套餐", "商品名【" . $params['goods_name'] . "】");
  699. return json_success('success');
  700. }
  701. public static function update($params)
  702. {
  703. Db::beginTransaction();
  704. try {
  705. // 主表
  706. self::mainUpdate($params);
  707. // 商品详情表
  708. self::detailUpdate($params);
  709. // 商品标签表
  710. self::labelUpdate($params);
  711. // 产品运行控制信息表
  712. self::goodsRunningUpdate($params);
  713. // sku表
  714. self::goodsSkuSet($params, 'update');
  715. Db::commit();
  716. } catch (BusinessException $e) {
  717. Db::rollBack();
  718. return json_fail($e->getMessage());
  719. } catch (\Exception $e) {
  720. Db::rollBack();
  721. return json_fail('数据更新失败~');
  722. }
  723. _syslog("编辑商品", "商品名【" . $params['goods_name'] . "】" ?? "商品ID:【" . $params['goods_id'] . "】");
  724. return json_success('success');
  725. }
  726. public static function changeStatus($params)
  727. {
  728. try {
  729. Goods::where('goods_id', $params['goods_id'])->update(['goods_status' => $params['goods_status']]);
  730. return json_success('修改成功');
  731. } catch (\Exception $e) {
  732. return json_fail('修改状态失败');
  733. }
  734. }
  735. public static function updatePackage($params)
  736. {
  737. Db::beginTransaction();
  738. try {
  739. // 主表
  740. self::mainUpdate($params);
  741. // 商品详情表
  742. self::detailUpdate($params);
  743. // 套包组件表
  744. self::componentUpdate($params, 'update');
  745. // 商品标签表
  746. self::labelUpdate($params);
  747. // 产品运行控制信息表
  748. self::goodsRunningUpdate($params);
  749. Db::commit();
  750. } catch (BusinessException $e) {
  751. Db::rollBack();
  752. return json_fail($e->getMessage());
  753. } catch (\Exception $e) {
  754. Db::rollBack();
  755. dump($e->getTrace());
  756. return json_fail('数据更新失败~');
  757. }
  758. _syslog("编辑套餐", "商品名【" . $params['goods_name'] . "】" ?? "商品ID:【" . $params['goods_id'] . "】");
  759. return json_success('success');
  760. }
  761. /**
  762. * @Desc 删除商品
  763. * @Author Gorden
  764. * @Date 2024/3/28 13:20
  765. *
  766. * @param $ids
  767. * @return Response
  768. */
  769. public static function delete($ids)
  770. {
  771. if (!$ids) {
  772. return json_fail("数据错误~");
  773. }
  774. if (!is_array($ids)) {
  775. $ids = [$ids];
  776. }
  777. $goods = Goods::whereIn('goods_id', $ids)->get()->toArray();
  778. if (!$goods) {
  779. return json_fail("数据错误~");
  780. }
  781. Db::beginTransaction();
  782. try {
  783. Goods::whereIn('goods_id', $ids)->delete();
  784. GoodsDetail::whereIn('join_detail_goods_id', $ids)->delete();
  785. GoodsLabel::whereIn('join_label_goods_id', $ids)->delete();
  786. GoodsRunning::whereIn('join_running_goods_id', $ids)->delete();
  787. GoodsSku::whereIn('join_sku_goods_id', $ids)->delete();
  788. Db::commit();
  789. _syslog("删除商品 / 套餐", "ID:【" . implode(',', $ids) . "】", $goods);
  790. return json_success("商品删除成功");
  791. } catch (\Exception $e) {
  792. Db::rollBack();
  793. return json_fail("商品删除失败~");
  794. }
  795. }
  796. /**
  797. * @Desc 商品主表
  798. * @Author Gorden
  799. * @Date 2024/3/11 11:20
  800. *
  801. * @param $params
  802. * @return mixed|string
  803. * @throws BusinessException
  804. */
  805. public static function mainInsert($params)
  806. {
  807. if (!empty($params['goods_cover'])) {
  808. $params['goods_cover'] = str_replace(getenv('STORAGE_DOMAIN'), '', $params['goods_cover']);
  809. }
  810. // 如果产品是待处理状态
  811. if ($params['goods_status'] == 'PENDING') {
  812. if (strtotime($params['goods_on_addtimes']) <= time()) {
  813. $params['goods_status'] = 'ON';
  814. }
  815. }
  816. $category = SysCategory::where('category_id', $params['join_goods_category_id'])->first();
  817. if (!$category) {
  818. throw new BusinessException("产品分类不存在~");
  819. }
  820. if (empty($params['goods_category'])) {
  821. $params['goods_category'] = $category->category_classify;
  822. }
  823. try {
  824. $model = new Goods();
  825. $model->goods_id = $params['goods_id'];
  826. $model->join_goods_category_id = $params['join_goods_category_id'] ?? 0;
  827. $model->join_goods_supplier_id = $params['join_goods_supplier_id'] ?? 0;
  828. $model->goods_classify = $params['goods_classify'] ?? '';
  829. $model->goods_status = $params['goods_status'] ?? '';
  830. $model->goods_category = $params['goods_category'] ?? '';
  831. $model->goods_prefix = $params['goods_prefix'] ?? ($category->category_name ? '【' . $category->category_name . '】' : '');
  832. $model->goods_name = $params['goods_name'];
  833. $model->goods_market_price = $params['goods_market_price'] ?? 0;
  834. $model->goods_sales_price = $params['goods_sales_price'] ?? 0;
  835. $model->goods_sku_json = !empty($params['goods_sku_json_label']) ? json_encode($params['goods_sku_json_label']) : json_encode(['规格' => []]);
  836. $model->goods_attribute_json = !empty($params['goods_attribute_json']) ? $params['goods_attribute_json'] : '{}';
  837. // $model->goods_service_json = !empty($params['goods_service_json']) ? $params['goods_service_json'] : '{}';
  838. $model->goods_title = $params['goods_title'] ?? '';
  839. $model->goods_cover = $params['goods_cover'] ?? '';
  840. $model->goods_on_addtimes = isset($params['goods_on_addtimes']) ? strtotime($params['goods_on_addtimes']) : null;
  841. $model->goods_sort = $params['goods_sort'] ?? null;
  842. $model->goods_groupby = $params['goods_groupby'] ?? '';
  843. $model->goods_remark = $params['goods_remark'] ?? '';
  844. $model->goods_extend_json = $params['goods_extend_json'] ?? '{}';
  845. $model->is_support_appointment = $params['is_support_appointment'] ?? 'N';
  846. $model->goods_addtimes = time();
  847. if (!empty($params['is_support_appointment']) && $params['is_support_appointment'] == 'Y' && !empty($params['appointment_times'])) {
  848. $times = [];
  849. $attributeJsonTimeArr = [];
  850. $personTotal = 0;
  851. foreach ($params['appointment_times'] as $time) {
  852. $attributeJsonTimeArr[] = strtotime(date('Y-m-d ') . $time['appointmentTimeStart']);
  853. $attributeJsonTimeArr[] = strtotime(date('Y-m-d ') . $time['appointmentTimeEnd']);
  854. $personTotal += $time['person'];
  855. $times[$time['appointmentTimeStart']] = [
  856. 'person' => $time['person'],
  857. 'duration' => $time['appointmentTimeStart'] . '-' . $time['appointmentTimeEnd']
  858. ];
  859. }
  860. $attributeJsonTime = date('H:i', min($attributeJsonTimeArr)) . '至' . date('H:i', max($attributeJsonTimeArr));
  861. $newDates = [];
  862. foreach ($params['dates'] as $date) {
  863. $key = self::$week[$date];
  864. $newDates[$key] = $date;
  865. }
  866. ksort($newDates);
  867. $currentDate = current($newDates);
  868. $lastDate = end($newDates);
  869. $attributeJsonDate = '';
  870. if (self::$week[$lastDate] - self::$week[$currentDate] + 1 > count($newDates)) {
  871. $attributeJsonDate = implode(',', $newDates);
  872. } else if (self::$week[$lastDate] - self::$week[$currentDate] + 1 == count($newDates)) {
  873. $attributeJsonDate = $currentDate . '至' . $lastDate;
  874. }
  875. $attributeJson = [
  876. 'icon' => '',
  877. 'date' => $attributeJsonDate,
  878. 'time' => $attributeJsonTime,
  879. 'dates' => $newDates ? array_values($newDates) : [],
  880. 'times' => $times,
  881. 'person' => $personTotal
  882. ];
  883. if (!empty($params['appointment_label'])) {
  884. $attributeJson['label'] = $params['appointment_label'];
  885. }
  886. if (!empty($params['address'])) {
  887. $attributeJson['address'] = $params['address'];
  888. $attributeJson['position'] = $params['position'];
  889. }
  890. $model->goods_attribute_json = json_encode($attributeJson, JSON_UNESCAPED_UNICODE);
  891. // $times = [];
  892. // foreach ($params['appointment_times'] as $time) {
  893. // $times[$time['appointmentTimeStart']] = [
  894. // 'person' => $time['person'],
  895. // 'duration' => $time['appointmentTimeStart'] . '-' . $time['appointmentTimeEnd']
  896. // ];
  897. // }
  898. // $model->goods_attribute_json = json_encode([
  899. // 'icon' => '',
  900. // 'dates' => $params['dates'] ?? [],
  901. // 'times' => $times
  902. // ]);
  903. }
  904. if (!empty($params['goods_json']) && $params['join_goods_category_id'] == 65) {
  905. $goodsJson = json_decode($params['goods_json'], true);
  906. foreach ($goodsJson as $key => $item) {
  907. $goodsJson[$key]['color'] = hexToRgb($item['color']);
  908. }
  909. $model->goods_json = json_encode($goodsJson);
  910. } elseif (!empty($params['goods_json']) && $params['join_goods_category_id'] == 43) {
  911. $goodsJson = json_decode($params['goods_json'], true);
  912. $newGoodsJson = [];
  913. foreach ($goodsJson as $item1) {
  914. if (empty($item1['title'])) {
  915. continue;
  916. }
  917. $newItem1 = [];
  918. foreach ($item1['items'] as $item2) {
  919. $newParams = [];
  920. foreach ($item2['params'] as $param) {
  921. if (!empty($param[0]) || !empty($param[1])) {
  922. $newParams[] = $param;
  923. }
  924. }
  925. $newItem1['items'][$item2['key']] = $newParams;
  926. }
  927. $newItem1['service'] = $item1['service'] ?? '';
  928. $newGoodsJson[$item1['title']] = $newItem1;
  929. }
  930. $model->goods_json = json_encode($newGoodsJson);
  931. } else {
  932. $model->goods_json = '[]';
  933. }
  934. if ($model->save()) {
  935. return $model->goods_id;
  936. }
  937. // 异常
  938. throw new BusinessException("数据写入失败~");
  939. } catch (\Exception $e) {
  940. dump($e->getMessage());
  941. throw new BusinessException("数据写入失败~");
  942. }
  943. }
  944. /**
  945. * @Desc 详情表
  946. * @Author Gorden
  947. * @Date 2024/3/11 11:19
  948. *
  949. * @param $params
  950. * @return void
  951. * @throws BusinessException
  952. */
  953. public static function detailInsert($params)
  954. {
  955. if (!empty($params['goods_detail_slider_json'])) {
  956. $params['goods_detail_slider_json'] = str_replace(getenv('STORAGE_DOMAIN'), '', $params['goods_detail_slider_json']);
  957. $params['goods_detail_slider_json'] = json_encode(['slider' => $params['goods_detail_slider_json']]);
  958. }
  959. try {
  960. $model = new GoodsDetail();
  961. $model->join_detail_goods_id = $params['goods_id'];
  962. $model->goods_detail_code_json = $params['goods_detail_code_json'] ?? '{}';
  963. $model->goods_detail_slider_json = $params['goods_detail_slider_json'] ?? '{}';
  964. $model->goods_detail_specs_json = $params['goods_detail_specs_json'] ?? '{}';
  965. $model->goods_detail_content = $params['goods_detail_content'] ?? '';
  966. if (!$model->save()) {
  967. // 异常
  968. throw new BusinessException("数据写入失败~");
  969. }
  970. } catch (\Exception $e) {
  971. dump($e->getMessage());
  972. throw new BusinessException("数据写入失败~");
  973. }
  974. }
  975. public static function componentUpdate($params, $type = 'insert')
  976. {
  977. Db::beginTransaction();
  978. try {
  979. // 有先删除
  980. if ($type == 'update') {
  981. GoodsComponent::where('join_component_master_goods_id', $params['goods_id'])->delete();
  982. } else {
  983. Goods::where('goods_id', $params['goods_id'])->update(['goods_sku_json' => '{"规格": ["标准"]}']);
  984. $skuData = [
  985. 'join_sku_goods_id' => $params['goods_id'],
  986. 'goods_sku_status' => 'ON',
  987. 'goods_sku_specs_json' => '{"规格": "标准"}',
  988. 'goods_sku_title' => "标准" . $params['goods_name'],
  989. 'goods_sku_market_price' => $params['goods_market_price'] ?? 0,
  990. 'goods_sku_sales_price' => $params['goods_sales_price'] ?? 0,
  991. ];
  992. GoodsSku::insert($skuData);
  993. }
  994. $data = [];
  995. foreach ($params['goods_content_list'] as $item) {
  996. $goods = Goods::where('goods_id', $params['goods_id'])->first();
  997. if (!$goods) {
  998. continue;
  999. }
  1000. $data[] = [
  1001. 'join_component_master_goods_id' => $params['goods_id'],
  1002. 'join_component_goods_id' => $item['goods_id'] ?? '',
  1003. 'goods_component_price' => $item['goods_sales_price'] ?? '',
  1004. 'goods_component_cover' => $goods->goods_cover,
  1005. 'goods_component_json' => json_encode(['goods_name' => $item['goods_name'], 'nbr' => $item['nbr'], 'sku_id' => $item['sku_id'] ?? '']),
  1006. 'goods_component_addtimes' => time()
  1007. ];
  1008. }
  1009. if ($data) {
  1010. GoodsComponent::insert($data);
  1011. }
  1012. Db::commit();
  1013. } catch (\Exception $e) {
  1014. Db::rollBack();
  1015. dump($e->getMessage());
  1016. throw new BusinessException("数据写入失败~");
  1017. }
  1018. }
  1019. /**
  1020. * @Desc 标签表
  1021. * @Author Gorden
  1022. * @Date 2024/3/11 11:32
  1023. *
  1024. * @param $params
  1025. * @return void
  1026. * @throws BusinessException
  1027. */
  1028. public static function labelInsert($params)
  1029. {
  1030. $model = new GoodsLabel();
  1031. $model->join_label_goods_id = $params['goods_id'];
  1032. $model->goods_label = $params['goods_label'] ? implode(',', $params['goods_label']) : '';
  1033. $model->goods_label_extend_json = !empty($params['goods_label_extend_json']) ? $params['goods_label_extend_json'] : '{}';
  1034. if (!$model->save()) {
  1035. // 异常
  1036. throw new BusinessException('数据写入失败~');
  1037. }
  1038. }
  1039. /**
  1040. * @Desc 产品运行控制信息表
  1041. * @Author Gorden
  1042. * @Date 2024/3/11 11:38
  1043. *
  1044. * @param $params
  1045. * @return void
  1046. * @throws BusinessException
  1047. */
  1048. public static function goodsRunningInsert($params)
  1049. {
  1050. try {
  1051. $model = new GoodsRunning();
  1052. $model->join_running_goods_id = $params['goods_id'];
  1053. $model->goods_running_storage = $params['goods_running_storage'] ?? 0;
  1054. $model->goods_running_sale = $params['goods_running_sale'] ?? 0;
  1055. $model->goods_running_off_type = !empty($params['goods_running_off_type']) ? $params['goods_running_off_type'] : '';
  1056. $model->goods_running_off_json = !empty($params['goods_running_off_type']) && $params['goods_running_off_type'] == 'T' && !empty($params['goods_off_addtimes']) ? json_encode(['time' => strtotime($params['goods_off_addtimes'])]) : '[]';
  1057. if (!$model->save()) {
  1058. throw new BusinessException('数据写入失败');
  1059. }
  1060. } catch (\Exception $e) {
  1061. dump($e->getMessage());
  1062. throw new BusinessException('数据写入失败');
  1063. }
  1064. }
  1065. /**
  1066. * @Desc SKU
  1067. * @Author Gorden
  1068. * @Date 2024/3/11 12:01
  1069. *
  1070. * @param $params
  1071. * @return void
  1072. * @throws BusinessException
  1073. */
  1074. public static function skuInsert($params)
  1075. {
  1076. $model = new GoodsSku();
  1077. $model->join_sku_goods_id = $params['goods_id'];
  1078. $model->goods_sku_status = $params['goods_sku_status'];
  1079. $model->goods_sku_specs_json = $params['goods_sku_specs_json'];
  1080. $model->goods_sku_title = $params['goods_sku_title'];
  1081. $model->goods_sku_images_json = $params['goods_sku_images_json'];
  1082. $model->goods_sku_content = $params['goods_sku_content'];
  1083. $model->goods_sku_market_price = $params['goods_sku_market_price'];
  1084. $model->goods_sku_sales_price = $params['goods_sku_sales_price'];
  1085. $model->goods_sku_storage_json = $params['goods_sku_storage_json'];
  1086. $model->goods_sku_service_json = $params['goods_sku_service_json'];
  1087. $model->goods_sku_extend_json = $params['goods_sku_extend_json'];
  1088. if (!$model->save()) {
  1089. throw new BusinessException('数据写入失败~');
  1090. }
  1091. }
  1092. /**
  1093. * @Desc
  1094. * @Author Gorden
  1095. * @Date 2024/3/12 8:44
  1096. *
  1097. * @param $params
  1098. * @return void
  1099. * @throws BusinessException
  1100. */
  1101. public static function mainUpdate($params)
  1102. {
  1103. try {
  1104. $data = self::inputFilter($params, new Goods());
  1105. if (!empty($data['goods_cover'])) {
  1106. $data['goods_cover'] = str_replace(getenv('STORAGE_DOMAIN'), '', $data['goods_cover']);
  1107. }
  1108. $data['goods_on_addtimes'] = isset($data['goods_on_addtimes']) ? strtotime($data['goods_on_addtimes']) : 0;
  1109. $data['goods_sku_json'] = !empty($params['goods_sku_json_label']) ? json_encode($params['goods_sku_json_label']) : json_encode(['规格' => []]);
  1110. $row = Goods::find($data['goods_id']);
  1111. if ($row->join_goods_category_id != $data['join_goods_category_id']) {
  1112. $category = SysCategory::where('category_id', $params['join_goods_category_id'])->first();
  1113. if (!$category) {
  1114. throw new BusinessException("产品分类不存在~");
  1115. }
  1116. $data['goods_category'] = $category->category_classify ?? '';
  1117. $data['goods_prefix'] = $data['goods_prefix'] ?? ($category->category_name ? '【' . $category->category_name . '】' : '');
  1118. }
  1119. // 上架时间有变动
  1120. if ($data['goods_status'] == 'PENDING' && $row->goods_on_addtimes != $data['goods_on_addtimes']) {
  1121. $redis = Redis::connection();
  1122. // 删掉原来的
  1123. $oldKey = Goods::LISTING_KEY_PREFIX . date('YmdHi', $row->goods_on_addtimes);
  1124. $redis->srem($oldKey, $data['goods_id']);
  1125. // 加入新的
  1126. $newKey = Goods::LISTING_KEY_PREFIX . date('YmdHi', $data['goods_on_addtimes']);
  1127. $redis->sadd($newKey, $data['goods_id']);
  1128. }
  1129. if (!empty($params['is_support_appointment']) && $params['is_support_appointment'] == 'Y' && !empty($params['appointment_times'])) {
  1130. $times = [];
  1131. $attributeJsonTimeArr = [];
  1132. $personTotal = 0;
  1133. foreach ($params['appointment_times'] as $time) {
  1134. $attributeJsonTimeArr[] = strtotime(date('Y-m-d ') . $time['appointmentTimeStart']);
  1135. $attributeJsonTimeArr[] = strtotime(date('Y-m-d ') . $time['appointmentTimeEnd']);
  1136. $personTotal += $time['person'];
  1137. $times[$time['appointmentTimeStart']] = [
  1138. 'person' => $time['person'],
  1139. 'duration' => $time['appointmentTimeStart'] . '-' . $time['appointmentTimeEnd']
  1140. ];
  1141. }
  1142. $attributeJsonTime = date('H:i', min($attributeJsonTimeArr)) . '至' . date('H:i', max($attributeJsonTimeArr));
  1143. $newDates = [];
  1144. foreach ($params['dates'] as $date) {
  1145. $key = self::$week[$date];
  1146. $newDates[$key] = $date;
  1147. }
  1148. ksort($newDates);
  1149. $currentDate = current($newDates);
  1150. $lastDate = end($newDates);
  1151. $attributeJsonDate = '';
  1152. if (self::$week[$lastDate] - self::$week[$currentDate] + 1 > count($newDates)) {
  1153. $attributeJsonDate = implode(',', $newDates);
  1154. } else if (self::$week[$lastDate] - self::$week[$currentDate] + 1 == count($newDates)) {
  1155. $attributeJsonDate = $currentDate . '至' . $lastDate;
  1156. }
  1157. $attributeJson = [
  1158. 'icon' => '',
  1159. 'date' => $attributeJsonDate,
  1160. 'time' => $attributeJsonTime,
  1161. 'dates' => $newDates ? array_values($newDates) : [],
  1162. 'times' => $times,
  1163. 'person' => $personTotal
  1164. ];
  1165. if (!empty($params['appointment_label'])) {
  1166. $attributeJson['label'] = $params['appointment_label'];
  1167. }
  1168. if (!empty($params['address'])) {
  1169. $attributeJson['address'] = $params['address'];
  1170. $attributeJson['position'] = $params['position'];
  1171. }
  1172. $data['goods_attribute_json'] = json_encode($attributeJson, JSON_UNESCAPED_UNICODE);
  1173. }
  1174. if (!empty($data['goods_json']) && $data['join_goods_category_id'] == 65) {
  1175. $goodsJson = json_decode($data['goods_json'], true);
  1176. foreach ($goodsJson as $key => $item) {
  1177. $goodsJson[$key]['color'] = hexToRgb($item['color']);
  1178. }
  1179. $data['goods_json'] = json_encode($goodsJson);
  1180. } elseif (!empty($data['goods_json']) && $data['join_goods_category_id'] == 43) {
  1181. $goodsJson = json_decode($data['goods_json'], true);
  1182. $newGoodsJson = [];
  1183. foreach ($goodsJson as $item1) {
  1184. if (empty($item1['title'])) {
  1185. continue;
  1186. }
  1187. $newItem1 = [];
  1188. foreach ($item1['items'] as $item2) {
  1189. $newParams = [];
  1190. foreach ($item2['params'] as $param) {
  1191. if (!empty($param[0]) || !empty($param[1])) {
  1192. $newParams[] = $param;
  1193. }
  1194. }
  1195. $newItem1['items'][$item2['key']] = $newParams;
  1196. }
  1197. $newItem1['service'] = $item1['service'];
  1198. $newGoodsJson[$item1['title']] = $newItem1;
  1199. }
  1200. $data['goods_json'] = json_encode($newGoodsJson);
  1201. } else {
  1202. $data['goods_json'] = '[]';
  1203. }
  1204. foreach ($data as $key => $val) {
  1205. $row->{$key} = $val;
  1206. }
  1207. $row->save();
  1208. } catch (BusinessException $e) {
  1209. throw new BusinessException($e->getMessage());
  1210. } catch (\Exception $e) {
  1211. dump($e->getTrace());
  1212. throw new BusinessException('数据更新异常~1');
  1213. }
  1214. }
  1215. /**
  1216. * @Desc
  1217. * @Author Gorden
  1218. * @Date 2024/3/12 9:57
  1219. *
  1220. * @param $params
  1221. * @return void
  1222. * @throws BusinessException
  1223. */
  1224. public static function detailUpdate($params)
  1225. {
  1226. try {
  1227. $data = self::inputFilter($params, new GoodsDetail());
  1228. if (!empty($data['goods_detail_slider_json'])) {
  1229. $data['goods_detail_slider_json'] = str_replace(getenv('STORAGE_DOMAIN'), '', $data['goods_detail_slider_json']);
  1230. $data['goods_detail_slider_json'] = json_encode(['slider' => $data['goods_detail_slider_json']]);
  1231. }
  1232. // 根据goods_id 查详情ID
  1233. $detail = GoodsDetail::where('join_detail_goods_id', $params['goods_id'])->first();
  1234. if ($detail) {
  1235. self::doUpdate($detail->join_detail_goods_id, $data, new GoodsDetail());
  1236. } else {
  1237. $data['join_detail_goods_id'] = $params['goods_id'];
  1238. GoodsDetail::insert($data);
  1239. }
  1240. } catch (BusinessException $e) {
  1241. throw new BusinessException($e->getMessage());
  1242. } catch (\Exception $e) {
  1243. dump($e->getMessage());
  1244. throw new BusinessException('数据更新异常~2');
  1245. }
  1246. }
  1247. /**
  1248. * @Desc
  1249. * @Author Gorden
  1250. * @Date 2024/3/12 9:58
  1251. *
  1252. * @param $params
  1253. * @return void
  1254. * @throws BusinessException
  1255. */
  1256. public static function labelUpdate($params)
  1257. {
  1258. try {
  1259. $data = self::inputFilter($params, new GoodsLabel());
  1260. // 根据goods_id 查详情ID
  1261. $detail = GoodsLabel::where('join_label_goods_id', $params['goods_id'])->first();
  1262. if ($detail) {
  1263. self::doUpdate($detail->goods_label_id, $data, new GoodsLabel());
  1264. } else {
  1265. $data['join_label_goods_id'] = $params['goods_id'];
  1266. GoodsLabel::insert($data);
  1267. }
  1268. } catch (BusinessException $e) {
  1269. throw new BusinessException($e->getMessage());
  1270. } catch (\Exception $e) {
  1271. dump($e->getMessage());
  1272. throw new BusinessException('数据更新异常~3');
  1273. }
  1274. }
  1275. /**
  1276. * @Desc
  1277. * @Author Gorden
  1278. * @Date 2024/3/12 9:59
  1279. *
  1280. * @param $params
  1281. * @return void
  1282. * @throws BusinessException
  1283. */
  1284. public static function goodsRunningUpdate($params)
  1285. {
  1286. try {
  1287. $data = self::inputFilter($params, new GoodsRunning());
  1288. // 根据goods_id 查详情ID
  1289. $detail = GoodsRunning::where('join_running_goods_id', $params['goods_id'])->first();
  1290. if (!empty($params['goods_running_off_type']) && $params['goods_running_off_type'] == 'T') {
  1291. $redis = Redis::connection();
  1292. if (!empty($detail->goods_running_off_json)) {
  1293. $goodsRunningOffJson = json_decode($detail->goods_running_off_json, true);
  1294. if (isset($goodsRunningOffJson['time'])) {
  1295. $oldKey = Goods::LISTING_OFF_KEY_PREFIX . date('YmdHi', $goodsRunningOffJson['time']);
  1296. $goodsRunningOffJson['time'] = strtotime($params['goods_off_addtimes']);
  1297. $data['goods_running_off_json'] = json_encode($goodsRunningOffJson);
  1298. // 有老的下架时间,删除老的
  1299. $redis->srem($oldKey, $params['goods_id']);
  1300. } else {
  1301. $goodsRunningOffJson['time'] = strtotime($params['goods_off_addtimes']);
  1302. $data['goods_running_off_json'] = json_encode($goodsRunningOffJson);
  1303. }
  1304. } else {
  1305. $data['goods_running_off_json'] = json_encode(['time' => strtotime($params['goods_off_addtimes'])]);
  1306. }
  1307. // 加入自动下架
  1308. $newKey = Goods::LISTING_OFF_KEY_PREFIX . date('YmdHi', strtotime($params['goods_off_addtimes']));
  1309. $redis->sAdd($newKey, $params['goods_id']);
  1310. } else if (!empty($params['goods_running_off_type']) && !empty($detail->goods_running_off_type) && $params['goods_running_off_type'] == 'H' && $detail->goods_running_off_type == 'T') {
  1311. $goodsRunningOffJson = json_decode($detail->goods_running_off_json, true);
  1312. if (isset($goodsRunningOffJson['time'])) {
  1313. $oldKey = Goods::LISTING_OFF_KEY_PREFIX . date('YmdHi', $goodsRunningOffJson['time']);
  1314. $redis = Redis::connection();
  1315. $redis->srem($oldKey, $params['goods_id']);
  1316. }
  1317. }
  1318. if ($detail) {
  1319. self::doUpdate($detail->join_running_goods_id, $data, new GoodsRunning());
  1320. } else {
  1321. // 兼容老数据……
  1322. $data['join_running_goods_id'] = $params['goods_id'];
  1323. GoodsRunning::insert($data);
  1324. }
  1325. } catch (BusinessException $e) {
  1326. throw new BusinessException($e->getMessage());
  1327. } catch (\Exception $e) {
  1328. dump($e->getMessage());
  1329. throw new BusinessException('数据更新异常~4');
  1330. }
  1331. }
  1332. /**
  1333. * @Desc sku 设置
  1334. * @Author Gorden
  1335. * @Date 2024/4/10 10:43
  1336. *
  1337. * @param $params
  1338. * @return void
  1339. * @throws BusinessException
  1340. */
  1341. public static function goodsSkuSet($params, $operation = 'insert')
  1342. {
  1343. try {
  1344. Db::beginTransaction();
  1345. $skusOldIds = [];
  1346. if ($operation == 'update') {
  1347. // 查出所有的
  1348. $skusOldIds = GoodsSku::where('join_sku_goods_id', $params['goods_id'])->pluck('goods_sku_id', 'goods_sku_id');
  1349. // 删掉原有的
  1350. // GoodsSku::where('join_sku_goods_id', $params['goods_id'])->delete();
  1351. }
  1352. if (empty($skusOldIds) && empty($params['goods_sku_json_value'])) {
  1353. $skuData = [
  1354. 'join_sku_goods_id' => $params['goods_id'],
  1355. 'goods_sku_status' => 'ON',
  1356. 'goods_sku_specs_json' => '{"规格": "标准"}',
  1357. 'goods_sku_title' => "标准" . $params['goods_name'],
  1358. 'goods_sku_market_price' => $params['goods_market_price'] ?? 0,
  1359. 'goods_sku_sales_price' => $params['goods_sales_price'] ?? 0,
  1360. ];
  1361. GoodsSku::insert($skuData);
  1362. }
  1363. // 入新的
  1364. if (!empty($params['goods_sku_json_value'])) {
  1365. foreach ($params['goods_sku_json_value'] as $item) {
  1366. $skus = explode(',', $item['sku']);
  1367. $skuArr = [];
  1368. for ($i = 1; $i <= count($skus); $i++) {
  1369. $skuName = "skuName" . $i;
  1370. $key = $item[$skuName];
  1371. $skuArr[$key] = $skus[$i - 1];
  1372. }
  1373. $specsJson = json_encode($skuArr);
  1374. $skuTitle = str_replace('-', ',', $item['sku']) . $params['goods_name'];
  1375. if ($operation == 'update' && !empty($item['sku_id'])) {
  1376. $model = GoodsSku::where('goods_sku_id', $item['sku_id'])->first();
  1377. if (!$model) {
  1378. $model = new GoodsSku();
  1379. } else {
  1380. unset($skusOldIds[$model->goods_sku_id]);
  1381. }
  1382. } else {
  1383. $model = GoodsSku::where('join_sku_goods_id', $params['goods_id'])->where('goods_sku_title', $skuTitle)->first();
  1384. if (!$model) {
  1385. $model = new GoodsSku();
  1386. } else {
  1387. unset($skusOldIds[$model->goods_sku_id]);
  1388. }
  1389. }
  1390. $model->join_sku_goods_id = $params['goods_id'];
  1391. $model->goods_sku_status = $params['goods_status'];
  1392. $model->goods_sku_specs_json = $specsJson;
  1393. $model->goods_sku_title = $skuTitle;
  1394. $model->goods_sku_market_price = $params['goods_market_price'] ?? 0;
  1395. $model->goods_sku_sales_price = $item['price'];
  1396. $model->goods_sku_storage_json = json_encode(['storage' => $item['stock']]);
  1397. $model->save();
  1398. }
  1399. }
  1400. if ($operation == 'update' && !empty($skusOldIds)) {
  1401. GoodsSku::whereIn('goods_sku_id', $skusOldIds)->delete();
  1402. }
  1403. Db::commit();
  1404. } catch (\Exception $e) {
  1405. dump($e->getTrace());
  1406. Db::rollBack();
  1407. throw new BusinessException('数据更新异常~5');
  1408. }
  1409. }
  1410. /**
  1411. * @Desc
  1412. * @Author Gorden
  1413. * @Date 2024/3/12 8:45
  1414. *
  1415. * @param array $data
  1416. * @param $model
  1417. * @return array
  1418. * @throws BusinessException
  1419. */
  1420. private static function inputFilter(array $data, $model): array
  1421. {
  1422. $table = config('database.connections.mysql.prefix') . $model->getTable();
  1423. $allow_column = $model->getConnection()->select("desc `$table`");
  1424. if (!$allow_column) {
  1425. throw new BusinessException('表不存在', 2);
  1426. }
  1427. $columns = array_column($allow_column, 'Type', 'Field');
  1428. foreach ($data as $col => $item) {
  1429. if (!isset($columns[$col])) {
  1430. unset($data[$col]);
  1431. continue;
  1432. }
  1433. // 非字符串类型传空则为null
  1434. if ($item === '' && strpos(strtolower($columns[$col]), 'varchar') === false && strpos(strtolower($columns[$col]), 'text') === false) {
  1435. $data[$col] = null;
  1436. }
  1437. if (is_array($item)) {
  1438. $data[$col] = implode(',', $item);
  1439. }
  1440. if ($item != '' && (strpos(strtolower($columns[$col]), 'varchar') || strpos(strtolower($columns[$col]), 'text'))) {
  1441. $data[$col] = htmlspecialchars($item);
  1442. }
  1443. }
  1444. if (empty($data['created_at'])) {
  1445. unset($data['created_at']);
  1446. }
  1447. if (empty($data['updated_at'])) {
  1448. unset($data['updated_at']);
  1449. }
  1450. return $data;
  1451. }
  1452. /**
  1453. * @Desc 执行更新
  1454. * @Author Gorden
  1455. * @Date 2024/3/12 8:43
  1456. *
  1457. * @param $id
  1458. * @param $data
  1459. * @param $model
  1460. * @return void
  1461. */
  1462. private static function doUpdate($id, $data, $model)
  1463. {
  1464. $row = $model->find($id);
  1465. foreach ($data as $key => $val) {
  1466. $row->{$key} = $val;
  1467. }
  1468. $row->save();
  1469. }
  1470. /**
  1471. * @Desc 上架定时
  1472. * @Author Gorden
  1473. * @Date 2024/4/11 15:13
  1474. *
  1475. * @return void
  1476. */
  1477. public static function checkListing()
  1478. {
  1479. $key = Goods::LISTING_KEY_PREFIX . date('YmdHi');
  1480. $redis = Redis::connection();
  1481. if (!$redis->exists($key)) {
  1482. return;
  1483. }
  1484. $goodsIds = $redis->sMembers($key);
  1485. if (Goods::whereIn('goods_id', $goodsIds)->update(['goods_status' => 'ON'])) {
  1486. $redis->del($key);
  1487. }
  1488. }
  1489. /**
  1490. * @Desc 自动下架
  1491. * @Author Gorden
  1492. * @Date 2024/4/26 15:26
  1493. *
  1494. * @return void
  1495. */
  1496. public static function checkOffListing()
  1497. {
  1498. $key = Goods::LISTING_OFF_KEY_PREFIX . date('YmdHi');
  1499. $redis = Redis::connection();
  1500. if (!$redis->exists($key)) {
  1501. return;
  1502. }
  1503. $goodsIds = $redis->sMembers($key);
  1504. if (Goods::whereIn('goods_id', $goodsIds)->update(['goods_status' => 'OFF'])) {
  1505. $redis->del($key);
  1506. }
  1507. }
  1508. public static $week = [
  1509. '周一' => 1,
  1510. '周二' => 2,
  1511. '周三' => 3,
  1512. '周四' => 4,
  1513. '周五' => 5,
  1514. '周六' => 6,
  1515. '周日' => 7,
  1516. ];
  1517. }