misskey/src/client/app/common/views/components/messaging-room.vue

441 lines
9.9 KiB
Vue
Raw Normal View History

2018-02-13 06:17:59 +00:00
<template>
2018-02-26 19:36:16 +00:00
<div class="mk-messaging-room"
@dragover.prevent.stop="onDragover"
@drop.prevent.stop="onDrop"
>
2018-09-01 00:29:59 +00:00
<div class="body">
2019-05-18 11:36:33 +00:00
<p class="init" v-if="init"><fa icon="spinner" pulse fixed-width/>{{ $t('@.loading') }}</p>
<p class="empty" v-if="!init && messages.length == 0"><fa icon="info-circle"/>{{ user ? $t('not-talked-user') : $t('not-talked-group') }}</p>
2019-04-18 12:34:56 +00:00
<p class="no-history" v-if="!init && messages.length > 0 && !existMoreMessages"><fa :icon="faFlag"/>{{ $t('no-history') }}</p>
2018-02-18 15:18:01 +00:00
<button class="more" :class="{ fetching: fetchingMoreMessages }" v-if="existMoreMessages" @click="fetchMoreMessages" :disabled="fetchingMoreMessages">
2018-11-13 13:45:28 +00:00
<template v-if="fetchingMoreMessages"><fa icon="spinner" pulse fixed-width/></template>{{ fetchingMoreMessages ? $t('@.loading') : $t('@.load-more') }}
2018-02-13 06:17:59 +00:00
</button>
2018-02-13 11:53:45 +00:00
<template v-for="(message, i) in _messages">
2019-05-18 11:36:33 +00:00
<x-message :message="message" :key="message.id" :is-group="group != null"/>
2018-02-20 16:39:51 +00:00
<p class="date" v-if="i != messages.length - 1 && message._date != _messages[i + 1]._date">
<span>{{ _messages[i + 1]._datetext }}</span>
</p>
2018-02-13 06:17:59 +00:00
</template>
</div>
<footer>
2018-05-23 19:55:29 +00:00
<transition name="fade">
<div class="new-message" v-show="showIndicator">
2018-11-14 16:09:50 +00:00
<button @click="onIndicatorClick"><i><fa :icon="faArrowCircleDown"/></i>{{ $t('new-message') }}</button>
2018-05-23 19:55:29 +00:00
</div>
</transition>
2019-05-18 11:36:33 +00:00
<x-form :user="user" :group="group" ref="form"/>
2018-02-13 06:17:59 +00:00
</footer>
</div>
</template>
<script lang="ts">
import Vue from 'vue';
import i18n from '../../../i18n';
2018-02-20 16:39:51 +00:00
import XMessage from './messaging-room.message.vue';
import XForm from './messaging-room.form.vue';
2018-03-04 09:50:30 +00:00
import { url } from '../../../config';
2019-05-18 11:36:33 +00:00
import { faArrowCircleDown, faFlag } from '@fortawesome/free-solid-svg-icons';
2018-02-13 06:17:59 +00:00
export default Vue.extend({
i18n: i18n('common/views/components/messaging-room.vue'),
2019-05-18 11:36:33 +00:00
2018-02-20 16:39:51 +00:00
components: {
XMessage,
XForm
},
2018-02-26 19:36:16 +00:00
2019-05-18 11:36:33 +00:00
props: {
user: {
type: Object,
requird: false,
},
group: {
type: Object,
requird: false,
},
isNaked: {
type: Boolean,
requird: false,
},
},
2018-02-26 19:36:16 +00:00
2018-02-13 06:17:59 +00:00
data() {
return {
init: true,
fetchingMoreMessages: false,
messages: [],
existMoreMessages: false,
2018-05-23 19:55:29 +00:00
connection: null,
showIndicator: false,
2018-11-14 16:09:50 +00:00
timer: null,
2019-04-18 12:34:56 +00:00
faArrowCircleDown, faFlag
2018-02-13 06:17:59 +00:00
};
},
2018-02-26 19:36:16 +00:00
2018-02-13 06:17:59 +00:00
computed: {
_messages(): any[] {
return (this.messages as any).map(message => {
2018-03-29 05:48:47 +00:00
const date = new Date(message.createdAt).getDate();
const month = new Date(message.createdAt).getMonth() + 1;
2018-02-13 06:17:59 +00:00
message._date = date;
message._datetext = this.$t('@.month-and-day').replace('{month}', month.toString()).replace('{day}', date.toString());
2018-02-13 06:17:59 +00:00
return message;
});
2018-02-26 19:36:16 +00:00
},
form(): any {
return this.$refs.form;
2018-02-13 06:17:59 +00:00
}
},
mounted() {
2019-05-18 11:36:33 +00:00
this.connection = this.$root.stream.connectToChannel('messaging', {
otherparty: this.user ? this.user.id : undefined,
group: this.group ? this.group.id : undefined,
});
2018-02-13 06:17:59 +00:00
this.connection.on('message', this.onMessage);
this.connection.on('read', this.onRead);
this.connection.on('deleted', this.onDeleted);
2018-02-13 06:17:59 +00:00
2018-09-01 00:29:59 +00:00
if (this.isNaked) {
window.addEventListener('scroll', this.onScroll, { passive: true });
} else {
this.$el.addEventListener('scroll', this.onScroll, { passive: true });
}
2018-02-13 06:17:59 +00:00
document.addEventListener('visibilitychange', this.onVisibilitychange);
this.fetchMessages().then(() => {
this.init = false;
this.scrollToBottom();
});
},
2018-02-26 19:36:16 +00:00
2018-02-13 06:17:59 +00:00
beforeDestroy() {
this.connection.dispose();
2018-02-13 06:17:59 +00:00
2018-09-01 00:29:59 +00:00
if (this.isNaked) {
window.removeEventListener('scroll', this.onScroll);
} else {
this.$el.removeEventListener('scroll', this.onScroll);
}
2018-02-13 06:17:59 +00:00
document.removeEventListener('visibilitychange', this.onVisibilitychange);
},
2018-02-26 19:36:16 +00:00
2018-02-13 06:17:59 +00:00
methods: {
2018-02-26 19:36:16 +00:00
onDragover(e) {
2018-02-26 21:25:17 +00:00
const isFile = e.dataTransfer.items[0].kind == 'file';
const isDriveFile = e.dataTransfer.types[0] == 'mk_drive_file';
if (isFile || isDriveFile) {
e.dataTransfer.dropEffect = e.dataTransfer.effectAllowed == 'all' ? 'copy' : 'move';
} else {
e.dataTransfer.dropEffect = 'none';
}
2018-02-26 19:36:16 +00:00
},
onDrop(e): void {
// ファイルだったら
if (e.dataTransfer.files.length == 1) {
this.form.upload(e.dataTransfer.files[0]);
return;
} else if (e.dataTransfer.files.length > 1) {
this.$root.dialog({
type: 'error',
text: this.$t('only-one-file-attached')
});
2018-02-26 19:36:16 +00:00
return;
}
2018-02-26 21:25:17 +00:00
//#region ドライブのファイル
const driveFile = e.dataTransfer.getData('mk_drive_file');
if (driveFile != null && driveFile != '') {
const file = JSON.parse(driveFile);
this.form.file = file;
2018-02-26 19:36:16 +00:00
}
2018-02-26 21:25:17 +00:00
//#endregion
2018-02-26 19:36:16 +00:00
},
2018-02-13 06:17:59 +00:00
fetchMessages() {
return new Promise((resolve, reject) => {
const max = this.existMoreMessages ? 20 : 10;
2018-11-08 23:13:34 +00:00
this.$root.api('messaging/messages', {
2019-05-18 11:36:33 +00:00
userId: this.user ? this.user.id : undefined,
groupId: this.group ? this.group.id : undefined,
2018-02-13 06:17:59 +00:00
limit: max + 1,
2018-03-29 05:48:47 +00:00
untilId: this.existMoreMessages ? this.messages[0].id : undefined
2018-02-13 06:17:59 +00:00
}).then(messages => {
if (messages.length == max + 1) {
this.existMoreMessages = true;
messages.pop();
} else {
this.existMoreMessages = false;
}
this.messages.unshift.apply(this.messages, messages.reverse());
resolve();
});
});
},
2018-02-26 19:36:16 +00:00
2018-02-13 06:17:59 +00:00
fetchMoreMessages() {
this.fetchingMoreMessages = true;
this.fetchMessages().then(() => {
this.fetchingMoreMessages = false;
});
},
2018-02-26 19:36:16 +00:00
2018-02-13 06:17:59 +00:00
onMessage(message) {
2018-03-04 09:50:30 +00:00
// サウンドを再生する
2018-05-20 17:13:39 +00:00
if (this.$store.state.device.enableSounds) {
2018-03-06 09:06:45 +00:00
const sound = new Audio(`${url}/assets/message.mp3`);
2018-05-20 17:13:39 +00:00
sound.volume = this.$store.state.device.soundVolume;
2018-03-06 09:06:45 +00:00
sound.play();
2018-03-04 09:50:30 +00:00
}
2018-02-13 06:17:59 +00:00
const isBottom = this.isBottom();
this.messages.push(message);
2018-05-27 04:49:09 +00:00
if (message.userId != this.$store.state.i.id && !document.hidden) {
2018-10-08 16:50:49 +00:00
this.connection.send('read', {
2018-02-13 06:17:59 +00:00
id: message.id
});
}
if (isBottom) {
// Scroll to bottom
2018-02-22 19:01:22 +00:00
this.$nextTick(() => {
this.scrollToBottom();
});
2018-05-27 04:49:09 +00:00
} else if (message.userId != this.$store.state.i.id) {
2018-02-13 06:17:59 +00:00
// Notify
2018-05-23 19:55:29 +00:00
this.notifyNewMessage();
2018-02-13 06:17:59 +00:00
}
},
2018-02-26 19:36:16 +00:00
2019-05-18 11:36:33 +00:00
onRead(x) {
if (this.user) {
if (!Array.isArray(x)) x = [x];
for (const id of x) {
if (this.messages.some(x => x.id == id)) {
const exist = this.messages.map(x => x.id).indexOf(id);
this.messages[exist].isRead = true;
}
}
} else if (this.group) {
for (const id of x.ids) {
if (this.messages.some(x => x.id == id)) {
const exist = this.messages.map(x => x.id).indexOf(id);
this.messages[exist].reads.push(x.userId);
}
2018-02-13 06:17:59 +00:00
}
}
2018-02-13 06:17:59 +00:00
},
2018-02-26 19:36:16 +00:00
onDeleted(id) {
const msg = this.messages.find(m => m.id === id);
if (msg) {
this.messages = this.messages.filter(m => m.id !== msg.id);
}
},
2018-02-13 06:17:59 +00:00
isBottom() {
2018-02-22 19:01:22 +00:00
const asobi = 64;
2018-02-13 06:17:59 +00:00
const current = this.isNaked
? window.scrollY + window.innerHeight
: this.$el.scrollTop + this.$el.offsetHeight;
const max = this.isNaked
? document.body.offsetHeight
: this.$el.scrollHeight;
return current > (max - asobi);
},
2018-02-26 19:36:16 +00:00
2018-02-13 06:17:59 +00:00
scrollToBottom() {
if (this.isNaked) {
window.scroll(0, document.body.offsetHeight);
} else {
this.$el.scrollTop = this.$el.scrollHeight;
}
},
2018-02-26 19:36:16 +00:00
2018-05-23 19:55:29 +00:00
onIndicatorClick() {
this.showIndicator = false;
this.scrollToBottom();
},
notifyNewMessage() {
this.showIndicator = true;
if (this.timer) clearTimeout(this.timer);
this.timer = setTimeout(() => {
this.showIndicator = false;
2018-02-13 06:17:59 +00:00
}, 4000);
},
2018-02-26 19:36:16 +00:00
2018-09-01 00:29:59 +00:00
onScroll() {
const el = this.isNaked ? window.document.documentElement : this.$el;
const current = el.scrollTop + el.clientHeight;
if (current > el.scrollHeight - 1) {
this.showIndicator = false;
}
},
2018-02-13 06:17:59 +00:00
onVisibilitychange() {
if (document.hidden) return;
for (const message of this.messages) {
2018-05-27 04:49:09 +00:00
if (message.userId !== this.$store.state.i.id && !message.isRead) {
2018-10-08 16:50:49 +00:00
this.connection.send('read', {
2018-02-13 06:17:59 +00:00
id: message.id
});
}
}
2018-02-13 06:17:59 +00:00
}
}
});
</script>
<style lang="stylus" scoped>
2018-09-27 12:43:11 +00:00
.mk-messaging-room
2018-02-22 19:01:22 +00:00
display flex
flex 1
flex-direction column
height 100%
2018-09-27 12:43:11 +00:00
background var(--messagingRoomBg)
2018-02-22 19:01:22 +00:00
2018-09-01 00:29:59 +00:00
> .body
2018-02-22 19:01:22 +00:00
width 100%
2018-02-13 06:17:59 +00:00
max-width 600px
margin 0 auto
flex 1
2018-02-13 06:17:59 +00:00
2018-09-27 12:43:11 +00:00
> .init,
2018-02-13 06:17:59 +00:00
> .empty
width 100%
margin 0
padding 16px 8px 8px 8px
text-align center
font-size 0.8em
2018-09-27 12:43:11 +00:00
color var(--messagingRoomInfo)
opacity 0.5
2018-02-13 06:17:59 +00:00
[data-icon]
2018-02-13 06:17:59 +00:00
margin-right 4px
> .no-history
display block
margin 0
padding 16px
text-align center
font-size 0.8em
2018-09-27 12:43:11 +00:00
color var(--messagingRoomInfo)
opacity 0.5
2018-02-13 06:17:59 +00:00
[data-icon]
2018-02-13 06:17:59 +00:00
margin-right 4px
> .more
display block
margin 16px auto
padding 0 12px
line-height 24px
color #fff
2018-04-28 23:51:17 +00:00
background rgba(#000, 0.3)
2018-02-13 06:17:59 +00:00
border-radius 12px
&:hover
2018-04-28 23:51:17 +00:00
background rgba(#000, 0.4)
2018-02-13 06:17:59 +00:00
&:active
2018-04-28 23:51:17 +00:00
background rgba(#000, 0.5)
2018-02-13 06:17:59 +00:00
&.fetching
cursor wait
> [data-icon]
2018-02-13 06:17:59 +00:00
margin-right 4px
> .message
// something
> .date
display block
margin 8px 0
text-align center
&:before
content ''
display block
position absolute
height 1px
width 90%
top 16px
left 0
right 0
margin 0 auto
2018-09-27 12:43:11 +00:00
background var(--messagingRoomDateDividerLine)
2018-02-13 06:17:59 +00:00
> span
display inline-block
margin 0
padding 0 16px
//font-weight bold
line-height 32px
2018-09-27 12:43:11 +00:00
color var(--messagingRoomDateDividerText)
background var(--messagingRoomBg)
2018-02-13 06:17:59 +00:00
> footer
position -webkit-sticky
position sticky
z-index 2
bottom 0
2018-02-13 06:17:59 +00:00
width 100%
max-width 600px
margin 0 auto
padding 0
2018-10-13 09:08:30 +00:00
background var(--messagingRoomBg)
2018-02-13 06:17:59 +00:00
background-clip content-box
2018-05-23 19:55:29 +00:00
> .new-message
2018-02-13 06:17:59 +00:00
position absolute
top -48px
width 100%
padding 8px 0
text-align center
2018-05-23 19:55:29 +00:00
> button
2018-02-13 06:17:59 +00:00
display inline-block
margin 0
2018-05-23 19:56:58 +00:00
padding 0 12px 0 30px
2018-02-13 06:17:59 +00:00
cursor pointer
line-height 32px
font-size 12px
2018-09-26 11:19:35 +00:00
color var(--primaryForeground)
background var(--primary)
2018-02-13 06:17:59 +00:00
border-radius 16px
2018-05-23 19:55:29 +00:00
&:hover
2018-09-26 11:19:35 +00:00
background var(--primaryLighten10)
2018-05-23 19:55:29 +00:00
&:active
2018-09-26 11:19:35 +00:00
background var(--primaryDarken10)
2018-02-13 06:17:59 +00:00
> i
2018-02-13 06:17:59 +00:00
position absolute
top 0
left 10px
line-height 32px
font-size 16px
2018-05-23 19:55:29 +00:00
.fade-enter-active, .fade-leave-active
transition opacity 0.1s
.fade-enter, .fade-leave-to
transition opacity 0.5s
opacity 0
2018-02-13 06:17:59 +00:00
</style>