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">
|
2018-02-13 06:17:59 +00:00
|
|
|
<p class="init" v-if="init">%fa:spinner .spin%%i18n:common.loading%</p>
|
2018-04-14 16:04:40 +00:00
|
|
|
<p class="empty" v-if="!init && messages.length == 0">%fa:info-circle%%i18n:@empty%</p>
|
|
|
|
<p class="no-history" v-if="!init && messages.length > 0 && !existMoreMessages">%fa:flag%%i18n:@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-05-20 11:26:38 +00:00
|
|
|
<template v-if="fetchingMoreMessages">%fa:spinner .pulse .fw%</template>{{ fetchingMoreMessages ? '%i18n:common.loading%' : '%i18n:@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">
|
2018-02-20 16:39:51 +00:00
|
|
|
<x-message :message="message" :key="message.id"/>
|
|
|
|
<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">
|
|
|
|
<button @click="onIndicatorClick">%fa:arrow-circle-down%%i18n:@new-message%</button>
|
|
|
|
</div>
|
|
|
|
</transition>
|
2018-02-26 19:36:16 +00:00
|
|
|
<x-form :user="user" ref="form"/>
|
2018-02-13 06:17:59 +00:00
|
|
|
</footer>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
|
|
|
import Vue from 'vue';
|
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';
|
2018-02-13 06:17:59 +00:00
|
|
|
|
|
|
|
export default Vue.extend({
|
2018-02-20 16:39:51 +00:00
|
|
|
components: {
|
|
|
|
XMessage,
|
|
|
|
XForm
|
|
|
|
},
|
2018-02-26 19:36:16 +00:00
|
|
|
|
2018-02-13 06:17:59 +00:00
|
|
|
props: ['user', 'isNaked'],
|
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,
|
|
|
|
timer: null
|
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;
|
2018-08-07 04:25:50 +00:00
|
|
|
message._datetext = '%i18n:common.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() {
|
2018-10-10 11:58:42 +00:00
|
|
|
this.connection = (this as any).os.stream.connectToChannel('messaging', { otherparty: this.user.id });
|
2018-02-13 06:17:59 +00:00
|
|
|
|
|
|
|
this.connection.on('message', this.onMessage);
|
|
|
|
this.connection.on('read', this.onRead);
|
|
|
|
|
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() {
|
2018-10-07 02:06:17 +00:00
|
|
|
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) {
|
2018-08-06 19:53:21 +00:00
|
|
|
alert('%i18n:@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-02-18 03:35:18 +00:00
|
|
|
(this as any).api('messaging/messages', {
|
2018-03-29 05:48:47 +00:00
|
|
|
userId: this.user.id,
|
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
|
|
|
|
2018-02-13 06:17:59 +00:00
|
|
|
onRead(ids) {
|
|
|
|
if (!Array.isArray(ids)) ids = [ids];
|
|
|
|
ids.forEach(id => {
|
|
|
|
if (this.messages.some(x => x.id == id)) {
|
|
|
|
const exist = this.messages.map(x => x.id).indexOf(id);
|
2018-03-29 05:48:47 +00:00
|
|
|
this.messages[exist].isRead = true;
|
2018-02-13 06:17:59 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
2018-02-26 19:36:16 +00:00
|
|
|
|
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;
|
|
|
|
this.messages.forEach(message => {
|
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
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
</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
|
2018-07-14 16:07:17 +00:00
|
|
|
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-fa]
|
|
|
|
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-fa]
|
|
|
|
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-fa]
|
|
|
|
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
|
2018-07-14 16:07:17 +00:00
|
|
|
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
|
|
|
|
|
|
|
> [data-fa]
|
|
|
|
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>
|