Produceorder.php 42 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096
  1. <?php
  2. namespace app\admin\controller;
  3. use app\admin\library\Auth;
  4. use app\admin\model\Admin;
  5. use app\admin\model\MobileOrderAdmin;
  6. use app\admin\model\MobileOrderRefundLog;
  7. use app\admin\model\MobileOrderSubAnchor;
  8. use app\admin\model\ShortcutContent;
  9. use app\common\controller\Backend;
  10. use app\common\model\Area;
  11. use app\common\model\LogisticsCompany;
  12. use app\common\model\MobileOrder;
  13. use app\common\model\MobileOrderOperation;
  14. use app\common\model\Produce;
  15. use app\common\service\MobileOrderExport;
  16. use app\common\service\ProduceOrderService;
  17. use app\common\service\TransferCheck;
  18. use app\common\service\ZopOrderService;
  19. use app\common\validate\RefundValidate;
  20. use app\service\byte_dance\ByteDanceSettle;
  21. use Exception;
  22. use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
  23. use PhpOffice\PhpSpreadsheet\Reader\Csv;
  24. use PhpOffice\PhpSpreadsheet\Reader\Xls;
  25. use PhpOffice\PhpSpreadsheet\Reader\Xlsx;
  26. use think\Db;
  27. use think\db\Query;
  28. use think\exception\PDOException;
  29. use think\exception\ValidateException;
  30. use think\Loader;
  31. /**
  32. *
  33. *
  34. * @icon fa fa-circle-o
  35. */
  36. class Produceorder extends Backend
  37. {
  38. protected $noNeedRight=['status','pay_type'];
  39. /**
  40. * MobileOrder模型对象
  41. * @var \app\admin\model\MobileOrder
  42. */
  43. protected $model = null;
  44. protected $proxy = 0;
  45. public function _initialize()
  46. {
  47. parent::_initialize();
  48. $this->model = new \app\admin\model\MobileOrder;
  49. $this->assign('status',\app\common\model\MobileOrder::$status);
  50. $this->assign('logistics_list',json_encode(LogisticsCompany::all()));
  51. if(in_array(2, $this->auth->getGroupIds())){
  52. $this->proxy = 1;
  53. $this->assign('status',\app\common\model\MobileOrder::$GongYSStatus);
  54. }
  55. $platformSourceList = [
  56. '1' => '自动生产',
  57. '0' => '手动生产',
  58. ];
  59. $this->assign('platformSourceList', $platformSourceList);
  60. }
  61. public function import_old()
  62. {
  63. $file = $this->request->request('file');
  64. if (!$file) {
  65. $this->error(__('Parameter %s can not be empty', 'file'));
  66. }
  67. $file=parse_url($file,PHP_URL_PATH);
  68. $filePath = ROOT_PATH . DS . 'public' . DS . $file;
  69. if (!is_file($filePath)) {
  70. $this->error(__('No results were found'));
  71. }
  72. //实例化reader
  73. $ext = pathinfo($filePath, PATHINFO_EXTENSION);
  74. if (!in_array($ext, ['csv', 'xls', 'xlsx'])) {
  75. $this->error(__('Unknown data format'));
  76. }
  77. if ($ext === 'csv') {
  78. $file = fopen($filePath, 'r');
  79. $filePath = tempnam(sys_get_temp_dir(), 'import_csv');
  80. $fp = fopen($filePath, "w");
  81. $n = 0;
  82. while ($line = fgets($file)) {
  83. $line = rtrim($line, "\n\r\0");
  84. $encoding = mb_detect_encoding($line, ['utf-8', 'gbk', 'latin1', 'big5']);
  85. if ($encoding != 'utf-8') {
  86. $line = mb_convert_encoding($line, 'utf-8', $encoding);
  87. }
  88. if ($n == 0 || preg_match('/^".*"$/', $line)) {
  89. fwrite($fp, $line . "\n");
  90. } else {
  91. fwrite($fp, '"' . str_replace(['"', ','], ['""', '","'], $line) . "\"\n");
  92. }
  93. $n++;
  94. }
  95. fclose($file) || fclose($fp);
  96. $reader = new Csv();
  97. } elseif ($ext === 'xls') {
  98. $reader = new Xls();
  99. } else {
  100. $reader = new Xlsx();
  101. }
  102. //导入文件首行类型,默认是注释,如果需要使用字段名称请使用name
  103. $importHeadType = isset($this->importHeadType) ? $this->importHeadType : 'comment';
  104. $table = $this->model->getQuery()->getTable();
  105. $database = \think\Config::get('database.database');
  106. $fieldArr = [];
  107. $list = db()->query("SELECT COLUMN_NAME,COLUMN_COMMENT FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = ? AND TABLE_SCHEMA = ?", [$table, $database]);
  108. foreach ($list as $k => $v) {
  109. if ($importHeadType == 'comment') {
  110. $fieldArr[$v['COLUMN_COMMENT']] = $v['COLUMN_NAME'];
  111. } else {
  112. $fieldArr[$v['COLUMN_NAME']] = $v['COLUMN_NAME'];
  113. }
  114. }
  115. //加载文件
  116. $insert = [];
  117. try {
  118. if (!$PHPExcel = $reader->load($filePath)) {
  119. $this->error(__('Unknown data format'));
  120. }
  121. $currentSheet = $PHPExcel->getSheet(0); //读取文件中的第一个工作表
  122. $allColumn = $currentSheet->getHighestDataColumn(); //取得最大的列号
  123. $allRow = $currentSheet->getHighestRow(); //取得一共有多少行
  124. $maxColumnNumber = Coordinate::columnIndexFromString($allColumn);
  125. $fields = [];
  126. for ($currentRow = 1; $currentRow <= 1; $currentRow++) {
  127. for ($currentColumn = 1; $currentColumn <= $maxColumnNumber; $currentColumn++) {
  128. $val = $currentSheet->getCellByColumnAndRow($currentColumn, $currentRow)->getValue();
  129. $fields[] = $val;
  130. }
  131. }
  132. for ($currentRow = 2; $currentRow <= $allRow; $currentRow++) {
  133. $values = [];
  134. for ($currentColumn = 1; $currentColumn <= $maxColumnNumber; $currentColumn++) {
  135. $val = $currentSheet->getCellByColumnAndRow($currentColumn, $currentRow)->getValue();
  136. $values[] = is_null($val) ? '' : $val;
  137. }
  138. $row = [];
  139. $temp = array_combine($fields, $values);
  140. foreach ($temp as $k => $v) {
  141. if (isset($fieldArr[$k]) && $k !== '') {
  142. $row[$fieldArr[$k]] = $v;
  143. }
  144. }
  145. if ($row) {
  146. $insert[] = $row;
  147. }
  148. }
  149. } catch (Exception $exception) {
  150. $this->error($exception->getMessage());
  151. }
  152. if (!$insert) {
  153. $this->error(__('No rows were updated'));
  154. }
  155. try {
  156. //是否包含admin_id字段
  157. $has_admin_id = false;
  158. foreach ($fieldArr as $name => $key) {
  159. if ($key == 'admin_id') {
  160. $has_admin_id = true;
  161. break;
  162. }
  163. }
  164. if ($has_admin_id) {
  165. $auth = Auth::instance();
  166. foreach ($insert as &$val) {
  167. if (!isset($val['admin_id']) || empty($val['admin_id'])) {
  168. $val['admin_id'] = $auth->isLogin() ? $auth->id : 0;
  169. }
  170. $val['is_auto'] = 0;
  171. }
  172. }
  173. (new MobileOrder)->insertAll($insert);
  174. } catch (PDOException $exception) {
  175. $msg = $exception->getMessage();
  176. if (preg_match("/.+Integrity constraint violation: 1062 Duplicate entry '(.+)' for key '(.+)'/is", $msg, $matches)) {
  177. $msg = "导入失败,包含【{$matches[1]}】的记录已存在";
  178. };
  179. $this->error($msg);
  180. } catch (Exception $e) {
  181. $this->error($e->getMessage());
  182. }
  183. $this->success();
  184. }
  185. /**
  186. * 导入
  187. *
  188. * @return void
  189. */
  190. public function import()
  191. {
  192. ProduceOrderService::saveFile(input('file'),$this->auth->id);
  193. $this->success('上传成功,请等待导入');
  194. }
  195. /**
  196. * 默认生成的控制器所继承的父类中有index/add/edit/del/multi五个基础方法、destroy/restore/recyclebin三个回收站方法
  197. * 因此在当前控制器中可不用编写增删改查的代码,除非需要自己控制这部分逻辑
  198. * 需要将application/admin/library/traits/Backend.php中对应的方法复制到当前控制器,然后进行修改
  199. */
  200. /**
  201. * 查看
  202. */
  203. public function index()
  204. {
  205. //当前是否为关联查询
  206. $this->relationSearch = true;
  207. //设置过滤方法
  208. $this->request->filter(['strip_tags', 'trim']);
  209. $map=[];
  210. if($this->admin('is_sub')){
  211. $map['s_id']=$this->admin('id');
  212. }
  213. //if(!$this->admin('is_manager')){
  214. // $map['s_id']=$this->admin('id');
  215. //}
  216. $export=input('export');
  217. if($export){
  218. list($where, $sort, $order, $offset, $limit) = $this->buildparams();
  219. $list = $this->model
  220. ->where($where)
  221. ->where($map)
  222. ->where('type',1)
  223. // ->where(['platform_source'=>'敢探号','api_goods_id'=>123564678989,'or'])
  224. // ->whereOr('platform_source','敢探号')
  225. // ->whereOr('api_goods_id','123564678989')
  226. ->where('is_need_push', 1)
  227. ->where(function($q) {
  228. //供应商
  229. if($this->proxy){
  230. return $q->where('mobile_order.admin_id', $this->auth->id);
  231. }
  232. })
  233. ->with(['produce'])
  234. ->order($sort, $order)
  235. ->paginate($limit);
  236. return MobileOrderExport::export($list,$this->admin());
  237. }
  238. if ($this->request->isAjax()) {
  239. //如果发送的来源是Selectpage,则转发到Selectpage
  240. if ($this->request->request('keyField')) {
  241. return $this->selectpage();
  242. }
  243. // status需要组装的搜索条件
  244. $filter = json_decode($this->request->get('filter'),true);
  245. $op = json_decode($this->request->get('op'),true);
  246. list($where, $sort, $order, $offset, $limit) = $this->buildparams();
  247. if($this->admin('is_manager')){
  248. // 注销关联 ,'refund_log.refunduser'
  249. $relation=['operation','operation.admin','admin','admin.admin','refund_log',
  250. 'sub_remark','sub_remark.admin','anchor','sub_anchor','sub_anchor.mobileAnchor','proxy','payment'];
  251. }else{
  252. // 注销关联 ,'refund_log.refunduser'
  253. $relation=['admin','admin.admin','refund_log','sub_remark'=>function(Query $query){
  254. $query->where('sub_admin_id',$this->auth->id);
  255. },'sub_remark.admin','anchor','sub_anchor','sub_anchor.mobileAnchor','proxy','payment'];
  256. }
  257. $list = $this->model
  258. ->where($where)
  259. ->where($map)
  260. ->where('type',1)
  261. // ->where(['platform_source'=>'敢探号','api_goods_id'=>123564678989,'or'])
  262. ->where(function($q) {
  263. //供应商
  264. if($this->proxy){
  265. return $q->where('mobile_order.admin_id', $this->auth->id);
  266. }
  267. })
  268. // ->where(function ($q){
  269. // $q->whereOr('platform_source','敢探号')->whereOr('api_goods_id','123564678989');
  270. // })
  271. ->where('is_need_push', 1)
  272. // ->whereOr('platform_source','敢探号')
  273. ->order($sort, $order)
  274. ->with(['produce'])
  275. ->paginate($limit);
  276. // print_r($list->getLastSql());
  277. $upStatus=[];
  278. if($list->items()) {
  279. $upStatus = \app\admin\model\Mobile::whereIn('id', array_column($list->items(), 'mobile_id'))->where('status', '<>',0)->column('id');
  280. }
  281. foreach ($list as $row) {
  282. $row['pay_link']=h5_link("oder-detail/{$row['id']}");
  283. $row->append(['phone_order_count']);
  284. $row['mobile_can_up']=!in_array($row['mobile_id'],$upStatus);
  285. if($row['s_id']) {
  286. $row['s_name'] = Admin::where('id', $row['s_id'])->value('nickname');
  287. }else{
  288. $row['s_name']='';
  289. }
  290. $row['create_time'] = date('Y-m-d H:i:s', $row['create_time']);
  291. $row['update_time'] = date('Y-m-d H:i:s', $row['update_time']);
  292. $row['produce_status_name'] = $row['produce_status'] == 1 ? '已开户' : '未开户';
  293. $row['produce_activation_name'] = $row['produce_activation'] == 1 ? '已激活' : '未激活';
  294. $row['produce_is_recharge_name'] = $row['produce_is_recharge'] == 1 ? '已首充' : '未首充';
  295. $row['is_arrearage_name'] = $row['is_arrearage'] == 1 ? '已欠费' : '未欠费';
  296. $row['logistics_numbers'] = $row['logistics_numbers'] ?: '';
  297. $row['first_amount'] = $row['first_amount'] ?: '';
  298. }
  299. $extend=[];
  300. $amountMap=[];
  301. $amountMap['type']=1;
  302. if(!$this->admin('is_manager')){
  303. $amountMap['s_id']=$this->admin('id');
  304. }
  305. /*$amountBase=$this->model->filterSaled()->where($where)->where($amountMap)->sum('amount_base');
  306. $amountDi=$this->model->filterSaled()->where($where)->where($amountMap)->sum('amount_di');
  307. $extend['total']=bcsub($amountBase,$amountDi);*/
  308. $extend['total']=$this->model->filterSaled()->where($where)->where($amountMap)->sum('amount_base');
  309. $extend['alipay']=$this->model->filterSaled()->where($where)->where($amountMap)->where('pay_type',2)->sum('amount');
  310. $extend['wechat']=$this->model->filterSaled()->where($where)->where($amountMap)->where('pay_type',1)->sum('amount');
  311. $extend['jd']=$this->model->filterSaled()->where($where)->where($amountMap)->where('pay_type',3)->sum('amount');
  312. $extend['dy']=$this->model->filterSaled()->where($where)->where($amountMap)->where('pay_type',4)->sum('amount');
  313. $extend['ks']=$this->model->filterSaled()->where($where)->where($amountMap)->where('pay_type',5)->sum('amount');
  314. $extend['di']=$this->model->filterSaled()->where($where)->where($amountMap)->sum('amount_di');
  315. $extend['profit']=$this->model->filterSaled()->where($where)->where($amountMap)->sum('amount_profit');
  316. $extend['refund']=$this->model->where($where)->where($amountMap)->sum('amount_refund');
  317. // select count(id) as total,status from mobile_order where type=1 GROUP BY status; 角标查询
  318. if (isset($filter['status'])){
  319. unset($filter['status']);
  320. unset($op['status']);
  321. }
  322. $this->request->get(['filter'=>json_encode($filter)]);
  323. $this->request->get(['op'=>json_encode($op)]);
  324. list($where, $sort, $order, $offset, $limit) = $this->buildparams();
  325. // if($this->admin('is_manager')){
  326. // $relation=['mobile','anchor'];
  327. // }else{
  328. // $relation=['mobile'];
  329. // }
  330. $status_list = $this->model
  331. //->with($relation)
  332. ->field('count(mobile_order.id) as total,mobile_order.status')
  333. ->where($where)
  334. ->where($map)
  335. ->where('mobile_order.type',1)
  336. ->where(function($q) {
  337. //供应商
  338. if($this->proxy){
  339. return $q->where('mobile_order.admin_id', $this->auth->id)->whereIn('status',[15,17,20,25,60]);
  340. }
  341. })
  342. ->group('mobile_order.status')
  343. ->select();
  344. $status_10_count=0;
  345. if (!$this->proxy) {
  346. // 状态10的数据统计
  347. $status_10_count = $this->model
  348. // ->with($relation)
  349. ->where($where)
  350. ->whereNotIn('status', [0, 50, 90])
  351. ->where($map)
  352. ->where('mobile_order.type', 1)
  353. ->with(['produce'])
  354. ->where(function ($q) {
  355. //供应商
  356. if ($this->proxy) {
  357. return $q->where('mobile_order.admin_id', $this->auth->id);
  358. }
  359. })->count();
  360. }
  361. $result = array("total" => $list->total(), "rows" => $list->items(),'extend'=>$extend,'status_list'=>$status_list,'status_10_count'=>$status_10_count);
  362. $result['link']=$this->request->url().'&export=1';
  363. return json($result);
  364. }
  365. // print_r(123);
  366. // exit();
  367. $this->assign('admins',Admin::getAdmins());
  368. $this->assign('subs',Admin::getSubs());
  369. $this->assign('is_proxy',$this->proxy);
  370. $this->assignconfig('is_proxy',$this->proxy);
  371. return $this->view->fetch();
  372. }
  373. protected function buildparams($searchfields = null, $relationSearch = null)
  374. {
  375. $searchfields = is_null($searchfields) ? $this->searchFields : $searchfields;
  376. $relationSearch = is_null($relationSearch) ? $this->relationSearch : $relationSearch;
  377. $search = $this->request->get("search", '');
  378. $filter = $this->request->get("filter", '');
  379. $op = $this->request->get("op", '', 'trim');
  380. $sort = $this->request->get("sort", !empty($this->model) && $this->model->getPk() ? $this->model->getPk() : 'id');
  381. $order = $this->request->get("order", "DESC");
  382. $offset = $this->request->get("offset/d", 0);
  383. $limit = $this->request->get("limit/d", 999999);
  384. //新增自动计算页码
  385. $page = $limit ? intval($offset / $limit) + 1 : 1;
  386. if ($this->request->has("page")) {
  387. $page = $this->request->get("page/d", 1);
  388. }
  389. $this->request->get([config('paginate.var_page') => $page]);
  390. $filter = (array)json_decode($filter, true);
  391. $op = (array)json_decode($op, true);
  392. $filter = $filter ? $filter : [];
  393. $where = [];
  394. $alias = [];
  395. $bind = [];
  396. $name = '';
  397. $aliasName = '';
  398. if (!empty($this->model) && $this->relationSearch) {
  399. $name = $this->model->getTable();
  400. $alias[$name] = Loader::parseName(basename(str_replace('\\', '/', get_class($this->model))));
  401. $aliasName = $alias[$name] . '.';
  402. }
  403. $sortArr = explode(',', $sort);
  404. foreach ($sortArr as $index => & $item) {
  405. $item = stripos($item, ".") === false ? $aliasName . trim($item) : $item;
  406. }
  407. unset($item);
  408. $sort = implode(',', $sortArr);
  409. $adminIds = $this->getDataLimitAdminIds();
  410. if (is_array($adminIds)) {
  411. $where[] = [$aliasName . $this->dataLimitField, 'in', $adminIds];
  412. }
  413. if ($search) {
  414. $searcharr = is_array($searchfields) ? $searchfields : explode(',', $searchfields);
  415. foreach ($searcharr as $k => &$v) {
  416. $v = stripos($v, ".") === false ? $aliasName . $v : $v;
  417. }
  418. unset($v);
  419. $where[] = [implode("|", $searcharr), "LIKE", "%{$search}%"];
  420. }
  421. $index = 0;
  422. foreach ($filter as $k => $v) {
  423. if (!preg_match('/^[a-zA-Z0-9_\-\.]+$/', $k)) {
  424. continue;
  425. }
  426. $sym = isset($op[$k]) ? $op[$k] : '=';
  427. if (stripos($k, ".") === false) {
  428. $k = $aliasName . $k;
  429. }
  430. $v = !is_array($v) ? trim($v) : $v;
  431. $sym = strtoupper(isset($op[$k]) ? $op[$k] : $sym);
  432. //null和空字符串特殊处理
  433. if (!is_array($v)) {
  434. if (in_array(strtoupper($v), ['NULL', 'NOT NULL'])) {
  435. $sym = strtoupper($v);
  436. }
  437. if (in_array($v, ['""', "''"])) {
  438. $v = '';
  439. $sym = '=';
  440. }
  441. }
  442. switch ($sym) {
  443. case '=':
  444. case '<>':
  445. $where[] = [$k, $sym, (string)$v];
  446. break;
  447. case 'LIKE':
  448. case 'NOT LIKE':
  449. case 'LIKE %...%':
  450. case 'NOT LIKE %...%':
  451. $where[] = [$k, trim(str_replace('%...%', '', $sym)), "%{$v}%"];
  452. break;
  453. case '>':
  454. case '>=':
  455. case '<':
  456. case '<=':
  457. $where[] = [$k, $sym, intval($v)];
  458. break;
  459. case 'FINDIN':
  460. case 'FINDINSET':
  461. case 'FIND_IN_SET':
  462. $v = is_array($v) ? $v : explode(',', str_replace(' ', ',', $v));
  463. $findArr = array_values($v);
  464. foreach ($findArr as $idx => $item) {
  465. $bindName = "item_" . $index . "_" . $idx;
  466. $bind[$bindName] = $item;
  467. $where[] = "FIND_IN_SET(:{$bindName}, `" . str_replace('.', '`.`', $k) . "`)";
  468. }
  469. break;
  470. case 'IN':
  471. case 'IN(...)':
  472. case 'NOT IN':
  473. case 'NOT IN(...)':
  474. $where[] = [$k, str_replace('(...)', '', $sym), is_array($v) ? $v : explode(',', $v)];
  475. break;
  476. case 'BETWEEN':
  477. case 'NOT BETWEEN':
  478. $arr = array_slice(explode(',', $v), 0, 2);
  479. if (stripos($v, ',') === false || !array_filter($arr)) {
  480. continue 2;
  481. }
  482. //当出现一边为空时改变操作符
  483. if ($arr[0] === '') {
  484. $sym = $sym == 'BETWEEN' ? '<=' : '>';
  485. $arr = $arr[1];
  486. } elseif ($arr[1] === '') {
  487. $sym = $sym == 'BETWEEN' ? '>=' : '<';
  488. $arr = $arr[0];
  489. }
  490. $where[] = [$k, $sym, $arr];
  491. break;
  492. case 'RANGE':
  493. case 'NOT RANGE':
  494. $v = str_replace(' - ', ',', $v);
  495. $arr = array_slice(explode(',', $v), 0, 2);
  496. if (stripos($v, ',') === false || !array_filter($arr)) {
  497. continue 2;
  498. }
  499. //当出现一边为空时改变操作符
  500. if ($arr[0] === '') {
  501. $sym = $sym == 'RANGE' ? '<=' : '>';
  502. $arr = $arr[1];
  503. } elseif ($arr[1] === '') {
  504. $sym = $sym == 'RANGE' ? '>=' : '<';
  505. $arr = $arr[0];
  506. }
  507. $tableArr = explode('.', $k);
  508. if (count($tableArr) > 1 && $tableArr[0] != $name && !in_array($tableArr[0], $alias) && !empty($this->model)) {
  509. //修复关联模型下时间无法搜索的BUG
  510. $relation = Loader::parseName($tableArr[0], 1, false);
  511. $alias[$this->model->$relation()->getTable()] = $tableArr[0];
  512. }
  513. $where[] = [$k, str_replace('RANGE', 'BETWEEN', $sym) . ' TIME', $arr];
  514. break;
  515. case 'NULL':
  516. case 'IS NULL':
  517. case 'NOT NULL':
  518. case 'IS NOT NULL':
  519. $where[] = [$k, strtolower(str_replace('IS ', '', $sym))];
  520. break;
  521. default:
  522. break;
  523. }
  524. $index++;
  525. }
  526. if (!empty($this->model)) {
  527. $this->model->alias($alias);
  528. }
  529. $model = $this->model;
  530. $where = function (Query $query) use ($where, $alias, $bind, &$model) {
  531. if (!empty($model)) {
  532. $model->alias($alias);
  533. $model->bind($bind);
  534. }
  535. foreach ($where as $k => $v) {
  536. if (is_array($v)) {
  537. if($v[0]=='rules.rule'){
  538. $query->whereExists("select * from mobile_order_rules where mobile_order_rules.mobile_order_id=mobile_order.id and rule='{$v[2]}'");
  539. continue;
  540. }elseif ($v[0]=='admin.admin_id'){
  541. $query->whereExists("select * from mobile_order_admin where mobile_order_admin.mobile_order_id=mobile_order.id and admin_id={$v[2]}");
  542. continue;
  543. }elseif ($v[0]=='mobile_order.sub_mobile_anchor_id'){
  544. $query->whereExists("select * from mobile_order_sub_anchor where mobile_order_sub_anchor.mobile_order_id=mobile_order.id and sub_mobile_anchor_id={$v[2]}");
  545. continue;
  546. }elseif ($v[0]=='status'){
  547. $v[0]='mobile_order.status';
  548. }elseif ($v[0]=='create_time'){
  549. $v[0]='mobile_order.create_time';
  550. }elseif ($v[0]=='update_time'){
  551. $v[0]='mobile_order.update_time';
  552. }elseif ($v[0]=='mobile_order.status' && $v[2]==10){
  553. $query->whereNotIn('status',[0,50,90]);
  554. continue;
  555. }
  556. call_user_func_array([$query, 'where'], $v);
  557. } else {
  558. $query->where($v);
  559. }
  560. }
  561. };
  562. return [$where, $sort, $order, $offset, $limit, $page, $alias, $bind];
  563. }
  564. /**
  565. * 添加
  566. */
  567. public function add()
  568. {
  569. if ($this->request->isPost()) {
  570. $params = $this->request->post("row/a");
  571. if ($params) {
  572. $params = $this->preExcludeFields($params);
  573. if ($this->dataLimit && $this->dataLimitFieldAutoFill) {
  574. $params[$this->dataLimitField] = $this->auth->id;
  575. }
  576. $result = false;
  577. Db::startTrans();
  578. try {
  579. //是否采用模型验证
  580. if ($this->modelValidate) {
  581. $name = str_replace("\\model\\", "\\validate\\", get_class($this->model));
  582. $validate = is_bool($this->modelValidate) ? ($this->modelSceneValidate ? $name . '.add' : $name) : $this->modelValidate;
  583. $this->model->validateFailException(true)->validate($validate);
  584. }
  585. // $params['city'] = implode('/', Area::getTreeId($params['city']));
  586. if (!empty($params['city'])) {
  587. $params['city'] = implode(',',Area::getTreeId(explode('/',$params['city'])[2]));
  588. }
  589. $params['status'] = 10;
  590. $params['pay_time'] = strtotime($params['pay_time']);
  591. $params['create_time'] = time();
  592. $params['update_time'] = time();
  593. $params['order_no'] = order_no();
  594. $params['is_need_push'] = 1;
  595. $params['is_push_zop'] = 0;
  596. $params['is_auto'] = 0;
  597. $result = MobileOrder::insertGetId($params);
  598. // exit();
  599. // $result = $this->model->allowField(true)->save($params);
  600. Db::commit();
  601. } catch (ValidateException $e) {
  602. Db::rollback();
  603. $this->error($e->getMessage());
  604. } catch (PDOException $e) {
  605. Db::rollback();
  606. $this->error($e->getMessage());
  607. } catch (Exception $e) {
  608. Db::rollback();
  609. throw $e;
  610. //$this->error($e->getMessage());
  611. }
  612. if ($result !== false) {
  613. // 推送订单
  614. ZopOrderService::push($result);
  615. $this->success();
  616. } else {
  617. $this->error(__('No rows were inserted'));
  618. }
  619. }
  620. $this->error(__('Parameter %s can not be empty', ''));
  621. }
  622. $produces = Produce::all();
  623. $this->assign('produces', $produces);
  624. return $this->view->fetch();
  625. }
  626. /**
  627. * 详情
  628. *
  629. * @param [type] $ids
  630. * @return void
  631. */
  632. public function detail($ids = null)
  633. {
  634. $model = $this->model->with('produce')->where('id', $ids)->find();
  635. $row = $model->toArray();
  636. $row['produce_activation_name'] = $row['produce_activation'] == 1 ? '已激活' : '未激活';
  637. $row['produce_is_recharge_name'] = $row['produce_is_recharge'] == 1 ? '已首充' : '未首充';
  638. $row['create_time'] = date('Y-m-d H:i:s', $row['create_time']);
  639. $row['update_time'] = date('Y-m-d H:i:s', $row['update_time']);
  640. $this->assign('row', $row);
  641. $this->assign('produce', $model->produce ?? []);
  642. return $this->view->fetch();
  643. }
  644. public function edit($ids = null)
  645. {
  646. $model=$this->model->find($ids);
  647. $originStatus=$model['status'];
  648. $row=$model->toArray();
  649. if($this->request->isGet()){
  650. $row['address']=$model->originData()['address'];
  651. $row['city']=\app\common\model\Area::getNameString($row['city'],'/');
  652. $this->assign('row',$row);
  653. return view();
  654. }else{
  655. $field=[
  656. 'name',
  657. 'phone',
  658. 'id_no',
  659. 'address',
  660. 'trans_no',
  661. 'trans_id',
  662. 'status',
  663. 'city',
  664. 'address_name',
  665. ];
  666. $data=input('row/a');
  667. Db::startTrans();
  668. if(isset($data['city'])){
  669. $data['city']=\app\common\model\Area::whereIn('name|shortname',str_replace('/',',',$data['city']))->column('id');
  670. }
  671. $this->validate($data,[
  672. 'status'=>'in:'.implode(',',array_keys(\app\common\model\MobileOrder::$status))
  673. ]);
  674. foreach ($data as $key=>$value){
  675. if(in_array($key,$field)) {
  676. $model[$key] = $value;
  677. }
  678. }
  679. MobileOrderAdmin::add($model,$originStatus!=$model['status'],$this->auth->id);
  680. $model->save();
  681. // 只有未推送过, 并且产品信息已绑定的订单才推送
  682. if ($model->is_push_zop == 0) {
  683. ZopOrderService::push($ids);
  684. }
  685. Db::commit();
  686. $this->success('');
  687. }
  688. }
  689. public function refund($ids){
  690. $model=$this->model->find($ids);
  691. $this->assign('row',$model);
  692. if($this->request->isGet()){
  693. return view();
  694. }else{
  695. $data=input('row/a');
  696. $this->validate($data,RefundValidate::class);
  697. Db::startTrans();
  698. $model=$this->model->where('id',$ids)->lock(true)->findOrFail();
  699. $model->makeRefund('admin',$this->admin(),$data);
  700. Db::commit();
  701. $this->success();
  702. }
  703. }
  704. public function status(){
  705. return \app\common\model\MobileOrder::$status;
  706. }
  707. public function pay_type(){
  708. return \app\common\model\MobileOrder::$payTypes;
  709. }
  710. public function add_operation(){
  711. $id=input('ids/d');
  712. if($this->request->isGet()){
  713. $this->assign('shortcut',ShortcutContent::getList());
  714. return view();
  715. }else{
  716. $content=input('row.content');
  717. $this->validate(compact('content'),[
  718. 'content'=>'max:250',
  719. ]);
  720. MobileOrderOperation::create([
  721. 'mobile_order_id'=>$id,
  722. 'admin_id'=>$this->auth->id,
  723. 'content'=>$content,
  724. ]);
  725. $this->success();
  726. }
  727. }
  728. public function send(){
  729. $id=input('ids/d');
  730. if($this->request->isGet()){
  731. $row=$this->model->find($id);
  732. $this->assign('row',$row);
  733. return view();
  734. }else{
  735. $data=input('row/a');
  736. Db::startTrans();
  737. $row=$this->model->lock(true)->findOrFail($id);
  738. $row->dealSend($data['trans_no'],$data['trans_id']);
  739. Db::commit();
  740. $this->success();
  741. }
  742. }
  743. #展示开卡资料
  744. public function show_open_data($id){
  745. $row=$this->model->find($id);
  746. return view('',compact('row'));
  747. }
  748. public function see_logistics($ids){
  749. $order=$this->model->find($ids);
  750. $data=TransferCheck::instance()->setNo($order['trans_no'])->setName($order['name'])->setPhone($order['phone'])->setLogisticsCompany(LogisticsCompany::get($order['trans_id']))->get();
  751. $this->assign('data',$data);
  752. return view('mobile_order/see_logistics');
  753. }
  754. /**
  755. * 添加备注等信息
  756. *
  757. * @param [type] $id
  758. * @return void
  759. */
  760. public function sub_remark($id)
  761. {
  762. $order=$this->model->find($id);
  763. if($this->request->isGet()){
  764. // 获取旗子数据
  765. $flags = [
  766. '#FF0000' => "<span style='color: #FF0000;font-size: 24px; '>&#127987;</span>",
  767. '#FFFF00' => "<span style='color: #FFFF00;font-size: 24px; '>&#127987;</span>",
  768. '#00FF33' => "<span style='color: #00FF33;font-size: 24px; '>&#127987;</span>",
  769. '#0033FF' => "<span style='color: #0033FF;font-size: 24px; '>&#127987;</span>",
  770. '#9900FF' => "<span style='color: #9900FF;font-size: 24px; '>&#127987;</span>",
  771. '#00CCFF' => "<span style='color: #00CCFF;font-size: 24px; '>&#127987;</span>",
  772. ];
  773. $this->assign('flags', $flags);
  774. $this->assign('row', $order);
  775. return view();
  776. }else{
  777. $data=input('row/a');
  778. $order->produce_id = $data['produce_id'] ?? 0;
  779. $order->produce_type = $data['produce_type'] ?? '';
  780. $order->remark = $data['remark'] ?? '';
  781. $order->flag = $data['flag'] ?? '';
  782. $order->save();
  783. $this->success();
  784. }
  785. }
  786. public function set_anchor($ids){
  787. $order=$this->model->find($ids);
  788. if(!$order){
  789. $this->error('订单不存在');
  790. }
  791. if($this->request->isGet()){
  792. if($this->admin('is_manager')){
  793. $anchor_id=$order['mobile_anchor_id'];
  794. $this->assign('mobile_anchor_id',$anchor_id);
  795. }
  796. $sub_anchor_id=$order->subAnchor()->value('sub_mobile_anchor_id');
  797. $this->assign('sub_mobile_anchor_id',$sub_anchor_id);
  798. $this->assign('order',$order);
  799. return view('sub_anchor');
  800. }else{
  801. $data=input('row/a');
  802. if(!empty($data['mobile_anchor_id']) && !\app\admin\model\MobileAnchor::find($data['mobile_anchor_id'])){
  803. $this->error('主播不存在');
  804. }
  805. if(!empty($data['sub_mobile_anchor_id']) && !\app\admin\model\MobileAnchor::find($data['sub_mobile_anchor_id'])){
  806. $this->error('主播不存在');
  807. }
  808. if($this->admin('is_manager')){
  809. if(!empty($data['mobile_anchor_id'])){
  810. $order['mobile_anchor_id']=$data['mobile_anchor_id'];
  811. if(!$order->save()){
  812. $this->error('保存失败');
  813. }
  814. }
  815. if(!empty($data['sub_mobile_anchor_id'])){
  816. MobileOrderSubAnchor::sync($order,$data['sub_mobile_anchor_id']);
  817. }
  818. }elseif ($this->admin('is_sub') && $this->admin('id')==$order['s_id']){
  819. if(!empty($data['sub_mobile_anchor_id'])){
  820. if($order->subAnchor()->find()){
  821. $this->error('无法再次设置');
  822. }
  823. MobileOrderSubAnchor::sync($order,$data['sub_mobile_anchor_id']);
  824. }
  825. }
  826. $this->success();
  827. }
  828. }
  829. public function mobile_up($ids){
  830. $order=$this->model->find($ids);
  831. $mobile=$order->mobile()->find();
  832. if($this->request->isGet()) {
  833. $this->assign('row', $mobile);
  834. $this->assign('disabled', 1);
  835. $this->assign('otherSubDisabled', 1);
  836. return view();
  837. }else{
  838. if($mobile){
  839. $mobile->again_sell();
  840. }
  841. $this->success();
  842. }
  843. }
  844. public function refund_log(){
  845. if($this->request->isAjax()){
  846. list($where, $sort, $order, $offset, $limit) = $this->buildparams();
  847. $this->relationSearch=true;
  848. /*$limit=input('limit',10);
  849. $offset=input('offset',0);*/
  850. $filter=json_decode(input('filter'),true)?:[];
  851. $map=[];
  852. if(!empty($filter['od.mobile_id'])){
  853. $map['mobile_order.mobile_id']=['eq',$filter['od.mobile_id']];
  854. }
  855. if(!empty($filter['od.s_id'])){
  856. $map['mobile_order.s_id']=['eq',$filter['od.s_id']];
  857. }
  858. if(isset($filter['pass'])){
  859. $map['pass']=['eq',$filter['pass']];
  860. }
  861. if(!empty($filter['od.no'])){
  862. $map['mobile_order.no']=['like',"%{$filter['od.no']}%"];
  863. }
  864. if(!empty($filter['od.order_no'])){
  865. $map['mobile_order.order_no']=['like',"%{$filter['od.order_no']}%"];
  866. }
  867. if(!empty($filter['reason'])){
  868. $map['reason']=['like',"%{$filter['reason']}%"];
  869. }
  870. if(!empty($filter['admin.nickname'])){
  871. $map['admin.nickname']=['like',"%{$filter['admin.nickname']}%"];
  872. }
  873. if(!empty($filter['od.amount'])){
  874. $amountArr=explode(',',$filter['od.amount']);
  875. if(empty($amountArr[0])){
  876. $amountArr[0]=0;
  877. }
  878. if(empty($amountArr[1])){
  879. $amountArr[1]=10000000000000;
  880. }
  881. $map['mobile_order.amount']=['between',$amountArr];
  882. }
  883. if(!empty($filter['amount_backend'])){
  884. $amountArr=explode(',',$filter['amount_backend']);
  885. if(empty($amountArr[0])){
  886. $amountArr[0]=0;
  887. }
  888. if(empty($amountArr[1])){
  889. $amountArr[1]=10000000000000;
  890. }
  891. $map['amount_backend']=['between',$amountArr];
  892. }
  893. $log=(new MobileOrderRefundLog)->getTable();
  894. if(!empty($filter['create_time'])){
  895. list($s,$e)=explode(' - ',$filter['create_time']);
  896. $map["{$log}.create_time"]=['between',[strtotime($s),strtotime($e)]];
  897. }
  898. //$page=$offset/$limit+1;
  899. $data=MobileOrderRefundLog::where($map)
  900. ->where(function (Query $query){
  901. if(!$this->admin('is_manager')){
  902. $query->where('mobile_order.s_id',$this->admin('id'));
  903. }
  904. })
  905. ->with(['od','admin'])
  906. ->order('id','desc')
  907. ->paginate($limit,false);
  908. /* foreach ($data as $model){
  909. $model->setAttr('s_name',Admin::where('id',$model['od']['s_id'])->value('nickname'));
  910. }*/
  911. return json([
  912. 'total'=>$data->total(),
  913. 'rows'=>$data->items(),
  914. 'filter'=>$filter,
  915. ]);
  916. }
  917. return $this->fetch();
  918. }
  919. #dy结算
  920. public function dy_settle(){
  921. $id=input('ids');
  922. if(empty($id)){
  923. $this->error('id必须');
  924. }
  925. Db::startTrans();
  926. $order=$this->model->lock(true)->find($id);
  927. if(!$order){
  928. $this->error('订单不存在');
  929. }
  930. $order->dySettle();
  931. Db::commit();
  932. $this->success();
  933. }
  934. public function aaa(){
  935. $admin = admin::where('proxy',0)
  936. ->where('sub',1)
  937. ->field(['id','nickname'])
  938. ->select();
  939. return "$admin";
  940. }
  941. /**
  942. * 重新提交
  943. *
  944. * @return void
  945. */
  946. public function resubmit()
  947. {
  948. $id=input('ids');
  949. if(empty($id)){
  950. $this->error('id必须');
  951. }
  952. $result = ZopOrderService::push($id);
  953. if ($result['code'] != 0) {
  954. $this->error($result['message']);
  955. }
  956. $this->success();
  957. }
  958. /**
  959. * 取消订单
  960. *
  961. * @return void
  962. */
  963. public function cancel()
  964. {
  965. $id=input('ids');
  966. if(empty($id)){
  967. $this->error('id必须');
  968. }
  969. $order = MobileOrder::where('id', $id)->find();
  970. $order->is_cancel = 1;
  971. $order->canceled_by = $this->auth->id;
  972. $order->canceled_at = date('Y-m-d H:i:s');
  973. $order->save();
  974. $this->success();
  975. }
  976. /**
  977. * 批量编辑
  978. *
  979. * @param [type] $ids
  980. * @return void
  981. */
  982. public function multi_edit($ids)
  983. {
  984. if($this->request->isGet()) {
  985. // 获取旗子数据
  986. $flags = [
  987. '#FF0000' => "<span style='color: #FF0000;font-size: 24px; '>&#127987;</span>",
  988. '#FFFF00' => "<span style='color: #FFFF00;font-size: 24px; '>&#127987;</span>",
  989. '#00FF33' => "<span style='color: #00FF33;font-size: 24px; '>&#127987;</span>",
  990. '#0033FF' => "<span style='color: #0033FF;font-size: 24px; '>&#127987;</span>",
  991. '#9900FF' => "<span style='color: #9900FF;font-size: 24px; '>&#127987;</span>",
  992. '#00CCFF' => "<span style='color: #00CCFF;font-size: 24px; '>&#127987;</span>",
  993. ];
  994. $this->assign('flags', $flags);
  995. return view();
  996. }else{
  997. $tempData=input('row/a');
  998. $data=[];
  999. $infoData=[];
  1000. if(!empty($tempData['flag'])){
  1001. $data['flag']=$tempData['flag'];
  1002. }
  1003. if(!empty($tempData['produce_id'])){
  1004. $data['produce_id']=$tempData['produce_id'];
  1005. }
  1006. if(!empty($tempData['produce_type'])){
  1007. $data['produce_type']=$tempData['produce_type'];
  1008. }
  1009. if(!empty($tempData['remark'])){
  1010. $data['remark']=$tempData['remark'];
  1011. }
  1012. if($data) {
  1013. $orders = $this->model->whereIn('id', $ids)->select();
  1014. foreach ($orders as $order) {
  1015. foreach ($data as $key => $value) {
  1016. $order->$key = $value;
  1017. $order->save();
  1018. }
  1019. // 只有未推送过, 并且产品信息已绑定的订单才推送
  1020. if ($order->is_push_zop == 0) {
  1021. ZopOrderService::push($ids);
  1022. }
  1023. }
  1024. }
  1025. $this->success();
  1026. }
  1027. }
  1028. }