123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476 |
- <?php
- namespace app\store\model;
- use app\common\model\Order as OrderModel;
- use app\store\model\User as UserModel;
- use app\store\model\UserCoupon as UserCouponModel;
- use app\store\service\order\Export as Exportservice;
- use app\common\library\helper;
- use app\common\enum\OrderType as OrderTypeEnum;
- use app\common\enum\DeliveryType as DeliveryTypeEnum;
- use app\common\service\Message as MessageService;
- use app\common\service\order\Refund as RefundService;
- use app\common\service\wechat\wow\Order as WowService;
- use app\common\service\goods\source\Factory as FactoryStock;
- use app\common\enum\order\PayStatus as PayStatusEnum;
- use app\common\enum\order\Status as OrderStatusEnum;
- /**
- * 订单管理
- * Class Order
- * @package app\store\model
- */
- class Order extends OrderModel
- {
- /**
- * 订单列表
- * @param string $dataType
- * @param array $query
- * @return \think\Paginator
- * @throws \think\exception\DbException
- */
- public function getList($dataType, $query = [])
- {
- // 检索查询条件
- !empty($query) && $this->setWhere($query);
- // 获取数据列表
- return $this->with(['goods.image', 'address', 'user'])
- ->alias('order')
- ->field('order.*')
- ->join('user', 'user.user_id = order.user_id')
- ->where($this->transferDataType($dataType))
- ->where('order.is_delete', '=', 0)
- ->order(['order.create_time' => 'desc'])
- ->paginate(10, false, [
- 'query' => \request()->request()
- ]);
- }
- /**
- * 订单列表(全部)
- * @param $dataType
- * @param array $query
- * @return false|\PDOStatement|string|\think\Collection
- * @throws \think\db\exception\DataNotFoundException
- * @throws \think\db\exception\ModelNotFoundException
- * @throws \think\exception\DbException
- */
- public function getListAll($dataType, $query = [])
- {
- // 检索查询条件
- !empty($query) && $this->setWhere($query);
- // 获取数据列表
- return $this->with(['goods.image', 'address', 'user', 'extract', 'extract_shop'])
- ->alias('order')
- ->field('order.*')
- ->join('user', 'user.user_id = order.user_id')
- ->where($this->transferDataType($dataType))
- ->where('order.is_delete', '=', 0)
- ->order(['order.create_time' => 'desc'])
- ->select();
- }
- /**
- * 订单导出
- * @param $dataType
- * @param $query
- * @throws \think\db\exception\DataNotFoundException
- * @throws \think\db\exception\ModelNotFoundException
- * @throws \think\exception\DbException
- */
- public function exportList($dataType, $query)
- {
- // 获取订单列表
- $list = $this->getListAll($dataType, $query);
- // 导出csv文件
- return (new Exportservice)->orderList($list);
- }
- /**
- * 批量发货模板
- */
- public function deliveryTpl()
- {
- return (new Exportservice)->deliveryTpl();
- }
- /**
- * 设置检索查询条件
- * @param $query
- */
- private function setWhere($query)
- {
- if (isset($query['search']) && !empty($query['search'])) {
- $this->where('order_no|user.nickName', 'like', '%' . trim($query['search']) . '%');
- }
- if (isset($query['start_time']) && !empty($query['start_time'])) {
- $this->where('order.create_time', '>=', strtotime($query['start_time']));
- }
- if (isset($query['end_time']) && !empty($query['end_time'])) {
- $this->where('order.create_time', '<', strtotime($query['end_time']) + 86400);
- }
- if (isset($query['delivery_type']) && !empty($query['delivery_type'])) {
- $query['delivery_type'] > -1 && $this->where('delivery_type', '=', $query['delivery_type']);
- }
- if (isset($query['extract_shop_id']) && !empty($query['extract_shop_id'])) {
- $query['extract_shop_id'] > -1 && $this->where('extract_shop_id', '=', $query['extract_shop_id']);
- }
- // 用户id
- if (isset($query['user_id']) && $query['user_id'] > 0) {
- $this->where('order.user_id', '=', (int)$query['user_id']);
- }
- }
- /**
- * 转义数据类型条件
- * @param $dataType
- * @return array
- */
- private function transferDataType($dataType)
- {
- // 数据类型
- $filter = [];
- switch ($dataType) {
- case 'delivery':
- $filter = [
- 'pay_status' => 20,
- 'delivery_status' => 10,
- 'order_status' => ['in', [10, 21]]
- ];
- break;
- case 'receipt':
- $filter = [
- 'pay_status' => 20,
- 'delivery_status' => 20,
- 'receipt_status' => 10
- ];
- break;
- case 'pay':
- $filter = ['pay_status' => 10, 'order_status' => 10];
- break;
- case 'complete':
- $filter = ['order_status' => 30];
- break;
- case 'cancel':
- $filter = ['order_status' => 20];
- break;
- case 'all':
- $filter = [];
- break;
- }
- return $filter;
- }
- /**
- * 确认发货(单独订单)
- * @param $data
- * @return array|bool|false
- * @throws \app\common\exception\BaseException
- * @throws \think\Exception
- * @throws \think\exception\DbException
- * @throws \Exception
- */
- public function delivery($data)
- {
- // 转义为订单列表
- $orderList = [$this];
- // 验证订单是否满足发货条件
- if (!$this->verifyDelivery($orderList)) {
- return false;
- }
- // 整理更新的数据
- $updateList = [[
- 'order_id' => $this['order_id'],
- 'express_id' => $data['express_id'],
- 'express_no' => $data['express_no']
- ]];
- // 更新订单发货状态
- if ($status = $this->updateToDelivery($updateList)) {
- // 获取已发货的订单
- $completed = self::detail($this['order_id'], ['user', 'address', 'goods', 'express']);
- // 发送消息通知
- $this->sendDeliveryMessage([$completed]);
- // 同步好物圈订单
- (new WowService($this['wxapp_id']))->update([$completed]);
- }
- return $status;
- }
- /**
- * 批量发货
- * @param $data
- * @return bool
- * @throws \think\db\exception\DataNotFoundException
- * @throws \think\db\exception\ModelNotFoundException
- * @throws \think\exception\DbException
- * @throws \Exception
- */
- public function batchDelivery($data)
- {
- // 获取csv文件中的数据
- if (!$csvData = $this->getCsvData()) {
- return false;
- }
- // 整理订单id集
- $orderNos = helper::getArrayColumn($csvData, 0);
- // 获取订单列表数据
- $orderList = helper::arrayColumn2Key($this->getListByOrderNos($orderNos), 'order_no');
- // 验证订单是否存在
- $tempArr = array_values(array_diff($orderNos, array_keys($orderList)));
- if (!empty($tempArr)) {
- $this->error = "订单号[{$tempArr[0]}] 不存在!";
- return false;
- }
- // 整理物流单号
- $updateList = [];
- foreach ($csvData as $item) {
- $updateList[] = [
- 'order_id' => $orderList[$item[0]]['order_id'],
- 'express_id' => $data['express_id'],
- 'express_no' => $item[1],
- ];
- }
- // 验证订单是否满足发货条件
- if (!$this->verifyDelivery($orderList)) {
- return false;
- }
- // 更新订单发货状态(批量)
- if ($status = $this->updateToDelivery($updateList)) {
- // 获取已发货的订单
- $completed = $this->getListByOrderNos($orderNos, ['user', 'address', 'goods', 'express']);
- // 发送消息通知
- $this->sendDeliveryMessage($completed);
- // 同步好物圈订单
- (new WowService(self::$wxapp_id))->update($completed);
- }
- return $status;
- }
- /**
- * 确认发货后发送消息通知
- * @param $orderList
- * @return bool
- */
- private function sendDeliveryMessage($orderList)
- {
- // 发送消息通知
- foreach ($orderList as $item) {
- MessageService::send('order.delivery', [
- 'order' => $item,
- 'order_type' => OrderTypeEnum::MASTER,
- ]);
- }
- return true;
- }
- /**
- * 更新订单发货状态(批量)
- * @param $orderList
- * @return array|false
- * @throws \Exception
- */
- private function updateToDelivery($orderList)
- {
- $data = [];
- foreach ($orderList as $item) {
- $data[] = [
- 'order_id' => $item['order_id'],
- 'express_no' => $item['express_no'],
- 'express_id' => $item['express_id'],
- 'delivery_status' => 20,
- 'delivery_time' => time(),
- ];
- }
- return $this->isUpdate()->saveAll($data);
- }
- /**
- * 验证订单是否满足发货条件
- * @param $orderList
- * @return bool
- */
- private function verifyDelivery($orderList)
- {
- foreach ($orderList as $order) {
- if (
- $order['pay_status']['value'] != 20
- || $order['delivery_type']['value'] != DeliveryTypeEnum::EXPRESS
- || $order['delivery_status']['value'] != 10
- ) {
- $this->error = "订单号[{$order['order_no']}] 不满足发货条件!";
- return false;
- }
- }
- return true;
- }
- /**
- * 获取csv文件中的数据
- * @return array|bool
- */
- private function getCsvData()
- {
- // 获取表单上传文件 例如上传了001.jpg
- $file = \request()->file('iFile');
- if (empty($file)) {
- $this->error = '请上传发货模板';
- return false;
- }
- // 设置区域信息
- setlocale(LC_ALL, 'zh_CN');
- // 打开上传的文件
- $csvFile = fopen($file->getInfo()['tmp_name'], 'r');
- // 忽略第一行(csv标题)
- fgetcsv($csvFile);
- // 遍历并记录订单信息
- $orderList = [];
- while ($item = fgetcsv($csvFile)) {
- if (!isset($item[0]) || empty($item[0]) || !isset($item[1]) || empty($item[1])) {
- $this->error = '模板文件数据不合法';
- return false;
- }
- $orderList[] = $item;
- }
- if (empty($orderList)) {
- $this->error = '模板文件中没有订单数据';
- return false;
- }
- return $orderList;
- }
- /**
- * 修改订单价格
- * @param $data
- * @return bool
- */
- public function updatePrice($data)
- {
- if ($this['pay_status']['value'] != 10) {
- $this->error = '该订单不合法';
- return false;
- }
- // 实际付款金额
- $payPrice = bcadd($data['update_price'], $data['update_express_price'], 2);
- if ($payPrice <= 0) {
- $this->error = '订单实付款价格不能为0.00元';
- return false;
- }
- return $this->save([
- 'order_no' => $this->orderNo(), // 修改订单号, 否则微信支付提示重复
- 'order_price' => $data['update_price'],
- 'pay_price' => $payPrice,
- 'update_price' => helper::bcsub($data['update_price'], helper::bcsub($this['total_price'], $this['coupon_money'])),
- 'express_price' => $data['update_express_price']
- ]) !== false;
- }
- /**
- * 审核:用户取消订单
- * @param $data
- * @return bool|mixed
- * @throws \app\common\exception\BaseException
- * @throws \think\exception\DbException
- */
- public function confirmCancel($data)
- {
- // 判断订单是否有效
- if ($this['pay_status']['value'] != 20) {
- $this->error = '该订单不合法';
- return false;
- }
- // 订单取消事件
- $status = $this->transaction(function () use ($data) {
- if ($data['is_cancel'] == true) {
- // 执行退款操作
- (new RefundService)->execute($this);
- // 回退商品库存
- FactoryStock::getFactory($this['order_source'])->backGoodsStock($this['goods'], true);
- // 回退用户优惠券
- $this['coupon_id'] > 0 && UserCouponModel::setIsUse($this['coupon_id'], false);
- // 回退用户积分
- $User = UserModel::detail($this['user_id']);
- $describe = "订单取消:{$this['order_no']}";
- $this['points_num'] > 0 && $User->setIncPoints($this['points_num'], $describe);
- }
- // 更新订单状态
- return $this->save(['order_status' => $data['is_cancel'] ? 20 : 10]);
- });
- if ($status == true) {
- // 同步好物圈订单
- (new WowService(self::$wxapp_id))->update([$this]);
- }
- return $status;
- }
- /**
- * 获取已付款订单总数 (可指定某天)
- * @param null $startDate
- * @param null $endDate
- * @return int|string
- * @throws \think\Exception
- */
- public function getPayOrderTotal($startDate = null, $endDate = null)
- {
- $filter = [
- 'pay_status' => PayStatusEnum::SUCCESS,
- 'order_status' => ['<>', OrderStatusEnum::CANCELLED],
- ];
- if (!is_null($startDate) && !is_null($endDate)) {
- $filter['pay_time'] = [
- ['>=', strtotime($startDate)],
- ['<', strtotime($endDate) + 86400],
- ];
- }
- return $this->getOrderTotal($filter);
- }
- /**
- * 获取订单总数量
- * @param array $filter
- * @return int|string
- * @throws \think\Exception
- */
- private function getOrderTotal($filter = [])
- {
- return $this->where($filter)
- ->where('is_delete', '=', 0)
- ->count();
- }
- /**
- * 获取某天的总销售额
- * @param null $startDate
- * @param null $endDate
- * @return float|int
- */
- public function getOrderTotalPrice($startDate = null, $endDate = null)
- {
- if (!is_null($startDate) && !is_null($endDate)) {
- $this->where('pay_time', '>=', strtotime($startDate))
- ->where('pay_time', '<', strtotime($endDate) + 86400);
- }
- return $this->where('pay_status', '=', 20)
- ->where('order_status', '<>', 20)
- ->where('is_delete', '=', 0)
- ->sum('pay_price');
- }
- /**
- * 获取某天的下单用户数
- * @param $day
- * @return float|int
- */
- public function getPayOrderUserTotal($day)
- {
- $startTime = strtotime($day);
- $userIds = $this->distinct(true)
- ->where('pay_time', '>=', $startTime)
- ->where('pay_time', '<', $startTime + 86400)
- ->where('pay_status', '=', 20)
- ->where('is_delete', '=', 0)
- ->column('user_id');
- return count($userIds);
- }
- }
|