src/Entity/JournalVolume.php line 87
<?php
namespace App\Entity;
use Doctrine\ORM\Mapping as ORM,
Doctrine\DBAL\Types\Types,
Doctrine\Common\Collections\Collection,
Doctrine\Common\Collections\ArrayCollection;
use Symfony\Component\Serializer\Annotation\Groups,
Symfony\Component\Validator\Constraints as Assert;
use ApiPlatform\Metadata\ApiResource,
ApiPlatform\Metadata\ApiProperty,
ApiPlatform\Metadata\Get,
ApiPlatform\Metadata\GetCollection,
ApiPlatform\Metadata\Post,
ApiPlatform\Metadata\Put,
ApiPlatform\Metadata\Delete,
ApiPlatform\Metadata\ApiFilter,
ApiPlatform\Doctrine\Orm\Filter\OrderFilter,
ApiPlatform\Doctrine\Orm\Filter\BooleanFilter;
use App\Entity\Trait\IdTrait,
App\Entity\Trait\UuidTrait,
App\Entity\Trait\OrdStatTrait,
App\Entity\Trait\TimestampableTrait,
App\Entity\Trait\TranslatableTrait,
App\Entity\Trait\Languages\LanguageableChildTrait,
App\Entity\Trait\ArchivedTrait,
App\Entity\Trait\EditorialGroupableTrait,
App\Entity\Trait\ImportIdTrait,
App\Entity\Interface\TranslatableInterface,
App\Entity\Interface\OrdStatableInterface,
App\Entity\Interface\LanguageableChildInterface,
App\Entity\Interface\ArchivableInterface,
App\Entity\Interface\EditorialGroupableInterface,
App\Entity\Interface\CloneableInterface;
use App\Attribute\Sanitizeable,
App\Enum\JournalDateFormat,
App\Enum\JournalLicence,
App\Enum\AuthorProperty,
App\Doctrine\DBAL\Types\AuthorPropertiesType,
App\Repository\JournalVolumeRepository,
App\Security\Voter\ArchivableVoter,
App\Lib\Actions,
App\DTO\CloneDTO,
App\StateProcessor\CloneProcessor,
App\Security\Voter\CloneVoter,
App\Filter\IriFilter;
use App\Util\ClassUtils;
#[ApiResource(
description: 'Journal volumes',
normalizationContext: ['groups' => [
'read',
'read:' . self::class,
'read:' . self::class . 'Translation'
]],
denormalizationContext: ['groups' => ['write']],
security: 'is_granted("' . Journal::class . '")',
order: ['ord' => 'desc'],
operations: [
new GetCollection(),
new Post(
uriTemplate: '/journal_volumes/clone',
input: CloneDTO::class,
processor: CloneProcessor::class,
security: 'is_granted("' . Actions::CLONE .'")',
securityMessage: CloneVoter::MESSAGE
),
new Post(denormalizationContext: ['groups' => ['write', 'post']]),
new Get(),
new Put(),
new Delete(
securityPostDenormalize: 'is_granted("' . Actions::DELETE . '", object)',
securityPostDenormalizeMessage: ArchivableVoter::MESSAGE
),
],
extraProperties: ['standard_put' => false],
)]
#[ApiFilter(IriFilter::class, properties: ['parent'])]
#[ApiFilter(BooleanFilter::class, properties: ['isArchived', 'stat'])]
#[ApiFilter(OrderFilter::class, properties: ['ord'])]
#[ORM\Entity(repositoryClass: JournalVolumeRepository::class)]
class JournalVolume implements
TranslatableInterface,
OrdStatableInterface,
LanguageableChildInterface,
EditorialGroupableInterface,
ArchivableInterface,
CloneableInterface
{
use IdTrait,
UuidTrait,
OrdStatTrait,
TimestampableTrait,
TranslatableTrait,
LanguageableChildTrait,
ArchivedTrait,
EditorialGroupableTrait,
ImportIdTrait;
#[ApiProperty(description: 'Parent', readableLink: false, writableLink: false)]
#[Groups(['read', 'post'])]
#[ORM\ManyToOne(targetEntity: Journal::class, inversedBy: 'volumes')]
#[ORM\JoinColumn(onDelete: 'cascade', nullable: false)]
private Journal $parent;
#[ApiProperty(description: 'Planned numbers amount')]
#[Groups(['read:' . self::class, 'write'])]
#[Assert\Positive]
#[ORM\Column(type: Types::INTEGER, options: ['default' => 1])]
private int $plannedNumbersAmount = 1;
#[ApiProperty(description: 'Published at')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::DATE_IMMUTABLE, nullable: true)]
private ?\DateTimeImmutable $publishedAt = null;
#[ApiProperty(description: 'Is published at visible')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isPublishedAtVisible = false;
#[ApiProperty(description: 'Published at format', writableLink: false)]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(
type: Types::STRING,
enumType: JournalDateFormat::class,
length: 255,
options: ['default' => JournalDateFormat::COMPLETE]
)]
private JournalDateFormat $publishedAtFormat = JournalDateFormat::COMPLETE;
#[ApiProperty(description: 'DOI')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::STRING, length: 255, nullable: true)]
private ?string $doi = null;
#[ApiProperty(description: 'Is DOI visible')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isDoiVisible = false;
#[ApiProperty(description: 'Licence', writableLink: false)]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(
type: Types::STRING,
enumType: JournalLicence::class,
length: 255,
options: ['default' => JournalLicence::CC_BY]
)]
private JournalLicence $licence = JournalLicence::CC_BY;
#[ApiProperty(description: 'Is licence visible')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isLicenceVisible = false;
#[ApiProperty(description: 'Funding number')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::STRING, length: 255, nullable: true)]
private ?string $fundingNumber = null;
#[ApiProperty(description: 'Additional info')]
#[Groups(['read:' . self::class, 'write'])]
#[Sanitizeable]
#[ORM\Column(type: Types::TEXT, nullable: true)]
private ?string $additionalInfo = null;
#[ApiProperty(description: 'Meta link')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::STRING, length: 255, nullable: true)]
private ?string $metaLink = null;
#[ApiProperty(description: 'Visible editorial group member properties', writableLink: false)]
#[Groups(['read:' . self::class, 'write'])]
#[Assert\Choice(callback: [AuthorProperty::class, 'cases'], multiple: true)]
#[Assert\Unique]
#[Assert\Count(min: 1)]
#[ORM\Column(type: AuthorPropertiesType::NAME)]
private array $visibleEditorialGroupMemberProperties = [
AuthorProperty::PREFIX,
AuthorProperty::NAME,
AuthorProperty::SURNAME,
AuthorProperty::AFFILIATION
];
#[ApiProperty(description: 'Issues', writableLink: false)]
#[Groups(['read:' . self::class])]
#[ORM\OneToMany(targetEntity: JournalIssue::class, mappedBy: 'volume', cascade: ['persist'])]
#[ORM\OrderBy(['ord' => 'desc'])]
private Collection $issues;
#[ApiProperty(description: 'Import link')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::STRING, length: 255, nullable: true)]
private ?string $importLink = null;
#[ApiProperty(description: 'Import link en')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::STRING, length: 255, nullable: true)]
private ?string $importLinkEn = null;
public function __construct(Journal $parent)
{
$this->setUuid();
$this->parent = $parent;
$this->translations = new ArrayCollection();
$this->issues = new ArrayCollection();
$this->editorialGroups = new ArrayCollection();
foreach ($this->parent->getLanguages() as $lang) {
new JournalVolumeTranslation($this, $lang);
}
foreach($this->parent->getEditorialGroups() as $editorialGroup) {
$copied = JournalVolumeEditorialGroup::constructFrom(parent: $this, template: $editorialGroup);
$this->addEditorialGroup($copied);
}
$this->visibleEditorialGroupMemberProperties = $parent->getView()->getVisibleEditorialGroupMemberProperties();
$this->licence = $parent->getData()->getLicence();
$this->createdAt = new \DateTimeImmutable();
$this->updatedAt = new \DateTimeImmutable();
$parent->addVolume($this);
}
public function getParent(): Journal
{
return $this->parent;
}
public function getPlannedNumbersAmount(): int
{
return $this->plannedNumbersAmount;
}
public function setPlannedNumbersAmount(int $plannedNumbersAmount): self
{
$this->plannedNumbersAmount = $plannedNumbersAmount;
return $this;
}
#[Groups(['read:' . self::class])]
public function getNumbersRatio(): string
{
$issues = count($this->issues);
return "{$issues}/{$this->plannedNumbersAmount}";
}
public function getPublishedAt(): ?\DateTimeImmutable
{
return $this->publishedAt;
}
public function setPublishedAt(?\DateTimeImmutable $publishedAt): self
{
$this->publishedAt = $publishedAt;
return $this;
}
public function getIsPublishedAtVisible(): bool
{
return $this->isPublishedAtVisible;
}
public function setIsPublishedAtVisible(bool $isPublishedAtVisible): self
{
$this->isPublishedAtVisible = $isPublishedAtVisible;
return $this;
}
public function getPublishedAtFormat(): JournalDateFormat
{
return $this->publishedAtFormat;
}
public function setPublishedAtFormat(JournalDateFormat $publishedAtFormat): self
{
$this->publishedAtFormat = $publishedAtFormat;
return $this;
}
public function getDoi(): ?string
{
return $this->doi;
}
public function setDoi(?string $doi): self
{
$this->doi = $doi;
return $this;
}
public function getIsDoiVisible(): bool
{
return $this->isDoiVisible;
}
public function setIsDoiVisible(bool $isDoiVisible): self
{
$this->isDoiVisible = $isDoiVisible;
return $this;
}
public function getLicence(): JournalLicence
{
return $this->licence;
}
public function setLicence(JournalLicence $licence): self
{
$this->licence = $licence;
return $this;
}
public function getIsLicenceVisible(): bool
{
return $this->isLicenceVisible;
}
public function setIsLicenceVisible(bool $isLicenceVisible): self
{
$this->isLicenceVisible = $isLicenceVisible;
return $this;
}
public function getFundingNumber(): ?string
{
return $this->fundingNumber;
}
public function setFundingNumber(?string $fundingNumber): self
{
$this->fundingNumber = $fundingNumber;
return $this;
}
public function getAdditionalInfo(): ?string
{
return $this->additionalInfo;
}
public function setAdditionalInfo(?string $additionalInfo): self
{
$this->additionalInfo = $additionalInfo;
return $this;
}
public function getMetaLink(): ?string
{
return $this->metaLink;
}
public function setMetaLink(?string $metaLink): self
{
$this->metaLink = $metaLink;
return $this;
}
public function getVisibleEditorialGroupMemberProperties(): array
{
return $this->visibleEditorialGroupMemberProperties;
}
public function setVisibleEditorialGroupMemberProperties(array $visibleEditorialGroupMemberProperties): self
{
$this->visibleEditorialGroupMemberProperties = $visibleEditorialGroupMemberProperties;
return $this;
}
/**
* @return Collection|JournalIssue[]
*/
public function getIssues(): Collection
{
return $this->issues;
}
public function addIssue(JournalIssue $issue): self
{
if ($this->issues->contains($issue)) {
return $this;
}
$this->issues[] = $issue;
return $this;
}
public function removeIssue(JournalIssue $issue): self
{
$this->issues->removeElement($issue);
return $this;
}
public function resetIssues(): self
{
$this->issues = new ArrayCollection();
return $this;
}
public function clone(): self
{
$clone = clone $this;
$clone->id = null;
$clone->setUuid();
$clone->ord = 0;
$clone->stat = false;
$clone->importId = null;
ClassUtils::cloneCollection($this, $clone, 'translations');
$clone->synchronizeTranslations();
$nativeLanguage = $clone->parent->getNativeLanguage();
$nativeTranslation = $clone->getTranslation($nativeLanguage);
$nativeTranslation->setTitle($nativeTranslation->getTitle() . ' [KOPIA]');
ClassUtils::cloneCollection($this, $clone, 'editorialGroups');
$clone->resetIssues();
foreach ($this->getIssues() as $issue) {
$clone->addIssue(
$issue->clone(
$clone->parent
)
);
}
$clone->createdAt = new \DateTimeImmutable();
$clone->updatedAt = new \DateTimeImmutable();
return $clone;
}
public function getImportLink(): ?string
{
return $this->importLink;
}
public function setImportLink(?string $importLink): self
{
$this->importLink = $importLink;
return $this;
}
public function getImportLinkEn(): ?string
{
return $this->importLinkEn;
}
public function setImportLinkEn(?string $importLinkEn): self
{
$this->importLinkEn = $importLinkEn;
return $this;
}
}