src/Entity/Product.php line 34

Open in your IDE?
  1. <?php
  2. namespace App\Entity;
  3. use App\Entity\Media;
  4. use App\Repository\ProductRepository;
  5. use DateInterval;
  6. use DateTime;
  7. use App\SlugHandler\ProductNameSlugHandler;
  8. use Doctrine\Common\Collections\ArrayCollection;
  9. use Doctrine\Common\Collections\Collection;
  10. use Doctrine\Common\Collections\Criteria;
  11. use Doctrine\DBAL\Types\Types;
  12. use Gedmo\Mapping\Annotation\Slug;
  13. use Doctrine\ORM\Mapping as ORM;
  14. use Doctrine\ORM\Mapping\PrePersist;
  15. use Gedmo\Blameable\Traits\BlameableEntity;
  16. use Gedmo\Mapping\Annotation\SlugHandler;
  17. use Gedmo\Mapping\Annotation\Translatable;
  18. use Gedmo\Sluggable\Handler\InversedRelativeSlugHandler;
  19. use Gedmo\Sluggable\Handler\RelativeSlugHandler;
  20. use Gedmo\Timestampable\Traits\TimestampableEntity;
  21. use Knp\DoctrineBehaviors\Contract\Entity\TranslatableInterface;
  22. use Knp\DoctrineBehaviors\Model\Translatable\TranslatableTrait;
  23. use Symfony\Component\PropertyAccess\PropertyAccess;
  24. use Symfony\Bridge\Doctrine\Validator\Constraints\UniqueEntity;
  25. use Symfony\Component\Serializer\Annotation\SerializedName;
  26. use Symfony\Component\Validator\Constraints as Assert;
  27. #[ORM\MappedSuperclass()]
  28. #[UniqueEntity('slug')]
  29. #[ORM\HasLifecycleCallbacks]
  30. #[ORM\Entity(repositoryClassProductRepository::class)]
  31. class Product implements TranslatableInterface
  32. {
  33.     use BlameableEntity//Hook blameable behaviour. Updates createdBy, updatedBy fields
  34.     use TimestampableEntity//Hook timestampable behaviour. Updates createdAt, updatedAt fields 
  35.     use TranslatableTrait;
  36.     const STATUS_DRAFT 'DRAFT';
  37.     const STATUS_SCHEDULED 'SCHEDULED';
  38.     const STATUS_ACTIVE 'ACTIVE';
  39.     const STATUS_BLOCKED 'BLOCKED';
  40.     const STATUS_FINISHED 'FINISHED';
  41.     const DEFAULT_SHIPPING_INDIVIDUAL_DURANTION 3;
  42.     const DEFAULT_SHIPPING_SHARED_DURANTION 8;
  43.     #[Assert\Valid]
  44.     protected $translations;
  45.     #[ORM\Id]
  46.     #[ORM\GeneratedValue(strategy"IDENTITY")]
  47.     #[ORM\Column]
  48.     private ?int $id null;
  49.     #[ORM\Column(length255uniquetrueupdatable:true)]
  50.     #[Slug(fields: ['slug'])]
  51.     private ?string $slug null;
  52.     #[ORM\ManyToOne(inversedBy'products')]
  53.     #[ORM\JoinColumn(nullablefalse)]
  54.     #[Assert\NotNull()]
  55.     private ?ProductType $type null;
  56.     #[ORM\Column]
  57.     #[Assert\NotNull()]
  58.     private ?bool $isDraft true;
  59.     #[ORM\Column]
  60.     #[Assert\NotNull()]
  61.     private ?bool $isBlocked false;
  62.     #[ORM\Column]
  63.     #[Assert\NotNull()]
  64.     private ?bool $isSoldOut false;
  65.     #[ORM\Column(nullabletrue)]
  66.     #[Assert\Positive()]
  67.     #[Assert\Expression(
  68.         "!this.getShipping() || (this.getVolumetricWeight() || value)",
  69.     )]
  70.     private ?float $weight null;
  71.     #[ORM\Column(nullabletrue)]
  72.     #[Assert\Positive()]
  73.     private ?int $width null;
  74.     #[ORM\Column(nullabletrue)]
  75.     #[Assert\Positive()]
  76.     private ?int $length null;
  77.     #[ORM\Column(nullabletrue)]
  78.     #[Assert\Positive()]
  79.     private ?int $height null;
  80.     #[ORM\Column(nullabletrue)]
  81.     #[Assert\Positive()]
  82.     #[Assert\Expression(
  83.         "!this.getShipping() || (this.getWeight() || value)",
  84.     )]
  85.     private ?float $volumetricWeight null;
  86.     #[ORM\Column(nullabletrue)]
  87.     #[Assert\Positive()]
  88.     private ?int $unitsPerBox 1;
  89.     #[ORM\Column(nullabletrue)]
  90.     #[Assert\Positive()]
  91.     private ?int $unitsPerBoxGrouped 1;
  92.     #[ORM\Column]
  93.     #[Assert\NotNull()]
  94.     private ?bool $hideDiscount false;
  95.     #[ORM\Column(length50nullabletrue)]
  96.     #[Assert\Length(max50)]
  97.     private ?string $ean null;
  98.     #[ORM\ManyToMany(targetEntityCategory::class, inversedBy'products')]
  99.     #[Assert\Count(min1)]
  100.     private Collection $categories;
  101.     #[ORM\ManyToMany(targetEntitySubcategory::class, inversedBy'products')]
  102.     private Collection $subcategories;
  103.     #[ORM\Column(length50nullabletrue)]
  104.     #[Assert\Length(max50)]
  105.     private ?string $purchaseOrderNumber null;
  106.     #[ORM\OneToMany(mappedBy'productImage'targetEntityMedia::class, cascade: ["persist""remove"])]
  107.     #[ORM\OrderBy(["position" => "ASC"])]
  108.     private Collection $productImages;
  109.     #[ORM\OneToMany(mappedBy'productVideo'targetEntityMedia::class, cascade: ["persist""remove"])]
  110.     private Collection $productVideos;
  111.     #[ORM\ManyToMany(targetEntityLabel::class, inversedBy'products')]
  112.     private Collection $labels;
  113.     #[ORM\Column]
  114.     #[Assert\NotNull()]
  115.     private ?bool $isMandatoryProductFeatures false;
  116.     #[ORM\Column]
  117.     #[Assert\NotNull()]
  118.     private ?bool $isMandatoryProducer false;
  119.     #[ORM\Column(nullabletrue)]
  120.     private ?bool $isMandatoryTriweerList false;
  121.     #[ORM\Column(nullabletrue)]
  122.     private ?bool $isMandatoryTriweerComments false;
  123.     #[ORM\Column]
  124.     #[Assert\NotNull()]
  125.     private ?bool $isLiveShopping false;
  126.     #[ORM\Column]
  127.     #[Assert\NotNull()]
  128.     private ?bool $hasBlockchainTraceability false;
  129.     #[ORM\Column]
  130.     #[Assert\NotNull()]
  131.     private ?bool $showOnWall true;
  132.     #[ORM\Column]
  133.     #[Assert\NotNull()]
  134.     private ?bool $allowReturn false;
  135.     #[ORM\Column(nullabletrue)]
  136.     #[Assert\Expression(
  137.         "this.getAllowReturn() == false || value",
  138.     )]
  139.     #[Assert\Positive()]
  140.     private ?int $nDaysAllowedForReturn null;
  141.     #[ORM\Column]
  142.     #[Assert\NotNull()]
  143.     private ?bool $sendNotifications true;
  144.     #[ORM\Column(length255nullabletrue)]
  145.     #[Assert\Length(max255)]
  146.     private ?string $producerPurchaseOrderNumber null;
  147.     #[ORM\Column]
  148.     #[Assert\NotNull()]
  149.     #[Assert\Positive()]
  150.     private ?int $limitUnitsPerBuyer null;
  151.     #[ORM\Column(nullabletrue)]
  152.     #[Assert\Positive()]
  153.     private ?int $limitUnitsToSell null;
  154.     #[ORM\Column(typeTypes::DATETIME_MUTABLE)]
  155.     #[Assert\NotNull()]
  156.     private ?\DateTimeInterface $comingSoonDate null;
  157.     #[ORM\Column(typeTypes::DATETIME_MUTABLE)]
  158.     #[Assert\NotNull()]
  159.     private ?\DateTimeInterface $offerLaunchDate null;
  160.     #[ORM\Column(typeTypes::DATETIME_MUTABLE)]
  161.     #[Assert\NotNull()]
  162.     private ?\DateTimeInterface $countdownDate null;
  163.     #[ORM\Column(typeTypes::DATETIME_MUTABLE)]
  164.     #[Assert\NotNull()]
  165.     private ?\DateTimeInterface $endOfferDate null;
  166.     #[ORM\Column(typeTypes::DATETIME_MUTABLE)]
  167.     //#[Assert\NotNull()]
  168.     private ?\DateTimeInterface $shippingDate null;
  169.     #[ORM\Column(nullabletrue)]
  170.     #[Assert\PositiveOrZero()]
  171.     // #[Assert\NotNull()]
  172.     private ?int $individualShippingDuration null;
  173.     #[ORM\Column(nullabletrue)]
  174.     #[Assert\PositiveOrZero()]
  175.     // #[Assert\NotNull()]
  176.     private ?int $sharedShippingDuration null;
  177.     #[ORM\Column(length255nullabletrue)]
  178.     #[Assert\Length(max255)]
  179.     private ?string $individualPackageFormat null;
  180.     #[ORM\Column]
  181.     #[Assert\NotNull()]
  182.     #[Assert\PositiveOrZero()]
  183.     private ?int $nHoursOnWallAfterEnd null;
  184.     #[ORM\ManyToOne(inversedBy'products')]
  185.     #[ORM\JoinColumn(nullablefalse)]
  186.     #[Assert\NotNull()]
  187.     private ?Iva $iva null;
  188.     #[ORM\OneToMany(mappedBy'product'targetEntityProductPriceScale::class, cascade: ["persist""remove"], orphanRemovaltrue)]
  189.     #[ORM\OrderBy(["initialPrice" => "DESC"])]
  190.     #[Assert\Count(min1)]
  191.     #[Assert\Valid]
  192.     private Collection $priceScales;
  193.     #[ORM\Column]
  194.     #[Assert\NotNull()]
  195.     #[Assert\PositiveOrZero()]
  196.     private ?int $maxCoinsToSpent null;
  197.     #[ORM\ManyToOne(inversedBy'products')]
  198.     #[ORM\JoinColumn(nullabletrue)]
  199.     #[Assert\NotNull()]
  200.     private ?Brand $brand null;
  201.     #[ORM\ManyToOne(inversedBy'products')]
  202.     #[Assert\Expression(
  203.         "this.getIsMandatoryProducer() == false || value",
  204.     )]
  205.     private ?Producer $producer null;
  206.     #[ORM\ManyToOne(inversedBy'products')]
  207.     #[ORM\JoinColumn(nullablefalse)]
  208.     #[Assert\NotNull()]
  209.     private ?UnitMeasurement $unitMeasurement null;
  210.     #[ORM\OneToMany(mappedBy'product'targetEntityOrderDetail::class)]
  211.     private Collection $orderDetails;
  212.     #[ORM\Column]
  213.     #[Assert\NotNull()]
  214.     #[Assert\Positive()]
  215.     private ?float $originalPrice null;
  216.     #[ORM\Column(nullabletrue)]
  217.     #[Assert\Positive()]
  218.     private ?float $individualPrice null;
  219.     #[ORM\Column(options: ['default' => false])]
  220.     private ?bool $individualDisabled false;
  221.     #[ORM\Column(options: ['default' => 0])]
  222.     #[Assert\NotNull()]
  223.     #[Assert\PositiveOrZero()]
  224.     #[Assert\Expression(
  225.         "value <= this.getProductMaxQuantity()",
  226.     )]
  227.     private ?int $fakeUsers 0;
  228.     #[ORM\JoinTable(name'faketriweers_product')]
  229.     #[ORM\JoinColumn(name'fake_triweer_id'referencedColumnName'id')]
  230.     #[ORM\InverseJoinColumn(name'product_id'referencedColumnName'id')]
  231.     #[ORM\ManyToMany(targetEntityUser::class)]
  232.     private Collection $fakeTriweersInOffer;
  233.     #[ORM\Column(typeTypes::TEXTnullabletrue)]
  234.     private ?string $additionalInfo null;
  235.     /**
  236.      * @deprecated
  237.      */
  238.     #[ORM\Column(nullabletrue)]
  239.     #[Assert\PositiveOrZero()]
  240.     private ?float $shippingCost null;
  241.     #[ORM\Column(length500nullabletrue)]
  242.     private ?string $holdedId null;
  243.     #[ORM\OneToMany(mappedBy'product'targetEntityUserShared::class)]
  244.     private Collection $influencerShareds;
  245.     #[ORM\ManyToOne(inversedBy'products')]
  246.     #[ORM\JoinColumn(nullablefalse)]
  247.     #[Assert\NotNull()]
  248.     private ?ProductUnique $productUnique null;
  249.     #[ORM\Column(options: ['default' => false])]
  250.     private ?bool $onlyAdults false;
  251.     #[ORM\Column(nullabletrue)]
  252.     #[Assert\PositiveOrZero()]
  253.     private ?int $amountForFreeShipping null;
  254.     #[ORM\Column(nullabletrue)]
  255.     private ?bool $better_price null;
  256.     #[ORM\ManyToOne(inversedBy'products')]
  257.     #[Assert\NotNull()]
  258.     private ?Shipping $shipping null;
  259.     #[ORM\ManyToMany(targetEntityUser::class, mappedBy"favouriteProducts")]
  260.     private Collection $userFavourites;
  261.     private Collection $randomUserFavourites;
  262.     #[ORM\OneToMany(mappedBy'product'targetEntityReview::class)]
  263.     private Collection $userReviews;
  264.     #[ORM\Column(length255nullabletrue)]
  265.     private ?string $meta_title null;
  266.     #[ORM\Column(length255nullabletrue)]
  267.     private ?string $meta_description null;
  268.     #[ORM\Column(nullablefalseoptions: ['default' => false])]
  269.     private ?bool $showPrice false;
  270.     #[ORM\PrePersist]
  271.     
  272.     public function setSlugValue()
  273.     {
  274.         $this->setSlug($this->translate('es')->getName());
  275.     }
  276.     public function setSlugEditValue()
  277.     {
  278.         $name $this->translate('es')->getName() ?? '';
  279.         $this->setSlug($this->generateSlug($name)); 
  280.     }
  281.     private function generateSlug(string $name): string
  282.     {
  283.         return strtolower(trim(preg_replace('/[^A-Za-z0-9-]+/''-'$name)));
  284.     }
  285.     public function __call($method$arguments)
  286.     {
  287.         return $this->proxyCurrentLocaleTranslation($method$arguments);
  288.     }
  289.     public function __construct()
  290.     {
  291.         $this->categories          = new ArrayCollection();
  292.         $this->subcategories       = new ArrayCollection();
  293.         $this->productImages       = new ArrayCollection();
  294.         $this->productVideos       = new ArrayCollection();
  295.         $this->labels              = new ArrayCollection();
  296.         $this->priceScales         = new ArrayCollection();
  297.         $this->orderDetails        = new ArrayCollection();
  298.         $this->influencerShareds   = new ArrayCollection();
  299.         $this->userFavourites      = new ArrayCollection();
  300.         $this->fakeTriweersInOffer = new ArrayCollection();
  301.         $this->userReviews         = new ArrayCollection();
  302.     }
  303.     public function __clone()
  304.     {
  305.         $this->orderDetails      = new ArrayCollection();
  306.         $this->influencerShareds = new ArrayCollection();
  307.         $this->userFavourites    = new ArrayCollection();
  308.         $this->userReviews       = new ArrayCollection();
  309.         if ($this->id) {
  310.             $this->setId(null);
  311.         }
  312.         if ($this->holdedId) {
  313.             $this->setHoldedId('');
  314.         }
  315.         $this->setIsDraft(true);
  316.         foreach ($this->getTranslations() as $trans) {
  317.             /** @var \App\Entity\ProductTranslation $trans */
  318.             $clonedTrans = clone $trans;
  319.             $clonedTrans->setName('Clon ' $clonedTrans->getName());
  320.             $this->addTranslation($clonedTrans);
  321.         }
  322.         foreach ($this->getProductImages() as $idx => $media) {
  323.             $this->addProductImage((new Media())->clone($media'product'$idx));
  324.         }
  325.         foreach ($this->getProductVideos() as $idx => $media) {
  326.             $this->addProductVideo((new Media())->clone($media'product'$idx));
  327.         }
  328.         foreach ($this->getPriceScales() as $idx => $scale) {
  329.             $this->addPriceScale(clone $scale);
  330.         }
  331.     }
  332.     public function setId(?int $id): self
  333.     {
  334.         $this->id $id;
  335.         return $this;
  336.     }
  337.     public function getId(): ?int
  338.     {
  339.         return $this->id;
  340.     }
  341.     public function getSlug(): ?string
  342.     {
  343.         return $this->slug;
  344.     }
  345.     public function setSlug(string $slug): self
  346.     {
  347.         $this->slug $slug;
  348.         return $this;
  349.     }
  350.     public function getStatus(): string
  351.     {
  352.         if ($this->isDraft) {
  353.             return Product::STATUS_DRAFT;
  354.         } else if ($this->isBlocked) {
  355.             return Product::STATUS_BLOCKED;
  356.         } else if ($this->isSoldOut || $this->endOfferDate < (new \DateTime())) {
  357.             return Product::STATUS_FINISHED;
  358.         } else if ($this->offerLaunchDate > (new \DateTime())) {
  359.             return Product::STATUS_SCHEDULED;
  360.         }
  361.         return Product::STATUS_ACTIVE;
  362.     }
  363.     public function getIsScheduled(): bool
  364.     {
  365.         return $this->getStatus() == Product::STATUS_SCHEDULED;
  366.     }
  367.     public function getIsActive(): bool
  368.     {
  369.         return $this->getStatus() == Product::STATUS_ACTIVE;
  370.     }
  371.     public function getIsFinished(): bool
  372.     {
  373.         return $this->getStatus() == Product::STATUS_FINISHED;
  374.     }
  375.     public function getIsFinishedPlusExtra(): bool
  376.     {
  377.         if ($this->getStatus() != Product::STATUS_FINISHED) {
  378.             return false;
  379.         }
  380.         if ($this->nHoursOnWallAfterEnd 0) {
  381.             $endOfferDate DateTime::createFromInterface($this->endOfferDate);
  382.             $endOfferDate->add(new DateInterval("PT{$this->nHoursOnWallAfterEnd}H"));
  383.             return $endOfferDate <= (new \DateTime());
  384.         }
  385.         return true;
  386.     }
  387.     public function getStatusBgColor(): string
  388.     {
  389.         $status $this->getStatus();
  390.         switch ($status) {
  391.             case Product::STATUS_DRAFT:
  392.                 $color "#F1C14F";
  393.                 break;
  394.             case Product::STATUS_BLOCKED:
  395.                 $color "#000000";
  396.                 break;
  397.             case Product::STATUS_SCHEDULED:
  398.                 $color "#36A9E1";
  399.                 break;
  400.             case Product::STATUS_ACTIVE:
  401.                 $color "#32D29D";
  402.                 break;
  403.             case Product::STATUS_FINISHED:
  404.                 $color "#FF4863";
  405.                 break;
  406.         }
  407.         return $color;
  408.     }
  409.     public function getStatusColor(): string
  410.     {
  411.         $status $this->getStatus();
  412.         switch ($status) {
  413.             case Product::STATUS_DRAFT:
  414.                 $color "#FFFFFF";
  415.                 break;
  416.             case Product::STATUS_BLOCKED:
  417.                 $color "#FFFFFF";
  418.                 break;
  419.             case Product::STATUS_SCHEDULED:
  420.                 $color "#FFFFFF";
  421.                 break;
  422.             case Product::STATUS_ACTIVE:
  423.                 $color "#FFFFFF";
  424.                 break;
  425.             case Product::STATUS_FINISHED:
  426.                 $color "#FFFFFF";
  427.                 break;
  428.         }
  429.         return $color;
  430.     }
  431.     public function getNameWithId(): ?string
  432.     {
  433.         return $this->getId() . ' - ' $this->getName();
  434.     }
  435.     public function getType(): ?ProductType
  436.     {
  437.         return $this->type;
  438.     }
  439.     public function setType(?ProductType $type): self
  440.     {
  441.         $this->type $type;
  442.         return $this;
  443.     }
  444.     public function getIsDraft(): ?bool
  445.     {
  446.         return $this->isDraft;
  447.     }
  448.     public function setIsDraft(?bool $isDraft): self
  449.     {
  450.         $this->isDraft $isDraft;
  451.         return $this;
  452.     }
  453.     public function getIsBlocked(): ?bool
  454.     {
  455.         return $this->isBlocked;
  456.     }
  457.     public function setIsBlocked(?bool $isBlocked): self
  458.     {
  459.         $this->isBlocked $isBlocked;
  460.         return $this;
  461.     }
  462.     public function getIsSoldOut(): ?bool
  463.     {
  464.         return $this->isSoldOut;
  465.     }
  466.     public function setIsSoldOut(bool $isSoldOut): self
  467.     {
  468.         $this->isSoldOut $isSoldOut;
  469.         return $this;
  470.     }
  471.     public function getWeight(): ?float
  472.     {
  473.         return $this->weight;
  474.     }
  475.     public function setWeight(?float $weight): self
  476.     {
  477.         $this->weight $weight;
  478.         return $this;
  479.     }
  480.     public function getWidth(): ?int
  481.     {
  482.         return $this->width;
  483.     }
  484.     public function setWidth(?int $width): self
  485.     {
  486.         $this->width $width;
  487.         return $this;
  488.     }
  489.     public function getLength(): ?int
  490.     {
  491.         return $this->length;
  492.     }
  493.     public function setLength(?int $length): self
  494.     {
  495.         $this->length $length;
  496.         return $this;
  497.     }
  498.     public function getHeight(): ?int
  499.     {
  500.         return $this->height;
  501.     }
  502.     public function setHeight(?int $height): self
  503.     {
  504.         $this->height $height;
  505.         return $this;
  506.     }
  507.     public function getVolumetricWeight(): ?float
  508.     {
  509.         return $this->volumetricWeight;
  510.     }
  511.     public function setVolumetricWeight(?float $volumetricWeight): self
  512.     {
  513.         $this->volumetricWeight $volumetricWeight;
  514.         return $this;
  515.     }
  516.     /**
  517.      *  Max value between weight and volumetric weight
  518.      */
  519.     public function getShippingWeight(): ?float
  520.     {
  521.         return max($this->volumetricWeight$this->weight);
  522.     }
  523.     public function getEan(): ?string
  524.     {
  525.         return $this->ean;
  526.     }
  527.     public function setEan(?string $ean): self
  528.     {
  529.         $this->ean $ean;
  530.         return $this;
  531.     }
  532.     public function getUnitsPerBox(): ?int
  533.     {
  534.         return $this->unitsPerBox;
  535.     }
  536.     public function setUnitsPerBox(?int $unitsPerBox): self
  537.     {
  538.         $this->unitsPerBox $unitsPerBox;
  539.         return $this;
  540.     }
  541.     public function getUnitsPerBoxGrouped(): ?int
  542.     {
  543.         return $this->unitsPerBoxGrouped ?? 1;
  544.     }
  545.     public function setUnitsPerBoxGrouped(?int $unitsPerBoxGrouped): self
  546.     {
  547.         $this->unitsPerBoxGrouped $unitsPerBoxGrouped;
  548.         return $this;
  549.     }
  550.     public function getHideDiscount(): ?bool
  551.     {
  552.         return $this->hideDiscount ?? false;
  553.     }
  554.     public function setHideDiscount(?bool $hideDiscount): self
  555.     {
  556.         $this->hideDiscount $hideDiscount;
  557.         return $this;
  558.     }
  559.     /**
  560.      * @return Collection<int, Category>
  561.      */
  562.     public function getCategories(): Collection
  563.     {
  564.         return $this->categories;
  565.     }
  566.     public function getCategoriesText(): ?string
  567.     {
  568.         return implode(', '$this->categories->toArray());
  569.     }
  570.     public function addCategory(Category $category): self
  571.     {
  572.         if (!$this->categories->contains($category)) {
  573.             $this->categories->add($category);
  574.         }
  575.         return $this;
  576.     }
  577.     public function removeCategory(Category $category): self
  578.     {
  579.         $this->categories->removeElement($category);
  580.         return $this;
  581.     }
  582.     /**
  583.      * @return Collection<int, Subcategory>
  584.      */
  585.     public function getSubcategories(): Collection
  586.     {
  587.         return $this->subcategories;
  588.     }
  589.     public function getSubcategoriesText(): ?string
  590.     {
  591.         return implode(', '$this->subcategories->toArray());
  592.     }
  593.     public function addSubcategory(Subcategory $subcategory): self
  594.     {
  595.         if (!$this->subcategories->contains($subcategory)) {
  596.             $this->subcategories->add($subcategory);
  597.         }
  598.         return $this;
  599.     }
  600.     public function removeSubcategory(Subcategory $subcategory): self
  601.     {
  602.         $this->subcategories->removeElement($subcategory);
  603.         return $this;
  604.     }
  605.     public function getPurchaseOrderNumber(): ?string
  606.     {
  607.         return $this->purchaseOrderNumber;
  608.     }
  609.     public function setPurchaseOrderNumber(?string $purchaseOrderNumber): self
  610.     {
  611.         $this->purchaseOrderNumber $purchaseOrderNumber;
  612.         return $this;
  613.     }
  614.     /**
  615.      * Main image is the one that have position 0
  616.      * @return Media|null
  617.      */
  618.     public function getMainProductImage(): ?Media
  619.     {
  620.         if ($this->productImages) {
  621.             $filteredProductImages $this->productImages->filter(function (Media $productImage) {
  622.                 return $productImage->getPosition() == 0;
  623.             });
  624.             return (count($filteredProductImages) > 0) ? $filteredProductImages->first() : null;
  625.         } else {
  626.             return null;
  627.         }
  628.     }
  629.     /**
  630.      * @return Collection<int, Media>
  631.      */
  632.     public function getProductImages(): Collection
  633.     {
  634.         return $this->productImages;
  635.     }
  636.     public function addProductImage(Media $productImage): self
  637.     {
  638.         if (!$this->productImages->contains($productImage)) {
  639.             $productImage->setDirectory('product');
  640.             $this->productImages->add($productImage);
  641.             $productImage->setProductImage($this);
  642.         }
  643.         return $this;
  644.     }
  645.     public function removeProductImage(Media $productImage): self
  646.     {
  647.         if ($this->productImages->removeElement($productImage)) {
  648.             // set the owning side to null (unless already changed)
  649.             if ($productImage->getProductImage() === $this) {
  650.                 $productImage->setProductImage(null);
  651.             }
  652.         }
  653.         return $this;
  654.     }
  655.     /**
  656.      * @return Collection<int, Media>
  657.      */
  658.     public function getProductVideos(): Collection
  659.     {
  660.         return $this->productVideos;
  661.     }
  662.     public function addProductVideo(Media $productVideo): self
  663.     {
  664.         if (!$this->productVideos->contains($productVideo)) {
  665.             $productVideo->setDirectory('product');
  666.             $this->productVideos->add($productVideo);
  667.             $productVideo->setProductVideo($this);
  668.         }
  669.         return $this;
  670.     }
  671.     public function removeProductVideo(Media $productVideo): self
  672.     {
  673.         if ($this->productVideos->removeElement($productVideo)) {
  674.             // set the owning side to null (unless already changed)
  675.             if ($productVideo->getProductVideo() === $this) {
  676.                 $productVideo->setProductVideo(null);
  677.             }
  678.         }
  679.         return $this;
  680.     }
  681.     /**
  682.      * @return Collection<int, Label>
  683.      */
  684.     public function getLabels(): Collection
  685.     {
  686.         return $this->labels;
  687.     }
  688.     public function getLabelsText(): ?string
  689.     {
  690.         return implode(', '$this->labels->toArray());
  691.     }
  692.     public function addLabel(Label $label): self
  693.     {
  694.         if (!$this->labels->contains($label)) {
  695.             $this->labels->add($label);
  696.         }
  697.         return $this;
  698.     }
  699.     public function removeLabel(Label $label): self
  700.     {
  701.         $this->labels->removeElement($label);
  702.         return $this;
  703.     }
  704.     public function getIsMandatoryProductFeatures(): ?bool
  705.     {
  706.         return $this->isMandatoryProductFeatures;
  707.     }
  708.     public function setIsMandatoryProductFeatures(bool $isMandatoryProductFeatures): self
  709.     {
  710.         $this->isMandatoryProductFeatures $isMandatoryProductFeatures;
  711.         return $this;
  712.     }
  713.     public function getIsMandatoryProducer(): ?bool
  714.     {
  715.         return $this->isMandatoryProducer;
  716.     }
  717.     public function setIsMandatoryProducer(bool $isMandatoryProducer): self
  718.     {
  719.         $this->isMandatoryProducer $isMandatoryProducer;
  720.         return $this;
  721.     }
  722.     public function getIsMandatoryTriweerList(): ?bool
  723.     {
  724.         return $this->isMandatoryTriweerList;
  725.     }
  726.     public function setIsMandatoryTriweerList(?bool $isMandatoryTriweerList): self
  727.     {
  728.         $this->isMandatoryTriweerList $isMandatoryTriweerList;
  729.         return $this;
  730.     }
  731.     public function getIsMandatoryTriweerComments(): ?bool
  732.     {
  733.         return $this->isMandatoryTriweerComments;
  734.     }
  735.     public function setIsMandatoryTriweerComments(?bool $isMandatoryTriweerComments): self
  736.     {
  737.         $this->isMandatoryTriweerComments $isMandatoryTriweerComments;
  738.         return $this;
  739.     }
  740.     public function isIsLiveShopping(): ?bool
  741.     {
  742.         return $this->isLiveShopping;
  743.     }
  744.     public function setIsLiveShopping(bool $isLiveShopping): self
  745.     {
  746.         $this->isLiveShopping $isLiveShopping;
  747.         return $this;
  748.     }
  749.     public function isHasBlockchainTraceability(): ?bool
  750.     {
  751.         return $this->hasBlockchainTraceability;
  752.     }
  753.     public function setHasBlockchainTraceability(bool $hasBlockchainTraceability): self
  754.     {
  755.         $this->hasBlockchainTraceability $hasBlockchainTraceability;
  756.         return $this;
  757.     }
  758.     public function getShowOnWall(): ?bool
  759.     {
  760.         return $this->showOnWall;
  761.     }
  762.     public function setShowOnWall(bool $showOnWall): self
  763.     {
  764.         $this->showOnWall $showOnWall;
  765.         return $this;
  766.     }
  767.     public function getAllowReturn(): ?bool
  768.     {
  769.         return $this->allowReturn;
  770.     }
  771.     public function setAllowReturn(bool $allowReturn): self
  772.     {
  773.         $this->allowReturn $allowReturn;
  774.         return $this;
  775.     }
  776.     public function getNDaysAllowedForReturn(): ?int
  777.     {
  778.         return $this->nDaysAllowedForReturn;
  779.     }
  780.     public function setNDaysAllowedForReturn(?int $nDaysAllowedForReturn): self
  781.     {
  782.         $this->nDaysAllowedForReturn $nDaysAllowedForReturn;
  783.         return $this;
  784.     }
  785.     public function isInWarrantyPeriod(DateTime $dateBought): ?bool
  786.     {
  787.         $now = new DateTime('now');
  788.         if ($this->getAllowReturn() && $this->getNDaysAllowedForReturn()) {
  789.             $dayToCompare $dateBought->add(new DateInterval('P' $this->getNDaysAllowedForReturn() . 'D'));
  790.             if ($now $dateBought) {
  791.                 return true;
  792.             } else {
  793.                 return false;
  794.             }
  795.         } else {
  796.             return false;
  797.         }
  798.     }
  799.     public function isSendNotifications(): ?bool
  800.     {
  801.         return $this->sendNotifications;
  802.     }
  803.     public function setSendNotifications(bool $sendNotifications): self
  804.     {
  805.         $this->sendNotifications $sendNotifications;
  806.         return $this;
  807.     }
  808.     public function getProducerPurchaseOrderNumber(): ?string
  809.     {
  810.         return $this->producerPurchaseOrderNumber;
  811.     }
  812.     public function setProducerPurchaseOrderNumber(?string $producerPurchaseOrderNumber): self
  813.     {
  814.         $this->producerPurchaseOrderNumber $producerPurchaseOrderNumber;
  815.         return $this;
  816.     }
  817.     public function getLimitUnitsPerBuyer(): ?int
  818.     {
  819.         return $this->limitUnitsPerBuyer;
  820.     }
  821.     public function setLimitUnitsPerBuyer(int $limitUnitsPerBuyer): self
  822.     {
  823.         $this->limitUnitsPerBuyer $limitUnitsPerBuyer;
  824.         return $this;
  825.     }
  826.     public function getLimitUnitsToSell(): ?int
  827.     {
  828.         return $this->limitUnitsToSell;
  829.     }
  830.     public function setLimitUnitsToSell(?int $limitUnitsToSell): self
  831.     {
  832.         $this->limitUnitsToSell $limitUnitsToSell;
  833.         return $this;
  834.     }
  835.     public function getComingSoonDate(): ?\DateTimeInterface
  836.     {
  837.         return $this->comingSoonDate;
  838.     }
  839.     public function setComingSoonDate(?\DateTimeInterface $comingSoonDate): self
  840.     {
  841.         $this->comingSoonDate $comingSoonDate;
  842.         return $this;
  843.     }
  844.     public function getOfferLaunchDate(): ?\DateTimeInterface
  845.     {
  846.         return $this->offerLaunchDate;
  847.     }
  848.     public function setOfferLaunchDate(?\DateTimeInterface $offerLaunchDate): self
  849.     {
  850.         $this->offerLaunchDate $offerLaunchDate;
  851.         return $this;
  852.     }
  853.     /**
  854.      * Indicates if it should show the countdown in templates
  855.      *
  856.      * @return boolean
  857.      */
  858.     public function getShowCountdown(): bool
  859.     {
  860.         if ($this->countdownDate && $this->countdownDate < (new \DateTime())) {
  861.             return true;
  862.         }
  863.         return false;
  864.     }
  865.     public function getCountdownDate(): ?\DateTimeInterface
  866.     {
  867.         return $this->countdownDate;
  868.     }
  869.     public function setCountdownDate(?\DateTimeInterface $countdownDate): self
  870.     {
  871.         $this->countdownDate $countdownDate;
  872.         return $this;
  873.     }
  874.     public function getEndOfferDate(): ?\DateTimeInterface
  875.     {
  876.         return $this->endOfferDate;
  877.     }
  878.     public function setEndOfferDate(?\DateTimeInterface $endOfferDate): self
  879.     {
  880.         $this->endOfferDate $endOfferDate;
  881.         return $this;
  882.     }
  883.     public function getShippingDate(): ?\DateTimeInterface
  884.     {
  885.         return $this->shippingDate;
  886.     }
  887.     public function setShippingDate(?\DateTimeInterface $shippingDate): self
  888.     {
  889.         $this->shippingDate $shippingDate;
  890.         return $this;
  891.     }
  892.     public function getIndividualShippingDuration(): ?int
  893.     {
  894.         return $this->individualShippingDuration ?? Product::DEFAULT_SHIPPING_INDIVIDUAL_DURANTION;
  895.     }
  896.     public function setIndividualShippingDuration(?int $individualShippingDuration): self
  897.     {
  898.         $this->individualShippingDuration $individualShippingDuration;
  899.         return $this;
  900.     }
  901.     public function getSharedShippingDuration(): ?int
  902.     {
  903.         return $this->sharedShippingDuration ?? Product::DEFAULT_SHIPPING_SHARED_DURANTION;
  904.     }
  905.     public function setSharedShippingDuration(?int $sharedShippingDuration): self
  906.     {
  907.         $this->sharedShippingDuration $sharedShippingDuration;
  908.         return $this;
  909.     }
  910.     public function getIndividualPackageFormat(): ?string
  911.     {
  912.         return $this->individualPackageFormat;
  913.     }
  914.     public function setIndividualPackageFormat(?string $individualPackageFormat): self
  915.     {
  916.         $this->individualPackageFormat $individualPackageFormat;
  917.         return $this;
  918.     }
  919.     public function getNHoursOnWallAfterEnd(): ?int
  920.     {
  921.         return $this->nHoursOnWallAfterEnd;
  922.     }
  923.     public function setNHoursOnWallAfterEnd(int $nHoursOnWallAfterEnd): self
  924.     {
  925.         $this->nHoursOnWallAfterEnd $nHoursOnWallAfterEnd;
  926.         return $this;
  927.     }
  928.     public function getIva(): ?Iva
  929.     {
  930.         return $this->iva;
  931.     }
  932.     public function setIva(?Iva $iva): self
  933.     {
  934.         $this->iva $iva;
  935.         return $this;
  936.     }
  937.     public function getIndividualPrice(): ?float
  938.     {
  939.         return $this->individualPrice ?? $this->getInitialPrice();
  940.     }
  941.     public function setIndividualPrice(?float $individualPrice): self
  942.     {
  943.         $this->individualPrice $individualPrice;
  944.         return $this;
  945.     }
  946.     public function getIndividualDisabled(): ?bool
  947.     {
  948.         return $this->individualDisabled ?? $this->getInitialPrice();
  949.     }
  950.     public function setIndividualDisabled(?bool $individualDisabled): self
  951.     {
  952.         $this->individualDisabled $individualDisabled;
  953.         return $this;
  954.     }
  955.     public function getInitialPrice(): ?float
  956.     {
  957.         if ($this->priceScales->count() === 0) {
  958.             return $this->originalPrice;
  959.         }
  960.         return $this->priceScales[0]->getInitialPrice();
  961.     }
  962.     public function getCurrentPrice(): float
  963.     {
  964.         $currentQuantityOrdered $this->getCurrentQuantityOrdered();
  965.         $currentPriceScale $this->getCurrentPriceScale($currentQuantityOrdered);
  966.         if ($currentPriceScale->isSellProductIfNotComplete()) {
  967.             if ($currentQuantityOrdered  $currentPriceScale->getTotalMaxQuantity()) {
  968.                 return $currentPriceScale->getInitialPrice();
  969.             } else {
  970.                 return $currentPriceScale->getFinalPrice();
  971.             }
  972.         } else {
  973.             return $this->getNextPrice();
  974.         }
  975.     }
  976.     public function getNextPrice(?int $cartQuantity 0): float
  977.     {
  978.         $currentQuantityOrdered $this->getCurrentQuantityOrdered();
  979.         $currentPriceScale      $this->getCurrentPriceScale($currentQuantityOrdered);
  980.         // if (($this->priceScales->count() === 1) or ($this->currentScaleIsFirst($currentPriceScale))) {
  981.         //     if ($currentPriceScale->isSellProductIfNotComplete()) {
  982.         //         return $currentPriceScale->getInitialPrice();
  983.         //     }else{
  984.         //         return $currentPriceScale->getFinalPrice();
  985.         //     }
  986.         // }
  987.         if ($currentPriceScale->isSellProductIfNotComplete()) {
  988.             $belowMinQty = ($currentQuantityOrdered $cartQuantity) < $this->getScalesAccumulatedMinQuantity();
  989.             return $belowMinQty
  990.                 $currentPriceScale->getInitialPrice()
  991.                 : $currentPriceScale->getFinalPrice();
  992.         } else {
  993.             return $currentPriceScale->getFinalPrice();
  994.         }
  995.     }
  996.     /**
  997.      * Get the accumulated minimum quantity for multiple price scales.
  998.      */
  999.     public function getScalesAccumulatedMinQuantity(): int
  1000.     {
  1001.         $currentQuantityOrdered $this->getCurrentQuantityOrdered();
  1002.         $currentPriceScale      $this->getCurrentPriceScale($currentQuantityOrdered);
  1003.         return $this->getPriceScales()->reduce(
  1004.             fn ($acc$scale) =>
  1005.             $currentPriceScale->getFinalPrice() <= $scale->getFinalPrice()
  1006.                 ? $acc $scale->getScaleQuantity()
  1007.                 : $acc
  1008.         );
  1009.     }
  1010.     /**
  1011.      * Get the goal of orders (sum. of all scales units)
  1012.      *
  1013.      * @return float|null
  1014.      */
  1015.     public function getScaleOrdersGoal(): ?float
  1016.     {
  1017.         return $this->getPriceScales()->reduce(
  1018.             fn ($acc$scale) => $acc $scale->getScaleQuantity()
  1019.         );
  1020.     }
  1021.     /**
  1022.      * Get the current quantity of orders (sum. of all scales units until the current one)
  1023.      *
  1024.      * @return float|null
  1025.      */
  1026.     public function getScaleOrdersCurrent(): ?float
  1027.     {
  1028.         $current $this->getCurrentPriceScale();
  1029.         return $this->getPriceScales()->reduce(
  1030.             function ($acc$scale) use ($current) {
  1031.                 if ($current->getFinalPrice() <= $scale->getFinalPrice()) {
  1032.                     $acc += $scale->getScaleQuantity();
  1033.                 }
  1034.                 return $acc;
  1035.             }
  1036.         );
  1037.     }
  1038.     /**
  1039.      * Check if the current price scale is the first one.
  1040.      */
  1041.     public function currentScaleIsFirst($currentPriceScale): bool
  1042.     {
  1043.         return $this->priceScales->first()->getId() === $currentPriceScale->getId();
  1044.     }
  1045.     /**
  1046.      * Get the current price scale. It depends on the quantity already ordered
  1047.      *
  1048.      * @return ProductPriceScale
  1049.      */
  1050.     public function getCurrentPriceScale($currentQuantityOrdered null): ProductPriceScale
  1051.     {
  1052.         if ($currentQuantityOrdered === null) {
  1053.             $currentQuantityOrdered $this->getCurrentQuantityOrdered();
  1054.         }
  1055.         $currentPriceScale null;
  1056.         /** @var ProductPriceScale */
  1057.         foreach ($this->priceScales as $priceScale) {
  1058.             if (
  1059.                 ($priceScale->getTotalMinQuantity() <= $currentQuantityOrdered) &&
  1060.                 ($currentQuantityOrdered $priceScale->getTotalMaxQuantity())
  1061.             ) {
  1062.                 $currentPriceScale $priceScale;
  1063.                 break;
  1064.             }
  1065.         }
  1066.         if (!$currentPriceScale) {
  1067.             $currentPriceScale $priceScale;
  1068.         }
  1069.         return $currentPriceScale;
  1070.     }
  1071.     /**
  1072.      * @return boolean
  1073.      */
  1074.     public function getIsSellIsEnought()
  1075.     {
  1076.         return $this->getCurrentPriceScale()->isSellProductIfNotComplete();
  1077.     }
  1078.     /**
  1079.      * Get the current price scale. It depends on the quantity already ordered
  1080.      *
  1081.      * @return ProductPriceScale
  1082.      */
  1083.     public function getNextPriceScale($currentQuantityOrdered null): ProductPriceScale
  1084.     {
  1085.         if ($currentQuantityOrdered === null) {
  1086.             $currentQuantityOrdered $this->getCurrentQuantityOrdered();
  1087.         }
  1088.         $currentPriceScale null;
  1089.         $next False;
  1090.         $nextPriceScale null;
  1091.         /** @var ProductPriceScale */
  1092.         foreach ($this->priceScales as $priceScale) {
  1093.             if ($next) {
  1094.                 $nextPriceScale $priceScale;
  1095.                 $next False;
  1096.             }
  1097.             if (
  1098.                 ($priceScale->getTotalMinQuantity() <= $currentQuantityOrdered) &&
  1099.                 ($currentQuantityOrdered $priceScale->getTotalMaxQuantity())
  1100.             ) {
  1101.                 $next true;
  1102.                 $currentPriceScale $priceScale;
  1103.             }
  1104.         }
  1105.         if (!$currentPriceScale) {
  1106.             $currentPriceScale $priceScale;
  1107.         }
  1108.         if (!$nextPriceScale) {
  1109.             $nextPriceScale $currentPriceScale;
  1110.         }
  1111.         return $nextPriceScale;
  1112.     }
  1113.     /**
  1114.      * Max product quantity
  1115.      *
  1116.      * @return integer
  1117.      */
  1118.     public function getProductMaxQuantity(): int
  1119.     {
  1120.         $totalQuantity 0;
  1121.         if ($this->priceScales) {
  1122.             /** @var ProductPriceScale */
  1123.             foreach ($this->priceScales as $priceScale) {
  1124.                 $totalQuantity += $priceScale->getScaleQuantity();
  1125.             }
  1126.         }
  1127.         return $totalQuantity;
  1128.     }
  1129.     /**
  1130.      * Max product quantity
  1131.      *
  1132.      * @return integer
  1133.      */
  1134.     public function getProductMaxQuantityAvailable(): int
  1135.     {
  1136.         $totalQuantity 0;
  1137.         if ($this->priceScales) {
  1138.             /** @var ProductPriceScale */
  1139.             foreach ($this->priceScales as $priceScale) {
  1140.                 $totalQuantity += $priceScale->getScaleQuantityAvailable();
  1141.             }
  1142.         }
  1143.         return $totalQuantity;
  1144.     }
  1145.     /**
  1146.      * @return ProductPriceScale
  1147.      */
  1148.     public function getInitialPriceScale(): ProductPriceScale
  1149.     {
  1150.         return $this->priceScales->first();
  1151.     }
  1152.     /**
  1153.      * @return ProductPriceScale
  1154.      */
  1155.     public function getLastPriceScale(): ProductPriceScale
  1156.     {
  1157.         return $this->priceScales->last();
  1158.     }
  1159.     public function getOrderedPriceScales(): Collection
  1160.     {
  1161.         $criteria Criteria::create()
  1162.             ->orderBy(['scaleQuantity' => Criteria::ASC]);
  1163.         $orderedPrices $this->priceScales->matching($criteria);
  1164.         return $orderedPrices;
  1165.     }
  1166.     /**
  1167.      * @return Collection<int, ProductPriceScale>
  1168.      */
  1169.     public function getPriceScales(): Collection
  1170.     {
  1171.         return $this->priceScales;
  1172.     }
  1173.     public function addPriceScale(ProductPriceScale $priceScale): self
  1174.     {
  1175.         if (!$this->priceScales->contains($priceScale)) {
  1176.             $this->priceScales->add($priceScale);
  1177.             $priceScale->setProduct($this);
  1178.         }
  1179.         return $this;
  1180.     }
  1181.     public function removePriceScale(ProductPriceScale $priceScale): self
  1182.     {
  1183.         if ($this->priceScales->removeElement($priceScale)) {
  1184.             // set the owning side to null (unless already changed)
  1185.             if ($priceScale->getProduct() === $this) {
  1186.                 $priceScale->setProduct(null);
  1187.             }
  1188.         }
  1189.         return $this;
  1190.     }
  1191.     public function getMaxCoinsToSpent(): ?int
  1192.     {
  1193.         return $this->maxCoinsToSpent;
  1194.     }
  1195.     public function setMaxCoinsToSpent(int $maxCoinsToSpent): self
  1196.     {
  1197.         $this->maxCoinsToSpent $maxCoinsToSpent;
  1198.         return $this;
  1199.     }
  1200.     public function getBrand(): ?Brand
  1201.     {
  1202.         return $this->brand;
  1203.     }
  1204.     public function setBrand(?Brand $brand): self
  1205.     {
  1206.         $this->brand $brand;
  1207.         return $this;
  1208.     }
  1209.     public function getProducer(): ?Producer
  1210.     {
  1211.         return $this->producer;
  1212.     }
  1213.     public function setProducer(?Producer $producer): self
  1214.     {
  1215.         $this->producer $producer;
  1216.         return $this;
  1217.     }
  1218.     public function getUnitMeasurement(): ?UnitMeasurement
  1219.     {
  1220.         return $this->unitMeasurement;
  1221.     }
  1222.     public function setUnitMeasurement(?UnitMeasurement $unitMeasurement): self
  1223.     {
  1224.         $this->unitMeasurement $unitMeasurement;
  1225.         return $this;
  1226.     }
  1227.     /**
  1228.      * Get the current products's quantity ordered
  1229.      * @return int
  1230.      */
  1231.     public function getCurrentQuantityOrdered(): int
  1232.     {
  1233.         $currentQuantityOrdered $this->fakeUsers $this->fakeUsers 0;
  1234.         /** @var OrderDetail */
  1235.         foreach ($this->orderDetails as $orderDetail) {
  1236.             if (
  1237.                 $orderDetail->getHeader()->getStatus()->getKey() == OrderStatus::PENDING_PAYMENT ||
  1238.                 $orderDetail->getHeader()->getStatus()->getKey() == OrderStatus::CANCELED
  1239.             ) {
  1240.                 continue;
  1241.             }
  1242.             $currentQuantityOrdered += $orderDetail->getQuantity();
  1243.         }
  1244.         return $currentQuantityOrdered;
  1245.     }
  1246.     /**
  1247.      * Get the current products's quantity ordered
  1248.      * @return bool
  1249.      */
  1250.     public function hasBeenPurchasedRecently(?int $orderID null): bool
  1251.     {
  1252.         /** @var OrderDetail */
  1253.         foreach ($this->orderDetails as $orderDetail) {
  1254.             $order $orderDetail->getHeader();
  1255.             if ($orderID && $order->getId() === $orderID) continue;
  1256.             if ($order->getStatus()->getKey() == OrderStatus::MONEY_WITHHELD) {
  1257.                 return true;
  1258.             }
  1259.         }
  1260.         return false;
  1261.     }
  1262.     /**
  1263.      * Get the current products's quantity ordered in percentage
  1264.      * @return float
  1265.      */
  1266.     public function getCurrentOrderedPercentage(): float
  1267.     {
  1268.         $productMaxQuantity $this->getProductMaxQuantity();
  1269.         if ($productMaxQuantity) {
  1270.             return (100 $this->getCurrentQuantityOrdered() / $productMaxQuantity);
  1271.         } else {
  1272.             return 0;
  1273.         }
  1274.     }
  1275.     /**
  1276.      * @return Collection<int, OrderDetail>
  1277.      */
  1278.     public function getOrderDetails(): Collection
  1279.     {
  1280.         return $this->orderDetails;
  1281.     }
  1282.     public function addOrderDetail(OrderDetail $orderDetail): self
  1283.     {
  1284.         if (!$this->orderDetails->contains($orderDetail)) {
  1285.             $this->orderDetails->add($orderDetail);
  1286.             $orderDetail->setProduct($this);
  1287.         }
  1288.         return $this;
  1289.     }
  1290.     public function removeOrderDetail(OrderDetail $orderDetail): self
  1291.     {
  1292.         if ($this->orderDetails->removeElement($orderDetail)) {
  1293.             // set the owning side to null (unless already changed)
  1294.             if ($orderDetail->getProduct() === $this) {
  1295.                 $orderDetail->setProduct(null);
  1296.             }
  1297.         }
  1298.         return $this;
  1299.     }
  1300.     public function getPendingOrderDetails(): Collection
  1301.     {
  1302.         return $this->getOrderDetails()->filter(function($i) {
  1303.             $validStatus = !in_array($i->getHeader()->getStatus()->getKey(), [OrderStatus::PENDING_PAYMENTOrderStatus::CANCELEDOrderStatus::NOT_SOLD]);
  1304.             $validDate   $i->getCreatedAt() > $this->getOfferLaunchDate();
  1305.             
  1306.             return $validStatus && $validDate;
  1307.         });
  1308.     }
  1309.     public function getPendingOrderDetailsQuantity(): int
  1310.     {
  1311.         return $this->getOrderDetails()->reduce(function($acc$item) {
  1312.             if (in_array($item->getHeader()->getStatus()->getKey(), [OrderStatus::MONEY_WITHHELDOrderStatus::BANK_TRANSFER])) {
  1313.                 $acc += $item->getQuantity();
  1314.             }
  1315.             return $acc;
  1316.         }, 0);
  1317.     }
  1318.     public function getNumberOfOrders(): ?int
  1319.     {
  1320.         $ordersById = [];
  1321.         /** @var OrderDetail */
  1322.         foreach ($this->orderDetails as $orderDetail) {
  1323.             if (
  1324.                 $orderDetail->getHeader()->getStatus()->getKey() == OrderStatus::PENDING_PAYMENT ||
  1325.                 $orderDetail->getHeader()->getStatus()->getKey() == OrderStatus::CANCELED ||
  1326.                 $orderDetail->getHeader()->getStatus()->getKey() == OrderStatus::NOT_SOLD
  1327.             ) {
  1328.                 continue;
  1329.             }
  1330.             array_push($ordersById, [$orderDetail->getHeader()->getId() => $orderDetail]);
  1331.         }
  1332.         return count($ordersById);
  1333.     }
  1334.     public function getOriginalPrice(): ?float
  1335.     {
  1336.         return $this->originalPrice;
  1337.     }
  1338.     public function setOriginalPrice(?float $originalPrice): self
  1339.     {
  1340.         $this->originalPrice $originalPrice;
  1341.         return $this;
  1342.     }
  1343.     public function getFakeUsers(): ?int
  1344.     {
  1345.         return $this->fakeUsers;
  1346.     }
  1347.     public function setFakeUsers(?int $fakeUsers): self
  1348.     {
  1349.         $this->fakeUsers $fakeUsers;
  1350.         return $this;
  1351.     }
  1352.     /**
  1353.      * @return Collection<int, User>
  1354.      */
  1355.     public function getFakeTriweersInOffer(): Collection
  1356.     {
  1357.         return $this->fakeTriweersInOffer;
  1358.     }
  1359.     public function addFakeTriweersInOffer(User $user): self
  1360.     {
  1361.         if (!$this->fakeTriweersInOffer->contains($user)) {
  1362.             $this->fakeTriweersInOffer[] = $user;
  1363.         }
  1364.         return $this;
  1365.     }
  1366.     public function removeFakeTriweersInOffer(User $user): self
  1367.     {
  1368.         $this->fakeTriweersInOffer->removeElement($user);
  1369.         return $this;
  1370.     }
  1371.     /**
  1372.      * Calculate discount percentage dynamically based on field values
  1373.      *
  1374.      * @return float|null
  1375.      */
  1376.     public function getDiscountPercentage(): ?float 
  1377.     {
  1378.         if ($this->getUnitsPerBoxGrouped() > 0) {
  1379.             $originalUnitPrice $this->originalPrice / ($this->getUnitsPerBox() ?? 1);
  1380.             $sharedUnitPrice $this->getNextPrice() / $this->getUnitsPerBoxGrouped();
  1381.             return (100 - (100 $sharedUnitPrice $originalUnitPrice));
  1382.         }
  1383.         return (100 - (100 $this->getNextPrice() / $this->originalPrice));
  1384.     }
  1385.     public function getCartSavings($hasGroup)
  1386.     {
  1387.         $finalPrice $this->getNextPrice();
  1388.         $saving     0;
  1389.         if ($hasGroup) {
  1390.             $originalUnitPrice $this->getOriginalPrice() / $this->getUnitsPerBoxGrouped();
  1391.             $unitPrice         $finalPrice $this->getUnitsPerBoxGrouped();
  1392.             $saving            = ($this->getUnitsPerBoxGrouped() > 0)
  1393.                 ? (100 - (100 $unitPrice $originalUnitPrice))
  1394.                 : (100 - (100 $finalPrice $this->getOriginalPrice()));
  1395.         } else {
  1396.             $originalUnitPrice $this->getOriginalPrice() / $this->getUnitsPerBox();
  1397.             $unitPrice         $finalPrice $this->getUnitsPerBox();
  1398.             $saving            = ($this->getUnitsPerBox() > 0)
  1399.                  ? (100 - (100 $unitPrice $originalUnitPrice))
  1400.                  : (100 - (100 $finalPrice $this->getOriginalPrice()));
  1401.         }
  1402.         return $saving;
  1403.     }
  1404.     /**
  1405.      * Calculate max. discount percentage dynamically based on field values
  1406.      *
  1407.      * @return float|null
  1408.      */
  1409.     public function getMaxDiscountPercentage(): ?float
  1410.     {
  1411.         $lastScale $this->getPriceScales()->last();
  1412.         $originalUnitPrice $this->originalPrice $this->getUnitsPerBox();
  1413.         $sharedUnitPrice $lastScale->getFinalPrice() / ($this->getUnitsPerBoxGrouped() ?? 1);
  1414.         return (100 - (100 $sharedUnitPrice $originalUnitPrice));
  1415.     }
  1416.     /**
  1417.      * Calculate max. discount price dynamically based on field values
  1418.      *
  1419.      * @return float|null
  1420.      */
  1421.     public function getMaxDiscountUnitPrice(): ?float
  1422.     {
  1423.         $lastScale $this->getPriceScales()->last();
  1424.         $sharedUnitPrice $lastScale->getFinalPrice() / ($this->getUnitsPerBoxGrouped() ?? 1);
  1425.         return $sharedUnitPrice;
  1426.     }
  1427.     /**
  1428.      * Calculate discount percentage for collective purchase dynamically based on field values
  1429.      *
  1430.      * @return float|null
  1431.      */
  1432.     public function getIndividualDiscountPercentage(): ?float
  1433.     {
  1434.         return (100 - (100 $this->getIndividualPrice() / $this->originalPrice));
  1435.     }
  1436.     public function getNextDiscountPercentage(): ?float
  1437.     {
  1438.         if (count($this->getPriceScales()) > 1) {
  1439.             return (100 - (100 $this->getNextPrice() / $this->originalPrice));
  1440.         } else {
  1441.             // Return Discount relative to final price Scale if only 1 price
  1442.             // scale is defined. #316
  1443.             return (100 - (100 $this->getLastPriceScale()->getFinalPrice() / $this->originalPrice));
  1444.         };
  1445.     }
  1446.     public function getShippingCost(): ?float
  1447.     {
  1448.         return $this->getShipping() ? $this->getShipping()->getCostByKgs($this->weight) : null;
  1449.     }
  1450.     public function setShippingCost(?float $shippingCost): self
  1451.     {
  1452.         $this->shippingCost $shippingCost;
  1453.         return $this;
  1454.     }
  1455.     public function getHoldedId(): ?string
  1456.     {
  1457.         return $this->holdedId;
  1458.     }
  1459.     public function setHoldedId(?string $holdedId): self
  1460.     {
  1461.         $this->holdedId $holdedId;
  1462.         return $this;
  1463.     }
  1464.     /**
  1465.      * @return Collection<int, UserShared>
  1466.      */
  1467.     public function getInfluencerShareds(): Collection
  1468.     {
  1469.         return $this->influencerShareds;
  1470.     }
  1471.     public function addInfluencerShared(UserShared $influencerShared): self
  1472.     {
  1473.         if (!$this->influencerShareds->contains($influencerShared)) {
  1474.             $this->influencerShareds->add($influencerShared);
  1475.             $influencerShared->setProduct($this);
  1476.         }
  1477.         return $this;
  1478.     }
  1479.     public function removeInfluencerShared(UserShared $influencerShared): self
  1480.     {
  1481.         if ($this->influencerShareds->removeElement($influencerShared)) {
  1482.             // set the owning side to null (unless already changed)
  1483.             if ($influencerShared->getProduct() === $this) {
  1484.                 $influencerShared->setProduct(null);
  1485.             }
  1486.         }
  1487.         return $this;
  1488.     }
  1489.     public function getProductUnique(): ?ProductUnique
  1490.     {
  1491.         return $this->productUnique;
  1492.     }
  1493.     public function setProductUnique(?ProductUnique $productUnique): self
  1494.     {
  1495.         $this->productUnique $productUnique;
  1496.         return $this;
  1497.     }
  1498.     public function isOnlyAdults(): ?bool
  1499.     {
  1500.         return $this->onlyAdults;
  1501.     }
  1502.     public function setOnlyAdults(bool $onlyAdults): self
  1503.     {
  1504.         $this->onlyAdults $onlyAdults;
  1505.         return $this;
  1506.     }
  1507.     public function getAmountForFreeShipping(): ?int
  1508.     {
  1509.         return $this->amountForFreeShipping;
  1510.     }
  1511.     public function setAmountForFreeShipping(?int $amountForFreeShipping): self
  1512.     {
  1513.         $this->amountForFreeShipping $amountForFreeShipping;
  1514.         return $this;
  1515.     }
  1516.     public function isBetterPrice(): ?bool
  1517.     {
  1518.         return $this->better_price;
  1519.     }
  1520.     public function setBetterPrice(?bool $better_price): self
  1521.     {
  1522.         $this->better_price $better_price;
  1523.         return $this;
  1524.     }
  1525.     public function getShipping(): ?Shipping
  1526.     {
  1527.         return $this->shipping;
  1528.     }
  1529.     public function setShipping(?Shipping $shipping): self
  1530.     {
  1531.         $this->shipping $shipping;
  1532.         return $this;
  1533.     }
  1534.     /**
  1535.      * @return Collection<int, User>
  1536.      */
  1537.     public function getUserFavourites(): Collection
  1538.     {
  1539.         return $this->userFavourites;
  1540.     }
  1541.     /**
  1542.      * Shuffles the collection
  1543.      * 
  1544.      * @return Collection<int, User>
  1545.      */
  1546.     public function getRandomUserFavourites(): Collection
  1547.     {
  1548.         $array $this->userFavourites->toArray();
  1549.         shuffle($array);
  1550.         return new ArrayCollection($array);
  1551.     }
  1552.     /**
  1553.      * @return Collection<int, Review>
  1554.      */
  1555.     public function getUserReviews(): Collection
  1556.     {
  1557.         return $this->userReviews;
  1558.     }
  1559.     public function getMetaTitle(): ?string
  1560.     {
  1561.         return $this->meta_title;
  1562.     }
  1563.     public function setMetaTitle(?string $meta_title): self
  1564.     {
  1565.         $this->meta_title $meta_title;
  1566.         return $this;
  1567.     }
  1568.     public function getMetaDescription(): ?string
  1569.     {
  1570.         return $this->meta_description;
  1571.     }
  1572.     public function setMetaDescription(?string $meta_description): self
  1573.     {
  1574.         $this->meta_description $meta_description;
  1575.         return $this;
  1576.     }
  1577.     public function isShowPrice(): ?bool
  1578.     {
  1579.         return $this->showPrice;
  1580.     }
  1581.     public function setShowPrice(?bool $showPrice): self
  1582.     {
  1583.         $this->showPrice $showPrice;
  1584.         return $this;
  1585.     }
  1586. }