DeclarationUpgradeForm.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  1. <?php
  2. namespace common\models\forms;
  3. use common\components\Model;
  4. use common\helpers\Cache;
  5. use common\helpers\Date;
  6. use common\helpers\Form;
  7. use common\helpers\user\Cash;
  8. use common\helpers\user\Info;
  9. use common\models\Config;
  10. use common\models\DeclarationLevel;
  11. use common\models\DeclarationPackage;
  12. use common\models\DecLevelLog;
  13. use common\models\DecOrder;
  14. use common\models\EmployLevel;
  15. use common\models\Order;
  16. use common\models\OrderGoods;
  17. use common\models\Period;
  18. use common\models\ReceiveAddress;
  19. use common\models\Region;
  20. use common\models\ShopGoods;
  21. use common\models\User;
  22. use common\models\UserInfo;
  23. use common\models\UserNetwork;
  24. use yii\base\Exception;
  25. /**
  26. * 升级管理,进行升级
  27. */
  28. class DeclarationUpgradeForm extends Model
  29. {
  30. public $type;
  31. public $decLv;
  32. public $decWay;
  33. public $packageId;
  34. public $goodsId;
  35. public $goodsNum;
  36. public $insertUserName;
  37. public $consignee;
  38. public $acceptMobile;
  39. public $province;
  40. public $city;
  41. public $county;
  42. public $address;
  43. public $nowPerf;
  44. public $nextPerf;
  45. public $decUserName;
  46. public $remark;
  47. // 传过来的全部数据
  48. public $allData;
  49. private $_decId;
  50. public $_insertUserId;
  51. private $_decAmount;
  52. private $_decPv;
  53. private $_orderGoods;
  54. const TYPE_ZC = 'ZC';
  55. private $_userForm = null;
  56. // 全部的安置网上级
  57. private $_tempNetworkParentUser = [];
  58. /**
  59. * @inheritdoc
  60. */
  61. public function rules()
  62. {
  63. return [
  64. [['remark','type','decLv','decWay','packageId','goodsId', 'goodsNum', 'insertUserName','consignee','acceptMobile','province','city','county','address','nowPerf','nextPerf'], 'trim'],
  65. [['type','decLv','decWay','insertUserName','nowPerf','province','city','county','address','acceptMobile'], 'required'],
  66. [['decUserName'], 'issetDec'], // 必须是报单中心
  67. [['decWay'], 'hasProduct'],// 必须选择商品
  68. [['decLv'], 'alreadyMaxDec'], //判断要升级用户是否已经是最高级
  69. ];
  70. }
  71. public function attributeLabels()
  72. {
  73. return [
  74. 'type' => '升级类型',
  75. 'decLv' => '升级级别',
  76. 'decWay' => '报单方式',
  77. 'packageId' => '升级套餐',
  78. 'goodsId' => '商品ID',
  79. 'goodsNum' => '商品数量',
  80. 'insertUserName' => '要升级的会员编号',
  81. 'consignee' => '收货人',
  82. 'acceptMobile' => '收货人手机',
  83. 'province' => '收货省',
  84. 'city' => '收货市',
  85. 'county' => '收货区县',
  86. 'address' => '收货详细地址',
  87. ];
  88. }
  89. /**
  90. * 添加报单
  91. * @param $allData
  92. * @return bool|null
  93. * @throws Exception
  94. * @throws \yii\db\Exception
  95. */
  96. public function add($allData){
  97. if(!$this->validate()){
  98. return null;
  99. }
  100. $loginUserId = \Yii::$app->user->id;
  101. // 首购单
  102. if($this->type == self::TYPE_ZC){
  103. //报单商品及PV判断
  104. $decLevelConfig = Cache::getDecLevelConfig();
  105. $decLevel = $decLevelConfig[$this->decLv];
  106. $toDecLevel = $this->decLv;
  107. if(!$this->decLv){
  108. throw new Exception('请选择升级级别');
  109. }
  110. $baseInfo = Info::baseInfoZhByUserName($this->insertUserName);
  111. $userId = $baseInfo['ID'];
  112. $userDecPvSum = User::sumDevPvByUserId($userId); // 用户所有报单PV总和
  113. if ($userDecPvSum != $this->nowPerf) {
  114. throw new Exception('请联系客服人员核对升级会员业绩');
  115. }
  116. // 获取用户是否是观察期
  117. $observe = Config::getConfigByType('observe'); // 获取观察期配置信息
  118. $observeLimit = $observe['observePeriodLimit']['value']; // 月份限制
  119. $isObserve = User::checkIsObserve($baseInfo['CREATED_AT'], $observeLimit); // 判断用户是否再观察期中
  120. $diffPerf = $isObserve ? $this->nowPerf : 0; // 观察期内升级要加上用户累计的PV,全额则基础PV为0,全额购买
  121. if ($this->decWay != 2) {
  122. throw new Exception('升级方式不正确,请联系客服人员');
  123. }
  124. if($this->decWay==1) {
  125. // 先不加套餐升级方式
  126. // $decPackage = DeclarationPackage::findOneAsArray('ID=:ID', [':ID'=>$this->packageId]);
  127. // $this->_decAmount = $decPackage['AMOUNT'];
  128. // $this->_decPv = $decPackage['PV'];
  129. // $this->_orderGoods[] = [
  130. // 'GOODS_ID' => $this->packageId,
  131. // 'PRICE' => $this->_decAmount,
  132. // 'REAL_PRICE' => $this->_decAmount,
  133. // 'PV' => $this->_decPv,
  134. // 'REAL_PV' => $this->_decPv,
  135. // 'BUY_NUMS' => 1,
  136. // 'SKU_CODE' => $decPackage['PACKAGE_NO'],
  137. // 'GOODS_TITLE' => $decPackage['PACKAGE_NAME']
  138. // ];
  139. } else {
  140. $ids = $this->goodsId;
  141. $totalAmount = 0;
  142. $totalPv = 0;
  143. foreach ($this->goodsNum as $k => $v) {
  144. if ($v) {
  145. $goods = ShopGoods::findOneAsArray('ID=:ID AND STATUS=1',[':ID'=> $ids[$k]]);
  146. if($goods['STORE_NUMS']>0){
  147. $totalAmount += $goods['SELL_PRICE'] * intval($v);
  148. $totalPv += $goods['PRICE_PV'] * intval($v);
  149. $this->_orderGoods[] = [
  150. 'GOODS_ID' => $goods['ID'],
  151. 'PRICE' => $goods['SELL_PRICE'],
  152. 'REAL_PRICE' => $goods['SELL_PRICE'],
  153. 'PV' => $goods['PRICE_PV'],
  154. 'REAL_PV' => $goods['PRICE_PV'],
  155. 'POINT' => $goods['POINT'],
  156. 'BUY_NUMS' => intval($v),
  157. 'SKU_CODE' => $goods['GOODS_NO'],
  158. 'GOODS_TITLE' => $goods['GOODS_NAME']
  159. ];
  160. }
  161. }
  162. }
  163. // 这里特殊是用户原报单PV之和+用户购买的商品总PV
  164. $checkPv = $totalPv + $diffPerf;
  165. if($checkPv < $decLevel['PERF']) {
  166. throw new Exception('总PV不能小于所选级别PV');
  167. }
  168. foreach ($decLevelConfig as $key=>$val){
  169. if($checkPv>=$val['PERF']){
  170. $toDecLevel = $key;
  171. }
  172. }
  173. if($this->decLv!=$toDecLevel){
  174. throw new Exception('总PV不能超过已选级别下一个级别的PV值');
  175. }
  176. $this->_decAmount = $totalAmount;
  177. $this->_decPv = $totalPv;
  178. }
  179. //看现金余额是否充足
  180. if (Cash::getAvailableBalance($loginUserId) < $this->_decAmount){
  181. throw new Exception('报单人现金不足,无法完成报单');
  182. }
  183. $baseInfo = Info::baseInfoZhByUserName($this->insertUserName);
  184. $this->_insertUserId = $baseInfo['ID']; // 被报单人,通过insername 查找用户id
  185. $insertConId = $baseInfo['CON_UID'];
  186. $insertRecId = $baseInfo['REC_UID'];
  187. if(!($decResult = $this->addDecOrder($insertConId,$insertRecId, $baseInfo['DEC_LV'],$isObserve,$this->remark))) {
  188. throw new Exception("操作失败");
  189. }
  190. }
  191. return true;
  192. }
  193. /**
  194. * 添加报单订单
  195. * @return bool|UserInfo|null
  196. * @throws \yii\db\Exception
  197. */
  198. public function addDecOrder($insertConId,$insertRecId,$oriDecLv,$isObserve,$remark=''){
  199. $warehouse = Region::getWarehouseByCode($this->province);//仓库
  200. if(!$warehouse){
  201. throw new Exception('地区暂时不支持配送,具体联系客服');
  202. }
  203. $upgradeType = $isObserve ? 1 : 2; // 1补差 2全额
  204. $periodObj = Period::instance();
  205. $nowPeriodNum = $periodObj->getNowPeriodNum();
  206. $nowCalcMonth = $periodObj->getYearMonth($nowPeriodNum);
  207. $ord = date('Ymd').substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 10);
  208. // 加入报单信息
  209. $db = \Yii::$app->db;
  210. $transaction = $db->beginTransaction();
  211. try {
  212. $decOrderModel = new DecOrder();
  213. $decOrderModel->DEC_SN = 'DS'.$ord;
  214. $decOrderModel->ORDER_SN = 'OS'.$ord;
  215. $decOrderModel->TYPE = $this->type;
  216. $decOrderModel->USER_ID = \Yii::$app->user->id; // 报单人
  217. $decOrderModel->TO_USER_ID = $this->_insertUserId; // 被报单人
  218. $decOrderModel->DEC_AMOUNT = $this->_decAmount;
  219. $decOrderModel->DEC_PV = $this->_decPv;
  220. $decOrderModel->PERIOD_NUM = $nowPeriodNum;
  221. $decOrderModel->CALC_MONTH = $nowCalcMonth;
  222. $decOrderModel->P_CALC_MONTH = Date::ociToDate($nowCalcMonth, Date::OCI_TIME_FORMAT_SHORT_MONTH);
  223. $decOrderModel->PAID_WALLET = 'cash';
  224. $decOrderModel->CON_USER_ID = $insertConId;
  225. $decOrderModel->REC_USER_ID = $insertRecId;
  226. $decOrderModel->DEC_ID = $this->_decId;
  227. $decOrderModel->IS_DEL = 0;
  228. $decOrderModel->DETAIL_TYPE = 2;
  229. $decOrderModel->CREATED_AT = Date::nowTime();
  230. $decOrderModel->UPGRADE_TYPE = $upgradeType;
  231. $decOrderModel->REMARK = $remark;
  232. $decOrderModel->ORI_LV = $oriDecLv; // 变更前的级别
  233. $decOrderModel->UPGRADE_LV = $this->decLv; // 变更后的级别
  234. if(!$decOrderModel->save()){
  235. $transaction->rollBack();
  236. throw new Exception(Form::formatErrorsForApi($decOrderModel->getErrors()));
  237. }
  238. $orderModel = new Order();
  239. $orderModel->SN = 'OS'.$ord;
  240. $orderModel->DEC_SN = 'DS'.$ord;
  241. $orderModel->ORDER_TYPE = $this->type;
  242. $orderModel->USER_ID = $this->_insertUserId;
  243. $orderModel->USER_NAME = $this->insertUserName; // 要升级的用户
  244. $orderModel->ORDER_AMOUNT = $this->_decAmount;
  245. $orderModel->PV = $this->_decPv;
  246. $orderModel->PAY_AMOUNT = $this->_decAmount;
  247. $orderModel->PAY_PV = $this->_decPv;
  248. $orderModel->PAY_AT = Date::nowTime();
  249. $orderModel->PAY_TYPE = 'cash';
  250. $orderModel->PERIOD_NUM = $nowPeriodNum;
  251. $orderModel->P_CALC_MONTH = Date::ociToDate($nowCalcMonth, Date::OCI_TIME_FORMAT_SHORT_MONTH);
  252. $orderModel->FREIGHT = 0;
  253. $orderModel->PAY_FREIGHT = 0;
  254. $orderModel->CONSIGNEE = $this->consignee;
  255. $orderModel->MOBILE = $this->acceptMobile;
  256. $orderModel->PROVINCE = $this->province;
  257. $orderModel->CITY = $this->city;
  258. $orderModel->COUNTY = intval($this->county) ?? 0;
  259. $orderModel->ADDRESS = $this->address;
  260. $orderModel->WAREHOUSE = $warehouse;
  261. $orderModel->STATUS = 1;
  262. $orderModel->CREATED_AT = Date::nowTime();
  263. $orderModel->CREATE_USER = Info::getUserNameByUserId(\Yii::$app->user->id);
  264. if(!$orderModel->save()){
  265. $transaction->rollBack();
  266. throw new Exception(Form::formatErrorsForApi($orderModel->getErrors()));
  267. }
  268. /**
  269. * 2022-05-10
  270. * York
  271. */
  272. foreach ($this->goodsNum as $k => $v) {
  273. if ($v) {
  274. $goods = ShopGoods::findOneAsArray('ID=:ID',[':ID'=> $this->goodsId[$k]]);
  275. $storenums = $goods['STORE_NUMS'] - $this->goodsNum[$k];
  276. if($goods['STATUS']==1){
  277. if($goods['STORE_NUMS'] >=$this->goodsNum[$k]){
  278. $data = ShopGoods::find()->where(['ID' => $this->goodsId[$k]])->one();
  279. $data->STORE_NUMS = $storenums;
  280. $data->update();
  281. if ($storenums==0){
  282. $data->STATUS = 0;
  283. $data->UPDATED_AT = Date::nowTime();
  284. $data->update();
  285. }
  286. }else{
  287. throw new Exception("商品".$goods['GOODS_NAME']."库存不足");
  288. }
  289. }else{
  290. throw new Exception("商品".$goods['GOODS_NAME']."已下架");
  291. }
  292. }
  293. }
  294. // exit();
  295. // 加入商品到订单商品表
  296. foreach($this->_orderGoods as $key=>$value){
  297. $this->_orderGoods[$key]['ORDER_SN'] = $orderModel->SN;
  298. $this->_orderGoods[$key]['P_CALC_MONTH'] = Date::ociToDate($nowCalcMonth, Date::OCI_TIME_FORMAT_SHORT_MONTH);
  299. }
  300. OrderGoods::batchInsert($this->_orderGoods);
  301. //写入收货地址信息
  302. $addressModel = new ReceiveAddress();
  303. $addressModel->USER_ID = $this->_insertUserId;
  304. $addressModel->USER_NAME = $this->insertUserName;
  305. $addressModel->CONSIGNEE = $this->consignee;
  306. $addressModel->MOBILE = $this->acceptMobile;
  307. $addressModel->PROVINCE = $this->province;
  308. $addressModel->CITY = $this->city;
  309. $addressModel->COUNTY = intval($this->county) ?? 0;
  310. $addressModel->ADDRESS = $this->address;
  311. $addressModel->IS_DEFAULT = 0;
  312. if(!$addressModel->save()){
  313. $transaction->rollBack();
  314. throw new Exception(Form::formatErrorsForApi($addressModel->getErrors()));
  315. }
  316. // 扣报单人现金钱包
  317. Cash::changeUserCash(\Yii::$app->user->id, 'CASH', -abs($this->_decAmount), ['REMARK' =>'为'.$this->insertUserName.'升级报单']);
  318. // 为被升级人进行升级操作
  319. $decLevelLog = new DecLevelLog();
  320. $decLog = [
  321. 'userId' => $this->_insertUserId,//会员ID
  322. 'fromId' => $oriDecLv, // 变动前的级别
  323. 'levelId' => $this->decLv,// 变动后的级别
  324. 'actionId' => \Yii::$app->user->id,
  325. 'remark' => $this->remark,
  326. 'lvPv' => $this->_decPv
  327. ];
  328. $modifyDecLv = $decLevelLog->frontendChange($decLog);
  329. if (empty($modifyDecLv)) {
  330. $transaction->rollBack();
  331. throw new Exception("为会员升级失败");
  332. }
  333. $transaction->commit();
  334. } catch(Exception $e) {
  335. $transaction->rollBack();
  336. throw new Exception($e->getMessage());
  337. return false;
  338. }
  339. return $modifyDecLv;
  340. }
  341. /**
  342. * 判断报单中心是否存在
  343. * @param $attribute
  344. */
  345. public function issetDec($attribute){
  346. $decUser = User::find()
  347. ->select('ID')
  348. ->where('IS_DEC=1 AND USER_NAME=:USER_NAME', [':USER_NAME' => $this->decUserName])
  349. ->asArray()
  350. ->one();
  351. if (!$decUser) {
  352. $this->addError($attribute, '报单中心不存在');
  353. return false;
  354. } else {
  355. // 判断报单中心是否在新加入会员的安置网上级中
  356. $this->loopFindParentToNetwork($this->insertUserName);
  357. //反转数组,in_array搜索错误
  358. //in_array($this->decUserName, $this->_tempNetworkParentUser[$this->insertUserName]);
  359. $flipParent = array_flip(array_filter($this->_tempNetworkParentUser[$this->insertUserName]));
  360. if (!isset($flipParent[$this->decUserName])) {
  361. $this->addError($attribute, '为' . $this->insertUserName . '升级报单,报单中心' . $this->decUserName . '不在' . $this->insertUserName . '的安置网上级中');
  362. return ;
  363. }
  364. $this->_decId = $decUser['ID'];
  365. }
  366. }
  367. // 判断是否已选择商品或套餐
  368. public function hasProduct($attribute) {
  369. if ($this->decWay==1 && empty($this->packageId)) {
  370. $this->addError($attribute, '购买套餐升级,请选择套餐');
  371. return false;
  372. }
  373. if ($this->decWay!=1 && empty($this->goodsId)) {
  374. $this->addError($attribute, '购买商品升级,请选择商品');
  375. return false;
  376. }
  377. return true;
  378. }
  379. // 判断要升级的会员,是否已是最高级别
  380. public function alreadyMaxDec($attribute) {
  381. $baseInfo = Info::baseInfoZhByUserName($this->insertUserName);
  382. $userDecId = $baseInfo['DEC_LV'];// 用户当前的级别
  383. $maxPerfInfo = DeclarationLevel::getMaxDecPref();
  384. $maxDecId = $maxPerfInfo['ID']; // 级别配置中最高级别ID
  385. if ($maxDecId == $userDecId) {
  386. $this->addError($attribute, '已是最高级别,无需升级');
  387. return false;
  388. }
  389. return true;
  390. }
  391. /**
  392. * 寻找被升级人的上级
  393. * @param null $conUserName
  394. * @return bool
  395. */
  396. private function loopFindParentToNetwork($conUserName = null) {
  397. if($conUserName == null ){
  398. $conUserName = $this->insertUserName;
  399. }
  400. $baseUser = Info::getBaseUserByUserName($conUserName);
  401. $userNetworkInfo = UserNetwork::findOneAsArray('USER_ID=:USER_ID', [':USER_ID' => $baseUser['ID']]);
  402. unset($baseUser);
  403. $allParentUserIdsArr = explode(',', $userNetworkInfo['PARENT_UIDS']);
  404. unset($userNetworkInfo);
  405. $allParentUserIds = array_reverse($allParentUserIdsArr);
  406. unset($allParentUserIdsArr);
  407. if($allParentUserIds){
  408. foreach($allParentUserIds as $parentUserId) {
  409. $parentBaseUser = Info::getBaseUserById($parentUserId);
  410. $this->_tempNetworkParentUser[$this->insertUserName][] = $parentBaseUser['USER_NAME'] ;
  411. unset($parentUserId, $parentBaseUser);
  412. }
  413. }
  414. unset($allParentUserIds);
  415. return true;
  416. }
  417. // /**
  418. // * 删单
  419. // * @return bool
  420. // * @throws \yii\db\Exception
  421. // */
  422. // public function delete(){
  423. // if(!$this->validate()){
  424. // return false;
  425. // }
  426. // $transaction = \Yii::$app->db->beginTransaction();
  427. // try {
  428. // $oneOrder = $this->_oneOrder;
  429. // // 首购单要删除会员
  430. // if($this->type == self::TYPE_ZC){
  431. // UserInfo::deleteUser($oneOrder['TO_USER_ID']);
  432. // }
  433. // // 如果是复销单的话,还需要考虑给会员的复销池减去金额
  434. // elseif($this->type == self::TYPE_FX){
  435. // Reconsume::changePoolPV($oneOrder['TO_USER_ID'], -abs($oneOrder['DEC_PV']), ['REMARK'=>'删单扣除', 'DEAL_TYPE'=>Reconsume::TYPE_AUDIT_PV]);
  436. // }
  437. // $transaction->commit();
  438. // } catch (Exception $e) {
  439. // $transaction->rollBack();
  440. // $this->addError('delete', $e->getMessage());
  441. // return false;
  442. // }
  443. // return true;
  444. // }
  445. }