From fbc2a896cc64cb4482bc0d5c6170fa7fe10a86e6 Mon Sep 17 00:00:00 2001
From: jyhein <124268211+jyhein@users.noreply.github.com>
Date: Tue, 6 Feb 2024 18:58:18 +0200
Subject: [PATCH] Make submission language selection and metadata forms
independent from website language settings
---
classes/issue/IssueGalley.php | 4 +--
.../I9425_SeparateUIAndSubmissionLocales.php | 31 +++++++++++++++++++
classes/publication/Repository.php | 4 +--
classes/publication/maps/Schema.php | 4 ++-
classes/submission/Repository.php | 3 +-
classes/submission/maps/Schema.php | 8 ++++-
.../articleGalleys/form/ArticleGalleyForm.php | 13 ++++----
.../20-CreateContext.cy.js | 9 ++++--
.../60-content/AmwandengaSubmission.cy.js | 4 +--
.../tests/integration/SubmissionWizard.cy.js | 22 ++++++-------
dbscripts/xml/upgrade.xml | 1 +
pages/submission/SubmissionHandler.php | 4 +--
pages/workflow/WorkflowHandler.php | 12 ++++---
schemas/galley.json | 2 +-
14 files changed, 86 insertions(+), 35 deletions(-)
create mode 100644 classes/migration/upgrade/v3_5_0/I9425_SeparateUIAndSubmissionLocales.php
diff --git a/classes/issue/IssueGalley.php b/classes/issue/IssueGalley.php
index 5cbeed9badd..5b4bfdc24d8 100644
--- a/classes/issue/IssueGalley.php
+++ b/classes/issue/IssueGalley.php
@@ -60,8 +60,8 @@ public function isPdfGalley()
public function getGalleyLabel()
{
$label = $this->getLabel();
- if ($this->getLocale() != Locale::getLocale()) {
- $label .= ' (' . Locale::getMetadata($this->getLocale())->getDisplayName() . ')';
+ if ($this->getLocale() !== Locale::getLocale()) {
+ $label .= ' (' . Locale::getSubmissionLocaleDisplayNames([$this->getLocale()])[$this->getLocale()] . ')';
}
return $label;
}
diff --git a/classes/migration/upgrade/v3_5_0/I9425_SeparateUIAndSubmissionLocales.php b/classes/migration/upgrade/v3_5_0/I9425_SeparateUIAndSubmissionLocales.php
new file mode 100644
index 00000000000..4a95bd2c795
--- /dev/null
+++ b/classes/migration/upgrade/v3_5_0/I9425_SeparateUIAndSubmissionLocales.php
@@ -0,0 +1,31 @@
+getSupportedSubmissionLocales();
- $primaryLocale = $submission->getLocale();
+ $primaryLocale = $submission->getData('locale');
// Ensure that the specified section exists
$section = null;
@@ -77,6 +76,7 @@ public function validate($publication, array $props, Submission $submission, Con
}
// Check the word count on abstracts
+ $allowedLocales = $submission->getPublicationLanguages($context->getSupportedSubmissionMetadataLocales());
foreach ($allowedLocales as $localeKey) {
if (empty($props['abstract'][$localeKey])) {
continue;
diff --git a/classes/publication/maps/Schema.php b/classes/publication/maps/Schema.php
index e01c6337286..d8d813f48ce 100644
--- a/classes/publication/maps/Schema.php
+++ b/classes/publication/maps/Schema.php
@@ -49,7 +49,9 @@ protected function mapByProperties(array $props, Publication $publication, bool
);
}
- $output = $this->schemaService->addMissingMultilingualValues(PKPSchemaService::SCHEMA_PUBLICATION, $output, $this->context->getSupportedSubmissionLocales());
+ $locales = $this->submission->getPublicationLanguages($this->context->getSupportedSubmissionMetadataLocales());
+
+ $output = $this->schemaService->addMissingMultilingualValues(PKPSchemaService::SCHEMA_PUBLICATION, $output, $locales);
ksort($output);
diff --git a/classes/submission/Repository.php b/classes/submission/Repository.php
index 579c0c9a536..10d6ee5ac46 100644
--- a/classes/submission/Repository.php
+++ b/classes/submission/Repository.php
@@ -80,7 +80,8 @@ public function validateSubmit(Submission $submission, Context $context): array
$abstracts = $publication->getData('abstract');
if ($abstracts) {
$abstractErrors = [];
- foreach ($context->getSupportedSubmissionLocales() as $localeKey) {
+ $allowedLocales = $submission->getPublicationLanguages($context->getSupportedSubmissionMetadataLocales());
+ foreach ($allowedLocales as $localeKey) {
$abstract = $publication->getData('abstract', $localeKey);
$wordCount = $abstract ? PKPString::getWordCount($abstract) : 0;
if ($wordCount > $section->getAbstractWordCount()) {
diff --git a/classes/submission/maps/Schema.php b/classes/submission/maps/Schema.php
index 272fda89697..12f2097c384 100644
--- a/classes/submission/maps/Schema.php
+++ b/classes/submission/maps/Schema.php
@@ -36,7 +36,13 @@ protected function mapByProperties(array $props, Submission $submission): array
);
}
- $output = $this->schemaService->addMissingMultilingualValues($this->schemaService::SCHEMA_SUBMISSION, $output, $this->context->getSupportedSubmissionLocales());
+ $locales = $this->context->getSupportedSubmissionMetadataLocales();
+
+ if (!in_array($primaryLocale = $submission->getData('locale'), $locales)) {
+ $locales[] = $primaryLocale;
+ }
+
+ $output = $this->schemaService->addMissingMultilingualValues($this->schemaService::SCHEMA_SUBMISSION, $output, $locales);
ksort($output);
diff --git a/controllers/grid/articleGalleys/form/ArticleGalleyForm.php b/controllers/grid/articleGalleys/form/ArticleGalleyForm.php
index 0ad5450ca89..8896f7d741e 100644
--- a/controllers/grid/articleGalleys/form/ArticleGalleyForm.php
+++ b/controllers/grid/articleGalleys/form/ArticleGalleyForm.php
@@ -62,7 +62,7 @@ public function __construct($request, $submission, $publication, $articleGalley
$this->addCheck(new \PKP\form\validation\FormValidatorCSRF($this));
// Ensure a locale is provided and valid
- $journal = $request->getJournal();
+ $locales = $request->getJournal()->getSupportedSubmissionMetadataLocaleNames() + $submission->getPublicationLanguageNames() + ($articleGalley?->getLanguageNames() ?? []);
$this->addCheck(
new \PKP\form\validation\FormValidator(
$this,
@@ -70,9 +70,7 @@ public function __construct($request, $submission, $publication, $articleGalley
'required',
'editor.issues.galleyLocaleRequired'
),
- function ($locale) use ($journal) {
- return in_array($locale, $journal->getSupportedSubmissionLocaleNames());
- }
+ fn ($locale) => in_array($locale, $locales)
);
}
@@ -93,9 +91,12 @@ public function fetch($request, $template = null, $display = false)
'supportsDependentFiles' => $articleGalleyFile ? Repo::submissionFile()->supportsDependentFiles($articleGalleyFile) : null,
]);
}
- $context = $request->getContext();
+
+ $supportedLocales = $request->getContext()->getSupportedSubmissionMetadataLocaleNames() + $this->_submission->getPublicationLanguageNames() + ($this->_articleGalley?->getLanguageNames() ?? []);
+ ksort($supportedLocales);
+
$templateMgr->assign([
- 'supportedLocales' => $context->getSupportedSubmissionLocaleNames(),
+ 'supportedLocales' => $supportedLocales,
'submissionId' => $this->_submission->getId(),
'publicationId' => $this->_publication->getId(),
'formDisabled' => !$this->_isEditable
diff --git a/cypress/tests/data/10-ApplicationSetup/20-CreateContext.cy.js b/cypress/tests/data/10-ApplicationSetup/20-CreateContext.cy.js
index feac3e8ca6d..4d45190b0fb 100644
--- a/cypress/tests/data/10-ApplicationSetup/20-CreateContext.cy.js
+++ b/cypress/tests/data/10-ApplicationSetup/20-CreateContext.cy.js
@@ -70,8 +70,13 @@ describe('Data suite tests', function() {
cy.get('#appearance [role="status"]').contains('Saved');
cy.get('button[id="languages-button"]').click();
- cy.get('input[id^=select-cell-fr_CA-submissionLocale]').click();
- cy.contains('Locale settings saved.');
+ cy.get('input[id^="select-cell-fr_CA-formLocale"]').click();
+ cy.get('a[id^=component-grid-settings-languages-submissionlanguagegrid-addLanguageModal-button]').click();
+ cy.get('#locale-fr_CA').should('exist').click();
+ cy.get('#addLanguageForm button[name="submitFormButton"]').click();
+ cy.contains('Submission locales updated.').should('exist');
+ cy.get('input[id^="select-cell-fr_CA-submissionLocale"]').click();
+ cy.get('input[id^="select-cell-fr_CA-submissionMetadataLocale"]').should('be.checked');
cy.get('button[id="indexing-button"]').click();
cy.get('input[name="searchDescription-en"]').type(Cypress.env('contextDescriptions')['en'], {delay: 0});
diff --git a/cypress/tests/data/60-content/AmwandengaSubmission.cy.js b/cypress/tests/data/60-content/AmwandengaSubmission.cy.js
index a9c5fc14a45..a6de75ba6ba 100644
--- a/cypress/tests/data/60-content/AmwandengaSubmission.cy.js
+++ b/cypress/tests/data/60-content/AmwandengaSubmission.cy.js
@@ -224,7 +224,7 @@ describe('Data suite: Amwandenga', function() {
.find('h4').contains('Keywords').siblings('.submissionWizard__reviewPanel__item__value').contains('None provided')
.parents('.submissionWizard__reviewPanel')
.find('h4').contains('Abstract').siblings('.submissionWizard__reviewPanel__item__value').contains(submission.abstract);
- cy.get('h3').contains('Details (French)')
+ cy.get('h3').contains('Details (French (Canada))')
.parents('.submissionWizard__reviewPanel')
.find('h4').contains('Title').siblings('.submissionWizard__reviewPanel__item__value').contains('None provided')
.parents('.submissionWizard__reviewPanel')
@@ -234,7 +234,7 @@ describe('Data suite: Amwandenga', function() {
cy.get('h3').contains('For the Editors (English)')
.parents('.submissionWizard__reviewPanel')
.find('h4').contains('Comments for the Editor').siblings('.submissionWizard__reviewPanel__item__value').contains('None');
- cy.get('h3').contains('For the Editors (French)') // FIXME: Should be French
+ cy.get('h3').contains('For the Editors (French (Canada))')
// Save for later
cy.get('button').contains('Save for Later').click();
diff --git a/cypress/tests/integration/SubmissionWizard.cy.js b/cypress/tests/integration/SubmissionWizard.cy.js
index 1011d67375e..6b9e000ebd0 100644
--- a/cypress/tests/integration/SubmissionWizard.cy.js
+++ b/cypress/tests/integration/SubmissionWizard.cy.js
@@ -479,20 +479,20 @@ describe('Submission Wizard', function() {
cy.get('label:contains("Abstract *")');
cy.contains('Word Count: 0/500');
- // Change submission language to French and section to Reviews
+ // Change submission language to French (Canada) and section to Reviews
cy.contains('Submitting to the Articles section in English');
cy.get('button:contains("Change")').click();
cy.get('h2:contains("Change Submission Settings")')
.parents('.modal')
.within(() => {
- cy.get('label:contains("French")').click();
+ cy.get('label:contains("French (Canada)")').click();
cy.get('label:contains("Reviews")').click();
cy.get('button:contains("Save")').click();
});
- // Forms load with French fields displayed instead of English
- cy.contains('Submitting to the Reviews section in French');
- cy.get('span.pkpFormLocales__locale:contains("French")');
+ // Forms load with French (Canada) fields displayed instead of English
+ cy.contains('Submitting to the Reviews section in French (Canada)');
+ cy.get('span.pkpFormLocales__locale:contains("French (Canada)")');
cy.get('#titleAbstract-keywords-control-fr_CA').type('Transformation Sociale', {delay: 0});
cy.get('li:contains("Transformation Sociale")');
cy.get('#titleAbstract-keywords-control-fr_CA').type('{downarrow}{enter}', {delay: 0});
@@ -502,7 +502,7 @@ describe('Submission Wizard', function() {
cy.get('label:contains("Abstract *")').should('not.exist');
cy.get('*:contains("Word Count: 0/500")').should('not.exist');
- // Show English fields alongside French fields
+ // Show English fields alongside French (Canada) fields
cy.get('.pkpStep:contains("Submission Details") button.pkpFormLocales__locale:contains("English")').click();
cy.get('label:contains("Title in English")');
cy.get('label:contains("Keywords in English")');
@@ -523,7 +523,7 @@ describe('Submission Wizard', function() {
cy.get('button:contains("Continue")').click();
cy.get('button:contains("Continue")').click();
- // Check metadata form shows in French only at first
+ // Check metadata form shows in French (Canada) only at first
const metadata = {
subjects: "Subjects",
disciplines: "Disciplines",
@@ -539,7 +539,7 @@ describe('Submission Wizard', function() {
cy.get('label:contains("' + metadata[prop] + ' in English")').should('not.be.visible');
});
- // Show English fields alongside French fields
+ // Show English fields alongside French (Canada) fields
cy.get('.pkpStep:contains("For the Editors") button.pkpFormLocales__locale:contains("English")').click();
Object.keys(metadata).forEach((prop) => {
cy.get('label:contains("' + metadata[prop] + ' in English")');
@@ -551,13 +551,13 @@ describe('Submission Wizard', function() {
// Errors in review
cy.get('button:contains("Continue")').click();
cy.contains('There are one or more problems');
- cy.get('h3:contains("Details (French)")')
+ cy.get('h3:contains("Details (French (Canada))")')
.parents('.submissionWizard__reviewPanel')
.find('h4:contains("Title")')
.parent()
.contains('This field is required.');
- cy.contains('The given name is missing in French for one or more of the contributors.');
- cy.get('h3:contains("For the Editors (French)")')
+ cy.contains('The given name is missing in French (Canada) for one or more of the contributors.');
+ cy.get('h3:contains("For the Editors (French (Canada))")')
.parents('.submissionWizard__reviewPanel')
.find('h4:contains("Subjects")')
.parent()
diff --git a/dbscripts/xml/upgrade.xml b/dbscripts/xml/upgrade.xml
index a1ea787c3b2..5a437cc85e6 100644
--- a/dbscripts/xml/upgrade.xml
+++ b/dbscripts/xml/upgrade.xml
@@ -229,6 +229,7 @@
+
diff --git a/pages/submission/SubmissionHandler.php b/pages/submission/SubmissionHandler.php
index 6485286889c..81f81479355 100644
--- a/pages/submission/SubmissionHandler.php
+++ b/pages/submission/SubmissionHandler.php
@@ -94,7 +94,7 @@ protected function getSubmittingTo(Context $context, Submission $submission, arr
'submission.wizard.submittingToSectionInLanguage',
[
'section' => $section->getLocalizedTitle(),
- 'language' => Locale::getMetadata($submission->getData('locale'))->getDisplayName(),
+ 'language' => Locale::getSubmissionLocaleDisplayNames([$submission->getData('locale')])[$submission->getData('locale')],
]
);
} elseif ($sectionCount) {
@@ -108,7 +108,7 @@ protected function getSubmittingTo(Context $context, Submission $submission, arr
return __(
'submission.wizard.submittingInLanguage',
[
- 'language' => Locale::getMetadata($submission->getData('locale'))->getDisplayName(),
+ 'language' => Locale::getSubmissionLocaleDisplayNames([$submission->getData('locale')])[$submission->getData('locale')],
]
);
}
diff --git a/pages/workflow/WorkflowHandler.php b/pages/workflow/WorkflowHandler.php
index 98c9ef4b5c2..d0547e867fb 100644
--- a/pages/workflow/WorkflowHandler.php
+++ b/pages/workflow/WorkflowHandler.php
@@ -85,11 +85,15 @@ public function setupIndex($request)
$submissionContext = Services::get('context')->get($submission->getContextId());
}
- $locales = $submissionContext->getSupportedSubmissionLocaleNames();
- $locales = array_map(fn (string $locale, string $name) => ['key' => $locale, 'label' => $name], array_keys($locales), $locales);
-
$latestPublication = $submission->getLatestPublication();
+ $submissionLocale = $submission->getData('locale');
+ $locales = collect($submissionContext->getSupportedSubmissionMetadataLocaleNames() + $submission->getPublicationLanguageNames())
+ ->map(fn (string $name, string $locale) => ['key' => $locale, 'label' => $name])
+ ->sortBy('key')
+ ->values()
+ ->toArray();
+
$latestPublicationApiUrl = $request->getDispatcher()->url($request, Application::ROUTE_API, $submissionContext->getPath(), 'submissions/' . $submission->getId() . '/publications/' . $latestPublication->getId());
$temporaryFileApiUrl = $request->getDispatcher()->url($request, Application::ROUTE_API, $submissionContext->getPath(), 'temporaryFiles');
$issueApiUrl = $request->getDispatcher()->url($request, Application::ROUTE_API, $submissionContext->getData('urlPath'), 'issues/__issueId__');
@@ -113,7 +117,7 @@ class_exists(\APP\components\forms\publication\AssignToIssueForm::class); // For
]);
$components = $templateMgr->getState('components');
- $components[FORM_ISSUE_ENTRY] = $issueEntryForm->getConfig();
+ $components[FORM_ISSUE_ENTRY] = $this->getLocalizedForm($issueEntryForm, $submissionLocale, $locales);
$canEditPublication = Repo::submission()->canEditPublication($submission->getId(), $request->getUser()->getId());
diff --git a/schemas/galley.json b/schemas/galley.json
index b0aac2e72c5..5cbbfa08292 100644
--- a/schemas/galley.json
+++ b/schemas/galley.json
@@ -48,7 +48,7 @@
"description": "The primary locale of this galley.",
"apiSummary": true,
"validation": [
- "regex:/^([a-z]{2})((_[A-Z]{2})?)(@[a-z]{0,})?$/"
+ "regex:/^([A-Za-z]{2,4})([_-]([A-Za-z]{4,5}|[0-9]{4}))?([_-]([A-Za-z]{2}|[0-9]{3}))?(@[A-Za-z_]{2,30})?$/"
]
},
"label": {