Skip to content

Commit

Permalink
Merge pull request #10104 from owncloud/quick-link-modal
Browse files Browse the repository at this point in the history
feat: show link modal when password is enforced
  • Loading branch information
JammingBen authored Dec 7, 2023
2 parents 9485f02 + cdc18fd commit b99b51b
Show file tree
Hide file tree
Showing 26 changed files with 530 additions and 605 deletions.
5 changes: 5 additions & 0 deletions changelog/unreleased/enhancement-create-link-modal
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
Enhancement: Create link modal

When creating a link while passwords are enfoced, Web will now display a modal that lets the user not only set a password, but also the role and an optional expiration date.

https://github.com/owncloud/web/pull/10104
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ export default {
return store.getters['Files/selectedFiles']
})
const { actions: createLinkActions } = useFileActionsCreateLink({ store })
const { actions: createLinkActions } = useFileActionsCreateLink({ store, enforceModal: true })
const createLinkAction = computed<FileAction>(() => unref(createLinkActions)[0])
const areSelectActionsDisabled = computed<boolean>(() => selectedFiles.value.length < 1)
Expand Down
189 changes: 83 additions & 106 deletions packages/web-app-files/src/components/SideBar/Shares/FileLinks.vue
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@
<create-quick-link
v-else-if="canCreateLinks"
:expiration-rules="expirationRules"
@create-public-link="checkLinkToCreate"
@create-public-link="addNewLink"
/>
<details-and-edit
v-if="quicklink"
Expand Down Expand Up @@ -124,8 +124,11 @@ import {
useCapabilityFilesSharingPublicPasswordEnforcedFor,
useAbility,
usePasswordPolicyService,
getDefaultLinkPermissions,
useExpirationRules
useExpirationRules,
useDefaultLinkPermissions,
useFileActionsCreateLink,
FileAction,
useClientService
} from '@ownclouders/web-pkg'
import { shareViaLinkHelp, shareViaIndirectLinkHelp } from '../../../helpers/contextualHelpers'
import {
Expand All @@ -137,7 +140,6 @@ import {
Share,
SharePermissions
} from '@ownclouders/web-client/src/helpers/share'
import { showQuickLinkPasswordModal } from '@ownclouders/web-pkg'
import DetailsAndEdit from './Links/DetailsAndEdit.vue'
import NameAndCopy from './Links/NameAndCopy.vue'
import CreateQuickLink from './Links/CreateQuickLink.vue'
Expand All @@ -151,6 +153,7 @@ import {
import { isLocationSharesActive } from '@ownclouders/web-pkg'
import { useShares } from 'web-app-files/src/composables'
import { configurationManager } from '@ownclouders/web-pkg'
import { useGettext } from 'vue3-gettext'
export default defineComponent({
name: 'FileLinks',
Expand All @@ -161,11 +164,23 @@ export default defineComponent({
},
setup() {
const store = useStore()
const { $gettext } = useGettext()
const ability = useAbility()
const clientService = useClientService()
const { can } = ability
const { expirationRules } = useExpirationRules()
const passwordPolicyService = usePasswordPolicyService()
const hasResharing = useCapabilityFilesSharingResharing()
const hasShareJail = useCapabilityShareJailEnabled()
const { defaultLinkPermissions } = useDefaultLinkPermissions()
const { actions: createLinkActions } = useFileActionsCreateLink({ store })
const createLinkAction = computed<FileAction>(() =>
unref(createLinkActions).find(({ name }) => name === 'create-links')
)
const createQuicklinkAction = computed<FileAction>(() =>
unref(createLinkActions).find(({ name }) => name === 'create-quick-links')
)
const space = inject<Ref<SpaceResource>>('space')
const resource = inject<Ref<Resource>>('resource')
Expand Down Expand Up @@ -221,14 +236,71 @@ export default defineComponent({
)
}
const addNewLink = ({ link }) => {
const handlerArgs = { space: unref(space), resources: [unref(resource)] }
if (link?.quicklink) {
return unref(createQuicklinkAction)?.handler(handlerArgs)
}
return unref(createLinkAction)?.handler(handlerArgs)
}
const updatePublicLink = async ({ params }) => {
try {
await store.dispatch('Files/updateLink', {
id: params.id,
client: clientService.owncloudSdk,
params
})
store.dispatch('hideModal')
store.dispatch('showMessage', {
title: $gettext('Link was updated successfully')
})
} catch (e) {
// Human-readable error message is provided, for example when password is on banned list
if (e.statusCode === 400) {
return store.dispatch('setModalInputErrorMessage', $gettext(e.message))
}
store.dispatch('showErrorMessage', {
title: $gettext('Failed to update link'),
error: e
})
}
}
const showQuickLinkPasswordModal = (params) => {
const modal = {
variation: 'passive',
title: $gettext('Set password'),
cancelText: $gettext('Cancel'),
confirmText: $gettext('Set'),
hasInput: true,
inputDescription: $gettext('Passwords for links are required.'),
inputPasswordPolicy: passwordPolicyService.getPolicy(),
inputGeneratePasswordMethod: () => passwordPolicyService.generatePassword(),
inputLabel: $gettext('Password'),
inputType: 'password',
onInput: () => store.dispatch('setModalInputErrorMessage', ''),
onPasswordChallengeCompleted: () => store.dispatch('setModalConfirmButtonDisabled', false),
onPasswordChallengeFailed: () => store.dispatch('setModalConfirmButtonDisabled', true),
onCancel: () => store.dispatch('hideModal'),
onConfirm: (newPassword: string) => {
return updatePublicLink({ params: { ...params, password: newPassword } })
}
}
return store.dispatch('createModal', modal)
}
return {
$store: store,
ability,
space,
resource,
incomingParentShare: inject<Share>('incomingParentShare'),
hasSpaces: useCapabilitySpacesEnabled(),
hasShareJail: useCapabilityShareJailEnabled(),
hasShareJail,
hasPublicLinkEditing: useCapabilityFilesSharingPublicCanEdit(),
hasPublicLinkContribute: useCapabilityFilesSharingPublicCanContribute(),
hasPublicLinkAliasSupport: useCapabilityFilesSharingPublicAlias(),
Expand All @@ -241,10 +313,13 @@ export default defineComponent({
canCreatePublicLinks,
canDeleteReadOnlyPublicLinkPassword,
configurationManager,
passwordPolicyService,
canCreateLinks,
canEditLink,
expirationRules
expirationRules,
updatePublicLink,
showQuickLinkPasswordModal,
defaultLinkPermissions,
addNewLink
}
},
computed: {
Expand Down Expand Up @@ -380,39 +455,6 @@ export default defineComponent({
)
},
addNewLink() {
this.checkLinkToCreate({
link: {
name: this.$gettext('Link'),
permissions: getDefaultLinkPermissions({
ability: this.ability,
store: this.$store
}).toString(),
expiration: this.expirationRules.default,
password: false
}
})
},
checkLinkToCreate({ link }) {
const paramsToCreate = this.getParamsForLink(link)
if (this.isPasswordEnforcedFor(link)) {
showQuickLinkPasswordModal(
{
...this.$language,
store: this.$store,
passwordPolicyService: this.passwordPolicyService
},
(newPassword) => {
this.createLink({ params: { ...paramsToCreate, password: newPassword } })
}
)
} else {
this.createLink({ params: paramsToCreate })
}
},
checkLinkToUpdate({ link }) {
let params = this.getParamsForLink(link)
if (link.permissions === 0) {
Expand All @@ -424,16 +466,7 @@ export default defineComponent({
}
if (!link.password && !this.canDeletePublicLinkPassword(link)) {
showQuickLinkPasswordModal(
{
...this.$language,
store: this.$store,
passwordPolicyService: this.passwordPolicyService
},
(newPassword) => {
this.updatePublicLink({ params: { ...params, password: newPassword } })
}
)
this.showQuickLinkPasswordModal(params)
} else {
this.updatePublicLink({ params })
}
Expand Down Expand Up @@ -490,62 +523,6 @@ export default defineComponent({
}
},
async createLink({ params }) {
let path = this.resource.path
// sharing a share root from the share jail -> use resource name as path
if (this.hasShareJail && path === '/') {
path = `/${this.resource.name}`
}
try {
await this.addLink({
path,
client: this.$client,
storageId: this.resource.fileId || this.resource.id,
params
})
this.hideModal()
this.showMessage({
title: this.$gettext('Link was created successfully')
})
} catch (e) {
console.error(e)
// Human-readable error message is provided, for example when password is on banned list
if (e.statusCode === 400) {
return this.setModalInputErrorMessage(this.$gettext(e.message))
}
this.showErrorMessage({
title: this.$gettext('Failed to create link'),
error: e
})
}
},
async updatePublicLink({ params }) {
try {
await this.updateLink({
id: params.id,
client: this.$client,
params
})
this.hideModal()
this.showMessage({
title: this.$gettext('Link was updated successfully')
})
} catch (e) {
// Human-readable error message is provided, for example when password is on banned list
if (e.statusCode === 400) {
return this.setModalInputErrorMessage(this.$gettext(e.message))
}
this.showErrorMessage({
title: this.$gettext('Failed to update link'),
error: e
})
}
},
deleteLinkConfirmation({ link }) {
const modal = {
variation: 'danger',
Expand Down
4 changes: 2 additions & 2 deletions packages/web-app-files/src/extensions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import {
useRouter,
useSearch,
useFileActionsShowShares,
useFileActionsCreateQuickLink
useFileActionsCopyQuickLink
} from '@ownclouders/web-pkg'
import { computed, unref } from 'vue'
import { SDKSearch } from './search'
Expand All @@ -16,7 +16,7 @@ export const extensions = ({ applicationConfig }: ApplicationSetupOptions) => {
const { search: searchFunction } = useSearch()

const { actions: showSharesActions } = useFileActionsShowShares()
const { actions: quickLinkActions } = useFileActionsCreateQuickLink()
const { actions: quickLinkActions } = useFileActionsCopyQuickLink()

return computed(
() =>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ const quicklinkAction = {
handler: jest.fn(),
icon: 'link-add',
id: 'quicklink',
name: 'create-quicklink',
name: 'copy-quicklink',
label: () => 'Create and copy quicklink'
}

Expand Down Expand Up @@ -57,7 +57,7 @@ describe('QuickActions', () => {
})

it('should not display action buttons where "displayed" is set to false', () => {
const linkActionButton = wrapper.find('.files-quick-action-create-quicklink')
const linkActionButton = wrapper.find('.files-quick-action-copy-quicklink')

expect(linkActionButton.exists()).toBeFalsy()
})
Expand Down
Loading

0 comments on commit b99b51b

Please sign in to comment.