263 lines
8.4 KiB
PHP
263 lines
8.4 KiB
PHP
<?php
|
|
/**
|
|
* @file classes/author/Repository.php
|
|
*
|
|
* Copyright (c) 2014-2020 Simon Fraser University
|
|
* Copyright (c) 2000-2020 John Willinsky
|
|
* Distributed under the GNU GPL v3. For full terms see the file docs/COPYING.
|
|
*
|
|
* @class Repository
|
|
*
|
|
* @brief A repository to find and manage authors.
|
|
*/
|
|
|
|
namespace PKP\author;
|
|
|
|
use APP\author\Author;
|
|
use APP\author\DAO;
|
|
use APP\core\Request;
|
|
use APP\core\Services;
|
|
use APP\facades\Repo;
|
|
use APP\submission\Submission;
|
|
use PKP\context\Context;
|
|
use PKP\plugins\Hook;
|
|
use PKP\services\PKPSchemaService;
|
|
use PKP\submission\PKPSubmission;
|
|
use PKP\user\User;
|
|
use PKP\validation\ValidatorFactory;
|
|
|
|
class Repository
|
|
{
|
|
/** @var DAO */
|
|
public $dao;
|
|
|
|
/** @var string $schemaMap The name of the class to map this entity to its schema */
|
|
public $schemaMap = maps\Schema::class;
|
|
|
|
/** @var Request */
|
|
protected $request;
|
|
|
|
/** @var PKPSchemaService<Author> */
|
|
protected $schemaService;
|
|
|
|
public function __construct(DAO $dao, Request $request, PKPSchemaService $schemaService)
|
|
{
|
|
$this->dao = $dao;
|
|
$this->request = $request;
|
|
$this->schemaService = $schemaService;
|
|
}
|
|
|
|
/** @copydoc DAO::newDataObject() */
|
|
public function newDataObject(array $params = []): Author
|
|
{
|
|
$object = $this->dao->newDataObject();
|
|
if (!empty($params)) {
|
|
$object->setAllData($params);
|
|
}
|
|
return $object;
|
|
}
|
|
|
|
/** @copydoc DAO::get() */
|
|
public function get(int $id, int $publicationId = null): ?Author
|
|
{
|
|
return $this->dao->get($id, $publicationId);
|
|
}
|
|
|
|
/** @copydoc DAO::exists() */
|
|
public function exists(int $id, int $publicationId = null): bool
|
|
{
|
|
return $this->dao->exists($id, $publicationId);
|
|
}
|
|
|
|
/** @copydoc DAO::getCollector() */
|
|
public function getCollector(): Collector
|
|
{
|
|
return app(Collector::class);
|
|
}
|
|
|
|
/**
|
|
* Get an instance of the map class for mapping
|
|
* authors to their schema
|
|
*/
|
|
public function getSchemaMap(): maps\Schema
|
|
{
|
|
return app('maps')->withExtensions($this->schemaMap);
|
|
}
|
|
|
|
/**
|
|
* Validate properties for an author
|
|
*
|
|
* Perform validation checks on data used to add or edit an author.
|
|
*
|
|
* @param Author|null $author The author being edited. Pass `null` if creating a new author
|
|
* @param array $props A key/value array with the new data to validate
|
|
*
|
|
* @return array A key/value array with validation errors. Empty if no errors
|
|
*/
|
|
public function validate($author, $props, Submission $submission, Context $context)
|
|
{
|
|
$schemaService = Services::get('schema');
|
|
$allowedLocales = $context->getSupportedSubmissionLocales();
|
|
$primaryLocale = $submission->getData('locale');
|
|
|
|
$validator = ValidatorFactory::make(
|
|
$props,
|
|
$schemaService->getValidationRules(PKPSchemaService::SCHEMA_AUTHOR, $allowedLocales),
|
|
[
|
|
'country.regex' => __('validator.country.regex'),
|
|
]
|
|
);
|
|
|
|
// Check required fields
|
|
ValidatorFactory::required(
|
|
$validator,
|
|
$author,
|
|
$schemaService->getRequiredProps(PKPSchemaService::SCHEMA_AUTHOR),
|
|
$schemaService->getMultilingualProps(PKPSchemaService::SCHEMA_AUTHOR),
|
|
$allowedLocales,
|
|
$primaryLocale
|
|
);
|
|
|
|
// Check for input from disallowed locales
|
|
ValidatorFactory::allowedLocales($validator, $schemaService->getMultilingualProps(PKPSchemaService::SCHEMA_AUTHOR), $allowedLocales);
|
|
|
|
// The publicationId must match an existing publication that is not yet published
|
|
$validator->after(function ($validator) use ($props) {
|
|
if (isset($props['publicationId']) && !$validator->errors()->get('publicationId')) {
|
|
$publication = Repo::publication()->get($props['publicationId']);
|
|
if (!$publication) {
|
|
$validator->errors()->add('publicationId', __('author.publicationNotFound'));
|
|
} elseif ($publication->getData('status') === PKPSubmission::STATUS_PUBLISHED) {
|
|
$validator->errors()->add('publicationId', __('author.editPublishedDisabled'));
|
|
}
|
|
}
|
|
});
|
|
|
|
$errors = [];
|
|
if ($validator->fails()) {
|
|
$errors = $schemaService->formatValidationErrors($validator->errors());
|
|
}
|
|
|
|
Hook::call('Author::validate', [$errors, $author, $props, $allowedLocales, $primaryLocale]);
|
|
|
|
return $errors;
|
|
}
|
|
|
|
/**
|
|
* @copydoc \PKP\services\entityProperties\EntityWriteInterface::add()
|
|
*/
|
|
public function add(Author $author): int
|
|
{
|
|
$existingSeq = $author->getData('seq');
|
|
|
|
if (!isset($existingSeq)) {
|
|
$nextSeq = $this->dao->getNextSeq($author->getData('publicationId'));
|
|
$author->setData('seq', $nextSeq);
|
|
}
|
|
|
|
$authorId = $this->dao->insert($author);
|
|
$author = Repo::author()->get($authorId);
|
|
|
|
Hook::call('Author::add', [$author]);
|
|
|
|
return $author->getId();
|
|
}
|
|
|
|
/**
|
|
* @copydoc \PKP\services\entityProperties\EntityWriteInterface::edit()
|
|
*/
|
|
public function edit(Author $author, array $params)
|
|
{
|
|
$newAuthor = Repo::author()->newDataObject(array_merge($author->_data, $params));
|
|
|
|
Hook::call('Author::edit', [$newAuthor, $author, $params]);
|
|
|
|
$this->dao->update($newAuthor);
|
|
|
|
Repo::author()->get($newAuthor->getId());
|
|
}
|
|
|
|
/**
|
|
* @copydoc \PKP\services\entityProperties\EntityWriteInterface::delete()
|
|
*/
|
|
public function delete(Author $author)
|
|
{
|
|
Hook::call('Author::delete::before', [$author]);
|
|
$this->dao->delete($author);
|
|
|
|
$this->dao->resetContributorsOrder($author->getData('publicationId'));
|
|
|
|
Hook::call('Author::delete', [$author]);
|
|
}
|
|
|
|
/**
|
|
* Create an Author object from a User object
|
|
*
|
|
* This does not save the author in the database.
|
|
*/
|
|
public function newAuthorFromUser(User $user): Author
|
|
{
|
|
$author = Repo::author()->newDataObject();
|
|
$author->setGivenName($user->getGivenName(null), null);
|
|
$author->setFamilyName($user->getFamilyName(null), null);
|
|
$author->setAffiliation($user->getAffiliation(null), null);
|
|
$author->setCountry($user->getCountry());
|
|
$author->setEmail($user->getEmail());
|
|
$author->setUrl($user->getUrl());
|
|
$author->setBiography($user->getBiography(null), null);
|
|
$author->setIncludeInBrowse(1);
|
|
$author->setOrcid($user->getOrcid());
|
|
|
|
Hook::call('Author::newAuthorFromUser', [$author, $user]);
|
|
|
|
return $author;
|
|
}
|
|
|
|
/**
|
|
* Update author names when publication locale changes.
|
|
*
|
|
* @param int $publicationId
|
|
* @param string $oldLocale
|
|
* @param string $newLocale
|
|
*/
|
|
public function changePublicationLocale($publicationId, $oldLocale, $newLocale)
|
|
{
|
|
$authors = $this->getCollector()
|
|
->filterByPublicationIds([$publicationId])
|
|
->getMany();
|
|
|
|
foreach ($authors as $author) {
|
|
if (empty($author->getGivenName($newLocale))) {
|
|
if (empty($author->getFamilyName($newLocale)) && empty($author->getPreferredPublicName($newLocale))) {
|
|
// if no name exists for the new locale
|
|
// copy all names with the old locale to the new locale
|
|
$author->setGivenName($author->getGivenName($oldLocale), $newLocale);
|
|
$author->setFamilyName($author->getFamilyName($oldLocale), $newLocale);
|
|
$author->setPreferredPublicName($author->getPreferredPublicName($oldLocale), $newLocale);
|
|
} else {
|
|
// if the given name does not exist, but one of the other names do exist
|
|
// copy only the given name with the old locale to the new locale, because the given name is required
|
|
$author->setGivenName($author->getGivenName($oldLocale), $newLocale);
|
|
}
|
|
|
|
$this->dao->update($author);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reorders the authors of a publication according to the given order of the authors in the provided author array
|
|
*/
|
|
public function setAuthorsOrder(int $publicationId, array $authors)
|
|
{
|
|
$seq = 0;
|
|
foreach ($authors as $author) {
|
|
$author->setData('seq', $seq);
|
|
|
|
$this->dao->update($author);
|
|
|
|
$seq++;
|
|
}
|
|
}
|
|
}
|