|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774 |
- <?php
-
- /**
- * Copyright distrib (2018)
- *
- * contact@opendistrib.net
- *
- * Ce logiciel est un programme informatique servant à aider les producteurs
- * à distribuer leur production en circuits courts.
- *
- * Ce logiciel est régi par la licence CeCILL soumise au droit français et
- * respectant les principes de diffusion des logiciels libres. Vous pouvez
- * utiliser, modifier et/ou redistribuer ce programme sous les conditions
- * de la licence CeCILL telle que diffusée par le CEA, le CNRS et l'INRIA
- * sur le site "http://www.cecill.info".
- *
- * En contrepartie de l'accessibilité au code source et des droits de copie,
- * de modification et de redistribution accordés par cette licence, il n'est
- * offert aux utilisateurs qu'une garantie limitée. Pour les mêmes raisons,
- * seule une responsabilité restreinte pèse sur l'auteur du programme, le
- * titulaire des droits patrimoniaux et les concédants successifs.
- *
- * A cet égard l'attention de l'utilisateur est attirée sur les risques
- * associés au chargement, à l'utilisation, à la modification et/ou au
- * développement et à la reproduction du logiciel par l'utilisateur étant
- * donné sa spécificité de logiciel libre, qui peut le rendre complexe à
- * manipuler et qui le réserve donc à des développeurs et des professionnels
- * avertis possédant des connaissances informatiques approfondies. Les
- * utilisateurs sont donc invités à charger et tester l'adéquation du
- * logiciel à leurs besoins dans des conditions permettant d'assurer la
- * sécurité de leurs systèmes et ou de leurs données et, plus généralement,
- * à l'utiliser et l'exploiter dans les mêmes conditions de sécurité.
- *
- * Le fait que vous puissiez accéder à cet en-tête signifie que vous avez
- * pris connaissance de la licence CeCILL, et que vous en avez accepté les
- * termes.
- */
-
- namespace backend\controllers;
-
- use common\models\DeliveryNote;
- use common\models\Invoice;
- use common\models\PointSale;
- use common\models\Product;
- use common\models\Quotation;
- use common\models\ User;
- use common\models\Document;
- use common\helpers\GlobalParam;
- use common\models\Order;
- use common\logic\UserProducer\ UserProducer;
- use kartik\mpdf\Pdf;
- use yii\base\UserException;
- use yii;
-
- class DocumentController extends BackendController
- {
- public function behaviors()
- {
- return [
- 'verbs' => [
- 'class' => VerbFilter::className(),
- 'actions' => [
- ],
- ],
- 'access' => [
- 'class' => AccessControl::className(),
- 'rules' => [
- [
- 'allow' => true,
- 'roles' => ['@'],
- 'matchCallback' => function ($rule, $action) {
- return $this->getUserManager()->hasAccessBackend();
- }
- ]
- ],
- ],
- ];
- }
-
- public function actionGeneratePdfValidatedDocuments()
- {
- set_time_limit(0);
-
- $validatedDocumentsArray = array_merge(
- Quotation::find()->where(['status' => Document::STATUS_VALID])->all(),
- DeliveryNote::find()->where(['status' => Document::STATUS_VALID])->all(),
- Invoice::find()->where(['status' => Document::STATUS_VALID])->all()
- );
-
- foreach ($validatedDocumentsArray as $document) {
- if (!file_exists($document->getFilenameComplete())) {
- $document->generatePdf(Pdf::DEST_FILE);
- }
- }
- }
-
- public function actionCreate()
- {
- $class = $this->getClass();
- $model = new $class();
- $model->initTaxCalculationMethod();
-
- if ($model->load(Yii::$app->request->post())) {
- $model->id_producer = GlobalParam::getCurrentProducerId();
-
- if ($model->save()) {
- $this->processInvoiceViaDeliveryNotes($model);
-
- Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('create', $model));
- return $this->redirect(['/' . $this->getControllerUrl() . '/update', 'id' => $model->id]);
- } else {
- Yii::$app->getSession()->setFlash('error', 'Un problème est survenu lors de la création du document.');
- }
- }
-
- return $this->render('/document/create', [
- 'title' => $this->getTitle('Ajouter'),
- 'typeDocument' => $this->getDocumentType(),
- 'model' => $model,
- ]);
- }
-
- public function processInvoiceViaDeliveryNotes($model)
- {
- if ($model->getClass() == 'Invoice') {
- if ($model->deliveryNotes && is_array($model->deliveryNotes) && count($model->deliveryNotes)) {
- foreach ($model->deliveryNotes as $key => $idDeliveryNote) {
- Order::updateAll([
- 'id_invoice' => $model->id
- ], [
- 'id_delivery_note' => $idDeliveryNote
- ]);
- }
- }
- }
- }
-
- /**
- * Modifie un modèle Produit existant.
- * Si la modification réussit, le navigateur est redirigé vers la page 'index'.
- *
- * @param integer $id
- * @return mixed
- */
- public function actionUpdate($id)
- {
- $model = $this->findModel($id);
-
- if (!$model) {
- throw new NotFoundHttpException('Le document n\'a pas été trouvé.');
- }
-
- if ($model && $model->load(Yii::$app->request->post()) && $model->save()) {
- Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('update', $model));
- }
-
- return $this->render('/document/update', [
- 'title' => $this->getTitle('Modifier'),
- 'typeDocument' => $this->getDocumentType(),
- 'model' => $model,
- ]);
- }
-
- public function actionDelete($id)
- {
- $model = $this->findModel($id);
-
- if ($model->isStatusValid()) {
- throw new UserException('Vous ne pouvez pas supprimer un document validé.');
- }
-
- $model->delete();
-
- if ($this->getClass() == 'DeliveryNote') {
- Order::updateAll([
- 'order.id_delivery_note' => null
- ], [
- 'order.id_delivery_note' => $id
- ]);
- }
-
- if ($this->getClass() == 'Quotation') {
- Order::updateAll([
- 'order.id_quotation' => null
- ], [
- 'order.id_quotation' => $id
- ]);
- }
-
- if ($this->getClass() == 'Invoice') {
- Order::updateAll([
- 'order.id_invoice' => null
- ], [
- 'order.id_invoice' => $id
- ]);
- }
-
- Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('delete', $model));
- $this->redirect([$this->getControllerUrl() . '/index']);
- }
-
- public function actionExportCsvEvoliz($id)
- {
- $datas = [];
- $document = $this->findModel($id);
-
- // données
- $datas[] = [
- 'N° facture externe *',
- 'Date facture *',
- 'Client',
- 'Code client *',
- 'Total TVA',
- 'Total HT',
- 'Total TTC',
- 'Total réglé',
- 'Etat',
- 'Date Etat',
- 'Date de création',
- 'Objet',
- 'Date d\'échéance',
- 'Date d\'exécution',
- 'Taux de pénalité',
- 'Frais de recouvrement',
- 'Taux d\'escompte',
- 'Conditions de règlement *',
- 'Mode de paiement',
- 'Remise globale',
- 'Acompte',
- 'Nombre de relance',
- 'Commentaires',
- 'N° facture',
- 'Annulé',
- 'Catalogue',
- 'Réf.',
- 'Désignation *',
- 'Qté *',
- 'Unité',
- 'PU HT *',
- 'Remise',
- 'TVA',
- 'Total TVA',
- 'Total HT',
- 'Créateur',
- ];
-
- foreach ($document->getProductsOrders() as $productOrderArray) {
- foreach ($productOrderArray as $productOrder) {
-
- $price = $productOrder->getPrice();
- if ($document->isInvoicePrice() && $productOrder->getInvoicePrice()) {
- $price = $productOrder->getInvoicePrice();
- }
-
- $datas[] = [
- $document->reference, // N° facture externe *
- date('d/m/Y', strtotime($document->date)), // Date facture *
- '', // Client
- $document->user->evoliz_code, // Code client *
- '', // Total TVA
- '', // Total HT
- '', // Total TTC
- '', // Total réglé
- '', // Etat
- '', // Date Etat
- '', // Date de création
- $document->name, // Objet
- '', // Date d'échéance
- '', // Date d'exécution
- '', // Taux de pénalité
- '', // Frais de recouvrement
- '', // Taux d\'escompte
- 'A réception', // Conditions de règlement *
- '', // Mode de paiement
- '', // Remise globale
- '', // Acompte
- '', // Nombre de relance
- '', // Commentaires
- '', // N° facture
- '', // Annulé
- 'Non', // Catalogue
- '', // Réf.
- $productOrder->product->name, // Désignation *
- $productOrder->quantity, // Qté *
- '', // Product::strUnit($productOrder->unit, 'wording'), // Unité
- $price, // PU HT *
- '', // Remise
- $productOrder->taxRate->value * 100, // TVA
- '', // Total TVA
- '', // Total HT
- '', // Créateur
- ];
- }
- }
-
- // nom fichier
- $reference = $document->id;
- if ($document->reference && strlen($document->reference)) {
- $reference = $document->reference;
- }
-
- // status
- $status = '';
- if ($document->isStatusDraft()) {
- $status = 'brouillon_';
- }
-
- CSV::downloadSendHeaders(strtolower($this->getDocumentType()) . '_' . $status . $reference . '.csv');
- echo CSV::array2csv($datas);
- die();
- }
-
- public function actionDownload($id)
- {
- $document = $this->findModel($id);
- return $document->downloadPdf();
- }
-
- public function actionRegenerate($id)
- {
- $document = $this->findModel($id);
- $document->downloadPdf(true);
- Yii::$app->getSession()->setFlash('success', 'Le document PDF a bien été regénéré.');
- return $this->redirect([$this->getControllerUrl() . '/update', 'id' => $id]);
- }
-
- public function actionSend($id, $backUpdateForm = false)
- {
- $document = $this->findModel($id);
- if ($document->send()) {
- $document->is_sent = true;
- $document->save();
-
- Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('send', $document));
- } else {
- Yii::$app->getSession()->setFlash('danger', $this->getFlashMessage('send', $document));
- }
-
- if ($backUpdateForm) {
- return $this->redirect([$this->getControllerUrl() . '/update', 'id' => $id]);
- } else {
- return $this->redirect([$this->getControllerUrl() . '/index']);
- }
- }
-
- public function actionAjaxUserInfos($typeAction, $idUser, $classDocument, $idDocument = false)
- {
- \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
-
- if ($idUser > 0) {
- $user = User::searchOne([
- 'id' => $idUser
- ]);
-
- if ($user) {
- $document = null;
- if (Document::isValidClass($classDocument)) {
- $document = $classDocument::searchOne([
- 'id' => $idDocument,
- 'id_user' => $idUser
- ]);
- }
-
- if ($document && $document->id_user == $user->id) {
- $address = $document->address;
- } else {
- $address = $user->getFullAddress();
- }
-
- $json = [
- 'return' => 'success',
- 'address' => $address
- ];
-
- if ($classDocument == 'Invoice') {
- $options = [
- 'orderby' => 'distribution.date ASC',
- 'join_with' => ['user AS user_delivery_note', 'orders', 'producer']
- ];
- $deliveryNotesCreateArray = DeliveryNote::searchAll([
- 'id_user' => $user->id,
- 'status' => Document::STATUS_VALID
- ], $options);
- $deliveryNotesUpdateArray = DeliveryNote::searchAll([
- 'id_user' => $user->id,
- 'status' => Document::STATUS_VALID,
- 'order.id_invoice' => $idDocument
- ], $options);
- $json['delivery_note_create_array'] = $this->initDeliveryNoteArray('create', $deliveryNotesCreateArray);
- $json['delivery_note_update_array'] = $this->initDeliveryNoteArray('update', $deliveryNotesUpdateArray);
- }
-
- return $json;
- }
- }
-
- return ['return' => 'error'];
- }
-
- public function initDeliveryNoteArray($type, $deliveryNoteArrayResults)
- {
- $deliveryNoteArray = [];
- $isCreate = false;
- if ($type == 'create') {
- $isCreate = true;
- }
-
- if ($deliveryNoteArrayResults) {
- foreach ($deliveryNoteArrayResults as $deliveryNote) {
- $deliveryNoteData = $this->addDeliveryNoteToArray($deliveryNote, $isCreate);
- if ($deliveryNoteData) {
- $deliveryNoteArray[] = $deliveryNoteData;
- }
- }
- }
-
- return $deliveryNoteArray;
- }
-
- public function addDeliveryNoteToArray($deliveryNote, $isCreate = true)
- {
- $deliveryNoteData = array_merge(
- $deliveryNote->getAttributes(),
- [
- 'url' => Yii::$app->urlManager->createUrl(['delivery-note/update', 'id' => $deliveryNote->id]),
- 'total' => $deliveryNote->getAmountWithTax(Order::INVOICE_AMOUNT_TOTAL)
- ]
- );
-
- if (($isCreate && !$deliveryNote->isInvoiced()) || !$isCreate) {
- return $deliveryNoteData;
- }
-
- return false;
- }
-
- public function actionValidate($id, $backUpdateForm = false)
- {
- $classDocument = $this->getClass();
-
- if ($id > 0 && Document::isValidClass($classDocument)) {
- $document = $classDocument::searchOne([
- 'id' => $id
- ]);
-
- if ($document) {
- $document->changeStatus(Document::STATUS_VALID);
- $document->save();
-
- // génération PDF
- $document->generatePdf(Pdf::DEST_FILE);
-
- Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('validate', $document));
-
- if ($backUpdateForm) {
- return $this->redirect([$this->getControllerUrl() . '/update', 'id' => $id]);
- } else {
- return $this->redirect([$this->getControllerUrl() . '/index']);
- }
- }
- }
-
- Yii::$app->getSession()->setFlash('danger', 'Une erreur est survenue lors de la validation du document.');
- return $this->redirect([$this->getControllerUrl() . '/index']);
- }
-
- public function actionAjaxValidateDocument($idDocument, $classDocument)
- {
- \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
-
- if ($idDocument > 0 && Document::isValidClass($classDocument)) {
- $document = $classDocument::searchOne([
- 'id' => $idDocument
- ]);
-
- if ($document) {
- $document->changeStatus(Document::STATUS_VALID);
- $document->save();
- return [
- 'return' => 'success',
- 'alert' => [
- 'type' => 'success',
- 'message' => 'Document validé'
- ]
- ];
- }
- }
-
- return [
- 'return' => 'error',
- 'alert' => [
- 'type' => 'danger',
- 'message' => 'Une erreur est survenue lors de la validation du document.'
- ]
- ];
- }
-
- public function actionAjaxInit($idDocument, $classDocument)
- {
- \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
-
- if ($idDocument > 0 && Document::isValidClass($classDocument)) {
- $document = $classDocument::searchOne([
- 'id' => $idDocument
- ]);
-
- if ($document) {
- $productsArray = Product::searchAll([], [
- 'orderby' => 'product.order ASC'
- ]);
-
- $ordersArray = [];
- foreach ($document->orders as $order) {
- $order->init();
- $productsOrderArray = [];
- foreach ($order->productOrder as $productOrder) {
- $productsOrderArray[$productOrder->id] = array_merge($productOrder->getAttributes(), [
- 'url_order' => Yii::$app->urlManager->createUrl(['distribution/index', 'idOrderUpdate' => $productOrder->id_order])
- ]);
- }
- $ordersArray[$order->id] = array_merge(
- $order->getAttributes(),
- [
- 'username' => $order->getUsername(),
- 'distribution_date' => isset($order->distribution) ? date(
- 'd/m/Y',
- strtotime(
- $order->distribution->date
- )
- ) : null,
- 'point_sale_name' => isset($order->pointSale) ? $order->pointSale->name : null,
- 'productOrder' => $productsOrderArray,
- ]
- );
- }
-
- $userProducer = UserProducer::searchOne([
- 'id_user' => $document->user->id,
- 'id_producer' => GlobalParam::getCurrentProducerId()
- ]);
- $pointSale = PointSale::searchOne([
- 'id_user' => $document->user->id
- ]);
-
- $productsArray = yii\helpers\ArrayHelper::map(
- $productsArray,
- 'order',
- function ($product) use ($document, $userProducer, $pointSale) {
- return array_merge($product->getAttributes(), [
- 'unit_coefficient' => Product::$unitsArray[$product->unit]['coefficient'],
- 'prices' => $product->getPriceArray($userProducer->user, $pointSale),
- 'wording_unit' => $product->wording_unit,
- 'tax_rate' => $product->taxRate->value
- ]);
- }
- );
-
- return [
- 'return' => 'success',
- 'tax_rate_producer' => GlobalParam::getCurrentProducer()->taxRate->value,
- 'document' => array_merge($document->getAttributes(), [
- 'html_label' => $document->getHtmlLabel(),
- 'class' => $document->getClass()
- ]),
- 'id_user' => $document->user->id,
- 'products' => $productsArray,
- 'orders' => $ordersArray,
- 'total' => ($document->getClass() == 'Invoice' || $document->getClass() == 'DeliveryNote') ? $document->getAmount(
- Order::INVOICE_AMOUNT_TOTAL
- ) : $document->getAmount(Order::AMOUNT_TOTAL),
- 'total_with_tax' => ($document->getClass() == 'Invoice' || $document->getClass() == 'DeliveryNote') ? $document->getAmountWithTax(
- Order::INVOICE_AMOUNT_TOTAL
- ) : $document->getAmountWithTax(Order::AMOUNT_TOTAL),
- 'invoice_url' => ($document->getClass() == 'DeliveryNote' && $document->getInvoice()) ? Yii::$app->urlManager->createUrl(['invoice/update', 'id' => $document->getInvoice()->id]) : null
- ];
- }
- }
-
- return ['return' => 'error'];
- }
-
- public function actionAjaxAddProduct($idDocument, $classDocument, $idProduct, $quantity, $price)
- {
- \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
-
- if (Document::isValidClass($classDocument)) {
- $document = $classDocument::searchOne([
- 'id' => $idDocument
- ]);
- $product = Product::searchOne([
- 'id' => $idProduct
- ]);
-
- if ($document && $product) {
- if (count($document->orders) == 0) {
- $order = new Order;
- $order->id_user = $document->id_user;
- $order->id_point_sale = null;
- $order->id_distribution = null;
- $order->status = 'tmp-order';
- $order->origin = Order::ORIGIN_ADMIN;
- $order->date = date('Y-m-d H:i:s');
- $fieldIdDocument = 'id_' . $classDocument::tableName();
- $order->$fieldIdDocument = $document->id;
- $order->save();
- } else {
- $order = $document->orders[0];
- }
-
- if ($order) {
- $productOrder = new ProductOrder;
- $productOrder->id_order = $order->id;
- $productOrder->id_product = $idProduct;
- $quantity = $quantity / Product::$unitsArray[$product->unit]['coefficient'];
- $productOrder->quantity = $quantity;
- $productOrder->price = (float)$price;
- $productOrder->unit = $product->unit;
- $productOrder->step = $product->step;
- $productOrder->id_tax_rate = $product->taxRate->id;
- $productOrder->save();
-
- return [
- 'return' => 'success',
- 'alert' => [
- 'type' => 'success',
- 'message' => 'Produit ajouté'
- ]
- ];
- }
- }
- }
-
- return [
- 'return' => 'error',
- 'alert' => [
- 'type' => 'danger',
- 'message' => 'Une erreur est survenue lors de la suppression du produit.'
- ]
- ];
- }
-
- public function actionAjaxDeleteProductOrder($idProductOrder)
- {
- \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
-
- $productOrder = ProductOrder::searchOne([
- 'id' => $idProductOrder
- ]);
-
- if ($productOrder) {
- $productOrder->delete();
-
- return [
- 'return' => 'success',
- 'alert' => [
- 'type' => 'danger',
- 'message' => 'Produit supprimé'
- ]
- ];
- }
-
- return [
- 'return' => 'error',
- 'alert' => [
- 'type' => 'danger',
- 'message' => 'Une erreur est survenue lors de la suppression du produit.'
- ]
- ];
- }
-
- public function getClass()
- {
- $class = get_class($this);
- $class = str_replace('Controller', '', $class);
- $class = str_replace('backend\controllers\\', '', $class);
- return $class;
- }
-
- public function getDocumentType()
- {
- $class = $this->getClass();
-
- if ($class == 'Invoice') {
- $documentType = 'Facture';
- } elseif ($class == 'DeliveryNote') {
- $documentType = 'Bon de livraison';
- } elseif ($class == 'Quotation') {
- $documentType = 'Devis';
- }
-
- if (isset($documentType)) {
- return $documentType;
- }
-
- return '';
- }
-
- public function getFlashMessage($type = 'create', $model)
- {
- $class = $this->getClass();
-
- $message = $this->getDocumentType();
- $message .= ' <strong>' . Html::encode($model->name) . '</strong> ';
-
- if ($type == 'create') {
- $message .= 'ajouté';
- } elseif ($type == 'update') {
- $message .= 'modifié';
- } elseif ($type == 'delete') {
- $message .= 'supprimé';
- } elseif ($type == 'validate') {
- $message .= 'validé';
- } elseif ($type == 'send') {
- $message .= 'envoyé';
- }
-
- if ($class == 'Invoice') {
- $message .= 'e';
- }
-
- return $message;
- }
-
- protected function getTitle($prepend)
- {
- $class = $this->getClass();
-
- switch ($class) {
- case 'Invoice' :
- $title = $prepend . ' une facture';
- break;
- case 'DeliveryNote' :
- $title = $prepend . ' un bon de livraison';
- break;
- case 'Quotation' :
- $title = $prepend . ' un devis';
- break;
- }
-
- return $title;
- }
-
- public function getControllerUrl()
- {
- $path = strtolower($this->getClass());
- $path = str_replace('deliverynote', 'delivery-note', $path);
- return $path;
- }
-
- /**
- * Recherche un Document en fonction de son ID.
- *
- * @param integer $id
- * @return Document
- * @throws NotFoundHttpException si le modèle n'est pas trouvé
- */
- protected function findModel($id)
- {
- $class = $this->getClass();
-
- $model = $class::searchOne([
- 'id' => $id
- ], [
- 'orderby' => 'teshtygjhtyt'
- ]);
-
- if ($model) {
- return $model;
- } else {
- throw new NotFoundHttpException('The requested page does not exist.');
- }
- }
-
- }
|