[ 'class' => AccessControl::className(), 'rules' => [ [ 'allow' => true, 'roles' => ['@'], ] ], ], ]; } public function actionOrder() { return $this->render('order') ; } /** * Retourne au format JSON toutes les informations relatives à une * production donnée. * * @param integer $idDistribution * @return mixed */ public function actionInfosDistribution($idDistribution) { $distribution = Distribution::searchOne([ 'id' => $idDistribution ]); if ($distribution) { $arr = []; $productsArray = ProductDistribution::searchByDistribution($distribution->id) ; $data['products'] = $productsArray; $pointSaleArray = PointSale::find() ->joinWith(['pointSaleDistribution' => function($q) use ($distribution) { $q->where(['id_distribution' => $distribution->id]); }]) ->where([ 'id_producer' => $distribution->id_producer, ]) ->all(); $data['points_sale'] = []; foreach ($pointSaleArray as $pointSale) { if (isset($pointSale->pointSaleDistribution) && isset($pointSale->pointSaleDistribution[0])) { $data['points_sale'][$pointSale->id] = $pointSale->pointSaleDistribution[0]->delivery; } else { $data['points_sale'][$pointSale->id] = false; } } return json_encode($data); } return json_encode([]); } /** * Initialise le formulaire de création/modification de commande. * * @param Order $order * @return array */ public function initForm($order = null) { // Producteurs $producersArray = Yii::$app->user->identity->getBookmarkedProducers(); $idProducer = $this->getProducer()->id; // Producteur $producer = Producer::findOne($idProducer); // Points de vente $pointsSaleArray = PointSale::find() ->with('userPointSale') ->where(['id_producer' => $idProducer]) ->andWhere('restricted_access = 0 OR (restricted_access = 1 AND (SELECT COUNT(*) FROM user_point_sale WHERE point_sale.id = user_point_sale.id_point_sale AND user_point_sale.id_user = :id_user) > 0)') ->params([':id_user' => User::getCurrentId()]) ->all(); // jours de production $deadline = 20; $date = date('Y-m-d'); if (isset($producer)) { if($producer->order_deadline) { $deadline = $producer->order_deadline; } if (date('H') >= $deadline) { $date = date('Y-m-d', strtotime(date('Y-m-d')) + ($producer->order_delay) * (24 * 60 * 60)); } else { $date = date('Y-m-d', strtotime(date('Y-m-d')) + ($producer->order_delay - 1) * (24 * 60 * 60)); } } $distributionDays = Distribution::find() ->where(['active' => 1]) ->andWhere('date > :date') ->andWhere(['id_producer' => $idProducer]) ->addParams([':date' => $date]) ->all(); $distributionDaysArray = array('' => '--'); foreach ($distributionDays as $distribution) { $distributionDaysArray[$distribution->id] = date('d/m/Y', strtotime($distribution->date)); } // produits $products = Product::find() ->leftJoin('product_distribution', 'product.id = product_distribution.id_product') ->where(['product.active' => 1, 'id_producer' => $idProducer]) ->orderBy('product.order ASC')->all(); $productsArray = [] ; foreach ($products as $p) $productsArray[] = $p; // produits selec $posts = Yii::$app->request->post(); $selectedProducts = []; if (isset($posts['Product'])) { foreach ($posts['Product'] as $key => $quantity) { $key = (int) str_replace('product_', '', $key); $product = Product::find()->where(['id' => $key])->one(); if ($product && $quantity) $selectedProducts[$product->id] = (int) $quantity; } } elseif (!is_null($order)) { $productOrderArray = ProductOrder::searchAll([ 'id_order' => $order->id ]) ; foreach ($productOrderArray as $productOrder) { $selectedProducts[$productOrder->id_product] = (int) $productOrder->quantity; } } $availableProducts = [] ; $distribution = null; if (!is_null($order) && $order->id_distribution) { $availableProducts = ProductDistribution::searchByDistribution($order->id_distribution); $distribution = Distribution::find()->where(['id' => $order->id_distribution])->one(); } $orders = Order::searchAll([ 'id_user' => User::getCurrentId() ]) ; if ($idProducer) { $userProducer = UserProducer::searchOne([ 'id_producer' => $idProducer, 'id_user' => User::getCurrentId() ]) ; $credit = $userProducer->credit; } else { $credit = 0; } return [ 'pointsSaleArray' => $pointsSaleArray, 'distributionDaysArray' => $distributionDaysArray, 'productsArray' => $productsArray, 'selectedProducts' => $selectedProducts, 'availableProducts' => $availableProducts, 'distribution' => $distribution, 'ordersArray' => $orders, 'producersArray' => $producersArray, 'idProducer' => $idProducer, 'producer' => $producer, 'credit' => $credit ]; } /** * Affiche l'historique des commandes de l'utilisateur * * @return ProducerView */ public function actionHistory() { $dataProviderOrders = new ActiveDataProvider([ 'query' => Order::find() ->with('productOrder', 'pointSale', 'creditHistory') ->joinWith('distribution', 'distribution.producer') ->where([ 'id_user' => Yii::$app->user->id, 'distribution.id_producer' => $this->getProducer()->id ]) ->orderBy('distribution.date DESC'), 'pagination' => [ 'pageSize' => 10, ], ]); return $this->render('history', [ 'dataProviderOrders' => $dataProviderOrders, 'orderOk' => Yii::$app->getRequest()->get('orderOk', false), 'cancelOk' => Yii::$app->getRequest()->get('cancelOk', false), ]); } /** * Supprime un producteur. * * @param integer $id */ public function actionRemoveProducer($id = 0) { $userProducer = UserProducer::find() ->where(['id_producer' => $id, 'id_user' => User::getCurrentId()]) ->one(); $userProducer->active = 0; $userProducer->save(); $this->redirect(['order/index']); } /** * Crée une commande. * * @return mixed */ public function actionAjaxProcess() { \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON; $order = new Order ; $idProducer = $this->getProducer()->id ; $posts = Yii::$app->request->post(); if ($idProducer) { $this->_verifyProducerActive($idProducer); } if ($order->load($posts)) { $order = Order::find() ->where('id_distribution = :id_distribution') ->andWhere('id_user = :id_user') ->params([ ':id_distribution' => $posts['Order']['id_distribution'], ':id_user' => User::getCurrentId() ]) ->one(); if (!$order) { $order = new Order; $order->load(Yii::$app->request->post()); $order->id_user = User::getCurrentId(); $order->date = date('Y-m-d H:i:s'); $order->origin = Order::ORIGIN_USER; } $errors = $this->processForm($order); if(count($errors)) { return ['status' => 'error', 'errors' => $errors] ; } } return ['status' => 'success'] ; } /** * Modifie une commande. * * @param integer $id * @return mixed * @throws UserException */ public function actionUpdate($id) { $order = Order::searchOne([ 'id' => $id ]) ; if ($order->getState() != Order::STATE_OPEN) { throw new UserException('Cette commande n\'est pas modifiable.'); } $this->_verifyProducerActive($order->distribution->id_producer); if ($order && $order->load(Yii::$app->request->post())) { $order->date_update = date('Y-m-d H:i:s'); $this->processForm($order); } return $this->render('update', array_merge($this->initForm($order), [ 'model' => $order, 'orderNotfound' => !$order, ])); } /** * Vérifie si un producteur est actif. * * @param integer $idProducer * @throws NotFoundHttpException */ public function _verifyProducerActive($idProducer) { $producer = Producer::findOne($idProducer); if ($producer && !$producer->active) { throw new NotFoundHttpException('Ce producteur est actuellement hors ligne.'); } } /** * Traite le formulaire de création/modification de commande. * * @param Commande $order */ public function processForm($order) { $posts = Yii::$app->request->post(); $productsArray = []; $totalQuantity = 0; foreach ($posts['products'] as $key => $quantity) { $product = Product::find()->where(['id' => (int) $key])->one(); $totalQuantity += $quantity; if ($product && $quantity) { $productsArray[] = $product; } } // date $errorDate = false; if (isset($order->id_distribution)) { // date de commande $distribution = Distribution::find()->where(['id' => $order->id_distribution])->one(); if (date('H') >= 20) { $date = date('Y-m-d', strtotime(date('Y-m-d')) + 60 * 60 * 24); } else { $date = date('Y-m-d'); } if ($distribution->date < $date) { $errorDate = true; } } // point de vente $errorPointSale = false; if (isset($distribution) && $distribution) { $pointSaleDistribution = PointSaleDistribution::searchOne([ 'id_distribution' => $distribution->id, 'id_point_sale' => $posts['Order']['id_point_sale'] ]) ; if (!$pointSaleDistribution || !$pointSaleDistribution->delivery) { $errorPointSale = true; } $pointSale = PointSale::findOne($posts['Order']['id_point_sale']); if ($pointSale) { if (strlen($pointSale->code) && !$pointSale->validateCode($posts['code_point_sale'])) { $errorPointSale = true; } } else { $errorPointSale = true; } } $errors = [] ; if ($order->validate() && count($productsArray) && !$errorDate && !$errorPointSale) { // gestion point de vente $pointSale = PointSale::searchOne([ 'id' => $order->id_point_sale ]) ; $order->comment_point_sale = ($pointSale && strlen($pointSale->getComment())) ? $pv->getComment() : '' ; // la commande est automatiquement réactivée lors d'une modification $order->date_delete = null ; // sauvegarde de la commande $order->save(); // ajout de l'utilisateur à l'établissement Producer::addUser(User::getCurrentId(), $distribution->id_producer) ; // suppression de tous les enregistrements ProductOrder if (!is_null($order)) { ProductOrder::deleteAll(['id_order' => $order->id]); } // produits dispos $availableProducts = ProductDistribution::searchByDistribution($distribution->id) ; // sauvegarde des produits foreach ($productsArray as $product) { if (isset($availableProducts[$product->id])) { $productOrder = new ProductOrder(); $productOrder->id_order = $order->id; $productOrder->id_product = $product->id; $productOrder->price = $product->price; $quantity = (int) $posts['products'][$product->id] ; if ($availableProducts[$product->id]['quantity_max'] && $quantity > $availableProducts[$product->id]['quantity_remaining']) { $quantity = $availableProducts[$product->id]['quantity_remaining']; } $productOrder->quantity = $quantity; $productOrder->sale_mode = Product::SALE_MODE_UNIT ; $productOrder->save(); } } // credit $credit = Producer::getConfig('credit'); $order = Order::searchOne([ 'id' => $order->id ]) ; if ($credit && ($pointSale->credit || $order->getAmount(Order::AMOUNT_PAID))) { $amountPaid = $order->getAmount(Order::AMOUNT_PAID); // à payer if ($order->getPaymentStatus() == Order::PAYMENT_UNPAID) { $amountRemaining = $order->getAmount(Order::AMOUNT_REMAINING) ; $credit = Yii::$app->user->identity->getCredit($distribution->id_producer); if ($amountRemaining > $credit) { $amountRemaining = $credit; } if ($amountRemaining > 0) { $order->saveCreditHistory( CreditHistory::TYPE_PAYMENT, $amountRemaining, $distribution->id_producer, User::getCurrentId(), User::getCurrentId() ); } } // surplus à rembourser elseif ($order->getPaymentStatus() == Order::PAYMENT_SURPLUS) { $amountSurplus = $order->getAmount(Order::AMOUNT_SURPLUS) ; $order->saveCreditHistory( CreditHistory::TYPE_REFUND, $amountSurplus, $distribution->id_producer, User::getCurrentId(), User::getCurrentId() ); } } // redirection //$this->redirect(Yii::$app->urlManager->createUrl(['order/history', 'orderOk' => true])); } else { if (!count($productsArray)) { $errors[] = "Vous n'avez choisi aucun produit" ; } if ($errorDate) { $errors[] = "Vous ne pouvez pas commander pour cette date." ; } if ($errorPointSale) { $errors[] = "Point de vente invalide." ; } } return $errors ; } /** * Annule une commande. * * @param integer $id * @throws \yii\web\NotFoundHttpException * @throws UserException */ public function actionCancel($id) { $order = Order::searchOne([ 'id' => $id ]) ; if(!$order) { throw new \yii\web\NotFoundHttpException('Commande introuvable'); } if ($order->getState() != Order::STATE_OPEN) { throw new UserException('Vous ne pouvez plus annuler cette commande.'); } if ($order && User::getCurrentId() == $order->id_user) { // remboursement if ($order->getAmount(Order::AMOUNT_PAID)) { $order->saveCreditHistory( CreditHistory::TYPE_REFUND, $order->getAmount(Order::AMOUNT_PAID), $order->distribution->id_producer, User::getCurrentId(), User::getCurrentId() ); } // delete $order->date_delete = date('Y-m-d H:i:s'); $order->save() ; Yii::$app->session->setFlash('success','Votre commande a bien été annulée.') ; } $this->redirect(Yii::$app->urlManager->createUrl(['order/history'])); } /** * Vérifie le code saisi pour un point de vente. * * @param integer $idPointSale * @param string $code * @return boolean */ public function actionAjaxValidateCodePointSale($idPointSale, $code) { \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON; $pointSale = PointSale::findOne($idPointSale); if ($pointSale) { if ($pointSale->validateCode($code)) { return 1; } } return 0; } public function actionAjaxInfos($date = '') { \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON; $json = [] ; $format = 'Y-m-d' ; $dateObject = DateTime::createFromFormat($format, $date); // Producteur $producer = Producer::searchOne([ 'id' => $this->getProducer()->id ]) ; $json['producer'] = [ 'order_infos' => $producer->order_infos, 'credit' => $producer->credit ] ; // Distributions $deadline = 20; $dateMini = date('Y-m-d'); if (isset($producer)) { if($producer->order_deadline) { $deadline = $producer->order_deadline; } if (date('H') >= $deadline) { $dateMini = date('Y-m-d', strtotime(date('Y-m-d')) + ($producer->order_delay) * (24 * 60 * 60)); } else { $dateMini = date('Y-m-d', strtotime(date('Y-m-d')) + ($producer->order_delay - 1) * (24 * 60 * 60)); } } $distributionsArray = Distribution::searchAll([ 'active' => 1 ], [ 'conditions' => ['date > :date'], 'params' => [':date' => $dateMini], ]) ; $json['distributions'] = $distributionsArray ; // Commandes de l'utilisateur $ordersUserArray = Order::searchAll([ 'id_user' => User::getCurrentId() ], [ 'conditions' => [ 'distribution.date > :date' ], 'params' => [ ':date' => $dateMini ] ]); foreach($ordersUserArray as &$order) { $order = array_merge($order->getAttributes(), [ 'amount_total' => $order->getAmount(Order::AMOUNT_TOTAL), 'date_distribution' => $order->distribution->date, 'pointSale' => $order->pointSale->getAttributes() ]) ; } $json['orders'] = $ordersUserArray; // User $userProducer = UserProducer::searchOne([ 'id_producer' => $producer->id, 'id_user' => User::getCurrentId() ]) ; $json['credit'] = $userProducer->credit ; if($dateObject && $dateObject->format($format) === $date) { // Commande de l'utilisateur $orderUser = Order::searchOne([ 'distribution.date' => $date, 'id_user' => User::getCurrentId(), ]); if($orderUser) { $json['order'] = array_merge($orderUser->getAttributes(), [ 'amount_total' => $orderUser->getAmount(Order::AMOUNT_TOTAL), 'amount_paid' => $orderUser->getAmount(Order::AMOUNT_PAID), ]) ; } // distribution $distribution = Distribution::initDistribution($date) ; $json['distribution'] = $distribution ; $pointsSaleArray = PointSale::find() ->joinWith(['pointSaleDistribution' => function($query) use ($distribution) { $query->where(['id_distribution' => $distribution->id]); } ]) ->with(['userPointSale' => function($query) { $query->onCondition(['id_user' => User::getCurrentId()]) ; }]) ->where([ 'id_producer' => $distribution->id_producer, ]) ->all(); foreach($pointsSaleArray as &$pointSale) { $pointSale = array_merge($pointSale->getAttributes(),[ 'pointSaleDistribution' => [ 'id_distribution' => $pointSale->pointSaleDistribution[0]->id_distribution, 'id_point_sale' => $pointSale->pointSaleDistribution[0]->id_point_sale, 'delivery' => $pointSale->pointSaleDistribution[0]->delivery ], 'userPointSale' => ($pointSale->userPointSale ? $pointSale->userPointSale[0] : '') ]) ; } $json['points_sale'] = $pointsSaleArray; // Commandes totales $ordersArray = Order::searchAll([ 'distribution.date' => $date, ]); // Produits $productsArray = Product::find() ->where([ 'id_producer' => $this->getProducer()->id, ]) ->joinWith(['productDistribution' => function($query) use($distribution) { $query->andOnCondition('product_distribution.id_distribution = '.$distribution->id) ; }]) ->orderBy('product_distribution.active DESC, order ASC') ->asArray() ->all(); $indexProduct = 0 ; foreach($productsArray as &$product) { $quantityOrder = Order::getProductQuantity($product['id'], $ordersArray) ; $product['quantity_ordered'] = $quantityOrder ; $product['quantity_remaining'] = $product['quantity_max'] - $quantityOrder ; if($orderUser) { $quantityOrderUser = Order::getProductQuantity($product['id'], [$orderUser]) ; $product['quantity_ordered'] = $quantityOrder ; $product['quantity_remaining'] = $product['quantity_max'] - $quantityOrder + $quantityOrderUser ; $product['quantity_form'] = $quantityOrderUser ; } else { $product['quantity_form'] = 0 ; } if($product['quantity_remaining'] < 0) $product['quantity_remaining'] = 0 ; $product['index'] = $indexProduct ++ ; } $json['products'] = $productsArray; } return $json ; } }