暫無描述
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Shop.php 33KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海口快推科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 陈风任 <491085389@qq.com>
  11. * Date: 2019-3-20
  12. */
  13. namespace app\user\model;
  14. use think\Model;
  15. use think\Db;
  16. use think\Config;
  17. use think\Page;
  18. /**
  19. * 商城
  20. */
  21. class Shop extends Model
  22. {
  23. //初始化
  24. protected function initialize()
  25. {
  26. // 需要调用`Model`的`initialize`方法
  27. parent::initialize();
  28. $this->home_lang = get_home_lang();
  29. }
  30. // 处理购买订单,超过指定时间修改为已订单过期,针对未付款订单
  31. public function UpdateShopOrderData($users_id){
  32. $time = getTime() - Config::get('global.get_shop_order_validity');
  33. $where = array(
  34. 'users_id' => $users_id,
  35. 'order_status' => 0,
  36. 'add_time' => array('<',$time),
  37. );
  38. $data = [
  39. 'order_status' => 4, // 状态修改为订单过期
  40. 'pay_name' => '', // 订单过期则清空支付方式标记
  41. 'wechat_pay_type' => '', // 订单过期则清空微信支付类型标记
  42. 'update_time' => getTime(),
  43. ];
  44. // 查询订单id数组用于添加订单操作记录
  45. $OrderIds = Db::name('shop_order')->field('order_id')->where($where)->select();
  46. // 订单过期,更新规格数量
  47. $productSpecValueModel = new \app\user\model\ProductSpecValue;
  48. $productSpecValueModel->SaveProducSpecValueStock($OrderIds, $users_id);
  49. //批量修改订单状态
  50. Db::name('shop_order')->where($where)->update($data);
  51. // 添加订单操作记录
  52. if (!empty($OrderIds)) {
  53. AddOrderAction($OrderIds,$users_id,'0','4','0','0','订单过期!','会员未在订单有效期内支付,订单过期!');
  54. }
  55. }
  56. // 通过商品名称模糊查询订单信息
  57. public function QueryOrderList($pagesize,$users_id,$keywords,$query_get){
  58. // 商品名称模糊查询订单明细表,获取订单主表ID
  59. $DetailsWhere = [
  60. 'users_id' => $users_id,
  61. 'lang' => $this->home_lang,
  62. ];
  63. $DetailsWhere['product_name'] = ['LIKE', "%{$keywords}%"];
  64. $DetailsData = Db::name('shop_order_details')->field('order_id')->where($DetailsWhere)->select();
  65. // 若查无数据,则返回false
  66. if (empty($DetailsData)) {
  67. return false;
  68. }
  69. $order_ids = '';
  70. // 处理订单ID,查询订单主表信息
  71. foreach ($DetailsData as $key => $value) {
  72. if ('0' < $key) {
  73. $order_ids .= ',';
  74. }
  75. $order_ids .= $value['order_id'];
  76. }
  77. // 查询条件
  78. $OrderWhere = [
  79. 'users_id' => $users_id,
  80. 'lang' => $this->home_lang,
  81. 'order_id' => ['IN', $order_ids],
  82. ];
  83. $paginate_type = 'userseyou';
  84. if (isMobile()) {
  85. $paginate_type = 'usersmobile';
  86. }
  87. $paginate = array(
  88. 'type' => $paginate_type,
  89. 'var_page' => config('paginate.var_page'),
  90. 'query' => $query_get,
  91. );
  92. $pages = Db::name('shop_order')
  93. ->field("*")
  94. ->where($OrderWhere)
  95. ->order('add_time desc')
  96. ->paginate($pagesize, false, $paginate);
  97. $data['list'] = $pages->items();
  98. $data['pages'] = $pages;
  99. return $data;
  100. }
  101. public function GetOrderIsEmpty($users_id,$keywords,$select_status){
  102. // 基础查询条件
  103. $OrderWhere = [
  104. 'users_id' => $users_id,
  105. 'lang' => $this->home_lang,
  106. ];
  107. // 应用搜索条件
  108. if (!empty($keywords)) {
  109. $OrderWhere['order_code'] = ['LIKE', "%{$keywords}%"];
  110. }
  111. // 订单状态搜索
  112. if (!empty($select_status)) {
  113. if ('dzf' === $select_status) {
  114. $select_status = 0;
  115. }
  116. $OrderWhere['order_status'] = $select_status;
  117. if (3 == $select_status){
  118. $OrderWhere['is_comment'] = 0;
  119. }
  120. }
  121. $order = Db::name('shop_order')->where($OrderWhere)->count();
  122. // 查询存在数据,则返回1
  123. if (!empty($order)) {
  124. return 1; exit;
  125. }
  126. // 查询订单明细表
  127. if (empty($order) && !empty($keywords)) {
  128. $DetailsWhere = [
  129. 'users_id' => $users_id,
  130. 'lang' => $this->home_lang,
  131. ];
  132. $DetailsWhere['product_name'] = ['LIKE', "%{$keywords}%"];
  133. $DetailsData = Db::name('shop_order_details')->field('order_id')->where($DetailsWhere)->select();
  134. // 查询无数据,则返回0
  135. if (empty($DetailsData)) {
  136. return 0; exit;
  137. }
  138. $order_ids = '';
  139. // 处理订单ID,查询订单主表信息
  140. foreach ($DetailsData as $key => $value) {
  141. if (0 < $key) {
  142. $order_ids .= ',';
  143. }
  144. $order_ids .= $value['order_id'];
  145. }
  146. // 查询条件
  147. $OrderWhere = [
  148. 'users_id' => $users_id,
  149. 'lang' => $this->home_lang,
  150. 'order_id' => ['IN', $order_ids],
  151. ];
  152. $order2 = Db::name('shop_order')->where($OrderWhere)->count();
  153. if (!empty($order2)) {
  154. return 1; exit;
  155. }else{
  156. return 0; exit;
  157. }
  158. }
  159. }
  160. // 获取微信公众号access_token
  161. // 传入微信公众号appid
  162. // 传入微信公众号secret
  163. // 返回data
  164. public function GetWeChatAccessToken($appid,$secret){
  165. // 获取公众号access_token,接口限制10万次/天
  166. $time = getTime();
  167. $get_token_url = 'https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid='.$appid.'&secret='.$secret;
  168. $TokenData = httpRequest($get_token_url);
  169. $TokenData = json_decode($TokenData, true);
  170. if (!empty($TokenData['access_token'])) {
  171. // 存入缓存配置
  172. $WechatData = [
  173. 'wechat_token_value' => $TokenData['access_token'],
  174. 'wechat_token_time' => $time,
  175. ];
  176. getUsersConfigData('wechat',$WechatData);
  177. $data = [
  178. 'status' => true,
  179. 'token' => $WechatData['wechat_token_value'],
  180. ];
  181. }else{
  182. $data = [
  183. 'status' => false,
  184. 'prompt' => '错误提示:101,后台配置配置AppId或AppSecret不正确,请检查!',
  185. ];
  186. }
  187. return $data;
  188. }
  189. // 获取微信公众号jsapi_ticket
  190. // 传入微信公众号accesstoken
  191. // 返回data
  192. public function GetWeChatJsapiTicket($accesstoken){
  193. // 获取公众号jsapi_ticket
  194. $time = getTime();
  195. $get_ticket_url = 'https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token='.$accesstoken.'&type=jsapi';
  196. $TicketData = httpRequest($get_ticket_url);
  197. $TicketData = json_decode($TicketData, true);
  198. if (!empty($TicketData['ticket'])) {
  199. // 存入缓存配置
  200. $WechatData = [
  201. 'wechat_ticket_value' => $TicketData['ticket'],
  202. 'wechat_ticket_time' => $time,
  203. ];
  204. getUsersConfigData('wechat',$WechatData);
  205. $data = [
  206. 'status' => true,
  207. 'ticket' => $WechatData['wechat_ticket_value'],
  208. ];
  209. }else{
  210. $data = [
  211. 'status' => false,
  212. 'prompt' => '错误提示:102,后台配置配置AppId或AppSecret不正确,请检查!',
  213. ];
  214. }
  215. return $data;
  216. }
  217. // 获取随机字符串
  218. // 长度 length
  219. // 结果 str
  220. public function GetRandomString($length){
  221. $chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
  222. $str = "";
  223. for ($i = 0; $i < $length; $i++) {
  224. $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
  225. }
  226. return $str;
  227. }
  228. // 旧产品属性处理
  229. public function ProductAttrProcessing($value = array())
  230. {
  231. $attr_value = '';
  232. $AttrWhere = [
  233. 'a.aid' => $value['aid'],
  234. 'b.lang' => $this->home_lang
  235. ];
  236. $attrData = Db::name('product_attr')
  237. ->alias('a')
  238. ->field('a.attr_value as value, b.attr_name as name')
  239. ->join('__PRODUCT_ATTRIBUTE__ b', 'a.attr_id = b.attr_id', 'LEFT')
  240. ->where($AttrWhere)
  241. ->order('b.sort_order asc, a.attr_id asc')
  242. ->select();
  243. foreach ($attrData as $val) {
  244. $attr_value .= $val['name'].':'.$val['value'].'<br/>';
  245. }
  246. return $attr_value;
  247. }
  248. // 新产品属性处理
  249. public function ProductNewAttrProcessing($value = array())
  250. {
  251. $attr_value = '';
  252. if (!empty($value['merchant_id'])) {
  253. // 多商家商品则查询自定义参数
  254. $where = [
  255. 'aid' => $value['aid'],
  256. ];
  257. $order = 'sort_order asc, param_id asc';
  258. $field = 'param_name as name, param_value as value';
  259. $attrData = Db::name('product_custom_param')->field($field)->where($where)->order($order)->select();
  260. } else {
  261. // 新版参数
  262. if (!empty($value['attrlist_id'])) {
  263. $where = [
  264. 'b.aid' => $value['aid'],
  265. 'a.list_id' => ['IN', [0, $value['attrlist_id']]],
  266. 'a.status' => 1,
  267. ];
  268. }else{
  269. $where = [
  270. 'b.aid' => $value['aid'],
  271. 'a.list_id' => 0,
  272. 'a.status' => 1,
  273. ];
  274. }
  275. $attrData = Db::name('shop_product_attribute')
  276. ->alias('a')
  277. ->field('a.attr_name as name, b.attr_value as value')
  278. ->join('__SHOP_PRODUCT_ATTR__ b', 'a.attr_id = b.attr_id', 'LEFT')
  279. ->where($where)
  280. ->order('a.sort_order asc, a.attr_id asc')
  281. ->select();
  282. }
  283. foreach ($attrData as $val) {
  284. $attr_value .= $val['name'].':'.$val['value'].'<br/>';
  285. }
  286. return $attr_value;
  287. }
  288. // 产品规格处理
  289. public function ProductSpecProcessing($value = array())
  290. {
  291. $spec_value_s = '';
  292. if (!empty($value['spec_value_id'])) {
  293. $spec_value_id = explode('_', $value['spec_value_id']);
  294. if (!empty($spec_value_id)) {
  295. $SpecWhere = [
  296. 'aid' => $value['aid'],
  297. 'lang' => $this->home_lang,
  298. 'spec_value_id' => ['IN',$spec_value_id],
  299. ];
  300. $ProductSpecData = Db::name("product_spec_data")->where($SpecWhere)->field('spec_name,spec_value')->select();
  301. foreach ($ProductSpecData as $spec_value) {
  302. $spec_value_s .= $spec_value['spec_name'].':'.$spec_value['spec_value'].'<br/>';
  303. }
  304. }
  305. }
  306. return $spec_value_s;
  307. }
  308. // 产品库存处理
  309. public function ProductStockProcessing($SpecValue = array())
  310. {
  311. $SpecUpData = []; // 有规格
  312. $ArcUpData = []; // 无规格
  313. foreach ($SpecValue as $key => $value) {
  314. if (!empty($value['value_id'])) {
  315. $SpecUpData[] = [
  316. 'value_id' => $value['value_id'],
  317. 'spec_stock' => Db::raw('spec_stock-'.($value['quantity'])),
  318. 'spec_sales_num' => Db::raw('spec_sales_num+'.($value['quantity'])),
  319. ];
  320. $ArcUpData[] = [
  321. 'aid' => $value['aid'],
  322. 'stock_count' => Db::raw('stock_count-' . ($value['quantity'])),
  323. 'sales_num' => Db::raw('sales_num+' . ($value['quantity'])),
  324. 'sales_all' => Db::raw('sales_all+' . ($value['quantity'])),
  325. ];
  326. }else{
  327. $ArcUpData[] = [
  328. 'aid' => $value['aid'],
  329. 'stock_count' => Db::raw('stock_count-'.($value['quantity'])),
  330. 'sales_num' => Db::raw('sales_num+' . ($value['quantity'])),
  331. 'sales_all' => Db::raw('sales_all+' . ($value['quantity'])),
  332. ];
  333. }
  334. }
  335. // 更新规格库存销量
  336. if (!empty($SpecUpData)) model('ProductSpecValue')->saveAll($SpecUpData);
  337. // 更新商品库存销量
  338. if (!empty($ArcUpData)) model('Archives')->saveAll($ArcUpData);
  339. }
  340. /**
  341. * 生成订单之前的产品数据整理 - 【第一步】
  342. * @return [type] [description]
  343. */
  344. public function handlerOrderData($opt = 'normal', $OrderData = [], $list = [], $users = [], $post = [])
  345. {
  346. $aid = 0;
  347. if ('fast' == $opt) {
  348. $aid = !empty($list[0]['aid']) ? $list[0]['aid'] : 0;
  349. }
  350. // 会员配置
  351. $usersConfig = getUsersConfigData('all');
  352. // 产品数据处理
  353. $PromType = $ContainsVirtual = 1; // 1表示为虚拟订单
  354. $TotalAmount = $TotalNumber = 0;
  355. $level_discount = $users['level_discount'];
  356. foreach ($list as $key => $value) {
  357. // 未开启多规格则执行
  358. if (!isset($usersConfig['shop_open_spec']) || empty($usersConfig['shop_open_spec'])) {
  359. $value['spec_value_id'] = $value['spec_price'] = $value['spec_stock'] = $value['value_id'] = 0;
  360. }
  361. // 购物车商品存在规格并且价格不为空,则覆盖商品原来的价格
  362. if (!empty($value['spec_value_id']) && $value['spec_price'] >= 0) $value['users_price'] = $value['spec_price'];
  363. // 计算折扣后的价格
  364. if (!empty($level_discount)) {
  365. // 折扣率百分比
  366. $discount_price = $level_discount / 100;
  367. // 会员折扣价
  368. $value['users_price'] = 2 === intval($value['users_discount_type']) ? $value['users_price'] : $value['users_price'] * $discount_price;
  369. }
  370. // 查询会员折扣价
  371. if (empty($value['spec_value_id']) && !empty($users['level_id']) && 1 === intval($value['users_discount_type'])) {
  372. $value['users_price'] = model('ShopPublicHandle')->handleUsersDiscountPrice($value['aid'], $users['level_id']);
  373. }
  374. $value['users_price'] = sprintf("%.2f", $value['users_price']);
  375. // 购物车商品存在规格并且库存不为空,则覆盖商品原来的库存
  376. if (!empty($value['spec_stock'])) $value['stock_count'] = $value['spec_stock'];
  377. // 若库存为空则返回提示
  378. if (empty($value['stock_count'])) return ['code'=>0, 'msg'=>"产品 《{$value['title']}》 库存不足!"];
  379. // 非法提交,请正规合法提交订单-提交生成订单时检测判断
  380. if (empty($value['product_num'])) return ['code'=>0, 'msg'=>'非法提交,请正规合法提交订单,非法代码:402'];
  381. // 金额、数量计算
  382. if ($value['users_price'] >= 0 && !empty($value['product_num'])) {
  383. // 合计金额
  384. $TotalAmount += sprintf("%.2f", $value['users_price'] * $value['product_num']);
  385. // 合计数量
  386. $TotalNumber += $value['product_num'];
  387. // 判断订单类型,目前逻辑:一个订单中,只要存在一个普通产品(实物产品,需要发货物流),则为普通订单
  388. if (empty($value['prom_type'])) $PromType = 0;// 0表示为普通订单
  389. // 判断是否包含虚拟商品,只要存在一个虚拟商品则表示包含虚拟商品
  390. if (!empty($value['prom_type']) && $value['prom_type'] >= 1) $ContainsVirtual = 2;
  391. }
  392. $list[$key] = $value;
  393. }
  394. // 添加到订单主表
  395. $time = getTime();
  396. $OrderData['order_code'] = date('Ymd').$time.rand(10,100); //订单生成规则
  397. $OrderData['users_id'] = $users['users_id'];
  398. $OrderData['order_status'] = 0; // 订单未付款
  399. $OrderData['add_time'] = $time;
  400. $OrderData['payment_method'] = !empty($post['payment_method']) ? intval($post['payment_method']) : 0; // 订单支付方式
  401. $OrderData['order_total_amount']= $TotalAmount;
  402. $OrderData['order_amount'] = $TotalAmount;
  403. $OrderData['order_total_num'] = $TotalNumber;
  404. $OrderData['contains_virtual'] = $ContainsVirtual;
  405. $OrderData['prom_type'] = $PromType;
  406. $OrderData['user_note'] = !empty($post['message']) ? $post['message'] : ''; // 会员备注
  407. $OrderData['lang'] = $this->home_lang;
  408. /*特定场景专用*/
  409. $opencodetype = config('global.opencodetype');
  410. if (1 == $opencodetype) {
  411. $aid = !empty($list[0]['aid']) ? $list[0]['aid'] : 0;
  412. $OrderData['spec_value_id'] = !empty($OrderData['spec_value_id']) ? "_".trim($OrderData['spec_value_id'], '_')."_" : '';
  413. // 订单分组
  414. $group = 'archives';
  415. if (74 == $aid) {
  416. $group = 'cloudminipro';
  417. } else if (75 == $aid) {
  418. $group = 'cloudminiproMall';
  419. }
  420. $OrderData['group'] = $group;
  421. // 规格值
  422. $spec_value_ids = $OrderData['spec_value_id'];
  423. // 购买套餐
  424. $buy_type = 1;
  425. if (stristr($spec_value_ids, '_1_')) {
  426. $buy_type = 2;
  427. } else if (stristr($spec_value_ids, '_2_')) {
  428. $buy_type = 3;
  429. }
  430. $OrderData['buy_type'] = $buy_type;
  431. // 购买天数
  432. $buy_day = 0;
  433. if (stristr($spec_value_ids, '_4_')) {
  434. $buy_day = 365;
  435. } else if (stristr($spec_value_ids, '_5_')) {
  436. $buy_day = 365 * 2;
  437. }
  438. $OrderData['buy_day'] = $buy_day;
  439. }
  440. /*end*/
  441. if (isMobile() && isWeixin()) {
  442. $OrderData['pay_name'] = 'wechat';// 如果在微信端中则默认为微信支付
  443. $OrderData['wechat_pay_type'] = 'WeChatInternal';// 如果在微信端中则默认为微信端调起支付
  444. }
  445. if (1 == $OrderData['payment_method']) {
  446. // 追加添加到订单主表的数组
  447. $OrderData['order_status'] = 1; // 标记已付款
  448. $OrderData['pay_time'] = $time;
  449. $OrderData['pay_name'] = 'delivery_pay';// 货到付款
  450. $OrderData['wechat_pay_type'] = ''; // 选择货到付款,则去掉微信端调起支付标记
  451. $OrderData['update_time'] = $time;
  452. }
  453. // 判断订单来源
  454. $isMobile = isMobile();
  455. $isWeixin = isWeixin();
  456. $isWeixinApplets = isWeixinApplets();
  457. if (empty($isMobile) && empty($isWeixin)) {
  458. $OrderData['order_terminal'] = 1;
  459. } else if (!empty($isMobile) && empty($isWeixinApplets)) {
  460. $OrderData['order_terminal'] = 2;
  461. } else if (!empty($isMobile) && !empty($isWeixinApplets)) {
  462. $OrderData['order_terminal'] = 3;
  463. }
  464. return ['code'=>1, 'msg'=>'ok', 'data'=>['OrderData'=>$OrderData, 'list'=>$list]];
  465. }
  466. /**
  467. * 生成订单之后的订单明细整理 - 【第二步】
  468. * @return [type] [description]
  469. */
  470. public function handlerDetailsData($opt = 'normal', $OrderData = [], $list = [], $users = [])
  471. {
  472. $OrderId = $OrderData['order_id'];
  473. $cart_ids = $UpSpecValue = [];
  474. $attr_value = $spec_value = '';
  475. if (1 == count($list) && !empty($list[0]['under_order_type'])) {
  476. /*----------删除用户旧的未付款的同类记录 start ------------*/
  477. $order_md5 = md5($list[0]['aid'].$list[0]['spec_value_id']);
  478. $where = [
  479. 'order_md5' => $order_md5,
  480. 'users_id' => intval($users['users_id']),
  481. 'order_id' => ['NEQ', $OrderId],
  482. 'order_status' => 0,
  483. ];
  484. $opencodetype = config('global.opencodetype');
  485. if (1 == $opencodetype) {
  486. $where['group'] = $OrderData['group'];
  487. }
  488. $order_id_arr = Db::name('shop_order')->where($where)->column('order_id');
  489. if (!empty($order_id_arr)) {
  490. $r = Db::name('shop_order')->where([
  491. 'order_id' => ['IN', $order_id_arr],
  492. ])->delete();
  493. if (false !== $r) {
  494. Db::name('shop_order_details')->where([
  495. 'order_id' => ['IN', $order_id_arr],
  496. ])->delete();
  497. }
  498. }
  499. /*----------删除用户旧的未付款的同类记录 end ------------*/
  500. }
  501. /*----------------订单副表添加数组--------------*/
  502. // 添加到订单明细表
  503. foreach ($list as $key => $value) {
  504. // 旧产品属性处理
  505. $attr_value = model('Shop')->ProductAttrProcessing($value);
  506. // 新产品属性处理
  507. $attr_value_new = model('Shop')->ProductNewAttrProcessing($value);
  508. // 产品规格处理
  509. $spec_value = model('Shop')->ProductSpecProcessing($value);
  510. $Data = [
  511. // 产品属性
  512. 'attr_value' => htmlspecialchars($attr_value),
  513. // 产品属性
  514. 'attr_value_new' => htmlspecialchars($attr_value_new),
  515. // 产品规格
  516. 'spec_value' => htmlspecialchars($spec_value),
  517. // 产品规格值ID
  518. 'spec_value_id' => $value['spec_value_id'],
  519. // 对应规格值ID的唯一标识ID,数据表主键ID
  520. 'value_id' => $value['value_id'],
  521. // 后续添加
  522. ];
  523. // 订单副表添加数组
  524. $OrderDetailsData[] = [
  525. 'order_id' => $OrderId,
  526. 'users_id' => intval($users['users_id']),
  527. 'product_id' => $value['aid'],
  528. 'product_name' => $value['title'],
  529. 'num' => $value['product_num'],
  530. 'data' => serialize($Data),
  531. 'product_price' => $value['users_price'],
  532. 'prom_type' => $value['prom_type'],
  533. 'litpic' => $value['litpic'],
  534. 'add_time' => getTime(),
  535. 'lang' => $this->home_lang,
  536. ];
  537. // 处理购物车ID
  538. if (empty($value['under_order_type'])) array_push($cart_ids, $value['cart_id']);
  539. // 产品库存处理
  540. $UpSpecValue[] = [
  541. 'aid' => $value['aid'],
  542. 'value_id' => $value['value_id'],
  543. 'quantity' => $value['product_num'],
  544. 'spec_value_id' => $value['spec_value_id'],
  545. ];
  546. }
  547. return [
  548. 'code' => 1,
  549. 'msg' => 'ok',
  550. 'data' => [
  551. 'cart_ids' => $cart_ids,
  552. 'OrderDetailsData' => $OrderDetailsData,
  553. 'UpSpecValue' => $UpSpecValue,
  554. ]
  555. ];
  556. }
  557. /*------陈风任---2021-1-12---售后服务(退换货)------开始------*/
  558. // 读取会员自身所有服务单信息
  559. public function GetAllServiceInfo($users_id = null, $order_code = null, $status = null,$keywords = null)
  560. {
  561. $where = [
  562. 'users_id' => $users_id
  563. ];
  564. if ('ing' == $status) {
  565. $where['status'] = ['IN', [1, 2, 3, 4, 5]];
  566. } else if ('ed' == $status) {
  567. $where['status'] = ['IN', [6, 7, 8]];
  568. }
  569. if (!empty($order_code)) $where['order_code'] = ['LIKE', "%{$order_code}%"];
  570. if (!empty($keywords)) $where['order_code|product_name'] = ['LIKE', "%{$keywords}%"];
  571. $count = Db::name('shop_order_service')->where($where)->count('service_id');
  572. $pageObj = new Page($count, config('paginate.list_rows'));
  573. // 订单主表数据查询
  574. $Service = Db::name('shop_order_service')->where($where)
  575. ->order('status asc, service_id desc')
  576. ->limit($pageObj->firstRow.','.$pageObj->listRows)
  577. ->select();
  578. $New = get_archives_data($Service, 'product_id');
  579. foreach ($Service as $key => $value) {
  580. $Service[$key]['status_old'] = intval($value['status']);
  581. $Service[$key]['handle'] = in_array($value['status'], [6, 7, 8]) ? '已完成' : '处理中';
  582. $Service[$key]['status'] = Config::get('global.order_service_status')[$value['status']];
  583. $Service[$key]['service_type'] = Config::get('global.order_service_type')[$value['service_type']];
  584. $Service[$key]['ArchivesUrl'] = urldecode(arcurl('home/Product/view', $New[$value['product_id']]));
  585. $Service[$key]['OrDetailsUrl'] = url('user/Shop/shop_order_details', ['order_id'=>$value['order_id']]);
  586. $Service[$key]['SeDetailsUrl'] = url('user/Shop/after_service_details', ['service_id'=>$value['service_id']]);
  587. $Service[$key]['product_spec'] = str_replace("&lt;br/&gt;", " &nbsp; ", $value['product_spec']);
  588. // 规格
  589. $product_spec = !empty($Service[$key]['product_spec']) ? htmlspecialchars_decode($Service[$key]['product_spec']) : '';
  590. $product_spec_list = [];
  591. $product_spec_arr = explode('<br/>', $product_spec);
  592. foreach ($product_spec_arr as $sp_key => $sp_val) {
  593. $sp_arr = explode(':', $sp_val);
  594. if (trim($sp_arr[0]) && !empty($sp_arr[0])) {
  595. $product_spec_list[] = [
  596. 'name' => !empty($sp_arr[0]) ? trim($sp_arr[0]) : '',
  597. 'value' => !empty($sp_arr[1]) ? trim($sp_arr[1]) : '',
  598. ];
  599. }
  600. }
  601. $Service[$key]['product_spec_list'] = $product_spec_list;
  602. }
  603. $Return['Service'] = $Service;
  604. $Return['pageStr'] = $pageObj->show();
  605. return $Return;
  606. }
  607. // 服务详情信息
  608. public function GetServiceDetailsInfo($service_id = null, $users_id = null)
  609. {
  610. $Return = [];
  611. if (empty($service_id) || empty($users_id)) return $Return;
  612. $where = [
  613. 'users_id' => $users_id,
  614. 'service_id' => $service_id
  615. ];
  616. $Service = Db::name('shop_order_service')->where($where)->select();
  617. if (empty($Service)) return $Return;
  618. $New = get_archives_data($Service, 'product_id');
  619. $Service = $Service[0];
  620. $Service['arcurl'] = urldecode(arcurl('home/Product/view', $New[$Service['product_id']]));
  621. $Service['CancelUrl'] = url('user/Shop/after_service_details', ['_ajax' => 1, 'details_id' => $Service['details_id'], 'order_id' => $Service['order_id']]);
  622. $Service['StatusName'] = Config::get('global.order_service_status')[$Service['status']];
  623. $Service['upload_img'] = !empty($Service['upload_img']) ? explode(',', $Service['upload_img']) : [];
  624. $Service['product_img'] = handle_subdir_pic(get_default_pic($Service['product_img']));
  625. if (isMobile()) {
  626. $Service['product_spec'] = htmlspecialchars_decode($Service['product_spec']);
  627. } else {
  628. $Service['product_spec'] = str_replace("&lt;br/&gt;", " <br/> ", $Service['product_spec']);
  629. }
  630. $Service['service_type_old'] = $Service['service_type'];
  631. $Service['service_type'] = Config::get('global.order_service_type')[$Service['service_type']];
  632. $Service['admin_delivery'] = !empty($Service['admin_delivery']) ? unserialize($Service['admin_delivery']) : ['name'=>'', 'code'=>''];
  633. $Service['product_total'] = floatval(sprintf("%.2f", $Service['refund_price'] * (string)$Service['product_num']));
  634. /*计算退还余额*/
  635. $field_new = 'b.details_id, b.product_price, b.data, b.num, a.shipping_fee, a.order_total_num';
  636. $where_new = [
  637. 'b.order_id' => $Service['order_id'],
  638. 'b.details_id' => $Service['details_id'],
  639. 'b.apply_service' => 1
  640. ];
  641. $Order = Db::name('shop_order')->alias('a')
  642. ->field($field_new)
  643. ->join('__SHOP_ORDER_DETAILS__ b', 'a.order_id = b.order_id', 'LEFT')
  644. ->where($where_new)
  645. ->find();
  646. // 运费计算
  647. $ShippingFee = 0;
  648. $Service['ShippingFee'] = floatval($ShippingFee);
  649. // if (!empty($Order['shipping_fee'])) {
  650. // $ShippingFee = sprintf("%.2f", ($Order['shipping_fee'] / (string)$Order['order_total_num']) * (string)$Service['product_num']);
  651. // $Service['ShippingFee'] = floatval($ShippingFee);
  652. // }
  653. $Service['refund_total_price'] = floatval((string)$Service['product_total'] - (string)$ShippingFee);
  654. // 规格
  655. $product_spec_list = [];
  656. $spec_data = unserialize($Order['data']);
  657. if (!empty($spec_data['spec_value'])) {
  658. $spec_value_arr = explode('<br/>', htmlspecialchars_decode($spec_data['spec_value']));
  659. foreach ($spec_value_arr as $sp_key => $sp_val) {
  660. $sp_arr = explode(':', $sp_val);
  661. if (trim($sp_arr[0]) && !empty($sp_arr[0])) {
  662. $product_spec_list[] = [
  663. 'name' => !empty($sp_arr[0]) ? trim($sp_arr[0]) : '',
  664. 'value' => !empty($sp_arr[1]) ? trim($sp_arr[1]) : '',
  665. ];
  666. }
  667. }
  668. }
  669. $Service['product_spec_list'] = $product_spec_list;
  670. /* END */
  671. return $Service;
  672. }
  673. // 服务详情记录
  674. public function GetOrderServiceLog($service_id = null, $Users = [], $users_id = 0) {
  675. if (empty($service_id) || empty($Users)) return [];
  676. $Log = Db::name('shop_order_service_log')->order('log_id desc')->where('service_id', $service_id)->select();
  677. foreach ($Log as $key => $value) {
  678. if (!empty($value['users_id'])) {
  679. if (intval($users_id) === intval($value['users_id'])) {
  680. $Log[$key]['name'] = '会员';
  681. } else {
  682. $Log[$key]['name'] = '商家';
  683. }
  684. } else if (!empty($value['admin_id'])) {
  685. $Log[$key]['name'] = '商家';
  686. }
  687. }
  688. return $Log;
  689. }
  690. // 查询订单数据
  691. public function GetOrderDetailsInfo($details_id = null, $users_id = null)
  692. {
  693. $Return = [];
  694. if (empty($details_id) || empty($users_id)) return $Return;
  695. // 查询订单明细数据并处理
  696. $where = [
  697. 'details_id' => $details_id,
  698. 'users_id' => $users_id
  699. ];
  700. $Details = Db::name('shop_order_details')->where($where)->find();
  701. if (empty($Details)) {
  702. $Return = [
  703. 'code' => 0,
  704. 'msg' => '售后服务单不存在'
  705. ];
  706. return $Return;
  707. }
  708. if (!empty($Details) && 1 === intval($Details['apply_service'])) {
  709. $Return = [
  710. 'code' => 0,
  711. 'msg' => '已提交过申请,请查阅退换货信息'
  712. ];
  713. return $Return;
  714. }
  715. $array_new = get_archives_data([$Details], 'product_id');
  716. // 商品封面图处理
  717. $Details['litpic'] = handle_subdir_pic(get_default_pic($Details['litpic']));
  718. // 商品详情页URL
  719. $Details['arcurl'] = urldecode(arcurl('home/Product/view', $array_new[$Details['product_id']]));
  720. // 获取订单商品规格列表(购买时的商品规格)
  721. $Details['product_spec_list'] = model('ShopPublicHandle')->getOrderGoodsSpecList($Details);
  722. // 商品其他处理
  723. $Details['data'] = unserialize($Details['data']);
  724. $Details['spec_value'] = htmlspecialchars_decode(htmlspecialchars_decode($Details['data']['spec_value']));
  725. $Details['spec_value_id'] = $Details['data']['spec_value_id'];
  726. $Details['value_id'] = $Details['data']['value_id'];
  727. unset($Details['data']);
  728. // 查询订单主表数据并处理
  729. $field = 'order_code, merchant_id, consignee, province, city, district, address, mobile, allow_service';
  730. $Order = Db::name('shop_order')->field($field)->where('order_id', $Details['order_id'])->find();
  731. // 判断订单是否允许申请售后
  732. if (isset($Order['allow_service']) && 1 === intval($Order['allow_service'])) {
  733. $Return = [
  734. 'code' => 0,
  735. 'msg' => '该订单不允许申请售后维权!'
  736. ];
  737. return $Return;
  738. }
  739. $Order['province'] = get_province_name($Order['province']);
  740. $Order['city'] = get_city_name($Order['city']);
  741. $Order['district'] = get_area_name($Order['district']);
  742. // 合并数据返回
  743. $Return = array_merge($Details, $Order);
  744. return $Return;
  745. }
  746. /*------陈风任---2021-1-12---售后服务(退换货)------结束------*/
  747. }