Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Preload images #8051

Merged
merged 15 commits into from
Dec 13, 2022
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions changelog/unreleased/enhancement-preview-app-image-preloading
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
Enhancement: Introduce image preloading to the preview app

https://github.com/owncloud/web/pull/8051
https://github.com/owncloud/web/issues/2052
111 changes: 85 additions & 26 deletions packages/web-app-preview/src/App.vue
Original file line number Diff line number Diff line change
Expand Up @@ -210,6 +210,8 @@ export const mimeTypes = () => {
]
}

const PRELOAD_IMAGE_COUNT = 10

export default defineComponent({
name: 'Preview',
components: {
Expand Down Expand Up @@ -319,11 +321,11 @@ export default defineComponent({
},

isActiveFileTypeAudio() {
return this.activeFilteredFile.mimeType.toLowerCase().startsWith('audio')
return this.isFileTypeAudio(this.activeFilteredFile)
},

isActiveFileTypeVideo() {
return this.activeFilteredFile.mimeType.toLowerCase().startsWith('video')
return this.isFileTypeVideo(this.activeFilteredFile)
},
enterFullScreenDescription() {
return this.$gettext('Enter full screen mode')
Expand Down Expand Up @@ -364,6 +366,7 @@ export default defineComponent({
activeIndex(newValue, oldValue) {
if (newValue !== oldValue) {
this.loadMedium()
this.preloadImages()
}

if (oldValue !== null) {
Expand Down Expand Up @@ -409,19 +412,16 @@ export default defineComponent({
this.isFileContentLoading = false
this.isFileContentError = true
},

// react to PopStateEvent ()
handleLocalHistoryEvent() {
const result = this.$router.resolve(document.location)
this.setActiveFile(result.route.params.driveAliasAndItem)
},

handleFullScreenChangeEvent() {
if (document.fullscreenElement === null) {
this.isFullScreenModeActivated = false
}
},

// update route and url
updateLocalHistory() {
const routeOptions = mergeFileRouteOptions(
Expand All @@ -430,7 +430,6 @@ export default defineComponent({
)
history.pushState({}, document.title, this.$router.resolve(routeOptions).href)
},

loadMedium() {
this.isFileContentLoading = true

Expand All @@ -448,7 +447,6 @@ export default defineComponent({

this.loadActiveFileIntoCache()
},

async loadActiveFileIntoCache() {
try {
const loadRawFile = !this.isActiveFileTypeImage
Expand All @@ -459,26 +457,10 @@ export default defineComponent({
this.activeFilteredFile
)
} else {
mediaUrl = await loadPreview({
resource: this.activeFilteredFile,
isPublic: this.isPublicLinkContext,
server: configurationManager.serverUrl,
userId: this.user.id,
token: this.accessToken,
dimensions: [this.thumbDimensions, this.thumbDimensions] as [number, number]
})
mediaUrl = await this.loadPreview(this.activeFilteredFile)
}

this.cachedFiles.push({
id: this.activeFilteredFile.id,
name: this.activeFilteredFile.name,
url: mediaUrl,
ext: this.activeFilteredFile.extension,
mimeType: this.activeFilteredFile.mimeType,
isVideo: this.isActiveFileTypeVideo,
isImage: this.isActiveFileTypeImage,
isAudio: this.isActiveFileTypeAudio
})
this.addPreviewToCache(this.activeFilteredFile, mediaUrl)
this.isFileContentLoading = false
this.isFileContentError = false
} catch (e) {
Expand All @@ -487,7 +469,6 @@ export default defineComponent({
console.error(e)
}
},

triggerActiveFileDownload() {
if (this.isFileContentLoading) {
return
Expand Down Expand Up @@ -543,6 +524,84 @@ export default defineComponent({
imageRotateRight() {
this.currentImageRotation =
this.currentImageRotation === 270 ? 0 : this.currentImageRotation + 90
},
isFileTypeImage(file) {
return !this.isFileTypeAudio(file) && !this.isFileTypeVideo(file)
},
isFileTypeAudio(file) {
return file.mimeType.toLowerCase().startsWith('audio')
},

isFileTypeVideo(file) {
return file.mimeType.toLowerCase().startsWith('video')
},
addPreviewToCache(file, url) {
this.cachedFiles.push({
id: file.id,
name: file.name,
url,
ext: file.extension,
mimeType: file.mimeType,
isVideo: this.isFileTypeVideo(file),
isImage: this.isFileTypeImage(file),
isAudio: this.isFileTypeAudio(file)
})
},
loadPreview(file) {
return loadPreview({
resource: file,
isPublic: this.isPublicLinkContext,
server: configurationManager.serverUrl,
userId: this.user.id,
token: this.accessToken,
dimensions: [this.thumbDimensions, this.thumbDimensions] as [number, number]
})
},
preloadImages() {
let toPreloadFiles = []
AlexAndBear marked this conversation as resolved.
Show resolved Hide resolved

const loadPreviewAsync = (file) => {
toPreloadFiles.push(file.id)
this.loadPreview(file)
.then((mediaUrl) => {
this.addPreviewToCache(file, mediaUrl)
})
.catch((e) => {
console.error(e)
toPreloadFiles = toPreloadFiles.filter((fileId) => fileId !== file.id)
})
}

const preloadFile = (preloadFileIndex) => {
let cycleIndex =
(((this.activeIndex + preloadFileIndex) % this.filteredFiles.length) +
this.filteredFiles.length) %
this.filteredFiles.length

const file = this.filteredFiles[cycleIndex]

if (!this.isFileTypeImage(file) || toPreloadFiles.includes(file.id)) {
return
}

loadPreviewAsync(file)
}

for (
let followingFileIndex = 1;
followingFileIndex <= PRELOAD_IMAGE_COUNT;
followingFileIndex++
) {
preloadFile(followingFileIndex)
}

for (
let previousFileIndex = -1;
previousFileIndex >= PRELOAD_IMAGE_COUNT * -1;
previousFileIndex--
) {
preloadFile(previousFileIndex)
}
AlexAndBear marked this conversation as resolved.
Show resolved Hide resolved
}
}
})
Expand Down