You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

546 line
17KB

  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\Debug;
  39. use common\helpers\GlobalParam;
  40. use common\helpers\Price;
  41. use Yii;
  42. use common\components\ActiveRecordCommon;
  43. /**
  44. * This is the model class for table "product".
  45. *
  46. * @property integer $id
  47. * @property string $name
  48. * @property string $description
  49. * @property integer $active
  50. * @property string $photo
  51. * @property double $price
  52. * @property double $pweight
  53. * @property string $recipe
  54. * @property string $unit
  55. * @property double $step
  56. */
  57. class Product extends ActiveRecordCommon
  58. {
  59. public $total = 0;
  60. public $apply_distributions = true;
  61. public $price_with_tax = 0;
  62. public $wording_unit = '';
  63. public $pointsSale;
  64. public static $unitsArray = [
  65. 'piece' => [
  66. 'unit' => 'piece',
  67. 'wording_unit' => 'la pièce',
  68. 'wording' => 'pièce(s)',
  69. 'wording_short' => 'p.',
  70. 'coefficient' => 1
  71. ],
  72. 'g' => [
  73. 'ref_unit' => 'kg',
  74. 'unit' => 'g',
  75. 'wording_unit' => 'le g',
  76. 'wording' => 'g',
  77. 'wording_short' => 'g',
  78. 'coefficient' => 1000
  79. ],
  80. 'kg' => [
  81. 'unit' => 'kg',
  82. 'wording_unit' => 'le kg',
  83. 'wording' => 'kg',
  84. 'wording_short' => 'kg',
  85. 'coefficient' => 1
  86. ],
  87. 'mL' => [
  88. 'ref_unit' => 'L',
  89. 'unit' => 'mL',
  90. 'wording_unit' => 'le mL',
  91. 'wording' => 'mL',
  92. 'wording_short' => 'mL',
  93. 'coefficient' => 1000
  94. ],
  95. 'L' => [
  96. 'unit' => 'L',
  97. 'wording_unit' => 'le litre',
  98. 'wording' => 'L',
  99. 'wording_short' => 'L',
  100. 'coefficient' => 1
  101. ],
  102. ];
  103. /**
  104. * @inheritdoc
  105. */
  106. public static function tableName()
  107. {
  108. return 'product';
  109. }
  110. /**
  111. * @inheritdoc
  112. */
  113. public function rules()
  114. {
  115. return [
  116. [['name', 'id_producer'], 'required'],
  117. [['active', 'order', 'id_producer', 'id_tax_rate', 'id_product_category'], 'integer'],
  118. [['monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday', 'sunday', 'unavailable', 'apply_distributions', 'available_on_points_sale'], 'boolean'],
  119. [['price', 'weight', 'step', 'quantity_max', 'quantity_max_monday', 'quantity_max_tuesday', 'quantity_max_wednesday', 'quantity_max_thursday', 'quantity_max_friday', 'quantity_max_saturday', 'quantity_max_sunday'], 'number'],
  120. [['photo'], 'file'],
  121. [['name', 'reference', 'description', 'photo', 'unit'], 'string', 'max' => 255],
  122. [['recipe'], 'string', 'max' => 1000],
  123. ['step', 'required', 'message' => 'Champs obligatoire', 'when' => function ($model) {
  124. if ($model->unit != 'piece') {
  125. return true;
  126. }
  127. return false;
  128. }],
  129. [['price_with_tax', 'wording_unit', 'pointsSale'], 'safe']
  130. ];
  131. }
  132. /**
  133. * @inheritdoc
  134. */
  135. public function attributeLabels()
  136. {
  137. return [
  138. 'id' => 'ID',
  139. 'name' => 'Nom',
  140. 'reference' => 'Référence',
  141. 'description' => 'Description',
  142. 'active' => 'Actif',
  143. 'photo' => 'Photo',
  144. 'price' => 'Prix (€) TTC',
  145. 'weight' => 'Poids',
  146. 'recipe' => 'Recette',
  147. 'monday' => 'Lundi',
  148. 'tuesday' => 'Mardi',
  149. 'wednesday' => 'Mercredi',
  150. 'thursday' => 'Jeudi',
  151. 'friday' => 'Vendredi',
  152. 'saturday' => 'Samedi',
  153. 'sunday' => 'Dimanche',
  154. 'order' => 'Ordre',
  155. 'quantity_max' => 'Quantité max par défaut',
  156. 'quantity_max_monday' => 'Quantité max : lundi',
  157. 'quantity_max_tuesday' => 'Quantité max : mardi',
  158. 'quantity_max_wednesday' => 'Quantité max : mercredi',
  159. 'quantity_max_thursday' => 'Quantité max : jeudi',
  160. 'quantity_max_friday' => 'Quantité max : vendredi',
  161. 'quantity_max_saturday' => 'Quantité max : samedi',
  162. 'quantity_max_sunday' => 'Quantité max : dimanche',
  163. 'unavailable' => 'Épuisé',
  164. 'apply_distributions' => 'Appliquer ces modifications dans les distributions futures',
  165. 'unit' => 'Unité',
  166. 'step' => 'Pas',
  167. 'id_tax_rate' => 'TVA',
  168. 'id_product_category' => 'Catégorie',
  169. 'available_on_points_sale' => 'Par défaut'
  170. ];
  171. }
  172. public function afterFind()
  173. {
  174. if ($this->taxRate == null) {
  175. $producer = Producer::searchOne(['id' => GlobalParam::getCurrentProducerId()]);
  176. if ($producer) {
  177. $this->populateRelation('taxRate', $producer->taxRate);
  178. }
  179. }
  180. $this->wording_unit = Product::strUnit($this->unit);
  181. $this->price_with_tax = $this->getPriceWithTax();
  182. parent::afterFind();
  183. }
  184. public function getProductDistribution()
  185. {
  186. return $this->hasMany(ProductDistribution::className(), ['id_product' => 'id']);
  187. }
  188. public function getProductSubscription()
  189. {
  190. return $this->hasMany(ProductSubscription::className(), ['id_product' => 'id']);
  191. }
  192. public function getTaxRate()
  193. {
  194. return $this->hasOne(TaxRate::className(), ['id' => 'id_tax_rate']);
  195. }
  196. public function getProductPrice()
  197. {
  198. return $this->hasMany(ProductPrice::className(), ['id_product' => 'id']);
  199. }
  200. public function getProductCategory()
  201. {
  202. return $this->hasOne(ProductCategory::className(), ['id' => 'id_product_category']);
  203. }
  204. public function getProductPointSale()
  205. {
  206. return $this->hasMany(ProductPointSale::className(), ['id_product' => 'id']);
  207. }
  208. /**
  209. * Retourne les options de base nécessaires à la fonction de recherche.
  210. *
  211. * @return array
  212. */
  213. public static function defaultOptionsSearch()
  214. {
  215. return [
  216. 'with' => ['taxRate', 'productPointSale'],
  217. 'join_with' => [],
  218. 'orderby' => 'order ASC',
  219. 'attribute_id_producer' => 'product.id_producer'
  220. ];
  221. }
  222. public function isAvailableOnPointSale($pointSale)
  223. {
  224. if(!$pointSale) {
  225. return false;
  226. }
  227. // disponible par défaut
  228. if ($this->available_on_points_sale) {
  229. foreach ($this->productPointSale as $productPointSale) {
  230. if ($pointSale->id == $productPointSale->id_point_sale
  231. //&& $productPointSale->id_product == $this->id
  232. && !$productPointSale->available) {
  233. return false;
  234. }
  235. }
  236. return true;
  237. } // indisponible par défaut
  238. else {
  239. foreach ($this->productPointSale as $productPointSale) {
  240. if ($pointSale->id == $productPointSale->id_point_sale
  241. //&& $productPointSale->id_product == $this->id
  242. && $productPointSale->available) {
  243. return true;
  244. }
  245. }
  246. return false;
  247. }
  248. }
  249. /**
  250. * Retourne la description du produit.
  251. *
  252. * @return string
  253. */
  254. public function getDescription()
  255. {
  256. $description = $this->description;
  257. if (isset($this->weight) && is_numeric($this->weight) && $this->weight > 0) {
  258. if ($this->weight >= 1000) {
  259. $description .= ' (' . ($this->weight / 1000) . 'kg)';
  260. } else {
  261. $description .= ' (' . $this->weight . 'g)';
  262. }
  263. }
  264. return $description;
  265. }
  266. /**
  267. * Retourne le libellé (admin) du produit.
  268. * @return type
  269. */
  270. public function getStrWordingAdmin()
  271. {
  272. return $this->name;
  273. }
  274. /**
  275. * Enregistre le produit.
  276. *
  277. * @param boolean $runValidation
  278. * @param array $attributeNames
  279. * @return boolean
  280. */
  281. public function save($runValidation = true, $attributeNames = NULL)
  282. {
  283. $this->id_producer = GlobalParam::getCurrentProducerId();
  284. return parent::save($runValidation, $attributeNames);
  285. }
  286. /**
  287. * Retourne les produits d'une production donnée.
  288. *
  289. * @param integer $idDistribution
  290. * @return array
  291. */
  292. public static function searchByDistribution($idDistribution)
  293. {
  294. return Product::find()
  295. ->leftJoin('product_distribution', 'product.id = product_distribution.id_product')
  296. ->where([
  297. 'id_producer' => GlobalParam::getCurrentProducerId(),
  298. 'product_distribution.id_distribution' => $idDistribution
  299. ])
  300. ->orderBy('product_distribution.active DESC, product.order ASC')
  301. ->all();
  302. }
  303. /**
  304. * Retourne le nombre de produits du producteur courant.
  305. *
  306. * @return integer
  307. */
  308. public static function count()
  309. {
  310. return self::searchCount();
  311. }
  312. /**
  313. * Retourne le produit "Don".
  314. *
  315. * @return Product
  316. */
  317. public static function getProductGift()
  318. {
  319. $productGift = Product::find()
  320. ->where([
  321. 'product.id_producer' => 0
  322. ])
  323. ->andFilterWhere(['like', 'product.name', 'Don'])
  324. ->one();
  325. return $productGift;
  326. }
  327. public function getRefUnit($unit)
  328. {
  329. if (isset(self::$unitsArray[$unit]) && isset(self::$unitsArray[$unit]['ref_unit'])) {
  330. return self::$unitsArray[$unit]['ref_unit'];
  331. }
  332. return $unit;
  333. }
  334. /**
  335. * Retourne le libellé d'une unité.
  336. *
  337. * @param $format wording_unit, wording, short
  338. * @param $unitInDb Unité stockée en base de données (ex: si g > kg, si mL > L)
  339. * @return $string Libellé de l'unité
  340. */
  341. public static function strUnit($unit, $format = 'wording_short', $unitInDb = false)
  342. {
  343. $strUnit = '';
  344. if ($unitInDb) {
  345. if ($unit == 'g') {
  346. $unit = 'kg';
  347. }
  348. if ($unit == 'mL') {
  349. $unit = 'L';
  350. }
  351. }
  352. if (isset(self::$unitsArray[$unit]) && isset(self::$unitsArray[$unit][$format])) {
  353. $strUnit = self::$unitsArray[$unit][$format];
  354. }
  355. return $strUnit;
  356. }
  357. public function getPriceArray($user, $pointSale)
  358. {
  359. $priceArray = [];
  360. $userProducer = null;
  361. if ($user) {
  362. $userProducer = UserProducer::searchOne([
  363. 'id_user' => $user->id,
  364. ]);
  365. }
  366. // specific prices
  367. $specificPriceArray = $this->getSpecificPricesFilterByPriorityMatch(
  368. $this->productPrice,
  369. $user,
  370. $pointSale
  371. );
  372. foreach ($specificPriceArray as $specificPrice) {
  373. $priceArray[] = [
  374. 'from_quantity' => $specificPrice->from_quantity ? $specificPrice->from_quantity : 0,
  375. 'price' => number_format($this->getPrice([
  376. 'user' => $user,
  377. 'user_producer' => $userProducer,
  378. 'point_sale' => $pointSale,
  379. 'quantity' => $specificPrice->from_quantity
  380. ]), 3),
  381. 'price_with_tax' => number_format($this->getPriceWithTax([
  382. 'user' => $user,
  383. 'user_producer' => $userProducer,
  384. 'point_sale' => $pointSale,
  385. 'quantity' => $specificPrice->from_quantity
  386. ]), 2),
  387. ];
  388. }
  389. if (!$this->hasPriceWithQuantityZero($priceArray)) {
  390. // base price
  391. $priceArray[] = [
  392. 'from_quantity' => 0,
  393. 'price' => $this->getPrice(),
  394. 'price_with_tax' => $this->getPriceWithTax(),
  395. ];
  396. }
  397. usort($priceArray, function ($a, $b) {
  398. if ($a['price'] < $b['price']) {
  399. return 1;
  400. } elseif ($a['price'] > $b['price']) {
  401. return -1;
  402. } else {
  403. return 0;
  404. }
  405. });
  406. return $priceArray;
  407. }
  408. public function hasPriceWithQuantityZero($priceArray)
  409. {
  410. foreach ($priceArray as $price) {
  411. if ($price['from_quantity'] == 0) {
  412. return true;
  413. }
  414. }
  415. return false;
  416. }
  417. public function getSpecificPricesFilterByPriorityMatch($specificPrices, $user, $pointSale)
  418. {
  419. $priorityMatchSpecificPrice = ProductPrice::getPriorityMatchOfSpecificPriceArray($specificPrices, $user, $pointSale);
  420. $specificPricesFilter = [];
  421. foreach ($specificPrices as $keySpecificPrice => $specificPrice) {
  422. if (($priorityMatchSpecificPrice && $specificPrice->$priorityMatchSpecificPrice($user, $pointSale))
  423. || $specificPrice->matchFromQuantityOnly()) {
  424. $specificPricesFilter[] = $specificPrice;
  425. }
  426. }
  427. return $specificPricesFilter;
  428. }
  429. public function getPrice($params = [])
  430. {
  431. $specificPrices = $this->productPrice;
  432. $user = isset($params['user']) ? $params['user'] : false;
  433. $userProducer = isset($params['user_producer']) ? $params['user_producer'] : false;
  434. $pointSale = isset($params['point_sale']) ? $params['point_sale'] : false;
  435. $quantity = (isset($params['quantity']) && $params['quantity']) ? $params['quantity'] : 1;
  436. if ($specificPrices && ($user || $pointSale)) {
  437. $specificPrices = $this->getSpecificPricesFilterByPriorityMatch($specificPrices, $user, $pointSale);
  438. $bestPrice = 9999;
  439. foreach ($specificPrices as $specificPrice) {
  440. $fromQuantity = $specificPrice->from_quantity;
  441. if ((($fromQuantity && $fromQuantity <= $quantity) || !$fromQuantity)
  442. && $specificPrice->price < $bestPrice) {
  443. $bestPrice = $specificPrice->price;
  444. }
  445. }
  446. if ($bestPrice != 9999) {
  447. return $bestPrice;
  448. }
  449. }
  450. if ($userProducer && $userProducer->product_price_percent) {
  451. return $this->price * (1 + $userProducer->product_price_percent / 100);
  452. }
  453. if ($pointSale && $pointSale->product_price_percent) {
  454. return $this->price * (1 + $pointSale->product_price_percent / 100);
  455. }
  456. return $this->price;
  457. }
  458. /**
  459. * Retourne le prix du produit avec taxe
  460. */
  461. public function getPriceWithTax($params = [])
  462. {
  463. $taxRateValue = $this->taxRate ? $this->taxRate->value : 0;
  464. return Price::getPriceWithTax($this->getPrice($params), $taxRateValue);
  465. }
  466. public function getTheTaxRate()
  467. {
  468. if ($this->id_tax_rate) {
  469. return $this->id_tax_rate;
  470. } else {
  471. return GlobalParam::getCurrentProducer()->taxRate->id;
  472. }
  473. }
  474. public function getNameExport()
  475. {
  476. $producer = GlobalParam::getCurrentProducer();
  477. if ($producer->option_export_display_product_reference && $this->reference && strlen($this->reference) > 0) {
  478. return $this->reference;
  479. }
  480. return $this->name;
  481. }
  482. }