iceshrimp/src/client/pages/messaging/messaging-room.vue

404 lines
9.5 KiB
Vue
Raw Normal View History

2018-02-13 15:17:59 +09:00
<template>
2020-03-20 22:37:44 +09:00
<div class="mk-messaging-room naked"
2018-02-27 04:36:16 +09:00
@dragover.prevent.stop="onDragover"
@drop.prevent.stop="onDrop"
>
<template v-if="!fetching && user">
<portal to="title"><mk-user-name :user="user" :nowrap="false" class="name"/></portal>
<portal to="avatar"><mk-avatar class="avatar" :user="user" :disable-preview="true"/></portal>
</template>
<template v-if="!fetching && group">
2020-02-09 19:31:23 +09:00
<portal to="icon"><fa :icon="faUsers"/></portal>
<portal to="title">{{ group.name }}</portal>
</template>
2018-09-01 09:29:59 +09:00
<div class="body">
<mk-loading v-if="fetching"/>
<p class="empty" v-if="!fetching && messages.length == 0"><fa :icon="faInfoCircle"/>{{ $t('noMessagesYet') }}</p>
<p class="no-history" v-if="!fetching && messages.length > 0 && !existMoreMessages"><fa :icon="faFlag"/>{{ $t('noMoreHistory') }}</p>
<button class="more _button" ref="loadMore" :class="{ fetching: fetchingMoreMessages }" v-show="existMoreMessages" @click="fetchMoreMessages" :disabled="fetchingMoreMessages">
<template v-if="fetchingMoreMessages"><fa icon="spinner" pulse fixed-width/></template>{{ fetchingMoreMessages ? $t('loading') : $t('loadMore') }}
2018-02-13 15:17:59 +09:00
</button>
2020-03-29 10:59:05 +09:00
<x-list class="messages" :items="messages" v-slot="{ item: message }" direction="up" reversed>
2020-02-06 14:23:01 +09:00
<x-message :message="message" :is-group="group != null" :key="message.id"/>
</x-list>
2018-02-13 15:17:59 +09:00
</div>
<footer>
2018-05-24 04:55:29 +09:00
<transition name="fade">
<div class="new-message" v-show="showIndicator">
<button class="_buttonPrimary" @click="onIndicatorClick"><i><fa :icon="faArrowCircleDown"/></i>{{ $t('newMessageExists') }}</button>
2018-05-24 04:55:29 +09:00
</div>
</transition>
<x-form v-if="!fetching" :user="user" :group="group" ref="form"/>
2018-02-13 15:17:59 +09:00
</footer>
</div>
</template>
<script lang="ts">
import Vue from 'vue';
import { faArrowCircleDown, faFlag, faUsers, faInfoCircle } from '@fortawesome/free-solid-svg-icons';
2020-03-22 14:38:33 +09:00
import XList from '../../components/date-separated-list.vue';
2018-02-21 01:39:51 +09:00
import XMessage from './messaging-room.message.vue';
import XForm from './messaging-room.form.vue';
2020-03-22 14:38:33 +09:00
import parseAcct from '../../../misc/acct/parse';
2018-02-13 15:17:59 +09:00
export default Vue.extend({
2018-02-21 01:39:51 +09:00
components: {
XMessage,
XForm,
XList,
2019-05-18 20:36:33 +09:00
},
2018-02-27 04:36:16 +09:00
2018-02-13 15:17:59 +09:00
data() {
return {
fetching: true,
user: null,
group: null,
2018-02-13 15:17:59 +09:00
fetchingMoreMessages: false,
messages: [],
existMoreMessages: false,
2018-05-24 04:55:29 +09:00
connection: null,
showIndicator: false,
2018-11-15 01:09:50 +09:00
timer: null,
ilObserver: new IntersectionObserver(
(entries) => entries.some((entry) => entry.isIntersecting)
&& !this.fetching
&& !this.fetchingMoreMessages
&& this.existMoreMessages
&& this.fetchMoreMessages()
),
faArrowCircleDown, faFlag, faUsers, faInfoCircle
2018-02-13 15:17:59 +09:00
};
},
2018-02-27 04:36:16 +09:00
2018-02-13 15:17:59 +09:00
computed: {
2018-02-27 04:36:16 +09:00
form(): any {
return this.$refs.form;
2018-02-13 15:17:59 +09:00
}
},
watch: {
$route: 'fetch'
},
2018-02-13 15:17:59 +09:00
mounted() {
this.fetch();
if (this.$store.state.device.enableInfiniteScroll) {
this.$nextTick(() => this.ilObserver.observe(this.$refs.loadMore as Element));
}
2018-02-13 15:17:59 +09:00
},
2018-02-27 04:36:16 +09:00
2018-02-13 15:17:59 +09:00
beforeDestroy() {
this.connection.dispose();
2018-02-13 15:17:59 +09:00
window.removeEventListener('scroll', this.onScroll);
2018-09-01 09:29:59 +09:00
2018-02-13 15:17:59 +09:00
document.removeEventListener('visibilitychange', this.onVisibilitychange);
this.ilObserver.disconnect();
2018-02-13 15:17:59 +09:00
},
2018-02-27 04:36:16 +09:00
2018-02-13 15:17:59 +09:00
methods: {
async fetch() {
this.fetching = true;
if (this.$route.params.user) {
const user = await this.$root.api('users/show', parseAcct(this.$route.params.user));
this.user = user;
} else {
const group = await this.$root.api('users/groups/show', { groupId: this.$route.params.group });
this.group = group;
}
this.connection = this.$root.stream.connectToChannel('messaging', {
otherparty: this.user ? this.user.id : undefined,
group: this.group ? this.group.id : undefined,
});
this.connection.on('message', this.onMessage);
this.connection.on('read', this.onRead);
this.connection.on('deleted', this.onDeleted);
window.addEventListener('scroll', this.onScroll, { passive: true });
document.addEventListener('visibilitychange', this.onVisibilitychange);
this.fetchMessages().then(() => {
this.scrollToBottom();
// もっと見るの交差検知を発火させないためにfetchは
// スクロールが終わるまでfalseにしておく
// scrollendのようなイベントはないのでsetTimeoutで
setTimeout(() => this.fetching = false, 300);
});
},
2018-02-27 04:36:16 +09:00
onDragover(e) {
2018-02-27 06:25:17 +09: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-27 04:36:16 +09: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('onlyOneFileCanBeAttached')
});
2018-02-27 04:36:16 +09:00
return;
}
2018-02-27 06:25:17 +09: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-27 04:36:16 +09:00
}
2018-02-27 06:25:17 +09:00
//#endregion
2018-02-27 04:36:16 +09:00
},
2018-02-13 15:17:59 +09:00
fetchMessages() {
return new Promise((resolve, reject) => {
const max = this.existMoreMessages ? 20 : 10;
2018-11-09 08:13:34 +09:00
this.$root.api('messaging/messages', {
2019-05-18 20:36:33 +09:00
userId: this.user ? this.user.id : undefined,
groupId: this.group ? this.group.id : undefined,
2018-02-13 15:17:59 +09:00
limit: max + 1,
2018-03-29 14:48:47 +09:00
untilId: this.existMoreMessages ? this.messages[0].id : undefined
2018-02-13 15:17:59 +09: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-27 04:36:16 +09:00
2018-02-13 15:17:59 +09:00
fetchMoreMessages() {
this.fetchingMoreMessages = true;
this.fetchMessages().then(() => {
this.fetchingMoreMessages = false;
});
},
2018-02-27 04:36:16 +09:00
2018-02-13 15:17:59 +09:00
onMessage(message) {
2020-02-20 06:08:49 +09:00
this.$root.sound('chat');
2018-03-04 18:50:30 +09:00
2018-02-13 15:17:59 +09:00
const isBottom = this.isBottom();
this.messages.push(message);
2018-05-27 13:49:09 +09:00
if (message.userId != this.$store.state.i.id && !document.hidden) {
2018-10-09 01:50:49 +09:00
this.connection.send('read', {
2018-02-13 15:17:59 +09:00
id: message.id
});
}
if (isBottom) {
// Scroll to bottom
2018-02-23 04:01:22 +09:00
this.$nextTick(() => {
this.scrollToBottom();
});
2018-05-27 13:49:09 +09:00
} else if (message.userId != this.$store.state.i.id) {
2018-02-13 15:17:59 +09:00
// Notify
2018-05-24 04:55:29 +09:00
this.notifyNewMessage();
2018-02-13 15:17:59 +09:00
}
},
2018-02-27 04:36:16 +09:00
2019-05-18 20:36:33 +09: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 15:17:59 +09:00
}
}
2018-02-13 15:17:59 +09:00
},
2018-02-27 04:36:16 +09: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 15:17:59 +09:00
isBottom() {
2018-02-23 04:01:22 +09:00
const asobi = 64;
2018-02-13 15:17:59 +09: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-27 04:36:16 +09:00
2018-02-13 15:17:59 +09:00
scrollToBottom() {
window.scroll(0, document.body.offsetHeight);
2018-02-13 15:17:59 +09:00
},
2018-02-27 04:36:16 +09:00
2018-05-24 04:55:29 +09: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 15:17:59 +09:00
}, 4000);
},
2018-02-27 04:36:16 +09:00
2018-09-01 09:29:59 +09: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 15:17:59 +09:00
onVisibilitychange() {
if (document.hidden) return;
for (const message of this.messages) {
2018-05-27 13:49:09 +09:00
if (message.userId !== this.$store.state.i.id && !message.isRead) {
2018-10-09 01:50:49 +09:00
this.connection.send('read', {
2018-02-13 15:17:59 +09:00
id: message.id
});
}
}
2018-02-13 15:17:59 +09:00
}
}
});
</script>
<style lang="scss" scoped>
.mk-messaging-room {
> .body {
width: 100%;
> .empty {
width: 100%;
margin: 0;
padding: 16px 8px 8px 8px;
text-align: center;
font-size: 0.8em;
opacity: 0.5;
[data-icon] {
margin-right: 4px;
}
}
> .no-history {
display: block;
margin: 0;
padding: 16px;
text-align: center;
font-size: 0.8em;
color: var(--messagingRoomInfo);
opacity: 0.5;
[data-icon] {
margin-right: 4px;
}
}
> .more {
display: block;
margin: 16px auto;
padding: 0 12px;
line-height: 24px;
color: #fff;
background: rgba(#000, 0.3);
border-radius: 12px;
&:hover {
background: rgba(#000, 0.4);
}
&:active {
background: rgba(#000, 0.5);
}
&.fetching {
cursor: wait;
}
> [data-icon] {
margin-right: 4px;
}
}
> .messages {
> ::v-deep * {
margin-bottom: 16px;
}
}
}
> footer {
width: 100%;
> .new-message {
position: absolute;
top: -48px;
width: 100%;
padding: 8px 0;
text-align: center;
> button {
display: inline-block;
margin: 0;
padding: 0 12px 0 30px;
line-height: 32px;
font-size: 12px;
border-radius: 16px;
> i {
position: absolute;
top: 0;
left: 10px;
line-height: 32px;
font-size: 16px;
}
}
}
}
}
.fade-enter-active, .fade-leave-active {
transition: opacity 0.1s;
}
2018-05-24 04:55:29 +09:00
.fade-enter, .fade-leave-to {
transition: opacity 0.5s;
opacity: 0;
}
2018-02-13 15:17:59 +09:00
</style>