Skip to content

Commit

Permalink
Add version naming UI
Browse files Browse the repository at this point in the history
Signed-off-by: Louis Chemineau <[email protected]>
  • Loading branch information
artonge committed Nov 14, 2022
1 parent b094abd commit 3c0ff2c
Show file tree
Hide file tree
Showing 8 changed files with 348 additions and 147 deletions.
212 changes: 212 additions & 0 deletions apps/files_versions/src/components/Version.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,212 @@
<!--
- @copyright 2022 Carl Schwan <[email protected]>
- @license AGPL-3.0-or-later
-
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU Affero General Public License as
- published by the Free Software Foundation, either version 3 of the
- License, or (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU Affero General Public License for more details.
-
- You should have received a copy of the GNU Affero General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
-->
<template>
<div>
<NcListItem class="version"
:title="version.title"
:href="version.url">
<template #icon>
<img lazy="true"
:src="version.preview"
alt=""
height="256"
width="256"
class="version__image">
</template>
<template #subtitle>
<div class="version__info">
<span>{{ version.mtime | humanDateFromNow }}</span>
<!-- Separate dot to improve alignement -->
<span class="version__info__size">•</span>
<span class="version__info__size">{{ version.size | humanReadableSize }}</span>
</div>
</template>
<template v-if="!isCurrent" #actions>
<NcActionButton v-if="experimental"
:close-after-click="true"
@click="openVersionNameModal">
<template #icon>
<Pen :size="22" />
</template>
{{ version.title === '' ? t('files_versions', 'Name this version') : t('files_versions', 'Edit version name') }}
</NcActionButton>
<NcActionLink :href="version.url"
:close-after-click="true"
:download="version.url">
<template #icon>
<Download :size="22" />
</template>
{{ t('files_versions', 'Download version') }}
</NcActionLink>
<NcActionButton :close-after-click="true"
@click="restoreVersion">
<template #icon>
<BackupRestore :size="22" />
</template>
{{ t('files_versions', 'Restore version') }}
</NcActionButton>
<NcActionButton v-if="experimental"
:close-after-click="true"
@click="deleteVersion">
<template #icon>
<Delete :size="22" />
</template>
{{ t('files_versions', 'Delete version') }}
</NcActionButton>
</template>
</NcListItem>
<NcModal v-if="showVersionNameEditor"
:title="t('files_versions', 'Name this version')"
:close-button-contained="false"
@close="showVersionNameEditor = false">
<div class="modal">
<NcTextField ref="nameInput"
:value.sync="versionName"
:label="t('photos', 'Named versions are persisted, and excluded from automatic cleanups when your storage quota is full.')"
:placeholder="t('photos', 'Version name')"
:label-visible="true" />
<div class="modal__actions">
<NcButton @click="setVersionName('')">
{{ t('files_versions', 'Remove version name') }}
</NcButton>
<NcButton type="primary" :disabled="versionName.trim().length === 0" @click="setVersionName(versionName)">
<template #icon>
<ContentSave />
</template>
{{ t('files_versions', 'Save version name') }}
</NcButton>
</div>
</div>
</NcModal>
</div>
</template>

<script>
import BackupRestore from 'vue-material-design-icons/BackupRestore.vue'
import Download from 'vue-material-design-icons/Download.vue'
import Pen from 'vue-material-design-icons/Pen.vue'
import ContentSave from 'vue-material-design-icons/ContentSave.vue'
import Delete from 'vue-material-design-icons/Delete'
import { NcActionButton, NcActionLink, NcListItem, NcModal, NcButton, NcTextField } from '@nextcloud/vue'
import moment from '@nextcloud/moment'
export default {
name: 'Version',
components: {
NcActionLink,
NcActionButton,
NcListItem,
NcModal,
NcButton,
NcTextField,
BackupRestore,
Download,
Pen,
ContentSave,
Delete,
},
filters: {
humanReadableSize(bytes) {
return OC.Util.humanFileSize(bytes)
},
humanDateFromNow(timestamp) {
return moment(timestamp).fromNow()
},
},
props: {
/** @type {Vue.PropOptions<import('../utils/versions.js').Version>} */
version: {
type: Object,
required: true,
},
isCurrent: {
type: Boolean,
default: false,
},
},
data() {
return {
showVersionNameEditor: false,
versionName: this.version.title,
experimental: OC.experimental ?? false,
}
},
methods: {
openVersionNameModal() {
this.showVersionNameEditor = true
this.$nextTick(() => {
this.$refs.nameInput.$el.getElementsByTagName('input')[0].focus()
})
},
restoreVersion() {
this.$emit('restore', this.version)
},
setVersionName(name) {
this.versionName = name
this.showVersionNameEditor = false
this.$emit('name-update', this.version, name)
},
deleteVersion() {
this.$emit('delete', this.version)
},
},
}
</script>
<style scoped lang="scss">
.version {
display: flex;
flex-direction: row;
&__info {
display: flex;
flex-direction: row;
align-items: center;
gap: 0.5rem;
&__size {
color: var(--color-text-lighter);
}
}
&__image {
width: 3rem;
height: 3rem;
border: 1px solid var(--color-border);
margin-right: 1rem;
border-radius: var(--border-radius-large);
}
}
.modal {
display: flex;
justify-content: space-between;
flex-direction: column;
height: 250px;
padding: 16px;
&__actions {
display: flex;
justify-content: space-between;
margin-top: 64px;
}
}
</style>
68 changes: 33 additions & 35 deletions apps/files_versions/src/utils/versions.js
Original file line number Diff line number Diff line change
Expand Up @@ -23,13 +23,13 @@ import { getCurrentUser } from '@nextcloud/auth'
import client from '../utils/davClient.js'
import davRequest from '../utils/davRequest.js'
import logger from '../utils/logger.js'
import { basename, joinPaths } from '@nextcloud/paths'
import { joinPaths } from '@nextcloud/paths'
import { generateUrl } from '@nextcloud/router'
import { translate } from '@nextcloud/l10n'
import moment from '@nextcloud/moment'

/**
* @typedef {object} Version
* @property {string} fileId - The id of the file associated to the version.
* @property {string} title - 'Current version' or ''
* @property {string} fileName - File name relative to the version DAV endpoint
* @property {string} mimeType - Empty for the current version, else the actual mime type of the version
Expand All @@ -39,7 +39,6 @@ import moment from '@nextcloud/moment'
* @property {string} preview - Preview URL of the version
* @property {string} url - Download URL of the version
* @property {string|null} fileVersion - The version id, null for the current version
* @property {boolean} isCurrent - Whether this is the current version of the file
*/

/**
Expand All @@ -54,7 +53,10 @@ export async function fetchVersions(fileInfo) {
const response = await client.getDirectoryContents(path, {
data: davRequest,
})
return response.map(version => formatVersion(version, fileInfo))
return response
// Filter out root
.filter(({ mime }) => mime !== '')
.map(version => formatVersion(version, fileInfo))
} catch (exception) {
logger.error('Could not fetch version', { exception })
throw exception
Expand All @@ -65,13 +67,12 @@ export async function fetchVersions(fileInfo) {
* Restore the given version
*
* @param {Version} version
* @param {object} fileInfo
*/
export async function restoreVersion(version, fileInfo) {
export async function restoreVersion(version) {
try {
logger.debug('Restoring version', { url: version.url })
await client.moveFile(
`/versions/${getCurrentUser()?.uid}/versions/${fileInfo.id}/${version.fileVersion}`,
`/versions/${getCurrentUser()?.uid}/versions/${version.fileId}/${version.fileVersion}`,
`/versions/${getCurrentUser()?.uid}/restore/target`
)
} catch (exception) {
Expand All @@ -88,37 +89,34 @@ export async function restoreVersion(version, fileInfo) {
* @return {Version}
*/
function formatVersion(version, fileInfo) {
const isCurrent = version.mime === ''
const fileVersion = isCurrent ? null : basename(version.filename)

let url = null
let preview = null

if (isCurrent) {
// https://nextcloud_server2.test/remote.php/webdav/welcome.txt?downloadStartSecret=hl5awd7tbzg
url = joinPaths('/remote.php/webdav', fileInfo.path, fileInfo.name)
preview = generateUrl('/core/preview?fileId={fileId}&c={fileEtag}&x=250&y=250&forceIcon=0&a=0', {
fileId: fileInfo.id,
fileEtag: fileInfo.etag,
})
} else {
url = joinPaths('/remote.php/dav', version.filename)
preview = generateUrl('/apps/files_versions/preview?file={file}&version={fileVersion}', {
file: joinPaths(fileInfo.path, fileInfo.name),
fileVersion,
})
}

return {
title: isCurrent ? translate('files_versions', 'Current version') : '',
fileId: fileInfo.id,
title: '',
fileName: version.filename,
mimeType: version.mime,
size: isCurrent ? fileInfo.size : version.size,
size: version.size,
type: version.type,
mtime: moment(isCurrent ? fileInfo.mtime : version.lastmod).unix(),
preview,
url,
fileVersion,
isCurrent,
mtime: moment(version.lastmod).unix() * 1000,
preview: generateUrl('/apps/files_versions/preview?file={file}&version={fileVersion}', {
file: joinPaths(fileInfo.path, fileInfo.name),
fileVersion: version.basename,
}),
url: joinPaths('/remote.php/dav', version.filename),
fileVersion: version.basename,
}
}

/**
* @param {Version} version
* @param {string} newName
*/
export async function setVersionName(version, newName) {
// await fetch('POST', '/setVersionName')
}

/**
* @param {Version} version
*/
export async function deleteVersion(version) {
// await fetch('DELETE', '/version')
}
Loading

0 comments on commit 3c0ff2c

Please sign in to comment.