. */ declare(strict_types=1); namespace App\Entity\ProjectSystem; use App\Entity\Base\AbstractDBElement; use App\Entity\Base\TimestampTrait; use App\Entity\Parts\Part; use Doctrine\ORM\Mapping as ORM; use Symfony\Bridge\Doctrine\Validator\Constraints\UniqueEntity; use Symfony\Component\Validator\Constraints as Assert; use Symfony\Component\Validator\Context\ExecutionContextInterface; /** * The ProjectBOMEntry class represents a entry in a project's BOM. * * @ORM\Table("device_parts") * @ORM\HasLifecycleCallbacks() * @ORM\Entity() * @UniqueEntity(fields={"part", "project"}, message="project.bom_entry.part_already_in_bom") * @UniqueEntity(fields={"name", "project"}, message="project.bom_entry.name_already_in_bom", ignoreNull=true) */ class ProjectBOMEntry extends AbstractDBElement { use TimestampTrait; /** * @var int * @ORM\Column(type="float", name="quantity") * @Assert\Positive() */ protected float $quantity; /** * @var string A comma separated list of the names, where this parts should be placed * @ORM\Column(type="text", name="mountnames") */ protected string $mountnames; /** * @var string An optional name describing this BOM entry (useful for non-part entries) * @ORM\Column(type="string", nullable=true) * @Assert\Expression( * "this.getPart() !== null or this.getName() !== null", * message="validator.project.bom_entry.name_or_part_needed" * ) */ protected ?string $name = null; /** * @var string An optional comment for this BOM entry * @ORM\Column(type="text") */ protected string $comment; /** * @var Project * @ORM\ManyToOne(targetEntity="Project", inversedBy="bom_entries") * @ORM\JoinColumn(name="id_device", referencedColumnName="id") */ protected ?Project $project = null; /** * @var Part|null The part associated with this * @ORM\ManyToOne(targetEntity="App\Entity\Parts\Part", inversedBy="project_bom_entries") * @ORM\JoinColumn(name="id_part", referencedColumnName="id", nullable=true) */ protected ?Part $part = null; /** * @return float */ public function getQuantity(): float { return $this->quantity; } /** * @param float $quantity * @return ProjectBOMEntry */ public function setQuantity(float $quantity): ProjectBOMEntry { $this->quantity = $quantity; return $this; } /** * @return string */ public function getMountnames(): string { return $this->mountnames; } /** * @param string $mountnames * @return ProjectBOMEntry */ public function setMountnames(string $mountnames): ProjectBOMEntry { $this->mountnames = $mountnames; return $this; } /** * @return string */ public function getName(): ?string { return $this->name; } /** * @param string $name * @return ProjectBOMEntry */ public function setName(?string $name): ProjectBOMEntry { $this->name = $name; return $this; } /** * @return string */ public function getComment(): string { return $this->comment; } /** * @param string $comment * @return ProjectBOMEntry */ public function setComment(string $comment): ProjectBOMEntry { $this->comment = $comment; return $this; } /** * @return Project|null */ public function getProject(): ?Project { return $this->project; } /** * @param Project|null $project * @return ProjectBOMEntry */ public function setProject(?Project $project): ProjectBOMEntry { $this->project = $project; return $this; } /** * @return Part|null */ public function getPart(): ?Part { return $this->part; } /** * @param Part|null $part * @return ProjectBOMEntry */ public function setPart(?Part $part): ProjectBOMEntry { $this->part = $part; return $this; } /** * @Assert\Callback */ public function validate(ExecutionContextInterface $context, $payload) { //Round quantity to whole numbers, if the part is not a decimal part if ($this->part) { if (!$this->part->getPartUnit() || $this->part->getPartUnit()->isInteger()) { $this->quantity = round($this->quantity); } } //Check that every part name in the mountnames list is unique (per bom_entry) $mountnames = explode(',', $this->mountnames); $mountnames = array_map('trim', $mountnames); $uniq_mountnames = array_unique($mountnames); //If the number of unique names is not the same as the number of names, there are duplicates if (count($mountnames) !== count($uniq_mountnames)) { $context->buildViolation('project.bom_entry.mountnames_not_unique') ->atPath('mountnames') ->addViolation(); } //Check that the number of mountnames is the same as the (rounded) quantity if (!empty($this->mountnames) && count($uniq_mountnames) !== (int) round ($this->quantity)) { $context->buildViolation('project.bom_entry.mountnames_quantity_mismatch') ->atPath('mountnames') ->addViolation(); } } }