Merge pull request #1223 from h3poteto/iss-1222

closes #1222 Read exif and rotate image for all attachment images
This commit is contained in:
AkiraFukushima 2020-01-12 23:39:32 +09:00 committed by GitHub
commit 72e165b38b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 115 additions and 17 deletions

5
package-lock.json generated
View File

@ -6232,6 +6232,11 @@
}
}
},
"blueimp-load-image": {
"version": "2.26.0",
"resolved": "https://registry.npmjs.org/blueimp-load-image/-/blueimp-load-image-2.26.0.tgz",
"integrity": "sha512-pL7SYdK9ZxBFmFiEoh91PIYccU9IzytQ8goIUgc+5RW6vYVyvpGRUltefceeoNyLARbF8QXiYimHp6kgjZ/IBg=="
},
"bn.js": {
"version": "4.11.8",
"resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz",

View File

@ -170,6 +170,7 @@
"animate.css": "^3.7.2",
"auto-launch": "^5.0.5",
"axios": "^0.19.1",
"blueimp-load-image": "^2.26.0",
"boom": "^7.3.0",
"deep-extend": "^0.6.0",
"electron-context-menu": "^0.15.1",

View File

@ -2,12 +2,13 @@
<div id="current-media" v-loading="loading" element-loading-background="rgba(0, 0, 0, 0.8)">
<video :src="src" v-if="isMovieFile()" autoplay loop controls v-on:loadstart="loaded()"></video>
<video :src="src" v-else-if="isGIF()" autoplay loop v-on:loadstart="loaded()"></video>
<img :src="src" v-else v-on:load="loaded()" />
<img :src="imageSrc" v-else v-on:load="loaded()" />
</div>
</template>
<script>
import { mapState } from 'vuex'
import exifImageUrl from '@/components/utils/exifImageUrl'
export default {
props: {
@ -20,6 +21,24 @@ export default {
default: ''
}
},
data() {
return {
imageSrc: this.src
}
},
watch: {
src: async function(newSrc, _oldSrc) {
this.imageSrc = newSrc
if (newSrc && !this.isMovieFile() && !this.isGIF()) {
try {
const transformed = await exifImageUrl(newSrc)
this.imageSrc = transformed
} catch (err) {
console.error(err)
}
}
}
},
computed: {
...mapState({
loading: state => state.TimelineSpace.Modals.ImageViewer.loading

View File

@ -1,15 +1,18 @@
<template>
<img
:src="originalSrc"
:title="title"
:alt="alt"
v-on:error="failover"
v-on:load="loading = false"
:class="loading ? 'loading' : ''"
@click="$emit('click')" />
<img
:src="originalSrc"
:title="title"
:alt="alt"
v-on:error="failover"
v-on:load="loading = false"
:class="loading ? 'loading' : ''"
@click="$emit('click')"
/>
</template>
<script>
import exifImageUrl from '@/components/utils/exifImageUrl'
export default {
name: 'FailoverImg',
props: {
@ -25,24 +28,47 @@ export default {
type: String,
default: ''
},
readExif: {
type: Boolean,
default: false
},
failoverSrc: {
type: String,
default: ''
default:
''
}
},
data () {
data() {
return {
loading: true,
originalSrc: this.src
}
},
async mounted() {
if (this.readExif) {
try {
const transformed = await exifImageUrl(this.src)
this.originalSrc = transformed
} catch (err) {
console.error(err)
}
}
},
watch: {
src: function (newSrc, _oldSrc) {
src: async function(newSrc, _oldSrc) {
this.originalSrc = newSrc
if (this.readExif) {
try {
const transformed = await exifImageUrl(newSrc)
this.originalSrc = transformed
} catch (err) {
console.error(err)
}
}
}
},
methods: {
failover () {
failover() {
this.originalSrc = this.failoverSrc
}
}
@ -55,7 +81,7 @@ img {
min-height: 1em;
&.loading {
background-image: url("../../assets/images/loading-spinner.svg");
background-image: url('../../assets/images/loading-spinner.svg');
background-position: center center;
background-repeat: no-repeat;
background-size: contain;

View File

@ -83,7 +83,7 @@
<icon name="eye" class="hide"></icon>
</el-button>
<div class="media" v-bind:key="media.preview_url" v-for="media in mediaAttachments(message.status)">
<FailoverImg :src="media.preview_url" :title="media.description" />
<FailoverImg :src="media.preview_url" :title="media.description" :readExif="true" />
<el-tag class="media-label" size="mini" v-if="media.type == 'gifv'">GIF</el-tag>
<el-tag class="media-label" size="mini" v-else-if="media.type == 'video'">VIDEO</el-tag>
</div>

View File

@ -85,7 +85,7 @@
<icon name="eye" class="hide"></icon>
</el-button>
<div class="media" v-bind:key="media.preview_url" v-for="media in mediaAttachments(message.status)">
<FailoverImg :src="media.preview_url" :title="media.description" />
<FailoverImg :src="media.preview_url" :title="media.description" :readExif="true" />
<el-tag class="media-label" size="mini" v-if="media.type == 'gifv'">GIF</el-tag>
<el-tag class="media-label" size="mini" v-else-if="media.type == 'video'">VIDEO</el-tag>
</div>

View File

@ -77,7 +77,12 @@
<icon name="eye" class="hide"></icon>
</el-button>
<div class="media" v-bind:key="media.preview_url" v-for="media in mediaAttachments">
<FailoverImg :src="media.preview_url" @click="openImage(media.url, mediaAttachments)" :title="media.description" />
<FailoverImg
:src="media.preview_url"
@click="openImage(media.url, mediaAttachments)"
:title="media.description"
:readExif="true"
/>
<el-tag class="media-label" size="mini" v-if="media.type == 'gifv'">GIF</el-tag>
<el-tag class="media-label" size="mini" v-else-if="media.type == 'video'">VIDEO</el-tag>
</div>

View File

@ -0,0 +1,42 @@
import loadImage from 'blueimp-load-image'
const parseExtension = url => {
if (!url) {
return null
}
if (url.match(/\.jpg$/) || url.match(/\.jpeg$/) || url.match(/\.JPG$/) || url.match(/\.JPEG$/)) {
return 'image/jpeg'
} else if (url.match(/\.png$/) || url.match(/\.PNG$/)) {
return 'image/png'
} else if (url.match(/\.gif$/) || url.match(/\.GIF$/)) {
return 'image/gif'
} else if (url.match(/\.webp$/) || url.match(/\.WEBP$/)) {
return 'image/webp'
} else if (url.match(/\.svg$/) || url.match(/\.SVG$/)) {
return 'image/svg+xml'
}
return null
}
const exifImageUrl = url => {
return new Promise((resolve, reject) => {
const extension = parseExtension(url)
if (!extension) {
reject(Error(`url is not image: ${url}`))
}
loadImage(
url,
canvas => {
const data = canvas.toDataURL(extension)
resolve(data)
},
{
canvas: true,
meta: true,
orientation: true
}
)
})
}
export default exifImageUrl