first commit
This commit is contained in:
@@ -0,0 +1,430 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @file pages/issue/IssueHandler.php
|
||||
*
|
||||
* Copyright (c) 2014-2021 Simon Fraser University
|
||||
* Copyright (c) 2003-2021 John Willinsky
|
||||
* Distributed under the GNU GPL v3. For full terms see the file docs/COPYING.
|
||||
*
|
||||
* @class IssueHandler
|
||||
*
|
||||
* @ingroup pages_issue
|
||||
*
|
||||
* @brief Handle requests for issue functions.
|
||||
*/
|
||||
|
||||
namespace APP\pages\issue;
|
||||
|
||||
use APP\core\Application;
|
||||
use APP\facades\Repo;
|
||||
use APP\file\IssueFileManager;
|
||||
use APP\handler\Handler;
|
||||
use APP\issue\Collector;
|
||||
use APP\issue\IssueAction;
|
||||
use APP\issue\IssueGalleyDAO;
|
||||
use APP\observers\events\UsageEvent;
|
||||
use APP\payment\ojs\OJSCompletedPaymentDAO;
|
||||
use APP\payment\ojs\OJSPaymentManager;
|
||||
use APP\security\authorization\OjsIssueRequiredPolicy;
|
||||
use APP\security\authorization\OjsJournalMustPublishPolicy;
|
||||
use APP\template\TemplateManager;
|
||||
use PKP\config\Config;
|
||||
use PKP\db\DAORegistry;
|
||||
use PKP\facades\Locale;
|
||||
use PKP\plugins\Hook;
|
||||
use PKP\plugins\PluginRegistry;
|
||||
use PKP\security\authorization\ContextRequiredPolicy;
|
||||
use PKP\security\Validation;
|
||||
use PKP\submission\GenreDAO;
|
||||
use PKP\submission\PKPSubmission;
|
||||
|
||||
class IssueHandler extends Handler
|
||||
{
|
||||
/** @var \APP\issue\IssueGalley retrieved issue galley */
|
||||
public $_galley = null;
|
||||
|
||||
|
||||
/**
|
||||
* @copydoc PKPHandler::authorize()
|
||||
*/
|
||||
public function authorize($request, &$args, $roleAssignments)
|
||||
{
|
||||
$this->addPolicy(new ContextRequiredPolicy($request));
|
||||
$this->addPolicy(new OjsJournalMustPublishPolicy($request));
|
||||
|
||||
// the 'archives' op does not need this policy so it is left out of the operations array.
|
||||
$this->addPolicy(new OjsIssueRequiredPolicy($request, $args, ['view', 'download']));
|
||||
|
||||
return parent::authorize($request, $args, $roleAssignments);
|
||||
}
|
||||
|
||||
/**
|
||||
* @see PKPHandler::initialize()
|
||||
*
|
||||
* @param array $args Arguments list
|
||||
*/
|
||||
public function initialize($request, $args = [])
|
||||
{
|
||||
// Get the issue galley
|
||||
$galleyId = $args[1] ?? 0;
|
||||
if ($galleyId) {
|
||||
$issue = $this->getAuthorizedContextObject(Application::ASSOC_TYPE_ISSUE);
|
||||
$galleyDao = DAORegistry::getDAO('IssueGalleyDAO'); /** @var IssueGalleyDAO $galleyDao */
|
||||
$journal = $request->getJournal();
|
||||
$galley = $galleyDao->getByBestId($galleyId, $issue->getId());
|
||||
|
||||
// Invalid galley id, redirect to issue page
|
||||
if (!$galley) {
|
||||
$request->redirect(null, null, 'view', $issue->getId());
|
||||
}
|
||||
|
||||
$this->setGalley($galley);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Display about index page.
|
||||
*/
|
||||
public function index($args, $request)
|
||||
{
|
||||
$this->current($args, $request);
|
||||
}
|
||||
|
||||
/**
|
||||
* Display current issue page.
|
||||
*/
|
||||
public function current($args, $request)
|
||||
{
|
||||
$journal = $request->getJournal();
|
||||
$issue = Repo::issue()->getCurrent($journal->getId(), true);
|
||||
|
||||
if ($issue != null) {
|
||||
$request->redirect(null, 'issue', 'view', $issue->getBestIssueId());
|
||||
}
|
||||
|
||||
$this->setupTemplate($request);
|
||||
$templateMgr = TemplateManager::getManager($request);
|
||||
// consider public identifiers
|
||||
$pubIdPlugins = PluginRegistry::loadCategory('pubIds', true);
|
||||
$templateMgr->assign('pubIdPlugins', $pubIdPlugins);
|
||||
$templateMgr->display('frontend/pages/issue.tpl');
|
||||
}
|
||||
|
||||
/**
|
||||
* View an issue.
|
||||
*
|
||||
* @param array $args
|
||||
* @param \APP\core\Request $request
|
||||
*/
|
||||
public function view($args, $request)
|
||||
{
|
||||
$issue = $this->getAuthorizedContextObject(Application::ASSOC_TYPE_ISSUE);
|
||||
$this->setupTemplate($request);
|
||||
$templateMgr = TemplateManager::getManager($request);
|
||||
$journal = $request->getJournal();
|
||||
|
||||
if (($galley = $this->getGalley()) && $this->userCanViewGalley($request)) {
|
||||
if (!Hook::call('IssueHandler::view::galley', [&$request, &$issue, &$galley])) {
|
||||
$request->redirect(null, null, 'download', [$issue->getBestIssueId($journal), $galley->getBestGalleyId()]);
|
||||
}
|
||||
} else {
|
||||
self::_setupIssueTemplate($request, $issue, $request->getUserVar('showToc') ? true : false);
|
||||
$templateMgr->assign('issueId', $issue->getBestIssueId());
|
||||
|
||||
// consider public identifiers
|
||||
$pubIdPlugins = PluginRegistry::loadCategory('pubIds', true);
|
||||
$templateMgr->assign('pubIdPlugins', $pubIdPlugins);
|
||||
$templateMgr->display('frontend/pages/issue.tpl');
|
||||
event(new UsageEvent(Application::ASSOC_TYPE_ISSUE, $journal, null, null, null, $issue));
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Display the issue archive listings
|
||||
*
|
||||
* @param array $args
|
||||
* @param \APP\core\Request $request
|
||||
*/
|
||||
public function archive($args, $request)
|
||||
{
|
||||
$this->setupTemplate($request);
|
||||
$page = isset($args[0]) ? (int) $args[0] : 1;
|
||||
$templateMgr = TemplateManager::getManager($request);
|
||||
$context = $request->getContext();
|
||||
|
||||
$count = $context->getData('itemsPerPage') ? $context->getData('itemsPerPage') : Config::getVar('interface', 'items_per_page');
|
||||
$offset = $page > 1 ? ($page - 1) * $count : 0;
|
||||
|
||||
$collector = Repo::issue()->getCollector()
|
||||
->limit($count)
|
||||
->offset($offset)
|
||||
->filterByContextIds([$context->getId()])
|
||||
->orderBy(Collector::ORDERBY_SEQUENCE)
|
||||
->filterByPublished(true);
|
||||
|
||||
$issues = $collector->getMany()->toArray();
|
||||
$total = $collector->limit(null)->offset(null)->getCount();
|
||||
|
||||
$showingStart = $offset + 1;
|
||||
$showingEnd = min($offset + $count, $offset + count($issues));
|
||||
$nextPage = $total > $showingEnd ? $page + 1 : null;
|
||||
$prevPage = $showingStart > 1 ? $page - 1 : null;
|
||||
|
||||
$templateMgr->assign([
|
||||
'issues' => $issues,
|
||||
'showingStart' => $showingStart,
|
||||
'showingEnd' => $showingEnd,
|
||||
'total' => $total,
|
||||
'nextPage' => $nextPage,
|
||||
'prevPage' => $prevPage,
|
||||
]);
|
||||
|
||||
$templateMgr->display('frontend/pages/issueArchive.tpl');
|
||||
}
|
||||
|
||||
/**
|
||||
* Downloads an issue galley file
|
||||
*
|
||||
* @param array $args ($issueId, $galleyId)
|
||||
* @param \APP\core\Request $request
|
||||
*/
|
||||
public function download($args, $request)
|
||||
{
|
||||
if ($this->userCanViewGalley($request)) {
|
||||
$issue = $this->getAuthorizedContextObject(Application::ASSOC_TYPE_ISSUE);
|
||||
$galley = $this->getGalley();
|
||||
|
||||
if (!Hook::call('IssueHandler::download', [&$issue, &$galley])) {
|
||||
$issueFileManager = new IssueFileManager($issue->getId());
|
||||
if ($issueFileManager->downloadById($galley->getFileId(), $request->getUserVar('inline') ? true : false)) {
|
||||
event(new UsageEvent(Application::ASSOC_TYPE_ISSUE_GALLEY, $request->getContext(), null, null, null, $issue, $galley));
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the retrieved issue galley
|
||||
*
|
||||
* @return \APP\issue\IssueGalley
|
||||
*/
|
||||
public function getGalley()
|
||||
{
|
||||
return $this->_galley;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set a retrieved issue galley
|
||||
*
|
||||
* @param \APP\issue\IssueGalley $galley
|
||||
*/
|
||||
public function setGalley($galley)
|
||||
{
|
||||
$this->_galley = $galley;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines whether or not a user can view an issue galley.
|
||||
*
|
||||
* @param \APP\core\Request $request
|
||||
*/
|
||||
public function userCanViewGalley($request)
|
||||
{
|
||||
$issueAction = new IssueAction();
|
||||
|
||||
$journal = $request->getJournal();
|
||||
$user = $request->getUser();
|
||||
$userId = $user ? $user->getId() : 0;
|
||||
$issue = $this->getAuthorizedContextObject(Application::ASSOC_TYPE_ISSUE);
|
||||
$galley = $this->getGalley();
|
||||
|
||||
// If this is an editorial user who can view unpublished issue galleys,
|
||||
// bypass further validation
|
||||
if ($issueAction->allowedIssuePrePublicationAccess($journal, $user)) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// Ensure reader has rights to view the issue galley
|
||||
if ($issue->getPublished()) {
|
||||
$subscriptionRequired = $issueAction->subscriptionRequired($issue, $journal);
|
||||
$isSubscribedDomain = $issueAction->subscribedDomain($request, $journal, $issue->getId());
|
||||
|
||||
// Check if login is required for viewing.
|
||||
if (!$isSubscribedDomain && !Validation::isLoggedIn() && $journal->getData('restrictArticleAccess')) {
|
||||
Validation::redirectLogin();
|
||||
}
|
||||
|
||||
// If no domain/ip subscription, check if user has a valid subscription
|
||||
// or if the user has previously purchased the issue
|
||||
if (!$isSubscribedDomain && $subscriptionRequired) {
|
||||
// Check if user has a valid subscription
|
||||
$subscribedUser = $issueAction->subscribedUser($user, $journal, $issue->getId());
|
||||
if (!$subscribedUser) {
|
||||
// Check if payments are enabled,
|
||||
$paymentManager = Application::getPaymentManager($journal);
|
||||
|
||||
if ($paymentManager->purchaseIssueEnabled() || $paymentManager->membershipEnabled()) {
|
||||
// If only pdf files are being restricted, then approve all non-pdf galleys
|
||||
// and continue checking if it is a pdf galley
|
||||
if ($paymentManager->onlyPdfEnabled() && !$galley->isPdfGalley()) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (!Validation::isLoggedIn()) {
|
||||
Validation::redirectLogin('payment.loginRequired.forIssue');
|
||||
}
|
||||
|
||||
// If the issue galley has been purchased, then allow reader access
|
||||
$completedPaymentDao = DAORegistry::getDAO('OJSCompletedPaymentDAO'); /** @var OJSCompletedPaymentDAO $completedPaymentDao */
|
||||
$dateEndMembership = $user->getSetting('dateEndMembership', 0);
|
||||
if ($completedPaymentDao->hasPaidPurchaseIssue($userId, $issue->getId()) || (!is_null($dateEndMembership) && $dateEndMembership > time())) {
|
||||
return true;
|
||||
} else {
|
||||
// Otherwise queue an issue purchase payment and display payment form
|
||||
$queuedPayment = $paymentManager->createQueuedPayment($request, OJSPaymentManager::PAYMENT_TYPE_PURCHASE_ISSUE, $userId, $issue->getId(), $journal->getData('purchaseIssueFee'));
|
||||
$paymentManager->queuePayment($queuedPayment);
|
||||
|
||||
$paymentForm = $paymentManager->getPaymentForm($queuedPayment);
|
||||
$paymentForm->display($request);
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
if (!Validation::isLoggedIn()) {
|
||||
Validation::redirectLogin('reader.subscriptionRequiredLoginText');
|
||||
}
|
||||
$request->redirect(null, 'about', 'subscriptions');
|
||||
}
|
||||
}
|
||||
} else {
|
||||
$request->redirect(null, 'index');
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Given an issue, set up the template with all the required variables for
|
||||
* frontend/objects/issue_toc.tpl to function properly (i.e. current issue
|
||||
* and view issue).
|
||||
*
|
||||
* @param object $issue The issue to display
|
||||
* @param bool $showToc iff false and a custom cover page exists,
|
||||
* the cover page will be displayed. Otherwise table of contents
|
||||
* will be displayed.
|
||||
*/
|
||||
public static function _setupIssueTemplate($request, $issue, $showToc = false)
|
||||
{
|
||||
$journal = $request->getJournal();
|
||||
$user = $request->getUser();
|
||||
$templateMgr = TemplateManager::getManager($request);
|
||||
|
||||
// Determine pre-publication access
|
||||
// FIXME: Do that. (Bug #8278)
|
||||
|
||||
$templateMgr->assign([
|
||||
'issueIdentification' => $issue->getIssueIdentification(),
|
||||
'issueTitle' => $issue->getLocalizedTitle(),
|
||||
'issueSeries' => $issue->getIssueIdentification(['showTitle' => false]),
|
||||
]);
|
||||
|
||||
$locale = Locale::getLocale();
|
||||
|
||||
$templateMgr->assign([
|
||||
'locale' => $locale,
|
||||
]);
|
||||
|
||||
$issueGalleyDao = DAORegistry::getDAO('IssueGalleyDAO'); /** @var IssueGalleyDAO $issueGalleyDao */
|
||||
|
||||
$genreDao = DAORegistry::getDAO('GenreDAO'); /** @var GenreDAO $genreDao */
|
||||
$primaryGenres = $genreDao->getPrimaryByContextId($journal->getId())->toArray();
|
||||
$primaryGenreIds = array_map(function ($genre) {
|
||||
return $genre->getId();
|
||||
}, $primaryGenres);
|
||||
|
||||
// Show scheduled submissions if this is a preview
|
||||
$allowedStatuses = [PKPSubmission::STATUS_PUBLISHED];
|
||||
if (!$issue->getPublished()) {
|
||||
$allowedStatuses[] = PKPSubmission::STATUS_SCHEDULED;
|
||||
}
|
||||
|
||||
$issueSubmissions = Repo::submission()->getCollector()
|
||||
->filterByContextIds([$issue->getJournalId()])
|
||||
->filterByIssueIds([$issue->getId()])
|
||||
->filterByStatus($allowedStatuses)
|
||||
->orderBy(\APP\submission\Collector::ORDERBY_SEQUENCE, \APP\submission\Collector::ORDER_DIR_ASC)
|
||||
->getMany();
|
||||
|
||||
$sections = Repo::section()->getByIssueId($issue->getId());
|
||||
$issueSubmissionsInSection = [];
|
||||
foreach ($sections as $section) {
|
||||
$issueSubmissionsInSection[$section->getId()] = [
|
||||
'title' => $section->getHideTitle() ? null : $section->getLocalizedTitle(),
|
||||
'hideAuthor' => $section->getHideAuthor(),
|
||||
'articles' => [],
|
||||
];
|
||||
}
|
||||
foreach ($issueSubmissions as $submission) {
|
||||
if (!$sectionId = $submission->getCurrentPublication()->getData('sectionId')) {
|
||||
continue;
|
||||
}
|
||||
$issueSubmissionsInSection[$sectionId]['articles'][] = $submission;
|
||||
}
|
||||
|
||||
$authorUserGroups = Repo::userGroup()->getCollector()->filterByRoleIds([\PKP\security\Role::ROLE_ID_AUTHOR])->filterByContextIds([$journal->getId()])->getMany()->remember();
|
||||
$templateMgr->assign([
|
||||
'issue' => $issue,
|
||||
'issueGalleys' => $issueGalleyDao->getByIssueId($issue->getId()),
|
||||
'publishedSubmissions' => $issueSubmissionsInSection,
|
||||
'primaryGenreIds' => $primaryGenreIds,
|
||||
'authorUserGroups' => $authorUserGroups,
|
||||
]);
|
||||
|
||||
// Subscription Access
|
||||
$issueAction = new IssueAction();
|
||||
$subscriptionRequired = $issueAction->subscriptionRequired($issue, $journal);
|
||||
$subscribedUser = $issueAction->subscribedUser($user, $journal);
|
||||
$subscribedDomain = $issueAction->subscribedDomain($request, $journal);
|
||||
|
||||
if ($subscriptionRequired && !$subscribedUser && !$subscribedDomain) {
|
||||
$templateMgr->assign('subscriptionExpiryPartial', true);
|
||||
|
||||
// Partial subscription expiry for issue
|
||||
$partial = $issueAction->subscribedUser($user, $journal, $issue->getId());
|
||||
if (!$partial) {
|
||||
$issueAction->subscribedDomain($request, $journal, $issue->getId());
|
||||
}
|
||||
$templateMgr->assign('issueExpiryPartial', $partial);
|
||||
|
||||
// Partial subscription expiry for articles
|
||||
$articleExpiryPartial = [];
|
||||
foreach ($issueSubmissions as $issueSubmission) {
|
||||
$partial = $issueAction->subscribedUser($user, $journal, $issue->getId(), $issueSubmission->getId());
|
||||
if (!$partial) {
|
||||
$issueAction->subscribedDomain($request, $journal, $issue->getId(), $issueSubmission->getId());
|
||||
}
|
||||
$articleExpiryPartial[$issueSubmission->getId()] = $partial;
|
||||
}
|
||||
$templateMgr->assign('articleExpiryPartial', $articleExpiryPartial);
|
||||
}
|
||||
|
||||
$completedPaymentDao = DAORegistry::getDAO('OJSCompletedPaymentDAO'); /** @var OJSCompletedPaymentDAO $completedPaymentDao */
|
||||
$templateMgr->assign([
|
||||
'hasAccess' => !$subscriptionRequired ||
|
||||
$issue->getAccessStatus() == \APP\issue\Issue::ISSUE_ACCESS_OPEN ||
|
||||
$subscribedUser || $subscribedDomain ||
|
||||
($user && $completedPaymentDao->hasPaidPurchaseIssue($user->getId(), $issue->getId()))
|
||||
]);
|
||||
|
||||
$paymentManager = Application::getPaymentManager($journal);
|
||||
if ($paymentManager->onlyPdfEnabled()) {
|
||||
$templateMgr->assign('restrictOnlyPdf', true);
|
||||
}
|
||||
if ($paymentManager->purchaseArticleEnabled()) {
|
||||
$templateMgr->assign('purchaseArticleEnabled', true);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
/**
|
||||
* @defgroup pages_issue Issue Pages
|
||||
*/
|
||||
|
||||
/**
|
||||
* @file pages/issue/index.php
|
||||
*
|
||||
* Copyright (c) 2014-2021 Simon Fraser University
|
||||
* Copyright (c) 2003-2021 John Willinsky
|
||||
* Distributed under the GNU GPL v3. For full terms see the file docs/COPYING.
|
||||
*
|
||||
* @ingroup pages_issue
|
||||
*
|
||||
* @brief Handle requests for issue functions.
|
||||
*
|
||||
*/
|
||||
switch ($op) {
|
||||
case 'index':
|
||||
case 'current':
|
||||
case 'archive':
|
||||
case 'view':
|
||||
case 'download':
|
||||
define('HANDLER_CLASS', 'APP\pages\issue\IssueHandler');
|
||||
break;
|
||||
}
|
||||
Reference in New Issue
Block a user