Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

476 lines
19KB

  1. <?php
  2. /**
  3. * Copyright distrib (2018)
  4. *
  5. * contact@opendistrib.net
  6. *
  7. * Ce logiciel est un programme informatique servant à aider les producteurs
  8. * à distribuer leur production en circuits courts.
  9. *
  10. * Ce logiciel est régi par la licence CeCILL soumise au droit français et
  11. * respectant les principes de diffusion des logiciels libres. Vous pouvez
  12. * utiliser, modifier et/ou redistribuer ce programme sous les conditions
  13. * de la licence CeCILL telle que diffusée par le CEA, le CNRS et l'INRIA
  14. * sur le site "http://www.cecill.info".
  15. *
  16. * En contrepartie de l'accessibilité au code source et des droits de copie,
  17. * de modification et de redistribution accordés par cette licence, il n'est
  18. * offert aux utilisateurs qu'une garantie limitée. Pour les mêmes raisons,
  19. * seule une responsabilité restreinte pèse sur l'auteur du programme, le
  20. * titulaire des droits patrimoniaux et les concédants successifs.
  21. *
  22. * A cet égard l'attention de l'utilisateur est attirée sur les risques
  23. * associés au chargement, à l'utilisation, à la modification et/ou au
  24. * développement et à la reproduction du logiciel par l'utilisateur étant
  25. * donné sa spécificité de logiciel libre, qui peut le rendre complexe à
  26. * manipuler et qui le réserve donc à des développeurs et des professionnels
  27. * avertis possédant des connaissances informatiques approfondies. Les
  28. * utilisateurs sont donc invités à charger et tester l'adéquation du
  29. * logiciel à leurs besoins dans des conditions permettant d'assurer la
  30. * sécurité de leurs systèmes et ou de leurs données et, plus généralement,
  31. * à l'utiliser et l'exploiter dans les mêmes conditions de sécurité.
  32. *
  33. * Le fait que vous puissiez accéder à cet en-tête signifie que vous avez
  34. * pris connaissance de la licence CeCILL, et que vous en avez accepté les
  35. * termes.
  36. */
  37. namespace common\models;
  38. use common\helpers\GlobalParam;
  39. use Yii;
  40. use common\components\ActiveRecordCommon;
  41. use common\models\Producer;
  42. use common\models\PointSale;
  43. use common\models\UserPointSale;
  44. use common\models\Order;
  45. use common\models\ProductOrder;
  46. use common\models\User;
  47. /**
  48. * This is the model class for table "commande_auto".
  49. *
  50. * @property integer $id
  51. * @property integer $id_user
  52. * @property integer $id_producer
  53. * @property integer $id_point_sale
  54. * @property string $date_begin
  55. * @property string $date_end
  56. * @property integer $monday
  57. * @property integer $tuesday
  58. * @property integer $wednesday
  59. * @property integer $thursday
  60. * @property integer $friday
  61. * @property integer $saturday
  62. * @property integer $sunday
  63. * @property integer $week_frequency
  64. * @property string $username
  65. * @property string $auto_payment
  66. * @property string $comment
  67. */
  68. class Subscription extends ActiveRecordCommon
  69. {
  70. /**
  71. * @inheritdoc
  72. */
  73. public static function tableName()
  74. {
  75. return 'subscription';
  76. }
  77. /**
  78. * @inheritdoc
  79. */
  80. public function rules()
  81. {
  82. return [
  83. [['id_producer', 'id_point_sale'], 'required'],
  84. [['id_user', 'id_producer', 'id_point_sale', 'monday', 'tuesday',
  85. 'wednesday', 'thursday', 'friday', 'saturday', 'sunday', 'week_frequency'], 'integer'],
  86. [['auto_payment'], 'boolean'],
  87. [['date_begin', 'date_end', 'username', 'comment'], 'safe'],
  88. ];
  89. }
  90. /**
  91. * @inheritdoc
  92. */
  93. public function attributeLabels()
  94. {
  95. return [
  96. 'id' => 'ID',
  97. 'id_user' => 'Utilisateur',
  98. 'id_producer' => 'Etablissement',
  99. 'id_point_sale' => 'Point de vente',
  100. 'date_begin' => 'Date de début',
  101. 'date_end' => 'Date de fin',
  102. 'monday' => 'Lundi',
  103. 'tuesday' => 'Mardi',
  104. 'wednesday' => 'Mercredi',
  105. 'thursday' => 'Jeudi',
  106. 'friday' => 'Vendredi',
  107. 'saturday' => 'Samedi',
  108. 'sunday' => 'Dimanche',
  109. 'week_frequency' => 'Périodicité',
  110. 'auto_payment' => 'Paiement automatique',
  111. 'comment' => 'Commentaire'
  112. ];
  113. }
  114. /*
  115. * Relations
  116. */
  117. public function getUser()
  118. {
  119. return $this->hasOne(User::className(), ['id' => 'id_user']);
  120. }
  121. public function getProducer()
  122. {
  123. return $this->hasOne(
  124. Producer::className(),
  125. ['id' => 'id_producer']
  126. );
  127. }
  128. public function getPointSale()
  129. {
  130. return $this->hasOne(
  131. PointSale::className(),
  132. ['id' => 'id_point_sale']
  133. );
  134. }
  135. public function getProductSubscription()
  136. {
  137. return $this->hasMany(
  138. ProductSubscription::className(),
  139. ['id_subscription' => 'id']
  140. )->with('product');
  141. }
  142. /**
  143. * Retourne les options de base nécessaires à la fonction de recherche.
  144. *
  145. * @return array
  146. */
  147. public static function defaultOptionsSearch()
  148. {
  149. return [
  150. 'with' => ['producer'],
  151. 'join_with' => ['user', 'productSubscription', 'productSubscription.product', 'pointSale'],
  152. 'orderby' => 'user.name ASC',
  153. 'attribute_id_producer' => 'subscription.id_producer'
  154. ];
  155. }
  156. /**
  157. * Ajoute la commande pour une date donnée.
  158. *
  159. * @param string $date
  160. */
  161. public function add($date)
  162. {
  163. // distribution
  164. $distribution = Distribution::searchOne([
  165. 'distribution.date' => date('Y-m-d', strtotime($date))
  166. ]);
  167. if ($distribution && count($this->productSubscription)) {
  168. // commande
  169. $order = new Order;
  170. if (strlen($this->username)) {
  171. $order->username = $this->username;
  172. $order->id_user = 0;
  173. } else {
  174. $order->id_user = $this->id_user;
  175. }
  176. $order->date = date('Y-m-d H:i:s');
  177. $order->origin = Order::ORIGIN_AUTO;
  178. $order->id_point_sale = $this->id_point_sale;
  179. $order->id_distribution = $distribution->id;
  180. $order->id_subscription = $this->id;
  181. $order->status = 'tmp-order' ;
  182. if (strlen($this->comment)) {
  183. $order->comment = $this->comment;
  184. }
  185. $pointSale = PointSale::findOne($this->id_point_sale);
  186. $creditFunctioning = $pointSale->getCreditFunctioning();
  187. $order->auto_payment = 0;
  188. if ($order->id_user && Producer::getConfig('credit') && $pointSale->credit) {
  189. if ($creditFunctioning == Producer::CREDIT_FUNCTIONING_OPTIONAL) {
  190. $order->auto_payment = $this->auto_payment;
  191. } elseif ($creditFunctioning == Producer::CREDIT_FUNCTIONING_MANDATORY) {
  192. $order->auto_payment = 1;
  193. } elseif ($creditFunctioning == Producer::CREDIT_FUNCTIONING_USER) {
  194. $user = User::findOne($order->id_user);
  195. $userProducer = UserProducer::searchOne([
  196. 'id_user' => $order->id_user,
  197. 'id_producer' => $distribution->id_producer
  198. ]);
  199. if ($userProducer) {
  200. $order->auto_payment = $userProducer->credit_active;
  201. }
  202. }
  203. }
  204. $userPointSale = UserPointSale::searchOne([
  205. 'id_point_sale' => $this->id_point_sale,
  206. 'id_user' => $this->id_user
  207. ]);
  208. if ($userPointSale && strlen($userPointSale->comment)) {
  209. $order->comment_point_sale = $userPointSale->comment;
  210. }
  211. $order->save();
  212. // liaison utilisateur / point de vente
  213. if ($order->id_user) {
  214. $pointSale = PointSale::findOne($this->id_point_sale);
  215. $pointSale->linkUser($order->id_user);
  216. }
  217. // produits
  218. $amountTotal = 0;
  219. $productsAdd = false;
  220. foreach ($this->productSubscription as $productSubscription) {
  221. $productOrder = new ProductOrder;
  222. $productOrder->id_order = $order->id;
  223. $productOrder->id_product = $productSubscription->product->id;
  224. $productOrder->quantity = $productSubscription->quantity;
  225. $productOrder->price = $productSubscription->product->price;
  226. $productOrder->unit = $productSubscription->product->unit;
  227. $productOrder->step = $productSubscription->product->step;
  228. $productOrder->save();
  229. $productsAdd = true;
  230. }
  231. if (!$productsAdd) {
  232. $order->delete();
  233. }
  234. }
  235. }
  236. /**
  237. * Ajoute les commandes pour une date donnée à partir des abonnements.
  238. *
  239. * @param string $date
  240. * @param boolean $force
  241. */
  242. public static function addAll($date, $force = false)
  243. {
  244. $distribution = Distribution::searchOne([
  245. 'date' => date('Y-m-d', strtotime($date)),
  246. 'id_producer' => GlobalParam::getCurrentProducerId(),
  247. ]);
  248. if ($distribution) {
  249. $arrayOrdersDistribution = Order::searchAll([
  250. Order::tableName() . '.id_distribution' => $distribution->id
  251. ]);
  252. $arraySubscriptions = self::searchByDate($date);
  253. foreach ($arraySubscriptions as $subscription) {
  254. if (!$subscription->hasOrderAlreadyExist($arrayOrdersDistribution)) {
  255. $subscription->add($date);
  256. }
  257. }
  258. }
  259. }
  260. /**
  261. * Informe s'il existe une commande correspond à l'abonnement courant.
  262. *
  263. * @param array $arrayOrders
  264. * @return boolean
  265. */
  266. public function hasOrderAlreadyExist($arrayOrders)
  267. {
  268. if (is_array($arrayOrders) && count($arrayOrders) > 0) {
  269. foreach ($arrayOrders as $order) {
  270. if ((($order->id_user > 0 && $order->id_user == $this->id_user) ||
  271. (!$order->id_user && $order->username == $this->username)) &&
  272. $order->id_point_sale == $this->id_point_sale) {
  273. return true;
  274. }
  275. }
  276. }
  277. return false;
  278. }
  279. /**
  280. * Retourne les abonnements pour une date donnée.
  281. *
  282. * @param string $date
  283. * @return array
  284. */
  285. public static function searchByDate($date)
  286. {
  287. $date = date('Y-m-d', strtotime($date));
  288. $subscriptions = Subscription::searchAll();
  289. $arrSubscriptions = [];
  290. foreach ($subscriptions as $s) {
  291. if ($date >= $s->date_begin &&
  292. (!$s->date_end || $date <= $s->date_end) &&
  293. $s->matchWith($date)) {
  294. $arrSubscriptions[] = $s;
  295. }
  296. }
  297. return $arrSubscriptions;
  298. }
  299. /**
  300. * Valide le fait qu'un abonnement est bien compatible avec une date donnée.
  301. *
  302. * @param string $date
  303. * @return boolean
  304. */
  305. public function matchWith($date)
  306. {
  307. $arrayDays = [
  308. 1 => 'monday',
  309. 2 => 'tuesday',
  310. 3 => 'wednesday',
  311. 4 => 'thursday',
  312. 5 => 'friday',
  313. 6 => 'saturday',
  314. 7 => 'sunday'
  315. ];
  316. $nbDays = (strtotime($date) - strtotime($this->date_begin)) / (24 * 60 * 60);
  317. if (round($nbDays) % ($this->week_frequency * 7) < 7) {
  318. $numDay = date('N', strtotime($date));
  319. $day = $arrayDays[$numDay];
  320. if ($this->$day) {
  321. return true;
  322. }
  323. }
  324. return false;
  325. }
  326. /**
  327. * Recherche les distributions futures où l'abonnement peut s'appliquer.
  328. *
  329. * @return array
  330. */
  331. public function searchMatchedIncomingDistributions()
  332. {
  333. $producer = GlobalParam::getCurrentProducer();
  334. $params = [
  335. ':date_earliest_order' => $producer->getEarliestDateOrder(),
  336. ':date_begin' => date('Y-m-d', strtotime($this->date_begin)),
  337. ':id_producer' => GlobalParam::getCurrentProducerId()
  338. ];
  339. $incomingDistributions = Distribution::find()
  340. ->where('id_producer = :id_producer')
  341. ->andWhere('date >= :date_begin')
  342. ->andWhere('date >= :date_earliest_order');
  343. if ($this->date_end) {
  344. $incomingDistributions->andWhere('date < :date_end');
  345. $params[':date_end'] = date('Y-m-d', strtotime($this->date_end));
  346. }
  347. $incomingDistributions->orderBy('date ASC');
  348. $incomingDistributions->params($params);
  349. $incomingDistributionsArray = $incomingDistributions->all();
  350. $matchedIncomingDistributionsArray = [];
  351. foreach ($incomingDistributionsArray as $incomingDistribution) {
  352. if ($this->matchWith($incomingDistribution->date)) {
  353. $matchedIncomingDistributionsArray[] = $incomingDistribution;
  354. }
  355. }
  356. return $matchedIncomingDistributionsArray;
  357. }
  358. public function deleteOrdersIncomingDistributions()
  359. {
  360. $params = [
  361. ':id_producer' => GlobalParam::getCurrentProducerId(),
  362. ':date_today' => date('Y-m-d'),
  363. ':date_begin' => $this->date_begin,
  364. ':id_subscription' => $this->id
  365. ];
  366. $orderDeadline = Producer::getConfig('order_deadline');
  367. $hour = date('G');
  368. if ($hour >= $orderDeadline) {
  369. $conditionDistributionDate = 'distribution.date > :date_today';
  370. } else {
  371. $conditionDistributionDate = 'distribution.date >= :date_today';
  372. }
  373. $orders = Order::find()
  374. ->joinWith('distribution')
  375. ->where('distribution.id_producer = :id_producer')
  376. ->andWhere($conditionDistributionDate)
  377. ->andWhere('distribution.date >= :date_begin')
  378. ->andWhere('order.id_subscription = :id_subscription');
  379. if ($this->date_end) {
  380. $orders->andWhere('distribution.date <= :date_end');
  381. $params[':date_end'] = $this->date_end;
  382. }
  383. $orders->params($params);
  384. $ordersArray = $orders->all();
  385. $configCredit = Producer::getConfig('credit');
  386. if ($ordersArray && count($ordersArray)) {
  387. foreach ($ordersArray as $order) {
  388. $theOrder = Order::searchOne(['id' => $order->id]);
  389. // remboursement de la commande
  390. if ($theOrder->id_user && $theOrder->getAmount(Order::AMOUNT_PAID) && $configCredit) {
  391. $theOrder->saveCreditHistory(
  392. CreditHistory::TYPE_REFUND,
  393. $theOrder->getAmount(Order::AMOUNT_PAID),
  394. $theOrder->distribution->id_producer,
  395. $theOrder->id_user,
  396. User::getCurrentId()
  397. );
  398. }
  399. ProductOrder::deleteAll(['id_order' => $order->id]);
  400. $order->delete();
  401. }
  402. }
  403. }
  404. public function updateIncomingDistributions($update = false)
  405. {
  406. $matchedDistributionsArray = $this->searchMatchedIncomingDistributions();
  407. if ($update) {
  408. $this->deleteOrdersIncomingDistributions();
  409. }
  410. if (count($matchedDistributionsArray)) {
  411. foreach ($matchedDistributionsArray as $distribution) {
  412. $this->add($distribution->date);
  413. }
  414. }
  415. }
  416. }