mirror of
https://gitee.com/ShopeX/OMS
synced 2026-03-23 02:45:33 +08:00
486 lines
17 KiB
PHP
486 lines
17 KiB
PHP
<?php
|
|
/**
|
|
* Copyright 2012-2026 ShopeX (https://www.shopex.cn)
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
class tgkpi_mdl_analysis_delivery extends dbeav_model{
|
|
|
|
private $realTableName = 'sdb_tgkpi_pick';
|
|
|
|
/**
|
|
* count
|
|
* @param mixed $filter filter
|
|
* @return mixed 返回值
|
|
*/
|
|
public function count($filter=null){
|
|
$f = kernel::single('tgkpi_analysis_pick')->dealTime($filter);
|
|
$filter['time_from'] = sprintf('%s 00:00:00',$f['time_from']);
|
|
$filter['time_to'] = sprintf('%s 23:59:59',$f['time_to']);
|
|
$filter['time_from'] = strtotime($filter['time_from']);
|
|
$filter['time_to'] = strtotime($filter['time_to']);
|
|
$sql = "SELECT count(distinct pick_owner) as total FROM sdb_tgkpi_pick where ";
|
|
$sql .= $this->pickFilter($filter);
|
|
$rs = $this->db->select($sql);
|
|
return intval($rs[0]['total']);
|
|
}
|
|
|
|
/**
|
|
* @description where过滤条件
|
|
* @access public
|
|
* @param Array $filter
|
|
* @return String
|
|
*/
|
|
public function pickFilter($filter)
|
|
{
|
|
$where = array(1);
|
|
if ($filter['time_from'] && $filter['time_to']) {
|
|
$where[] = ' pick_start_time between '.$filter['time_from'].' AND '.$filter['time_to'];
|
|
}
|
|
#完成发货
|
|
$where[] = " pick_status='deliveryed'";
|
|
|
|
if ($filter['pick_owner']) {
|
|
if (is_array($filter['pick_owner'])) {
|
|
$where[] = ' pick_owner in(\''.implode('\',\'',$filter['pick_owner']).'\')';
|
|
}elseif(is_string($filter['pick_owner'])){
|
|
$where[] = ' pick_owner=\''.$filter['pick_owner'].'\'';
|
|
}
|
|
}
|
|
|
|
|
|
if ($filter['sku']=='single') {
|
|
$where[] = ' delivery_sku_num=1';
|
|
}elseif($filter['sku']=='multi'){
|
|
$where[] = ' delivery_sku_num>1';
|
|
}
|
|
|
|
|
|
return implode(' AND ',$where);
|
|
}
|
|
|
|
/**
|
|
* @description 获取单商品单数
|
|
* @access public
|
|
* @param Array $filter
|
|
* @return Array $single
|
|
*/
|
|
public function getSingleSku($filter)
|
|
{
|
|
$filter['sku'] = 'single';
|
|
$sql = 'SELECT COUNT(DISTINCT delivery_id) as singleSkuTotal , pick_owner FROM '.$this->realTableName.' WHERE ';
|
|
$sql .= $this->pickFilter($filter);
|
|
$sql .= ' GROUP BY pick_owner';
|
|
|
|
$single = array();
|
|
$rs = $this->db->select($sql);
|
|
foreach($rs as $v) {
|
|
$v['pick_owner'] = strtoupper(trim($v['pick_owner']));
|
|
$single[$v['pick_owner']] = $v;
|
|
}
|
|
unset($rs);
|
|
|
|
return $single;
|
|
}
|
|
|
|
/**
|
|
* @description 获取多商品单数
|
|
* @access public
|
|
* @param Array $filter
|
|
* @return Array $multi
|
|
*/
|
|
public function getMultiSku($filter)
|
|
{
|
|
$filter['sku'] = 'multi';
|
|
$sql = 'SELECT COUNT(DISTINCT delivery_id) as multiSkuTotal,pick_owner,SUM(pick_num) as pickBnTotal FROM '.$this->realTableName.' WHERE ';
|
|
$sql .= $this->pickFilter($filter);
|
|
$sql .= ' GROUP BY pick_owner';
|
|
|
|
$multi = array();
|
|
$rs = $this->db->select($sql);
|
|
foreach($rs as $v) {
|
|
$v['pick_owner'] = strtoupper(trim($v['pick_owner']));
|
|
$multi[$v['pick_owner']] = $v;
|
|
}
|
|
unset($rs);
|
|
|
|
return $multi;
|
|
}
|
|
|
|
/**
|
|
* @description 获取完成的单数
|
|
* @access public
|
|
* @param Array $filter
|
|
* @return Array $finish
|
|
*/
|
|
public function getFinishSku($filter)
|
|
{
|
|
$filter['pick_status'] = 'finish';
|
|
// 完成单数统计
|
|
$sql = 'SELECT COUNT(DISTINCT delivery_id) as _cdelivery , SUM(pick_num) as _cpick , pick_owner FROM '.$this->realTableName.' WHERE ';
|
|
$sql .= $this->pickFilter($filter);
|
|
$sql .= ' GROUP BY pick_owner';
|
|
|
|
$finish = array();
|
|
$rs = $this->db->select($sql);
|
|
foreach ($rs as $key=>$value) {
|
|
$value['pick_owner'] = strtoupper(trim($value['pick_owner']));
|
|
$finish[$value['pick_owner']] = $value;
|
|
}
|
|
|
|
return $finish;
|
|
}
|
|
|
|
public function getlist($cols='*', $filter=array(), $offset=0, $limit=-1, $orderType=null){
|
|
$f = kernel::single('tgkpi_analysis_pick')->dealTime($filter);
|
|
$filter['time_from'] = sprintf('%s 00:00:00',$f['time_from']);
|
|
$filter['time_to'] = sprintf('%s 23:59:59',$f['time_to']);
|
|
$filter['time_from'] = strtotime($filter['time_from']);
|
|
$filter['time_to'] = strtotime($filter['time_to']);
|
|
|
|
$sql = "SELECT pick_owner as picker, op_name, COUNT(DISTINCT delivery_id) as deliveryTotal , SUM(pick_num) as pickTotal , pick_error_num from ".$this->realTableName." where ";
|
|
$sql .= $this->pickFilter($filter);
|
|
$sql .= ' GROUP BY pick_owner';
|
|
if($orderType){
|
|
if (is_string($orderType)) {
|
|
$ot = $orderType;
|
|
$this->ot = array_filter(explode(' ',trim($ot)));
|
|
list($columns,$taxis) = $this->ot;
|
|
if ($columns == 'picker') {
|
|
$sql.=' ORDER BY '.(is_array($orderType) ? implode(' ',$orderType) : $orderType);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
$rs = $this->db->selectLimit($sql,$limit,$offset);;
|
|
if (empty($rs)) {
|
|
return array();
|
|
}
|
|
$all = $rs; unset($rs);
|
|
|
|
foreach ($all as $value) {
|
|
$pickOwners[] = $value['picker'];
|
|
}
|
|
$filter['pick_owner'] = $pickOwners;unset($pickOwners);
|
|
|
|
//所有的工号
|
|
$userModel = app::get('desktop')->model('users');
|
|
$temusers = $userModel->getList('op_no,name',array('op_no'=>$filter['pick_owner']));
|
|
foreach ($temusers as $v) {
|
|
$v['op_no'] = strtoupper(trim($v['op_no']));
|
|
$user[$v['op_no']] = $v;
|
|
}
|
|
unset($temusers);
|
|
|
|
//单商品单数
|
|
$single = $this->getSingleSku($filter);
|
|
|
|
//多商品单数
|
|
$multi = $this->getMultiSku($filter);
|
|
|
|
// 完成单数统计
|
|
$finish = $this->getFinishSku($filter);
|
|
|
|
$data = array();
|
|
foreach ($all as $key=>$value) {
|
|
$picker = strtoupper(trim($value['picker']));
|
|
$data[$key]['picker'] = $picker;
|
|
$data[$key]['name'] = $user[$picker]['name'];
|
|
$data[$key]['op_name'] = $value['op_name'];
|
|
$data[$key]['deliverys'] = $finish[$picker]['_cdelivery'] ? $finish[$picker]['_cdelivery'] : 0;
|
|
$data[$key]['single_nums'] = $single[$picker]['singleSkuTotal'] ? $single[$picker]['singleSkuTotal'] : 0;
|
|
$data[$key]['multi_deliverys'] = $multi[$picker]['multiSkuTotal'] ? $multi[$picker]['multiSkuTotal'] : 0;
|
|
$data[$key]['rate_of_single'] = $this->rate_format($data[$key]['single_nums'],$value['deliveryTotal']);
|
|
$data[$key]['rate_of_singlecomplete'] = $this->rate_format($data[$key]['deliverys'],$value['deliveryTotal']);
|
|
}
|
|
unset($all);
|
|
|
|
if ($columns != 'picker') {
|
|
usort($data,array($this,'ucmp'));
|
|
}
|
|
|
|
$this->tidy_data($data, $cols);
|
|
return $data;
|
|
}
|
|
|
|
/**
|
|
* @description
|
|
* @access public
|
|
* @param void
|
|
* @return void
|
|
*/
|
|
public function ucmp($a,$b)
|
|
{
|
|
if ($this->ot) {
|
|
$n = 1;
|
|
list($column,$taxis) = $this->ot;
|
|
if (strtolower($taxis) == 'desc') {
|
|
$n = -1;
|
|
}
|
|
if ($column=='name') {
|
|
return strcmp($a[$column],$b[$column]) * $n;
|
|
}elseif('%' == substr($a[$column],-1,1)){
|
|
if ($a[$column]==$b[$column]) {
|
|
return 0;
|
|
}
|
|
$atemp = substr($a[$column],0,-1);
|
|
$btemp = substr($b[$column],0,-1);
|
|
return ($atemp>$btemp ? 1 : -1)*$n;
|
|
}else{
|
|
if ($a[$column]==$b[$column]) {
|
|
return 0;
|
|
}
|
|
return ($a[$column]>$b[$column] ? 1 : -1) * $n;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @description 比率格式化
|
|
* @access public
|
|
* @param Float $numerator 分子
|
|
* @param Float $denominator 分母
|
|
* @return String
|
|
*/
|
|
public function rate_format($numerator,$denominator)
|
|
{
|
|
$numerator = (float)$numerator;$denominator = (float)$denominator;
|
|
try{
|
|
$quotient = round($numerator/$denominator*100,2);
|
|
$quotient = ''.$quotient;
|
|
$l = strlen($quotient);
|
|
if (false==strpos($quotient,'.')) {
|
|
$f = '%.0f%%';
|
|
}elseif(true==strpos($quotient,'.') && $quotient[$l-1]==0){
|
|
$f = '%.1f%%';
|
|
}else{
|
|
$f = '%.2f%%';
|
|
}
|
|
return sprintf($f,$quotient);
|
|
} catch (Exception $e) {
|
|
//分母不能为零
|
|
return sprintf('%d%%',0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* _filter
|
|
* @param mixed $filter filter
|
|
* @param mixed $tableAlias tableAlias
|
|
* @param mixed $baseWhere baseWhere
|
|
* @return mixed 返回值
|
|
*/
|
|
public function _filter($filter,$tableAlias=null,$baseWhere=null){
|
|
$config = app::get('eccommon')->getConf('analysis_config');
|
|
$filter['order_status'] = $config['filter']['order_status'];
|
|
$where = array(1);
|
|
if(isset($filter['time_from']) && $filter['time_from']){
|
|
$where[] = ' createtime >='.strtotime($filter['time_from']);
|
|
}
|
|
if(isset($filter['time_to']) && $filter['time_to']){
|
|
$where[] = ' createtime <'.(strtotime($filter['time_to'])+86400);
|
|
}
|
|
if(isset($filter['pick_nums']) && $filter['pick_nums']){
|
|
$where[] = ' pick_nums =\''.$filter['pick_nums'].'\'';
|
|
}
|
|
if(isset($filter['ship_status']) && $filter['ship_status']){
|
|
$where[] = ' ship_status =\''.$filter['ship_status'].'\'';
|
|
}
|
|
if(isset($filter['type_id']) && $filter['type_id']){
|
|
$where[] = ' shop_id =\''.$filter['type_id'].'\'';
|
|
}
|
|
return implode($where,' AND ');
|
|
}
|
|
|
|
/**
|
|
* 获取_schema
|
|
* @return mixed 返回结果
|
|
*/
|
|
public function get_schema(){
|
|
$schema = array (
|
|
'columns' => array (
|
|
'picker' => array (
|
|
'type' => 'varchar(10)',
|
|
'pkey' => true,
|
|
'label' => '工号',
|
|
'width' => 120,
|
|
'editable' => false,
|
|
'in_list' => true,
|
|
'default_in_list' => true,
|
|
'realtype' => 'mediumint(8) unsigned',
|
|
'order' => 10,
|
|
),
|
|
'name' => array (
|
|
'type' => 'varchar(10)',
|
|
'pkey' => true,
|
|
'label' => '姓名',
|
|
'width' => 100,
|
|
'editable' => false,
|
|
'in_list' => true,
|
|
'default_in_list' => true,
|
|
'order' => 20,
|
|
),
|
|
'op_name' => array (
|
|
'type' => 'varchar(30)',
|
|
'label' => '最终发货人',
|
|
'width' => 100,
|
|
'editable' => false,
|
|
'in_list' => true,
|
|
'default_in_list' => true,
|
|
'order' => 30,
|
|
),
|
|
'deliverys' => array (
|
|
'type' => 'number',
|
|
'label' => '总完成单数',
|
|
'width' => 80,
|
|
'editable' => true,
|
|
'filtertype' => 'normal',
|
|
'filterdefault' => 'true',
|
|
'in_list' => true,
|
|
'is_title' => true,
|
|
'default_in_list' => true,
|
|
'realtype' => 'varchar(50)',
|
|
'order' => 40,
|
|
),
|
|
'single_nums' => array (
|
|
'type' => 'number',
|
|
'label' => '单商品单数',
|
|
'width' => 120,
|
|
'editable' => true,
|
|
'filtertype' => 'normal',
|
|
'filterdefault' => 'true',
|
|
'in_list' => true,
|
|
'is_title' => true,
|
|
'default_in_list' => true,
|
|
'realtype' => 'varchar(50)',
|
|
'order' => 50,
|
|
),
|
|
'multi_deliverys' => array (
|
|
'type' => 'number',
|
|
'label' => '多商品单数',
|
|
'width' => 120,
|
|
'editable' => true,
|
|
'filtertype' => 'normal',
|
|
'filterdefault' => 'true',
|
|
'in_list' => true,
|
|
'is_title' => true,
|
|
'default_in_list' => true,
|
|
'realtype' => 'varchar(50)',
|
|
'order' => 60,
|
|
),
|
|
'rate_of_single' => array(
|
|
'type' => 'float',
|
|
'label' => '单商品占比',
|
|
'width' => 'auto',
|
|
'editable' => false,
|
|
'filtertype' => 'normal',
|
|
'default' => 0,
|
|
'in_list' => true,
|
|
'default_in_list' => true,
|
|
'order' => 70,
|
|
),
|
|
'rate_of_singlecomplete' => array(
|
|
'type' => 'float',
|
|
'label' => '单数完成率',
|
|
'width' => 'auto',
|
|
'editable' => false,
|
|
'filtertype' => 'normal',
|
|
'default' => 0,
|
|
'in_list' => true,
|
|
'default_in_list' => true,
|
|
'order' => 80,
|
|
),
|
|
),
|
|
'idColumn' => 'picker',
|
|
'in_list' => array (
|
|
0 => 'picker',
|
|
1 => 'name',
|
|
2 => 'op_name',
|
|
3 => 'deliverys',
|
|
4 => 'single_nums',
|
|
5 => 'multi_deliverys',
|
|
6 => 'rate_of_single',
|
|
7 => 'rate_of_singlecomplete',
|
|
|
|
),
|
|
'default_in_list' => array (
|
|
0 => 'picker',
|
|
1 => 'name',
|
|
2 => 'op_name',
|
|
3 => 'deliverys',
|
|
4 => 'single_nums',
|
|
5 => 'multi_deliverys',
|
|
6 => 'rate_of_single',
|
|
7 => 'rate_of_singlecomplete',
|
|
|
|
),
|
|
);
|
|
return $schema;
|
|
}
|
|
|
|
/**
|
|
* fetch_graph_data
|
|
* @param mixed $params 参数
|
|
* @return mixed 返回值
|
|
*/
|
|
public function fetch_graph_data($params){
|
|
$params['time_from'] = sprintf('%s 00:00:00',$params['time_from']);
|
|
$params['time_to'] = sprintf('%s 23:59:59',$params['time_to']);
|
|
|
|
if($params['report'] == 'month'):
|
|
$time_format = '%Y-%m';
|
|
else:
|
|
$time_format = '%Y-%m-%d';
|
|
endif;
|
|
$time_range = [];
|
|
if($params['target'] == 2):
|
|
$sql = "SELECT DATE_FORMAT(FROM_UNIXTIME(pick_start_time),'$time_format') as date,op_name,sum(pick_num) as pick_nums
|
|
FROM sdb_tgkpi_pick
|
|
WHERE pick_status='deliveryed' AND pick_start_time between ".strtotime($params['time_from'])." AND ".strtotime($params['time_to'])."
|
|
GROUP BY DATE_FORMAT(FROM_UNIXTIME(pick_start_time),'$time_format'),op_name
|
|
";
|
|
$rs = $this->db->selectLimit($sql,100,0);
|
|
foreach((array)$rs as $v) {
|
|
if(!in_array($v['date'],$time_range)) $time_range[] = $v['date'];
|
|
$data[$v['op_name']][$v['date']] = $v['pick_nums'];
|
|
}
|
|
else:
|
|
$sql = "SELECT DATE_FORMAT(FROM_UNIXTIME(pick_start_time),'$time_format') as date,op_name,count(distinct delivery_id) as deliverys
|
|
FROM sdb_tgkpi_pick
|
|
WHERE pick_status='deliveryed' AND pick_start_time between ".strtotime($params['time_from'])." AND ".strtotime($params['time_to'])."
|
|
GROUP BY DATE_FORMAT(FROM_UNIXTIME(pick_start_time),'$time_format'),op_name
|
|
";
|
|
$rs = $this->db->selectLimit($sql,100,0);
|
|
foreach((array)$rs as $v) {
|
|
if(!in_array($v['date'],$time_range)) $time_range[] = $v['date'];
|
|
$data[$v['op_name']][$v['date']] = $v['deliverys'];
|
|
}
|
|
endif;
|
|
|
|
if ($data) {
|
|
$pickers = array_keys($data);
|
|
$sql = 'SELECT name,op_no FROM sdb_desktop_users WHERE op_no in(\''.implode('\',\'',$pickers).'\')';
|
|
$pickers = $this->db->select($sql);
|
|
foreach ($pickers as $key=>$value) {
|
|
$d = $data[$value['op_no']];
|
|
$data[$value['name']] = $d;
|
|
unset($data[$value['op_no']]);
|
|
}
|
|
}
|
|
return array('categories'=>$time_range, 'data'=>$data);
|
|
}
|
|
}
|