Skip to content

Commit

Permalink
WIP shared resource view and routing
Browse files Browse the repository at this point in the history
  • Loading branch information
kulmann committed Apr 26, 2022
1 parent 970b47f commit 862b9cf
Show file tree
Hide file tree
Showing 7 changed files with 121 additions and 65 deletions.
57 changes: 48 additions & 9 deletions packages/web-app-files/src/components/FilesList/ResourceTable.vue
Original file line number Diff line number Diff line change
Expand Up @@ -177,6 +177,13 @@ import { defineComponent, PropType } from '@vue/composition-api'
import { Resource } from '../../helpers/resource'
import { ShareTypes } from '../../helpers/share'
const mapResourceFields = (resource: Resource, mapping = {}) => {
return Object.keys(mapping).reduce((result, resourceKey) => {
result[mapping[resourceKey]] = resource[resourceKey]
return result
}, {})
}
export default defineComponent({
mixins: [Rename],
model: {
Expand Down Expand Up @@ -250,6 +257,33 @@ export default defineComponent({
required: false,
default: null
},
/**
* Maps resource values to route params. Use `{ resourceFieldName: 'routeParamName' }` as format.
*
* An example would be `{ id: 'fileId' }` to map the value of the `id` field of a resource
* to the `fileId` param of the target route.
*
* Defaults to `{ storageId: 'storageId' } to map the value of the `storageId` field of a resource
* to the `storageId` param of the target route.
*/
targetRouteParamMapping: {
type: Object,
required: false,
default: () => ({ storageId: 'storageId' })
},
/**
* Maps resource values to route query options. Use `{ resourceFieldName: 'routeQueryName' }` as format.
*
* An example would be `{ id: 'fileId' }` to map the value of the `id` field of a resource
* to the `fileId` query option of the target route.
*
* Defaults to an empty object because no query options are expected as default.
*/
targetRouteQueryMapping: {
type: Object,
required: false,
default: () => ({})
},
/**
* Asserts whether clicking on the resource name triggers any action
*/
Expand Down Expand Up @@ -534,23 +568,28 @@ export default defineComponent({
this.openWithPanel('sharing-item')
},
folderLink(file) {
return this.createFolderLink(file.path, file.storageId)
return this.createFolderLink(file.path, file)
},
parentFolderLink(file) {
return this.createFolderLink(path.dirname(file.path), file.storageId)
return this.createFolderLink(path.dirname(file.path), file)
},
createFolderLink(path, storageId) {
createFolderLink(path, file) {
if (this.targetRoute === null) {
return {}
}
const params = {
item: path.replace(/^\//, ''),
...this.targetRoute.params,
...mapResourceFields(file, this.targetRouteParamMapping)
}
const query = {
...this.targetRoute.query,
...mapResourceFields(file, this.targetRouteQueryMapping)
}
return {
name: this.targetRoute.name,
query: this.targetRoute.query,
params: {
item: path.replace(/^\//, ''),
...this.targetRoute.params,
...(storageId && { storageId })
}
params,
query
}
},
fileDragged(file) {
Expand Down
4 changes: 3 additions & 1 deletion packages/web-app-files/src/router/spaces.ts
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,9 @@ export const buildRoutes = (components: RouteComponents): RouteConfig[] => [
}
},
{
path: 'shares/:item*',
// FIXME: this is cheating. We rely on shares having a drive alias of `shares/<shareName>` and hardcode it here until we have dynamic routes with drive aliases.
path: 'shares/:shareName?',
// path: 'shares/:shareName?/:item*',
name: locationSpacesShare.name,
component: components.SharedResource,
meta: {
Expand Down
11 changes: 5 additions & 6 deletions packages/web-app-files/src/services/folder/loaderSharedWithMe.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { FolderLoader, FolderLoaderTask, TaskContext } from '../folder'
import Router from 'vue-router'
import { useTask } from 'vue-concurrency'
import { aggregateResourceShares } from '../../helpers/resources'
import { aggregateResourceShares, buildWebDavSpacesPath } from '../../helpers/resources'
import { isLocationSharesActive } from '../../router'
import { Store } from 'vuex'
import get from 'lodash-es/get'
Expand Down Expand Up @@ -47,13 +47,12 @@ export class FolderLoaderSharedWithMe implements FolderLoader {
getToken
)

// FIXME, HACK 1: `/Shares` path prefix needs to be removed backend side. We remove it client side in the meantime.
// FIXME, HACK 2: webDavPath points to `files/<user>/Shares/xyz` but now needs to point to a shares webDavPath according to the storageId of the share. meh.
// FIXME, HACK 1: path needs to be empty because the share has it's own webdav endpoint (we access it's root and thus don't need any relative path). should ideally be removed backend side.
// FIXME, HACK 2: webDavPath points to `files/<user>/Shares/xyz` but now needs to point to a shares webdav root.
if (get(store, 'getters.capabilities.spaces.enabled', false)) {
resources.forEach((resource) => {
if (resource.path.startsWith('/Shares')) {
resource.path = resource.path.substring('/Shares'.length)
}
resource.path = ''
resource.webDavPath = buildWebDavSpacesPath(resource.storageId, '')
})
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,7 @@ import { FolderLoader, FolderLoaderTask, TaskContext } from '../../folder'
import Router from 'vue-router'
import { useTask } from 'vue-concurrency'
import { DavProperties } from 'web-pkg/src/constants'
import {
buildResource,
buildWebDavFilesPath,
buildWebDavSpacesPath
} from '../../../helpers/resources'
import { buildResource, buildWebDavSpacesPath } from '../../../helpers/resources'
import { isLocationSpacesActive } from '../../../router'
import { Store } from 'vuex'
import { fetchResources } from '../util'
Expand Down
13 changes: 7 additions & 6 deletions packages/web-app-files/src/services/folder/spaces/loaderShare.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ import { buildResource, buildWebDavSpacesPath } from '../../../helpers/resources
import { Store } from 'vuex'
import get from 'lodash-es/get'

const SHARE_JAIL_ID = 'a0ca6a90-a365-4782-871e-d44447bbc668'

export class FolderLoaderSpacesShare implements FolderLoader {
public isEnabled(store: Store<any>): boolean {
return get(store, 'getters.capabilities.spaces', false)
Expand All @@ -16,15 +18,14 @@ export class FolderLoaderSpacesShare implements FolderLoader {
}

public getTask(context: TaskContext): FolderLoaderTask {
const {
store,
clientService: { owncloudSdk: client }
} = context
const { store, clientService } = context

return useTask(function* (signal1, signal2, ref, storageId, path = null) {
return useTask(function* (signal1, signal2, ref, shareId, path = null) {
store.commit('Files/CLEAR_CURRENT_FILES_LIST')

const webDavResponse = yield client.files.list(buildWebDavSpacesPath(storageId, path || ''))
const webDavResponse = yield clientService.owncloudSdk.files.list(
buildWebDavSpacesPath([SHARE_JAIL_ID, shareId].join('!'), path || '')
)

const resources = webDavResponse.map(buildResource)
const currentFolder = resources.shift()
Expand Down
79 changes: 46 additions & 33 deletions packages/web-app-files/src/views/shares/SharedResource.vue
Original file line number Diff line number Diff line change
@@ -1,7 +1,17 @@
<template>
<div>
<list-loader v-if="loadResourcesTask.isRunning" />
<app-bar
:has-bulk-actions="true"
:breadcrumbs="breadcrumbs"
:breadcrumbs-context-actions-items="[currentFolder]"
>
<template #actions>
<create-and-upload />
</template>
</app-bar>
<app-loading-spinner v-if="loadResourcesTask.isRunning" />
<template v-else>
<progress-bar v-show="$_uploadProgressVisible" id="files-upload-progress" class="oc-p-s" />
<not-found-message v-if="folderNotFound" class="files-not-found oc-height-1-1" />
<no-content-message
v-else-if="isEmpty"
Expand Down Expand Up @@ -62,18 +72,21 @@
</div>
</template>

<script>
<script lang="ts">
// mixins
import MixinAccessibleBreadcrumb from '../../mixins/accessibleBreadcrumb'
import MixinFileActions from '../../mixins/fileActions'
import MixinFilesListFilter from '../../mixins/filesListFilter'
import MixinFilesListScrolling from '../../mixins/filesListScrolling'
import MixinMountSideBar from '../../mixins/sidebar/mountSideBar'
// components
import AppBar from '../../components/AppBar/AppBar.vue'
import ProgressBar from '../../components/Upload/ProgressBar.vue'
import CreateAndUpload from '../../components/AppBar/CreateAndUpload.vue'
import ResourceTable from '../../components/FilesList/ResourceTable.vue'
import QuickActions from '../../components/FilesList/QuickActions.vue'
import ListLoader from '../../components/FilesList/ListLoader.vue'
import NoContentMessage from '../../components/FilesList/NoContentMessage.vue'
import AppLoadingSpinner from 'web-pkg/src/components/AppLoadingSpinner.vue'
import NoContentMessage from 'web-pkg/src/components/NoContentMessage.vue'
import NotFoundMessage from '../../components/FilesList/NotFoundMessage.vue'
import ListInfo from '../../components/FilesList/ListInfo.vue'
import Pagination from '../../components/FilesList/Pagination.vue'
Expand All @@ -88,15 +101,22 @@ import { basename, join } from 'path'
import PQueue from 'p-queue'
import { createLocationSpaces } from '../../router'
import { useResourcesViewDefaults } from '../../composables'
import { defineComponent } from '@vue/composition-api'
import { fetchResources } from '../../services/folder'
import { Resource } from '../../helpers/resource'
import { breadcrumbsFromPath, concatBreadcrumbs } from '../../helpers/breadcrumbs'
import { useRouteQuery } from 'web-pkg/src/composables'
const visibilityObserver = new VisibilityObserver()
export default {
export default defineComponent({
components: {
AppBar,
ProgressBar,
CreateAndUpload,
ResourceTable,
QuickActions,
ListLoader,
AppLoadingSpinner,
NoContentMessage,
NotFoundMessage,
ListInfo,
Expand All @@ -113,8 +133,9 @@ export default {
],
setup() {
return {
...useResourcesViewDefaults(),
resourceTargetLocation: createLocationSpaces('files-spaces-share')
...useResourcesViewDefaults<Resource, any, any[]>(),
resourceTargetLocation: createLocationSpaces('files-spaces-share'),
shareId: useRouteQuery('shareId')
}
},
Expand All @@ -124,24 +145,28 @@ export default {
...mapState('Files/sidebar', { sidebarClosed: 'closed' }),
...mapGetters('Files', [
'highlightedFile',
'selectedFiles',
'currentFolder',
'inProgress',
'totalFilesCount',
'totalFilesSize'
]),
...mapGetters(['user', 'homeFolder', 'configuration']),
$_uploadProgressVisible() {
return this.inProgress.length > 0
},
isEmpty() {
return this.paginatedResources.length < 1
},
selected: {
get() {
return this.selectedFiles
},
set(resources) {
this.SET_FILE_SELECTION(resources)
}
breadcrumbs() {
return concatBreadcrumbs(
{
text: this.$gettext('Shared with me'),
to: '/files/shares/with-me'
},
...breadcrumbsFromPath(this.$route.path, this.$route.params.item)
)
},
folderNotFound() {
Expand All @@ -150,25 +175,22 @@ export default {
displayThumbnails() {
return !this.configuration.options.disablePreviews
},
storageId() {
return this.$route.params.storageId
}
},
watch: {
$route: {
handler: function () {
this.loadResourcesTask.perform(this, this.storageId)
// TODO: we also need to extract the share path from the URL. for now let's get loading a "share root" working at all...
this.loadResourcesTask.perform(this, this.shareId)
},
immediate: true
}
},
mounted() {
const loadResourcesEventToken = bus.subscribe('app.files.list.load', (path) => {
this.loadResourcesTask.perform(this, this.storageId, path)
this.loadResourcesTask.perform(this, this.shareId, path)
})
this.$on('beforeDestroy', () => bus.unsubscribe('app.files.list.load', loadResourcesEventToken))
Expand All @@ -181,12 +203,7 @@ export default {
methods: {
...mapActions('Files', ['loadPreview']),
...mapActions(['showMessage']),
...mapMutations('Files', [
'REMOVE_FILE',
'REMOVE_FILE_FROM_SEARCHED',
'SET_FILE_SELECTION',
'REMOVE_FILE_SELECTION'
]),
...mapMutations('Files', ['REMOVE_FILE', 'REMOVE_FILE_FROM_SEARCHED', 'REMOVE_FILE_SELECTION']),
fetchResources,
Expand Down Expand Up @@ -296,11 +313,7 @@ export default {
}
})
}
},
isResourceInSelection(resource) {
return this.selected?.includes(resource)
}
}
}
})
</script>
16 changes: 11 additions & 5 deletions packages/web-app-files/src/views/shares/SharedWithMe.vue
Original file line number Diff line number Diff line change
Expand Up @@ -113,6 +113,8 @@
:resources="sharesItems"
:are-resources-clickable="showsAcceptedShares"
:target-route="resourceTargetLocation"
:target-route-param-mapping="resourceTargetParamMapping"
:target-route-query-mapping="resourceTargetQueryMapping"
:header-position="fileListHeaderY"
:sort-by="sharesSortBy"
:sort-dir="sharesSortDir"
Expand Down Expand Up @@ -173,9 +175,8 @@ import MixinMountSideBar from '../../mixins/sidebar/mountSideBar'
import { VisibilityObserver } from 'web-pkg/src/observer'
import { ImageDimension, ImageType } from '../../constants'
import { useSort, useResourcesViewDefaults } from '../../composables'
import { useRouteQuery, useStore } from 'web-pkg/src/composables'
import { useCapabilitySpacesEnabled, useRouteQuery } from 'web-pkg/src/composables'
import debounce from 'lodash-es/debounce'
import get from 'lodash-es/get'
import AppLoadingSpinner from 'web-pkg/src/components/AppLoadingSpinner.vue'
import NoContentMessage from 'web-pkg/src/components/NoContentMessage.vue'
Expand Down Expand Up @@ -215,11 +216,14 @@ export default defineComponent({
any[]
>()
const store = useStore()
const hasSpaces = computed(() => get(store, 'getters.capabilities.spaces', false))
const hasSpaces = useCapabilitySpacesEnabled()
const resourceTargetLocation = computed(() =>
createLocationSpaces(unref(hasSpaces) ? 'files-spaces-share' : 'files-spaces-personal-home')
)
const resourceTargetParamMapping = computed(() =>
unref(hasSpaces) ? { name: 'shareName' } : null
)
const resourceTargetQueryMapping = computed(() => (unref(hasSpaces) ? { id: 'shareId' } : null))
const viewMode = computed(() =>
parseInt(String(unref(useRouteQuery('view-mode', ShareStatus.accepted.toString()))))
Expand Down Expand Up @@ -274,7 +278,9 @@ export default defineComponent({
sharesItems,
displayedFields,
resourceTargetLocation
resourceTargetLocation,
resourceTargetParamMapping,
resourceTargetQueryMapping
}
},
Expand Down

0 comments on commit 862b9cf

Please sign in to comment.