Produceorder.php 41 KB

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