src/Entity/JournalIssue.php line 138
<?php
namespace App\Entity;
use Doctrine\ORM\Mapping as ORM,
Doctrine\DBAL\Types\Types,
Doctrine\Common\Collections\Collection,
Doctrine\Common\Collections\ArrayCollection;
use Gedmo\Sluggable\Util\Urlizer;
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\SearchFilter,
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\Enum\JournalLicence,
App\Enum\JournalDateFormat,
App\Enum\AuthorProperty,
App\Enum\JournalIssueState,
App\Doctrine\DBAL\Types\AuthorPropertiesType,
App\Repository\JournalIssueRepository,
App\Lib\Actions,
App\DTO\CloneDTO,
App\Security\Voter\ArchivableVoter,
App\StateProcessor\CloneProcessor,
App\Security\Voter\CloneVoter;
use App\Util\ClassUtils,
App\Filter\IriFilter;
use App\Controller\Export\JournalIssue\JournalIssueCrossrefExporter,
App\Controller\Export\JournalIssue\JournalIssueArticlesCrossrefExporter,
App\Controller\Export\JournalIssue\JournalIssueSimcheckCrossrefExporter,
App\Controller\Export\JournalIssue\JournalIssueFundingCrossrefExporter,
App\Controller\Export\JournalIssue\JournalIssueLicenceFulltextCrossrefExporter,
App\Controller\Export\JournalIssue\JournalIssueArticlesCopernicusExporter,
App\Controller\Export\JournalIssue\JournalIssueArticlesPubMedExporter;
use App\Enum\Language;
use App\Util\RomanNumber;
#[ApiResource(
description: 'Journal issues',
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_issues/clone',
input: CloneDTO::class,
processor: CloneProcessor::class,
security: 'is_granted("' . Actions::CLONE .'")',
securityMessage: CloneVoter::MESSAGE
),
new Post(denormalizationContext: ['groups' => ['write', 'post']]),
new Get(),
new Get(
name: 'get_journal_issue_crossref',
uriTemplate: '/journal_issues/{uuid}/export/crossref',
controller: JournalIssueCrossrefExporter::class
),
new Get(
name: 'get_journal_issue_articles_crossref',
uriTemplate: '/journal_issues/{uuid}/articles/export/crossref',
controller: JournalIssueArticlesCrossrefExporter::class
),
new Get(
name: 'get_journal_issue_articles_copernicus',
uriTemplate: '/journal_issues/{uuid}/articles/export/copernicus',
controller: JournalIssueArticlesCopernicusExporter::class
),
new Get(
name: 'get_journal_issue_articles_pubmed',
uriTemplate: '/journal_issues/{uuid}/articles/export/pubmed',
controller: JournalIssueArticlesPubMedExporter::class
),
new Get(
name: 'get_journal_issue_simcheck_crossref',
uriTemplate: '/journal_issues/{uuid}/export/simcheck_crossref',
controller: JournalIssueSimcheckCrossrefExporter::class
),
new Get(
name: 'get_journal_issue_funding_crossref',
uriTemplate: '/journal_issues/{uuid}/export/funding_crossref',
controller: JournalIssueFundingCrossrefExporter::class
),
new Get(
name: 'get_journal_issue_licence_fulltext_crossref',
uriTemplate: '/journal_issues/{uuid}/export/licence_fulltext_crossref',
controller: JournalIssueLicenceFulltextCrossrefExporter::class
),
new Put(),
new Delete(
securityPostDenormalize: 'is_granted("' . Actions::DELETE . '", object)',
securityPostDenormalizeMessage: ArchivableVoter::MESSAGE
),
],
extraProperties: ['standard_put' => false],
)]
#[ApiFilter(SearchFilter::class, properties: [
'translations.title' => 'partial',
])]
#[ApiFilter(IriFilter::class, properties: ['parent', 'volume'])]
#[ApiFilter(BooleanFilter::class, properties: ['isArchived', 'stat'])]
#[ApiFilter(OrderFilter::class, properties: ['ord'])]
#[ORM\Entity(repositoryClass: JournalIssueRepository::class)]
class JournalIssue implements
TranslatableInterface,
OrdStatableInterface,
LanguageableChildInterface,
EditorialGroupableInterface,
ArchivableInterface,
CloneableInterface
{
use IdTrait,
UuidTrait,
OrdStatTrait,
TimestampableTrait,
TranslatableTrait,
LanguageableChildTrait,
ArchivedTrait,
EditorialGroupableTrait,
ImportIdTrait;
#[ApiProperty(description: 'Parent', writableLink: false)]
#[Groups(['read', 'post'])]
#[ORM\ManyToOne(targetEntity: Journal::class, inversedBy: 'issues')]
#[ORM\JoinColumn(onDelete: 'cascade', nullable: false)]
private Journal $parent;
#[ApiProperty(description: 'Volume', writableLink: false)]
#[Groups(['read', 'write'])]
#[Assert\Expression(
expression: '!value || value.getParent() === this.getParent()',
message: 'Cannot assign volume from the outside of current journal.'
)]
#[ORM\ManyToOne(targetEntity: JournalVolume::class, inversedBy: 'issues')]
#[ORM\JoinColumn(onDelete: 'cascade', nullable: false)]
private JournalVolume $volume;
#[ApiProperty(description: 'Is archived')]
#[Groups(['read:' . self::class, 'read:' . JournalVolume::class, 'read:list', 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
protected bool $isArchived = false;
#[ApiProperty(description: 'Planned articles amount')]
#[Groups(['read:' . self::class, 'write'])]
#[Assert\Positive]
#[ORM\Column(type: Types::INTEGER, options: ['default' => 1])]
private int $plannedArticlesAmount = 1;
#[ApiProperty(description: 'Is new')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isNew = false;
#[ApiProperty(description: 'Is newest issue')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => true])]
private bool $isNewestIssue = true;
#[ApiProperty(description: 'Is first view')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isFirstView = false;
#[ApiProperty(description: 'State', writableLink: false)]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(
type: Types::STRING,
enumType: JournalIssueState::class,
length: 255,
options: ['default' => JournalIssueState::LIST]
)]
private JournalIssueState $state = JournalIssueState::LIST;
#[ApiProperty(description: 'Is continous accession')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isContinousAccession = false;
#[ApiProperty(description: 'Number in year')]
#[Groups(['read:' . self::class, 'write'])]
#[Assert\Positive]
#[ORM\Column(type: Types::INTEGER, options: ['default' => 1])]
private int $numberInYear = 1;
#[ApiProperty(description: 'Is special issue')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isSpecialIssue = false;
#[ApiProperty(description: 'Is crossmark button visible')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isCrossmarkButtonVisible = false;
#[ApiProperty(description: 'Is open')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => true])]
private bool $isOpen = true;
#[ApiProperty(description: 'Price')]
#[Groups(['read:' . self::class])]
#[Assert\PositiveOrZero]
#[ORM\Column(type: Types::INTEGER, nullable: true)]
private ?int $price = null;
#[ApiProperty(description: 'Grace date')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::DATE_IMMUTABLE, nullable: true)]
private ?\DateTimeImmutable $graceDate = null;
#[ApiProperty(description: 'Is open access granted after newest issue expired')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
private bool $isOpenAccessGrantedAfterNewestIssueExpired = false;
#[ApiProperty(description: 'Published at')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::DATE_IMMUTABLE, nullable: true)]
private ?\DateTimeImmutable $publishedAt = null;
#[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: '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: 'Suggested citation type', writableLink: false)]
#[Groups(['read:' . Journal::class, 'write'])]
#[ORM\ManyToOne(targetEntity: CitationType::class)]
#[ORM\JoinColumn(onDelete: 'set null')]
private ?CitationType $suggestedCitationType = null;
#[ApiProperty(description: 'Funding number')]
#[Groups(['read:' . self::class, 'write'])]
#[ORM\Column(type: Types::STRING, length: 255, nullable: true)]
private ?string $fundingNumber = 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: 'Files')]
#[ORM\OneToMany(targetEntity: JournalIssueFile::class, mappedBy: 'parent', cascade: ['persist'])]
#[ORM\OrderBy(['ord' => 'desc'])]
private Collection $files;
#[ApiProperty(description: 'Sponsors')]
#[ORM\OneToMany(targetEntity: JournalIssueSponsor::class, mappedBy: 'parent', cascade: ['persist'])]
#[ORM\OrderBy(['ord' => 'desc'])]
private Collection $sponsors;
#[ApiProperty(description: 'Blocks')]
#[ORM\OneToMany(targetEntity: JournalIssueBlock::class, mappedBy: 'parent', cascade: ['persist'])]
#[ORM\OrderBy(['ord' => 'desc'])]
private Collection $blocks;
#[ApiProperty(description: 'Sections')]
#[ORM\OneToMany(targetEntity: JournalIssueSection::class, mappedBy: 'parent', cascade: ['persist'])]
#[ORM\OrderBy(['ord' => 'desc'])]
private Collection $sections;
#[ApiProperty(description: 'Articles', writableLink: false)]
#[Groups(['read:' . self::class])]
#[ORM\OneToMany(targetEntity: JournalArticle::class, mappedBy: 'issue', cascade: ['persist'])]
#[ORM\OrderBy(['pageFrom' => 'asc'])]
private Collection $articles;
private ?array $sortedArticles = null;
public function __construct(Journal $parent)
{
$this->setUuid();
$this->parent = $parent;
$this->translations = new ArrayCollection();
$this->files = new ArrayCollection();
$this->sponsors = new ArrayCollection();
$this->blocks = new ArrayCollection();
$this->sections = new ArrayCollection();
$this->articles = new ArrayCollection();
$this->editorialGroups = new ArrayCollection();
foreach ($this->parent->getLanguages() as $lang) {
new JournalIssueTranslation($this, $lang);
}
foreach($this->parent->getEditorialGroups() as $editorialGroup) {
$copied = JournalIssueEditorialGroup::constructFrom(parent: $this, template: $editorialGroup);
$this->addEditorialGroup($copied);
}
$this->isOpen = $parent->getData()->getIsOpen();
$this->price = $parent->getData()->getIssuePrice();
$this->isOpenAccessGrantedAfterNewestIssueExpired = $parent->getData()->getIsOpenAccessGrantedAfterNewestIssueExpired();
$this->visibleEditorialGroupMemberProperties = $parent->getView()->getVisibleEditorialGroupMemberProperties();
$this->suggestedCitationType = $parent->getData()->getSuggestedCitationType();
$this->licence = $parent->getData()->getLicence();
$this->createdAt = new \DateTimeImmutable();
$this->updatedAt = new \DateTimeImmutable();
$parent->addIssue($this);
}
public function getParent(): Journal
{
return $this->parent;
}
public function getVolume(): JournalVolume
{
return $this->volume;
}
public function setVolume(JournalVolume $volume): self
{
$this->volume = $volume;
return $this;
}
public function getPlannedArticlesAmount(): int
{
return $this->plannedArticlesAmount;
}
public function setPlannedArticlesAmount(int $plannedArticlesAmount): self
{
$this->plannedArticlesAmount = $plannedArticlesAmount;
return $this;
}
#[Groups(['read:' . self::class])]
public function getArticlesRatio(): string
{
$articles = count($this->articles);
return "{$articles}/{$this->plannedArticlesAmount}";
}
public function getIsNew(): bool
{
return $this->isNew;
}
public function setIsNew(bool $isNew): self
{
$this->isNew = $isNew;
return $this;
}
public function getIsNewestIssue(): bool
{
return $this->isNewestIssue;
}
public function setIsNewestIssue(bool $isNewestIssue): self
{
$wasNewestIssue = $this->isNewestIssue;
$this->isNewestIssue = $isNewestIssue;
if (
$this->isOpenAccessGrantedAfterNewestIssueExpired
&& $wasNewestIssue
&& ! $isNewestIssue
) {
$this->isOpen = true;
}
return $this;
}
public function getIsFirstView(): bool
{
return $this->isFirstView;
}
public function setIsFirstView(bool $isFirstView): self
{
$this->isFirstView = $isFirstView;
return $this;
}
public function getState(): JournalIssueState
{
return $this->state;
}
public function setState(JournalIssueState $state): self
{
$this->state = $state;
return $this;
}
public function getIsContinousAccession(): bool
{
return $this->isContinousAccession;
}
public function setIsContinousAccession(bool $isContinousAccession): self
{
$this->isContinousAccession = $isContinousAccession;
return $this;
}
public function getNumberInYear(): int
{
return $this->numberInYear;
}
public function setNumberInYear(int $numberInYear): self
{
$this->numberInYear = $numberInYear;
return $this;
}
public function getIsSpecialIssue(): bool
{
return $this->isSpecialIssue;
}
public function setIsSpecialIssue(bool $isSpecialIssue): self
{
$this->isSpecialIssue = $isSpecialIssue;
return $this;
}
public function getIsCrossmarkButtonVisible(): bool
{
return $this->isCrossmarkButtonVisible;
}
public function setIsCrossmarkButtonVisible(bool $isCrossmarkButtonVisible): self
{
$this->isCrossmarkButtonVisible = $isCrossmarkButtonVisible;
return $this;
}
public function getIsOpen(): bool
{
return $this->isOpen;
}
public function setIsOpen(bool $isOpen): self
{
$this->isOpen = $isOpen;
return $this;
}
public function getPrice(): ?int
{
return $this->price;
}
#[Groups(['read:' . self::class, 'read:' . Journal::class])]
public function getPriceReal(): ?float
{
return $this->price ? round($this->price / 1000, 2) : null;
}
public function setPrice(?int $price): self
{
$this->price = $price;
return $this;
}
#[Groups(['write'])]
public function setPriceReal(?float $priceReal): self
{
$this->price = (int) ($priceReal * 1000);
return $this;
}
public function getGraceDate(): ?\DateTimeImmutable
{
return $this->graceDate;
}
public function setGraceDate(?\DateTimeImmutable $graceDate): self
{
$this->graceDate = $graceDate;
return $this;
}
public function getIsOpenAccessGrantedAfterNewestIssueExpired(): bool
{
return $this->isOpenAccessGrantedAfterNewestIssueExpired;
}
public function setIsOpenAccessGrantedAfterNewestIssueExpired(bool $isOpenAccessGrantedAfterNewestIssueExpired): self
{
$this->isOpenAccessGrantedAfterNewestIssueExpired = $isOpenAccessGrantedAfterNewestIssueExpired;
return $this;
}
public function getPublishedAt(): ?\DateTimeImmutable
{
return $this->publishedAt;
}
public function setPublishedAt(?\DateTimeImmutable $publishedAt): self
{
if ($this->publishedAt === $publishedAt) {
return $this;
}
$this->publishedAt = $publishedAt;
if (! $publishedAt) {
return $this;
}
$period = $this->parent->getData()->getOpenAccessGrantedAfterPeriod();
if (! $period) {
return $this;
}
$graceDate = new \DateTime("@{$this->publishedAt->getTimestamp()}");
$graceDate->modify($period);
$this->graceDate = new \DateTimeImmutable("@{$graceDate->getTimestamp()}");
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 getLicence(): JournalLicence
{
return $this->licence;
}
public function setLicence(JournalLicence $licence): self
{
if ($this->licence !== $licence) {
foreach($this->articles as $article) {
$article->setLicence($licence);
}
}
$this->licence = $licence;
return $this;
}
public function getSuggestedCitationType(): ?CitationType
{
return $this->suggestedCitationType;
}
public function setSuggestedCitationType(?CitationType $suggestedCitationType): self
{
$this->suggestedCitationType = $suggestedCitationType;
return $this;
}
public function getFundingNumber(): ?string
{
return $this->fundingNumber;
}
public function setFundingNumber(?string $fundingNumber): self
{
$this->fundingNumber = $fundingNumber;
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|JournalIssueFile[]
*/
public function getFiles(): Collection
{
return $this->files;
}
public function addFile(JournalIssueFile $file): self
{
if ($this->files->contains($file)) {
return $this;
}
$this->files[] = $file;
return $this;
}
public function removeFile(JournalIssueFile $file): self
{
$this->files->removeElement($file);
return $this;
}
public function resetFiles(): self
{
$this->files = new ArrayCollection();
return $this;
}
public function getVisibleFiles(string $lang): Collection
{
return $this->files->filter(
fn (JournalIssueFile $file) =>
$file->getStat() === true &&
$file->readAvailableTranslation($lang, 'media')
);
}
/**
* @return Collection|JournalIssueSponsor[]
*/
public function getSponsors(): Collection
{
return $this->sponsors;
}
public function addSponsor(JournalIssueSponsor $sponsor): self
{
if ($this->sponsors->contains($sponsor)) {
return $this;
}
$this->sponsors[] = $sponsor;
return $this;
}
public function removeSponsor(JournalIssueSponsor $sponsor): self
{
$this->sponsors->removeElement($sponsor);
return $this;
}
public function resetSponsors(): self
{
$this->sponsors = new ArrayCollection();
return $this;
}
/**
* @return Collection|JournalIssueBlock[]
*/
public function getBlocks(): Collection
{
return $this->blocks;
}
public function addBlock(JournalIssueBlock $block): self
{
if ($this->blocks->contains($block)) {
return $this;
}
$this->blocks[] = $block;
return $this;
}
public function removeBlock(JournalIssueBlock $block): self
{
$this->blocks->removeElement($block);
return $this;
}
public function resetBlocks(): self
{
$this->blocks = new ArrayCollection();
return $this;
}
/**
* @return Collection|JournalIssueSection[]
*/
public function getSections(): Collection
{
return $this->sections;
}
public function addSection(JournalIssueSection $section): self
{
if ($this->sections->contains($section)) {
return $this;
}
$this->sections[] = $section;
return $this;
}
public function removeSection(JournalIssueSection $section): self
{
$this->sections->removeElement($section);
return $this;
}
public function resetSections(): self
{
$this->sections = new ArrayCollection();
return $this;
}
/**
* @return array|JournalArticle[]
*/
public function getArticles(): array
{
if (null === $this->sortedArticles) {
$this->sortedArticles = $this->articles->toArray();
if ($this->getParent()->isSortArticlesByOrd()) {
usort(
$this->sortedArticles,
function(JournalArticle $a, JournalArticle $b) {
return $b->getOrd() > $a->getOrd();
}
);
} else {
usort(
$this->sortedArticles,
function(JournalArticle $a, JournalArticle $b) {
$aPageFrom = RomanNumber::isRoman($a->getPageFrom())
? RomanNumber::convertToInt($a->getPageFrom())
: (int) $a->getPageFrom();
$bPageFrom = RomanNumber::isRoman($b->getPageFrom())
? RomanNumber::convertToInt($b->getPageFrom())
: (int) $b->getPageFrom();
return $aPageFrom > $bPageFrom;
}
);
}
}
return $this->sortedArticles;
}
public function addArticle(JournalArticle $article): self
{
if ($this->articles->contains($article)) {
return $this;
}
$this->articles[] = $article;
return $this;
}
public function removeArticle(JournalArticle $article): self
{
$this->articles->removeElement($article);
return $this;
}
public function resetArticles(): self
{
$this->articles = new ArrayCollection();
return $this;
}
public function getArticlesVisible(bool $sectionable = false): array
{
$articles = [];
/** @var JournalArticle $article */
foreach ($this->getArticles() as $article) {
if (! $article->getStat()) {
continue;
}
if (
($sectionable && ! $article->getSection()) ||
(! $sectionable && $article->getSection())
) {
continue;
}
$articles[] = $article;
}
return $articles;
}
public function getArticlesVisibleIgnoreSections(): array
{
$articles = [];
foreach ($this->getArticles() as $article) {
if (! $article->getStat()) {
continue;
}
$articles[] = $article;
}
return $articles;
}
public function getSectionGroupedArticles(): array
{
$articles = [];
/** @var JournalArticle $article */
foreach ($this->getArticles() as $article) {
if (! $article->getStat() || ! $article->getSection()?->getStat()) {
continue;
}
$sectionOrd = $this->getSectionOrd($article->getSection());
if (! isset($result[$sectionOrd])) {
$result[$sectionOrd] = [];
}
$articles[$sectionOrd][] = $article;
}
return $articles;
}
private function getSectionOrd(JournalSection $search): ?int
{
foreach($this->sections as $section) {
if ($search === $section->getSection()) {
return $section->getOrd();
}
}
return null;
}
public function getArticlesKeywords(string|Language $lang): array
{
$results = [];
/** @var JournalArticle $article */
foreach ($this->getArticles() as $article) {
if (! $article->getStat()) {
continue;
}
$keywords = $article->readAvailableMetadataTranslation($lang, 'keywords');
if (! $keywords) {
continue;
}
$results = array_merge($results, $keywords->toArray());
}
return $results;
}
public function getVisibleEditorialGroups(): Collection
{
return $this->editorialGroups;
}
#[Groups(['read:' . self::class, 'read:' . JournalVolume::class])]
public function getParentNativeLanguage(): Language
{
return $this->getLanguageableParent()->getNativeLanguage();
}
#[Groups(['read:' . self::class, 'read:' . JournalVolume::class])]
public function getParentLanguages(): array
{
return $this->getLanguageableParent()->getLanguages();
}
public function getNativeTitle(): string
{
return $this->getTranslation($this->parent->getNativeLanguage())?->getTitle();
}
public function getOaiId(): string
{
return Urlizer::urlize($this->getNativeTitle());
}
public function clone(Journal $parent = null): self
{
$clone = clone $this;
$clone->id = null;
$clone->setUuid();
$clone->importId = null;
ClassUtils::cloneCollection($this, $clone, 'translations');
$clone->synchronizeTranslations();
if (! $parent) {
$nativeLanguage = $clone->parent->getNativeLanguage();
$nativeTranslation = $clone->getTranslation($nativeLanguage);
$nativeTranslation->setTitle($nativeTranslation->getTitle() . ' [KOPIA]');
$clone->ord = 0;
$clone->stat = false;
} else {
$clone->parent = $parent;
$clone->parent->addIssue($clone);
}
ClassUtils::cloneCollection($this, $clone, 'sponsors');
ClassUtils::cloneCollection($this, $clone, 'blocks');
ClassUtils::cloneCollection($this, $clone, 'editorialGroups');
ClassUtils::cloneCollection($this, $clone, 'sections');
$this
->resetFiles()
->resetArticles();
$clone->createdAt = new \DateTimeImmutable();
$clone->updatedAt = new \DateTimeImmutable();
return $clone;
}
}