From 883418276a0225a6d524de78bdc1c73753d68fa5 Mon Sep 17 00:00:00 2001 From: Erik Hanson Date: Tue, 3 Dec 2024 11:05:03 -0800 Subject: [PATCH] pkp/pkp-lib#9661 Allow Journal Managers to invite users to adopt a role - ORCiD --- src/components/Modal/Dialog.vue | 2 +- ...ationManagerCancelInvitationDialogBody.vue | 8 +- .../UserInvitationManagerStore.js | 2 +- .../acceptInvitation/AcceptInvitationPage.vue | 2 +- .../AcceptInvitationPageStore.js | 101 +++++++++++++++++- .../AcceptInvitationUserAccountDetails.vue | 4 +- .../AcceptInvitationVerifyOrcid.vue | 42 +++++++- 7 files changed, 145 insertions(+), 16 deletions(-) diff --git a/src/components/Modal/Dialog.vue b/src/components/Modal/Dialog.vue index b04ca5729..fdedea28d 100644 --- a/src/components/Modal/Dialog.vue +++ b/src/components/Modal/Dialog.vue @@ -45,7 +45,7 @@
diff --git a/src/managers/UserInvitationManager/UserInvitationManagerCancelInvitationDialogBody.vue b/src/managers/UserInvitationManager/UserInvitationManagerCancelInvitationDialogBody.vue index 7ffc510eb..8c82861e5 100644 --- a/src/managers/UserInvitationManager/UserInvitationManagerCancelInvitationDialogBody.vue +++ b/src/managers/UserInvitationManager/UserInvitationManagerCancelInvitationDialogBody.vue @@ -2,10 +2,10 @@

{{ message }}

    -
  • {{ t('user.email') }}:{{ email }}
  • -
  • {{ t('userInvitation.roleTable.role') }}:{{ roles }}
  • -
  • {{ t('common.status') }}:{{ status }}
  • -
  • {{ t('user.affiliation') }}:{{ affiliation }}
  • +
  • {{ t('user.email') }}: {{ email }}
  • +
  • {{ t('userInvitation.roleTable.role') }}: {{ roles }}
  • +
  • {{ t('common.status') }}: {{ status }}
  • +
  • {{ t('user.affiliation') }}: {{ affiliation }}
