misskey-awawa/src/client/app/desktop/views/components/drive.file.vue

318 lines
6.2 KiB
Vue
Raw Normal View History

2018-02-14 10:03:48 +00:00
<template>
2018-02-22 22:56:35 +00:00
<div class="root file"
2018-02-14 10:03:48 +00:00
:data-is-selected="isSelected"
:data-is-contextmenu-showing="isContextmenuShowing"
@click="onClick"
draggable="true"
@dragstart="onDragstart"
@dragend="onDragend"
2018-02-18 03:35:18 +00:00
@contextmenu.prevent.stop="onContextmenu"
2018-02-14 10:03:48 +00:00
:title="title"
>
2018-03-29 05:48:47 +00:00
<div class="label" v-if="os.i.avatarId == file.id"><img src="/assets/label.svg"/>
2018-04-14 16:04:40 +00:00
<p>%i18n:@avatar%</p>
2018-02-14 10:03:48 +00:00
</div>
2018-03-29 05:48:47 +00:00
<div class="label" v-if="os.i.bannerId == file.id"><img src="/assets/label.svg"/>
2018-04-14 16:04:40 +00:00
<p>%i18n:@banner%</p>
2018-02-14 10:03:48 +00:00
</div>
2018-02-18 03:35:18 +00:00
<div class="thumbnail" ref="thumbnail" :style="`background-color: ${ background }`">
<img :src="`${file.url}?thumbnail&size=128`" alt="" @load="onThumbnailLoaded"/>
2018-02-14 10:03:48 +00:00
</div>
<p class="name">
2018-02-18 03:35:18 +00:00
<span>{{ file.name.lastIndexOf('.') != -1 ? file.name.substr(0, file.name.lastIndexOf('.')) : file.name }}</span>
<span class="ext" v-if="file.name.lastIndexOf('.') != -1">{{ file.name.substr(file.name.lastIndexOf('.')) }}</span>
2018-02-14 10:03:48 +00:00
</p>
</div>
</template>
<script lang="ts">
import Vue from 'vue';
import * as anime from 'animejs';
2018-02-18 03:35:18 +00:00
import contextmenu from '../../api/contextmenu';
import copyToClipboard from '../../../common/scripts/copy-to-clipboard';
2018-02-14 10:03:48 +00:00
export default Vue.extend({
2018-02-18 03:35:18 +00:00
props: ['file'],
2018-02-14 10:03:48 +00:00
data() {
return {
isContextmenuShowing: false,
isDragging: false
};
},
computed: {
2018-02-18 03:35:18 +00:00
browser(): any {
return this.$parent;
},
2018-02-14 10:03:48 +00:00
isSelected(): boolean {
return this.browser.selectedFiles.some(f => f.id == this.file.id);
},
title(): string {
2018-02-19 23:05:41 +00:00
return `${this.file.name}\n${this.file.type} ${Vue.filter('bytes')(this.file.datasize)}`;
2018-02-18 03:35:18 +00:00
},
background(): string {
2018-03-29 05:48:47 +00:00
return this.file.properties.avgColor
? `rgb(${this.file.properties.avgColor.join(',')})`
2018-02-18 03:35:18 +00:00
: 'transparent';
2018-02-14 10:03:48 +00:00
}
},
methods: {
onClick() {
this.browser.chooseFile(this.file);
},
onContextmenu(e) {
this.isContextmenuShowing = true;
2018-02-18 03:35:18 +00:00
contextmenu(e, [{
type: 'item',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.rename%',
2018-02-18 03:35:18 +00:00
icon: '%fa:i-cursor%',
onClick: this.rename
}, {
type: 'item',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.copy-url%',
2018-02-18 03:35:18 +00:00
icon: '%fa:link%',
onClick: this.copyUrl
}, {
type: 'link',
href: `${this.file.url}?download`,
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.download%',
2018-02-18 03:35:18 +00:00
icon: '%fa:download%',
}, {
type: 'divider',
}, {
type: 'item',
2018-04-15 22:07:32 +00:00
text: '%i18n:!common.delete%',
2018-02-18 03:35:18 +00:00
icon: '%fa:R trash-alt%',
onClick: this.deleteFile
}, {
type: 'divider',
}, {
type: 'nest',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.else-files%',
2018-02-18 03:35:18 +00:00
menu: [{
type: 'item',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.set-as-avatar%',
2018-02-18 03:35:18 +00:00
onClick: this.setAsAvatar
}, {
type: 'item',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.set-as-banner%',
2018-02-18 03:35:18 +00:00
onClick: this.setAsBanner
}]
}, {
type: 'nest',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.open-in-app%',
2018-02-18 03:35:18 +00:00
menu: [{
type: 'item',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.add-app%...',
2018-02-18 03:35:18 +00:00
onClick: this.addApp
}]
}], {
closed: () => {
this.isContextmenuShowing = false;
2018-02-14 10:03:48 +00:00
}
});
},
onDragstart(e) {
e.dataTransfer.effectAllowed = 'move';
2018-02-26 21:25:17 +00:00
e.dataTransfer.setData('mk_drive_file', JSON.stringify(this.file));
2018-02-14 10:03:48 +00:00
this.isDragging = true;
// 親ブラウザに対して、ドラッグが開始されたフラグを立てる
// (=あなたの子供が、ドラッグを開始しましたよ)
this.browser.isDragSource = true;
},
onDragend(e) {
this.isDragging = false;
this.browser.isDragSource = false;
},
onThumbnailLoaded() {
2018-03-29 05:48:47 +00:00
if (this.file.properties.avgColor) {
2018-02-14 10:03:48 +00:00
anime({
targets: this.$refs.thumbnail,
2018-03-29 05:48:47 +00:00
backgroundColor: `rgba(${this.file.properties.avgColor.join(',')}, 0)`,
2018-02-14 10:03:48 +00:00
duration: 100,
easing: 'linear'
});
}
2018-02-18 03:35:18 +00:00
},
rename() {
(this as any).apis.input({
2018-04-15 22:07:32 +00:00
title: '%i18n:!@contextmenu.rename-file%',
placeholder: '%i18n:!@contextmenu.input-new-file-name%',
2018-02-18 13:14:51 +00:00
default: this.file.name,
allowEmpty: false
2018-02-18 03:35:18 +00:00
}).then(name => {
(this as any).api('drive/files/update', {
2018-03-29 05:48:47 +00:00
fileId: this.file.id,
2018-02-18 03:35:18 +00:00
name: name
})
});
},
copyUrl() {
copyToClipboard(this.file.url);
(this as any).apis.dialog({
2018-04-14 16:04:40 +00:00
title: '%fa:check%%i18n:@contextmenu.copied%',
2018-04-15 22:07:32 +00:00
text: '%i18n:!@contextmenu.copied-url-to-clipboard%',
2018-02-18 03:35:18 +00:00
actions: [{
2018-04-15 22:07:32 +00:00
text: '%i18n:!common.ok%'
2018-02-18 03:35:18 +00:00
}]
});
},
setAsAvatar() {
(this as any).apis.updateAvatar(this.file);
},
setAsBanner() {
(this as any).apis.updateBanner(this.file);
},
addApp() {
alert('not implemented yet');
},
deleteFile() {
alert('not implemented yet');
2018-02-14 10:03:48 +00:00
}
}
});
</script>
<style lang="stylus" scoped>
2018-03-03 04:47:55 +00:00
@import '~const.styl'
2018-02-22 22:56:35 +00:00
.root.file
2018-02-14 10:03:48 +00:00
padding 8px 0 0 0
height 180px
border-radius 4px
&, *
cursor pointer
&:hover
background rgba(0, 0, 0, 0.05)
> .label
&:before
&:after
background #0b65a5
&:active
background rgba(0, 0, 0, 0.1)
> .label
&:before
&:after
background #0b588c
&[data-is-selected]
background $theme-color
&:hover
background lighten($theme-color, 10%)
&:active
background darken($theme-color, 10%)
> .label
&:before
&:after
display none
> .name
color $theme-color-foreground
&[data-is-contextmenu-showing]
&:after
content ""
pointer-events none
position absolute
top -4px
right -4px
bottom -4px
left -4px
border 2px dashed rgba($theme-color, 0.3)
border-radius 4px
> .label
position absolute
top 0
left 0
pointer-events none
&:before
content ""
display block
position absolute
z-index 1
top 0
left 57px
width 28px
height 8px
background #0c7ac9
&:after
content ""
display block
position absolute
z-index 1
top 57px
left 0
width 8px
height 28px
background #0c7ac9
> img
position absolute
z-index 2
top 0
left 0
> p
position absolute
z-index 3
top 19px
left -28px
width 120px
margin 0
text-align center
line-height 28px
color #fff
transform rotate(-45deg)
> .thumbnail
width 128px
height 128px
margin auto
> img
display block
position absolute
top 0
left 0
right 0
bottom 0
margin auto
max-width 128px
max-height 128px
pointer-events none
> .name
display block
margin 4px 0 0 0
font-size 0.8em
text-align center
word-break break-all
color #444
overflow hidden
> .ext
opacity 0.5
</style>