pleroma-fe/src/components/media_modal/media_modal.js

182 lines
5 KiB
JavaScript
Raw Normal View History

import StillImage from '../still-image/still-image.vue'
2019-01-26 15:45:03 +00:00
import VideoAttachment from '../video_attachment/video_attachment.vue'
import Modal from '../modal/modal.vue'
import PinchZoom from '../pinch_zoom/pinch_zoom.vue'
import fileTypeService from '../../services/file_type/file_type.service.js'
import GestureService from '../../services/gesture_service/gesture_service'
import Flash from 'src/components/flash/flash.vue'
2021-08-01 23:46:27 +00:00
import Vuex from 'vuex'
import { library } from '@fortawesome/fontawesome-svg-core'
import {
faChevronLeft,
2021-08-15 22:02:56 +00:00
faChevronRight,
faCircleNotch
} from '@fortawesome/free-solid-svg-icons'
library.add(
faChevronLeft,
2021-08-15 22:02:56 +00:00
faChevronRight,
faCircleNotch
)
2021-08-01 23:46:27 +00:00
const onlyXAxis = ([x, y]) => [x, 0]
2021-08-02 03:42:34 +00:00
const SCALING_RESET_MIN = 1.1
const SCALING_ENABLE_MOVE_THRESHOLD = 1
2021-08-01 23:46:27 +00:00
const MediaModal = {
components: {
2019-01-26 15:45:03 +00:00
StillImage,
VideoAttachment,
PinchZoom,
Modal,
Flash
},
2021-08-15 22:02:56 +00:00
data () {
return {
loading: false,
pinchZoomOptions: {}
2021-08-15 22:02:56 +00:00
}
},
computed: {
showing () {
return this.$store.state.mediaViewer.activated
},
media () {
return this.$store.state.mediaViewer.media
},
2021-08-15 16:45:48 +00:00
description () {
return this.currentMedia.description
},
currentIndex () {
return this.$store.state.mediaViewer.currentIndex
},
currentMedia () {
return this.media[this.currentIndex]
},
canNavigate () {
return this.media.length > 1
},
type () {
2021-08-15 22:02:56 +00:00
return this.currentMedia ? this.getType(this.currentMedia) : null
2021-08-01 23:46:27 +00:00
},
scaling () {
return this.$store.state.mediaViewer.swipeScaler.scaling
},
offsets () {
return this.$store.state.mediaViewer.swipeScaler.offsets
},
transform () {
2021-08-02 01:39:07 +00:00
return `translate(${this.offsets[0]}px, ${this.offsets[1]}px) scale(${this.scaling}, ${this.scaling})`
}
},
created () {
this.mediaGesture = new GestureService.SwipeAndScaleGesture({
direction: GestureService.DIRECTION_LEFT,
callbackPositive: this.goNext,
callbackNegative: this.goPrev,
2021-08-01 23:46:27 +00:00
swipePreviewCallback: this.handleSwipePreview,
swipeEndCallback: this.handleSwipeEnd,
2021-08-02 01:39:07 +00:00
pinchPreviewCallback: this.handlePinchPreview,
pinchEndCallback: this.handlePinchEnd,
threshold: 50
})
},
methods: {
2021-08-15 22:02:56 +00:00
getType (media) {
return fileTypeService.fileType(media.mimetype)
},
mediaTouchStart (e) {
this.mediaGesture.start(e)
},
mediaTouchMove (e) {
this.mediaGesture.move(e)
},
mediaTouchEnd (e) {
this.mediaGesture.end(e)
},
hide () {
this.$store.dispatch('closeMediaViewer')
},
goPrev () {
if (this.canNavigate) {
const prevIndex = this.currentIndex === 0 ? this.media.length - 1 : (this.currentIndex - 1)
2021-08-15 22:02:56 +00:00
const newMedia = this.media[prevIndex]
if (this.getType(newMedia) === 'image') {
this.loading = true
}
this.$store.dispatch('setCurrentMedia', newMedia)
}
},
goNext () {
if (this.canNavigate) {
const nextIndex = this.currentIndex === this.media.length - 1 ? 0 : (this.currentIndex + 1)
2021-08-15 22:02:56 +00:00
const newMedia = this.media[nextIndex]
if (this.getType(newMedia) === 'image') {
this.loading = true
}
this.$store.dispatch('setCurrentMedia', newMedia)
}
},
2021-08-15 22:02:56 +00:00
onImageLoaded () {
this.loading = false
},
2021-08-01 23:46:27 +00:00
handleSwipePreview (offsets) {
2021-08-02 03:42:34 +00:00
this.$store.dispatch('swipeScaler/apply', {
offsets: this.scaling > SCALING_ENABLE_MOVE_THRESHOLD ? offsets : onlyXAxis(offsets)
})
2021-08-01 23:46:27 +00:00
},
handleSwipeEnd (sign) {
2021-08-02 03:42:34 +00:00
if (this.scaling > SCALING_ENABLE_MOVE_THRESHOLD) {
this.$store.dispatch('swipeScaler/finish')
return
}
2021-08-01 23:46:27 +00:00
if (sign === 0) {
2021-08-02 03:42:34 +00:00
this.$store.dispatch('swipeScaler/reset')
2021-08-01 23:46:27 +00:00
} else if (sign > 0) {
this.goNext()
} else {
this.goPrev()
}
},
2021-08-02 01:39:07 +00:00
handlePinchPreview (offsets, scaling) {
console.log('handle pinch preview:', offsets, scaling)
this.$store.dispatch('swipeScaler/apply', { offsets, scaling })
},
handlePinchEnd () {
2021-08-02 03:42:34 +00:00
if (this.scaling > SCALING_RESET_MIN) {
this.$store.dispatch('swipeScaler/finish')
} else {
this.$store.dispatch('swipeScaler/reset')
}
2021-08-02 01:39:07 +00:00
},
handleKeyupEvent (e) {
if (this.showing && e.keyCode === 27) { // escape
this.hide()
}
},
handleKeydownEvent (e) {
if (!this.showing) {
return
}
if (e.keyCode === 39) { // arrow right
this.goNext()
} else if (e.keyCode === 37) { // arrow left
this.goPrev()
}
}
},
mounted () {
window.addEventListener('popstate', this.hide)
document.addEventListener('keyup', this.handleKeyupEvent)
document.addEventListener('keydown', this.handleKeydownEvent)
},
destroyed () {
window.removeEventListener('popstate', this.hide)
document.removeEventListener('keyup', this.handleKeyupEvent)
document.removeEventListener('keydown', this.handleKeydownEvent)
}
}
export default MediaModal