diff --git a/src/managers/UserInvitationManager/UserInvitationManagerStore.js b/src/managers/UserInvitationManager/UserInvitationManagerStore.js index 830414adf..f07042672 100755 --- a/src/managers/UserInvitationManager/UserInvitationManagerStore.js +++ b/src/managers/UserInvitationManager/UserInvitationManagerStore.js @@ -131,7 +131,7 @@ export const useUserInvitationManagerStore = defineComponentStore( }), affiliation: invitationObj.existingUser ? localize(invitationObj.existingUser.affiliation) - : invitationObj.affiliation, + : localize(invitationObj.affiliation), }, actions: [ { diff --git a/src/pages/acceptInvitation/AcceptInvitationPage.vue b/src/pages/acceptInvitation/AcceptInvitationPage.vue index 098eeefdd..2676d9124 100644 --- a/src/pages/acceptInvitation/AcceptInvitationPage.vue +++ b/src/pages/acceptInvitation/AcceptInvitationPage.vue @@ -57,7 +57,7 @@ {{ t('common.back') }} diff --git a/src/pages/acceptInvitation/AcceptInvitationPageStore.js b/src/pages/acceptInvitation/AcceptInvitationPageStore.js index 9445d2f98..741bda7dc 100644 --- a/src/pages/acceptInvitation/AcceptInvitationPageStore.js +++ b/src/pages/acceptInvitation/AcceptInvitationPageStore.js @@ -21,6 +21,7 @@ export const useAcceptInvitationPageStore = defineComponentStore( const email = ref(null); const userId = ref(null); + const existingUser = ref(null); /** All Errors */ const errors = ref({}); @@ -82,8 +83,37 @@ export const useAcceptInvitationPageStore = defineComponentStore( if (data.value.givenName) { updateAcceptInvitationPayload('givenName', data.value.givenName); //if not check this override the multilingual structure } + if (data.value.affiliation) { + updateAcceptInvitationPayload('affiliation', data.value.affiliation); + } updateAcceptInvitationPayload('userCountry', data.value.country); - updateAcceptInvitationPayload('userOrcid', data.value.orcid); + + updateAcceptInvitationPayload('orcid', data.value.orcid); + updateAcceptInvitationPayload( + 'orcidIsVerified', + data.value.orcidIsVerified, + ); + updateAcceptInvitationPayload( + 'orcidAccessDenied', + data.value.orcidAccessDenied, + ); + updateAcceptInvitationPayload( + 'orcidAccessToken', + data.value.orcidAccessToken, + ); + updateAcceptInvitationPayload( + 'orcidAccessScope', + data.value.orcidAccessScope, + ); + updateAcceptInvitationPayload( + 'orcidRefreshToken', + data.value.orcidRefreshToken, + ); + updateAcceptInvitationPayload( + 'orcidAccessExpiresOn', + data.value.orcidAccessExpiresOn, + ); + updateAcceptInvitationPayload( 'userGroupsToAdd', data.value.userGroupsToAdd, @@ -93,9 +123,11 @@ export const useAcceptInvitationPageStore = defineComponentStore( userId.value ? true : false, ); // add username to invitation payload for validations - updateAcceptInvitationPayload('username', null); + updateAcceptInvitationPayload('username', data.value.username); // add password to invitation payload for validations updateAcceptInvitationPayload('password', null); + + existingUser.value = data.existingUser; errors.value = []; if (steps.value.length === 0) { await submit(); @@ -107,6 +139,61 @@ export const useAcceptInvitationPageStore = defineComponentStore( acceptInvitationPayload.value[fieldName] = value; } + /** + * Sets ORCID data in invitation payload. If data is null, all ORCID related fields will be set to null/zero value. + * + * @param {Object|null} data - The ORCID OAuth data object. + * @param {string} data.orcid - The ORCID URL of the user. + * @param {boolean} data.orcidIsVerified - Indicates if the user's ORCID is verified. + * @param {null|string} data.orcidAccessDenied - Indicates if access to ORCID is denied (null if not denied). + * @param {string} data.orcidAccessToken - The access token for ORCID API. + * @param {string} data.orcidAccessScope - The scope of access for the ORCID API. + * @param {string} data.orcidRefreshToken - The refresh token for obtaining new access tokens. + * @param {string} data.orcidAccessExpiresOn - The expiration date and time of the access token in ISO format. + * @param data + */ + function setOrcidData(data) { + const fields = [ + 'orcid', + 'orcidIsVerified', + 'orcidAccessDenied', + 'orcidAccessToken', + 'orcidAccessScope', + 'orcidRefreshToken', + 'orcidAccessExpiresOn', + ]; + const isDataNull = data === null; + fields.forEach((fieldName) => { + acceptInvitationPayload.value[fieldName] = isDataNull + ? null + : data[fieldName]; + }); + } + + const hasValidOrcid = computed(() => { + if (acceptInvitationPayload.value['orcidIsVerified']) { + return true; + } else if (existingUser.value.orcidIsVerified) { + return true; + } + + return false; + }); + + const orcidUri = computed(() => { + const invitationOrcid = acceptInvitationPayload.value['orcid']; + if (invitationOrcid) { + return invitationOrcid; + } + + const userOrcid = existingUser.value['orcid']; + if (userOrcid) { + return userOrcid; + } + + return null; + }); + /** Steps */ const currentStepId = ref( pageInitConfig.steps[0] ? pageInitConfig.steps[0].id : '', @@ -311,7 +398,12 @@ export const useAcceptInvitationPageStore = defineComponentStore( method: 'PUT', body: {invitationData: invitationRequestPayload.value}, }); - if (!acceptInvitationPayload.value.privacyStatement) { + // FIXME: Privacy statement check blocks ORCID from saving before hand. + // Can this check be moved outside of update payload check? + if ( + !acceptInvitationPayload.value.privacyStatement && + currentStep.value.id !== 'verifyOrcid' + ) { errors.value = { privacyStatement: [t('acceptInvitation.privacyStatement.validation')], }; @@ -412,6 +504,7 @@ export const useAcceptInvitationPageStore = defineComponentStore( //computed currentStep, currentStepIndex, + hasValidOrcid, isOnFirstStep, isOnLastStep, isValid, @@ -419,6 +512,7 @@ export const useAcceptInvitationPageStore = defineComponentStore( startedSteps, stepTitle, openStep, + orcidUri, steps, pageTitleDescription, errors, @@ -428,6 +522,7 @@ export const useAcceptInvitationPageStore = defineComponentStore( //methods nextStep, previousStep, + setOrcidData, updateAcceptInvitationPayload, cancel, diff --git a/src/pages/acceptInvitation/AcceptInvitationUserAccountDetails.vue b/src/pages/acceptInvitation/AcceptInvitationUserAccountDetails.vue index edb2ff902..012d2172b 100644 --- a/src/pages/acceptInvitation/AcceptInvitationUserAccountDetails.vue +++ b/src/pages/acceptInvitation/AcceptInvitationUserAccountDetails.vue @@ -10,7 +10,7 @@
import PkpButton from '@/components/Button/Button.vue'; -import {defineProps} from 'vue'; +import {defineProps, onMounted} from 'vue'; import {useLocalize} from '@/composables/useLocalize'; import {useAcceptInvitationPageStore} from './AcceptInvitationPageStore'; -defineProps({}); +const props = defineProps({ + orcidUrl: {type: String, required: true}, + orcidOAuthUrl: {type: String, required: true}, +}); + const store = useAcceptInvitationPageStore(); const {t} = useLocalize(); +onMounted(() => { + pkp.eventBus.$on('addOrcidInvitationData', (data) => setOrcidData(data)); +}); + +/** + * Processes ORCID data for an invitation. + * + * @param {Object|null} data - The ORCID OAuth data object. + * @param {string} data.orcid - The ORCID URL of the user. + * @param {boolean} data.orcidIsVerified - Indicates if the user's ORCID is verified. + * @param {null|string} data.orcidAccessDenied - Indicates if access to ORCID is denied (null if not denied). + * @param {string} data.orcidAccessToken - The access token for ORCID API. + * @param {string} data.orcidAccessScope - The scope of access for the ORCID API. + * @param {string} data.orcidRefreshToken - The refresh token for obtaining new access tokens. + * @param {string} data.orcidAccessExpiresOn - The expiration date and time of the access token in ISO format. + * + * @returns {void} + */ +async function setOrcidData(data) { + store.setOrcidData(data); + await store.nextStep(); +} + /** * Go to the next step */ function skipOrcid() { - delete store.acceptInvitationPayload.userOrcid; store.openStep(store.steps[1 + store.currentStepIndex].id); } +/** + * Initiates ORCID OAuth granting flow + */ function verifyOrcid() { - store.openStep(store.steps[1 + store.currentStepIndex].id); + const oauthWindow = window.open( + props.orcidOAuthUrl, + '_blank', + 'toolbar=no, scrollbars=yes, width=540, height=700, top=500, left=500', + ); + oauthWindow.opener = self; }