CRMEB/crmeb/app/services/statistic/ProductStatisticServices.php
2022-12-19 18:40:49 +08:00

259 lines
11 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<?php
// +----------------------------------------------------------------------
// | CRMEB [ CRMEB赋能开发者助力企业发展 ]
// +----------------------------------------------------------------------
// | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
// +----------------------------------------------------------------------
// | Licensed CRMEB并不是自由软件未经许可不能去掉CRMEB相关版权
// +----------------------------------------------------------------------
// | Author: CRMEB Team <admin@crmeb.com>
// +----------------------------------------------------------------------
namespace app\services\statistic;
use app\services\BaseServices;
use app\services\other\export\ExportServices;
use app\services\order\StoreCartServices;
use app\services\order\StoreOrderServices;
use app\services\product\product\StoreProductLogServices;
use app\services\product\product\StoreVisitServices;
use app\services\user\UserBillServices;
use crmeb\exceptions\AdminException;
/**
* Class ProductStatisticServices
* @package app\services\statistic
*/
class ProductStatisticServices extends BaseServices
{
/**
* 商品基础
* @param $where
* @return array
*/
public function getBasic($where)
{
$time = explode('-', $where['time']);
if (count($time) != 2) throw new AdminException(100100);
//当前数据
$now = $this->basicInfo($where, $time);
//环比数据
$dayNum = (strtotime($time[1]) - strtotime($time[0])) / 86400 + 1;
$lastTime = array(
date("Y/m/d", strtotime("-$dayNum days", strtotime($time[0]))),
date("Y/m/d", strtotime("-1 days", strtotime($time[0])))
);
$where['time'] = implode('-', $lastTime);
$last = $this->basicInfo($where, $lastTime);
//组合数据,计算环比
$data = [];
foreach ($now as $key => $item) {
$data[$key]['num'] = $item;
$num = $last[$key] > 0 ? $last[$key] : 1;
$data[$key]['percent'] = bcmul((string)bcdiv((string)($item - $last[$key]), (string)$num, 4), 100, 2);
}
return $data;
}
/**
* 商品基础数据
* @param $where
* @param $time
* @return mixed
*/
public function basicInfo($where, $time)
{
/** @var StoreVisitServices $storeVisit */
$storeVisit = app()->make(StoreVisitServices::class);
/** @var StoreCartServices $storeCart */
$storeCart = app()->make(StoreCartServices::class);
/** @var StoreOrderServices $storeOrder */
$storeOrder = app()->make(StoreOrderServices::class);
/** @var StoreProductLogServices $productLog */
$productLog = app()->make(StoreProductLogServices::class);
$data['browse'] = $productLog->count($where + ['type' => 'visit']);//商品浏览量
$data['user'] = $productLog->getDistinctCount($where + ['type' => 'visit'], 'uid');//商品访客数
$data['cart'] = $storeCart->getSum($where, 'cart_num');//加入购物车件数
$data['order'] = $storeOrder->sum($where + ['pid' => 0], 'total_num', true);//下单件数
$data['pay'] = $storeOrder->sum([
['paid', '=', 1],
['pay_time', '>=', strtotime($time[0])],
['pay_time', '<', strtotime($time[1]) + 86400],
['pid', '>=', 0]
], 'total_num');//支付件数
$data['payPrice'] = $storeOrder->sum([
['paid', '=', 1],
['pay_time', '>=', strtotime($time[0])],
['pay_time', '<', strtotime($time[1]) + 86400],
['pid', '>=', 0]
], 'pay_price');//支付金额
$data['cost'] = $storeOrder->sum([
['paid', '=', 1],
['pay_time', '>=', strtotime($time[0])],
['pay_time', '<', strtotime($time[1]) + 86400],
['pid', '>=', 0]
], 'cost');//成本金额
$data['refundPrice'] = $storeOrder->sum($where + ['refund_status' => 2], 'pay_price', true);//退款金额
$data['refund'] = $storeOrder->sum($where + ['refund_status' => 2], 'total_num', true);//退款件数
$payPeople = $storeOrder->getDistinctCount($where + ['paid' => 1], 'uid');//成交用户数
$data['payPercent'] = $data['user'] > 0 ? bcmul(bcdiv($payPeople, $data['user'], 4), 100, 2) : 0;//访问-付款转化率
return $data;
}
/**
* 商品趋势
* @param $where
* @param $excel
* @return array
*/
public function getTrend($where, $excel = false)
{
$time = explode('-', $where['time']);
if (count($time) != 2) throw new AdminException(100100);
$dayCount = (strtotime($time[1]) - strtotime($time[0])) / 86400 + 1;
$data = [];
if ($dayCount == 1) {
$data = $this->trend($time, 0, $excel);
} elseif ($dayCount > 1 && $dayCount <= 31) {
$data = $this->trend($time, 1, $excel);
} elseif ($dayCount > 31 && $dayCount <= 92) {
$data = $this->trend($time, 3, $excel);
} elseif ($dayCount > 92) {
$data = $this->trend($time, 30, $excel);
}
return $data;
}
/**
* 商品趋势
* @param $time
* @param $num
* @param $excel
* @return array
*/
public function trend($time, $num, $excel = false)
{
/** @var StoreVisitServices $storeVisit */
$storeVisit = app()->make(StoreVisitServices::class);
/** @var StoreOrderServices $storeOrder */
$storeOrder = app()->make(StoreOrderServices::class);
/** @var StoreCartServices $storeCart */
$storeCart = app()->make(StoreCartServices::class);
/** @var StoreProductLogServices $productLog */
$productLog = app()->make(StoreProductLogServices::class);
if ($num == 0) {
$xAxis = ['00', '01', '02', '03', '04', '05', '06', '07', '08', '09', '10', '11', '12', '13', '14', '15', '16', '17', '18', '19', '20', '21', '22', '23'];
$timeType = '%H';
} elseif ($num != 0) {
$dt_start = strtotime($time[0]);
$dt_end = strtotime($time[1]);
while ($dt_start <= $dt_end) {
if ($num == 30) {
$xAxis[] = date('Y-m', $dt_start);
$dt_start = strtotime("+1 month", $dt_start);
$timeType = '%Y-%m';
} else {
$xAxis[] = date('m-d', $dt_start);
$dt_start = strtotime("+$num day", $dt_start);
$timeType = '%m-%d';
}
}
}
$browse = array_column($productLog->getProductTrend($time, $timeType, 'sum(visit_num)'), 'num', 'days');
$user = array_column($productLog->getProductTrend($time, $timeType, 'count(distinct(uid))'), 'num', 'days');
$pay = array_column($storeOrder->getProductTrend($time, $timeType, 'pay_time', 'sum(pay_price)'), 'num', 'days');
$refundList = $storeOrder->getProductTrend($time, $timeType, 'refund_reason_time', 'sum(pay_price)');
$refund = array_column($refundList, 'num', 'days');
if ($excel) {
$cart = array_column($storeCart->getProductTrend($time, $timeType, 'sum(cart_num)'), 'num', 'days');
$order = array_column($storeOrder->getProductTrend($time, $timeType, 'add_time', 'sum(total_num)'), 'num', 'days');
$payNum = array_column($storeOrder->getProductTrend($time, $timeType, 'pay_time', 'sum(total_num)'), 'num', 'days');
$payCountNum = array_column($storeOrder->getProductTrend($time, $timeType, 'pay_time', 'count(distinct(uid))'), 'num', 'days');
$cost = array_column($storeOrder->getProductTrend($time, $timeType, 'pay_time', 'sum(cost)'), 'num', 'days');
$orderIds = array_column($refundList, 'link_ids');
$ids = implode(',', $orderIds);
$totalNumList = $storeOrder->column(['refund_id' => $ids], 'total_num', 'id');
$refundNum = array_column($refundList, 'link_ids', 'days');
foreach ($refundNum as &$i) {
$oIds = explode(',', $i);
$i = array_map(function ($o) use ($totalNumList) {
if (isset($totalNumList[$o])) {
return $totalNumList[$o];
}
}, $oIds);
$i = array_sum($i);
}
$data = [];
foreach ($xAxis as &$item) {
if (isset($user[$item]) && isset($payCountNum[$item])) {
$changes = bcmul(bcdiv((string)$payCountNum[$item], (string)$user[$item], 4), 100, 2);
$changes = $changes > 100 ? 100 : $changes;
} else {
$changes = 0;
}
$data[] = [
'time' => $item,
'browse' => $browse[$item] ?? 0,
'user' => $user[$item] ?? 0,
'pay' => $pay[$item] ?? 0,
'refund' => $refund[$item] ?? 0,
'cart' => $cart[$item] ?? 0,
'order' => $order[$item] ?? 0,
'payNum' => $payNum[$item] ?? 0,
'cost' => $cost[$item] ?? 0,
'refundNum' => $refundNum[$item] ?? 0,
'changes' => $changes
];
}
/** @var ExportServices $exportService */
$exportService = app()->make(ExportServices::class);
$url = $exportService->productTrade($data);
return compact('url');
} else {
$data = $series = [];
foreach ($xAxis as $item) {
$data['商品浏览量'][] = isset($browse[$item]) ? floatval($browse[$item]) : 0;
$data['商品访客量'][] = isset($user[$item]) ? floatval($user[$item]) : 0;
$data['支付金额'][] = isset($pay[$item]) ? floatval($pay[$item]) : 0;
$data['退款金额'][] = isset($refund[$item]) ? floatval($refund[$item]) : 0;
}
foreach ($data as $key => $item) {
if ($key == '商品浏览量' || $key == '商品访客量') {
$series[] = [
'name' => $key,
'data' => $item,
'type' => 'line',
'smooth' => 'true',
'yAxisIndex' => 1,
];
} else {
$series[] = [
'name' => $key,
'data' => $item,
'type' => 'bar',
];
}
}
return compact('xAxis', 'series');
}
}
/**
* 商品排行
* @param $where
* @return mixed
*/
public function getProductRanking($where)
{
/** @var StoreProductLogServices $productLog */
$productLog = app()->make(StoreProductLogServices::class);
return $productLog->getRanking($where);
}
}