{"remainingRequest":"/data/jenkins/workspace/badp-bcxin-web-5.x-vuechat/node_modules/vue-loader/lib/index.js??vue-loader-options!/data/jenkins/workspace/badp-bcxin-web-5.x-vuechat/src/ui/main/conversation/MessageInputView.vue?vue&type=script&lang=js&","dependencies":[{"path":"/data/jenkins/workspace/badp-bcxin-web-5.x-vuechat/src/ui/main/conversation/MessageInputView.vue","mtime":1702016964385},{"path":"/data/jenkins/workspace/badp-bcxin-web-5.x-vuechat/node_modules/babel-loader/lib/index.js","mtime":1702017034360},{"path":"/data/jenkins/workspace/badp-bcxin-web-5.x-vuechat/node_modules/cache-loader/dist/cjs.js","mtime":1702017034107},{"path":"/data/jenkins/workspace/badp-bcxin-web-5.x-vuechat/node_modules/vue-loader/lib/index.js","mtime":1702017029242}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64://
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//

import wfc from "../../../wfc/client/wfc";
import TextMessageContent from "../../../wfc/messages/textMessageContent";
import store from "../../../store";
import {categoriesDefault, emojisDefault, VEmojiPicker} from "@imndx/v-emoji-picker"
import ClickOutside from "vue-click-outside";
import Tribute from "tributejs";
import '../../../tribute.css'
import ConversationType from "../../../wfc/model/conversationType";
import ConversationInfo from "../../../wfc/model/conversationInfo";
import GroupInfo from "../../../wfc/model/groupInfo";
import GroupMemberType from "../../../wfc/model/groupMemberType";
import QuoteInfo from "../../../wfc/model/quoteInfo";
import Draft from "../../util/draft";
import Mention from "../../../wfc/model/mention";
import {parser as emojiParse} from '../../util/emoji';
import QuoteMessageView from "../../main/conversation/message/QuoteMessageView";
import {fileFromDataUri} from "../../util/imageUtil";
import StickerMessageContent from "../../../wfc/messages/stickerMessageContent";
import {config as emojiConfig} from "../../main/conversation/EmojiAndStickerConfig";
import {ipcRenderer, isElectron} from "../../../platform";
import {copyText} from "../../util/clipboard";
import EventType from "../../../wfc/client/wfcEvent";
import IpcEventType from "../../../ipcEventType";
import ChannelMenuView from "./ChannelMenuView";
import IpcSub from "../../../ipc/ipcSub";
import pttClient from "../../../wfc/ptt/client/pttClient";
import TalkingCallback from "../../../wfc/ptt/client/talkingCallback";
import Config from "../../../config";
import SoundMessageContent from "../../../wfc/messages/soundMessageContent";
import BenzAMRRecorder from "benz-amr-recorder";
import TypingMessageContent from "../../../wfc/messages/typingMessageContent";
import {currentWindow, fs} from "../../../platform";

export default {
    name: "MessageInputView",
    props: {
        conversationInfo: {
            type: ConversationInfo,
            required: true,
            default: null,
        },
        inputOptions: {
            type: Object,
            required: false,
            default: () => ({}),
        },
        muted: {
            type: Boolean,
            required: true,
            default: false,
        }
    },
    data() {
        return {
            sharedConversationState: store.state.conversation,
            sharedContactState: store.state.contact,
            sharedMiscState: store.state.misc,
            showEmojiDialog: false,
            tribute: null,
            mentions: [],
            emojiCategories: categoriesDefault,
            emojis: emojisDefault,
            lastConversationInfo: null,
            storeDraftIntervalId: 0,
            tributeReplaced: false,
            enablePtt: wfc.isCommercialServer() && Config.ENABLE_PTT,
            amrRecorder: null,
            lastTypingMessageTimestamp: 0,

            isPttTalking: false,
            isRecording: false,
        }
    },
    methods: {
        onTributeReplaced(e) {
            // 正常下面这两行应当就生效了，不知道为啥不生效，所以采用了后面的 trick
            e.detail.event.preventDefault();
            e.detail.event.stopPropagation();

            this.tributeReplaced = true;
        },
        canisend() {
            let target = this.conversationInfo.conversation._target;
            if (target instanceof GroupInfo) {
                let groupInfo = target;
                let groupMember = wfc.getGroupMember(groupInfo.target, wfc.getUserId());
                if (groupInfo.mute === 1) {
                    return [GroupMemberType.Owner, GroupMemberType.Manager, GroupMemberType.Allowed].indexOf(groupMember.type) >= 0
                        || groupMember.type === GroupMemberType.Allowed;
                }
            }

            return true;
        },

        cancelQuoteMessage() {
            this.conversationInfo._quotedMessage = null;
            store.quoteMessage(null)
        },

        onInput(e) {
            this.notifyTyping(TypingMessageContent.TYPING_TEXT);
        },

        notifyTyping(type) {
            if ([ConversationType.Single, ConversationType.Group].indexOf(this.conversationInfo.conversation.type) >= 0) {
                let now = new Date().getTime();
                if (now - this.lastTypingMessageTimestamp > 10 * 1000) {
                    let typing = new TypingMessageContent(type);
                    wfc.sendConversationMessage(this.conversationInfo.conversation, typing)
                    this.lastTypingMessageTimestamp = now;
                }
            }
        },
        async handlePaste(e, source) {
            let text;
            e.preventDefault();

            if ((e.originalEvent || e).clipboardData) {
                text = (e.originalEvent || e).clipboardData.getData('text/plain');
            } else {
                text = await navigator.clipboard.readText();
            }
            console.log('handlePaste', e, source);
            if (isElectron() && false) {
                let args = ipcRenderer.sendSync(IpcEventType.FILE_PASTE);
                if (args.hasImage) {
                    document.execCommand('insertText', false, ' ');
                    document.execCommand('insertImage', false, 'local-resource://' + args.filename);
                    return;
                } else if (args.hasFile) {
                    args.files.forEach(file => {
                        store.sendFile(this.conversationInfo.conversation, file)
                    })
                    return;
                }
            } else {
                const dT = e.clipboardData || window.clipboardData;
                if (dT) {
                    let fileList = dT.files;
                    if (fileList.length > 0) {
                        for (let i = 0; i < fileList.length; i++) {
                            let file = fileList.item(i);
                            console.log('handle paste file', file);
                            if (file.type.indexOf('image') !== -1) {
                                // image
                                document.execCommand('insertImage', false, URL.createObjectURL(file));
                                this.styleImageInEditor();
                            } else {
                                // file
                                if (isElectron()) {
                                    if (fs.lstatSync(file.path).isDirectory()) {
                                        this.$notify({
                                            // title: '不支持',
                                            text: this.$t('conversation.not_support_send_folder'),
                                            type: 'warn'
                                        });
                                        break;
                                    }
                                } else {
                                    // TODO 浏览器端，不能判断是否是文件夹
                                    if (file.size < 1024 && file.type === '') {
                                        this.$notify({
                                            // title: '不支持',
                                            text: this.$t('conversation.not_support_send_such_file'),
                                            type: 'warn'
                                        });
                                        break;
                                    }
                                }
                                store.sendFile(this.conversationInfo.conversation, file)
                            }
                        }
                        return;
                    }
                } else {
                    const clipboardContents = await navigator.clipboard.read();
                    for (const item of clipboardContents) {
                        console.log('clipboard item', item.types, item)
                        if (item.types.includes("image/png")) {
                            const blob = await item.getType("image/png");
                            document.execCommand('insertImage', false, URL.createObjectURL(blob));
                            this.styleImageInEditor();
                            return;
                        }
                    }
                }
            }

            if (text && text.trim()) {
                document.execCommand('insertText', false, text);
                // Safari 浏览器 execCommand 失效，可以采用下面这种方式处理粘贴
                // this.$refs.input.innerText += text;
            }
        },

        styleImageInEditor() {
            let imgs = this.$refs.input.getElementsByTagName('img')
            for (let img of imgs) {
                img.style.maxWidth = '100px';
                img.style.maxHeight = '100px';
            }
        },

        mention(groupId, memberId) {
            let displayName = wfc.getGroupMemberDisplayName(groupId, memberId);
            this.mentions.push({
                key: displayName,
                value: '@' + memberId,
            })
            let text = this.$refs.input.innerText;
            let mentionValue;
            if (text.endsWith(' ')) {
                mentionValue = '@' + displayName + ' ';
            } else {
                mentionValue = ' @' + displayName + ' ';
            }
            document.execCommand('insertText', false, mentionValue);
        },

        insertText(text) {
            // this.$refs['input'].innerText = text;
            this.$refs.input.focus();
            document.execCommand('insertText', false, text);
        },

        copy() {
            let text = this.$refs['input'].innerText;
            if (text) {
                copyText(text)
            }
        },

        cut() {
            this.copy();
            this.$refs['input'].innerHTML = '';
        },

        async send(e) {
            if (this.tribute && this.tribute.isActive) {
                this.tributeReplaced = false;
                return;
            }

            // let text = this.$refs['input'].textContent;
            // if (!text.trim()) {
            //   return;
            // }
            // this.$refs['input'].textContent = '';
            // // 发送消息时，会话消息列表需要滚动到最后
            // store.setShouldAutoScrollToBottom(true)
            //
            // let textMessageContent = this.handleMention(text)
            // let conversation = this.conversationInfo.conversation;
            // wfc.sendConversationMessage(conversation, textMessageContent);
            //

            let input = this.$refs['input'];
            let message = input.innerHTML.trim();
            let conversation = this.conversationInfo.conversation;

            if (
                !conversation
                || !this.canisend()
                || !message
            ) return;

            if (e.ctrlKey) {
                // e.preventDefault();
                // this.refs.input.innerHTML = this.refs.input.innerHTML+ "<div><br></div>";
                if (window.getSelection) {
                    let nextChar = window.getSelection().focusNode.textContent.charAt(window.getSelection().focusOffset)
                    if (!nextChar) {
                        document.execCommand('InsertHTML', true, '<br>');
                    }

                    let selection = window.getSelection(),
                        range = selection.getRangeAt(0),
                        br = document.createElement("br");
                    range.deleteContents();
                    range.insertNode(br);
                    range.setStartAfter(br);
                    range.setEndAfter(br);
                    // range.collapse(false);
                    selection.removeAllRanges();
                    selection.addRange(range);
                    // return false;
                }
                return;
            }

            // if(!message.startsWith('<')){
            //     message = message.replace(/<br>/g, '\n').trim()
            // }

            let imgs = [...input.getElementsByTagName('img')];
            if (imgs) {
                for (const img of imgs) {
                    if (img.className.indexOf('emoji') >= 0) {
                        continue;
                    }
                    let src = img.src;
                    let file;
                    // 截图
                    if (isElectron() && src.startsWith('local-resource')) {
                        // 'local-resource://' + 绝对路径
                        file = decodeURI(src.substring(17, src.length));
                    } else {
                        if (src.startsWith('blob:')) {
                            let blob = await fetch(src).then(r => r.blob());
                            file = new File([blob], new Date().getTime() + '.png');
                        } else {
                            file = fileFromDataUri(src, new Date().getTime() + '.png');
                        }
                    }
                    this.$eventBus.$emit('uploadFile', file)
                    store.setShouldAutoScrollToBottom(true);
                    store.sendFile(this.conversationInfo.conversation, file)
                    // 会影响 input.getElementsByTagName 返回的数组，所以上面拷贝了一下
                    img.parentNode.removeChild(img);
                }
            }
            message = input.innerHTML.trim();
            message = message
                .replace(/<div><br><\/div>/g, '\n')
                .replace(/<br>/g, '\n')
                .replace(/<div>/g, '\n')
                .replace(/<\/div>/g, '')
                .replace(/<b>/g, '')
                .replace(/<\/b>/g, '')
                .replace(/&lt;/g, '<')
                .replace(/&gt;/g, '>')
                .replace(/&nbsp;/g, ' ')
                .replace(/&amp;/g, '&')


            //  自行部署表情时，需要手动替换下面的正则
            // TODO 在正则中使用变量，避免手动替换
            let p = `" src="${Config.emojiBaseUrl()}72x72\\/[0-9a-z-]+\\.png">`
            let re = new RegExp(p, 'g');
            message = message.replace(/<img class="emoji" draggable="false" alt="/g, '')
                .replace(re, '')

            if (message && message.trim()) {
                let textMessageContent = this.handleMention(message);
                let quotedMessage = this.sharedConversationState.quotedMessage;
                if (quotedMessage) {
                    let quoteInfo = QuoteInfo.initWithMessage(quotedMessage);
                    textMessageContent.setQuoteInfo(quoteInfo);
                }
                store.setShouldAutoScrollToBottom(true);
                wfc.sendConversationMessage(conversation, textMessageContent);
                this.$refs['input'].innerHTML = '';
            }

            input.innerHTML = '';
            store.quoteMessage(null);
            this.conversationInfo._quotedMessage = null;
            Draft.setConversationDraft(conversation, '', null, null);
            e.preventDefault();
        },

        toggleEmojiView() {
            this.showEmojiDialog = !this.showEmojiDialog;
            this.focusInput();
        },

        screenShot(hideCurrentWindow = false) {
            if (hideCurrentWindow) {
                currentWindow.hide();
            }
            console.log('screenShot', hideCurrentWindow);
            ipcRenderer.send(IpcEventType.START_SCREEN_SHOT, {});
        },
        showMessageHistory() {
            let hash = window.location.hash;
            let url = window.location.origin;
            if (hash) {
                url = window.location.href.replace(hash, '#/conversation-message-history');
            } else {
                url += "/conversation-message-history"
            }
            let conversation = this.conversationInfo.conversation;
            ipcRenderer.send(IpcEventType.showConversationMessageHistoryPage, {
                url: url,
                type: conversation.type,
                target: conversation.target,
                line: conversation.line,
            });
            console.log(IpcEventType.showConversationMessageHistoryPage, url)
        },

        hideEmojiView(e) {
            if (e.target.id !== 'showEmoji') {
                this.showEmojiDialog = false;
            }
        },

        onSelectEmoji(emoji) {
            this.showEmojiDialog = false;
            if (emoji.data.indexOf('http') >= 0) {
                let sticker = new StickerMessageContent('', emoji.data, 200, 200)
                wfc.sendConversationMessage(this.conversationInfo.conversation, sticker);

                return;
            }

            this.$refs.input.focus();
            this.insertHTML(emojiParse(emoji.data));
            this.focusInput();
        },

        createElementFromHTML(htmlString) {
            let div = document.createElement('div');
            div.innerHTML = htmlString.trim();

            // Change this to div.childNodes to support multiple top-level nodes
            return div.firstChild;
        },


        insertHTML(html) {
            let sel, range;

            if (window.getSelection && (sel = window.getSelection())) {
                range = sel.getRangeAt(0);
                range.collapse(true);
                let imgEmoji = this.createElementFromHTML(html);
                range.insertNode(imgEmoji);

                // Move the caret immediately after the inserted span
                range.setStartAfter(imgEmoji);
                range.collapse(true);
                sel.removeAllRanges();
                sel.addRange(range);
            } else if (document.selection && document.selection.createRange) {
                document.selection.createRange().text = html;
            }
        },

        pickFile() {
            this.$refs['fileInput'].click();
            this.notifyTyping(TypingMessageContent.TYPING_FILE);
        },

        startAudioCall() {
            console.log(`startAudioCall from mainWindow ${this.sharedMiscState.isMainWindow}`);
            let conversation = this.conversationInfo.conversation;
            this.$startVoipCall({audioOnly: true, conversation: conversation});
        },

        startVideoCall() {
            console.log(`startVideoCall from mainWindow ${this.sharedMiscState.isMainWindow}`);
            let conversation = this.conversationInfo.conversation;
            this.$startVoipCall({audioOnly: false, conversation: conversation});
        },

        toggleChannelMenu(toggle = true) {
            if (toggle) {
                this.$parent.$refs['conversationMessageList'].style.flexGrow = 1;
                this.storeDraft(this.lastConversationInfo);
            } else {
                if (this.$parent.messageInputViewResized) {
                    this.$parent.$refs['conversationMessageList'].style.flexGrow = 0;
                }
            }
            store.toggleChannelMenu(toggle);
        },

        onPickFile(event) {
            // this.batchProcess(e.target.files[0]);
            console.log('onPickFile', event.target.files[0]);
            let file = event.target.files[0];
            event.target.value = '';

            // TODO
            // var showMessage = snackbar.showMessage;
            //
            // if (!file || file.size === 0) {
            //   showMessage('You can\'t send an empty file.');
            //   return false;
            // }
            //
            // if (!file
            //     || file.size >= 100 * 1024 * 1024) {
            //   showMessage('Send file not allowed to exceed 100M.');
            //   return false;
            // }
            if (isElectron()) {
                if (new Date().getTime() - file.lastModified < 30 * 1000 && file.path.indexOf('/var/folders') === 0) {
                    console.log('not support file', file)
                    this.$notify({
                        text: ' 不支持的文件类型',
                        type: 'warn'
                    });
                    return;
                }
            }
            this.$eventBus.$emit('uploadFile', file)
            store.sendFile(this.conversationInfo.conversation, file);
        },

        initEmojiPicker() {
            window.__twemoji_base_url__ = Config.emojiBaseUrl();
            let config = emojiConfig();
            if (this.conversationInfo.conversation.type === ConversationType.SecretChat) {
                this.emojiCategories = config.emojiCategories.filter(c => !c.name.startsWith('Sticker'));
                this.emojis = config.emojis.filter(c => !c.category.startsWith('Sticker'));
                this.$refs.emojiPicker.changeCategory({name: 'Peoples'});
            } else {
                this.emojiCategories = config.emojiCategories;
                this.emojis = config.emojis;
            }
        },

        initMention(conversation) {
            // TODO group, channel

            if (this.tribute) {
                this.tribute.detach(this.$refs['input']);
                this.tribute = null;
            }
            let type = conversation.conversationType;
            if (type === ConversationType.Single
                || type === ConversationType.ChatRoom || type === ConversationType.Channel) {
                return
            }

            let mentionMenuItems = [];
            let groupInfo = wfc.getGroupInfo(conversation.target);
            mentionMenuItems.push({
                key: this.$t('conversation.all_people'),
                value: '@' + conversation.target,
                avatar: groupInfo.portrait ? groupInfo.portrait : Config.DEFAULT_GROUP_PORTRAIT_URL,
                //searchKey: '所有人' + pinyin.letter('所有人', '', null)
                searchKey: this.$t('conversation.all_people') + 'suoyouren' + 'syr'
            });

            let groupMemberUserInfos = store.getGroupMemberUserInfos(conversation.target, false);
            groupMemberUserInfos.forEach((e) => {
                mentionMenuItems.push({
                    key: e._displayName,
                    value: '@' + e.uid,
                    avatar: e.portrait,
                    searchKey: e._displayName + e._pinyin + e._firstLetters,
                });
            });

            this.tribute = new Tribute({
                values: mentionMenuItems,
                selectTemplate: (item) => {
                    if (typeof item === 'undefined') return null;
                    // if (this.range.isContentEditable(this.current.element)) {
                    //     return '<span contenteditable="false"><a href="http://zurb.com" target="_blank" title="' + item.original.email + '">' + item.original.value + '</a></span>';
                    // }
                    this.mentions.push({key: item.original.key, value: item.original.value});

                    return '@' + item.original.key;
                },
                menuItemTemplate: function (item) {
                    return '<img width="24" height="24" src="' + item.original.avatar + ' "> ' + item.original.key;
                },
                noMatchTemplate: function () {
                    return '<span style:"visibility: hidden;"></span>';
                },
                lookup: (item) => {
                    return item.searchKey;
                },
                menuContainer: document.getElementById('conversation-content'),
            });
            this.tribute.attach(this.$refs['input']);
        },

        handleMention(text) {
            let textMessageContent = new TextMessageContent();
            textMessageContent.content = text.trim();
            this.mentions.forEach(e => {
                if (text.indexOf(e.key) > -1) {
                    if (e.value === '@' + this.conversationInfo.conversation.target) {
                        textMessageContent.mentionedType = 2;
                    } else {
                        if (textMessageContent.mentionedType !== 2) {
                            textMessageContent.mentionedType = 1;
                            textMessageContent.mentionedTargets.push(e.value.substring(1));
                        }
                    }
                }
            });

            this.mentions.length = 0;
            return textMessageContent;
        },

        focusInput() {
            this.$nextTick(() => {
                if (this.$refs['input']) {
                    this.$refs['input'].focus();
                    console.log('focus end')
                }
            })
        },

        moveCursorToEnd(contentEditableDiv) {

            let range = document.createRange();
            range.selectNodeContents(contentEditableDiv);
            range.collapse(false);
            let sel = window.getSelection();
            sel.removeAllRanges();
            sel.addRange(range);
        },

        restoreDraft() {
            let draft = Draft.getConversationDraftEx(this.conversationInfo);
            if (!draft) {
                return;
            }
            console.log('restore draft', this.conversationInfo, draft);
            store.quoteMessage(draft.quotedMessage);
            let input = this.$refs['input'];
            if (input.innerHTML.trim()) {
                console.log('inputting, ignore', draft.text)
            } else {
                input.innerHTML = draft.text.replace(/ /g, '&nbsp').replace(/\n/g, '<br>');
                this.moveCursorToEnd(input);
            }
        },

        storeDraft(conversationInfo) {
            if (!this.$refs['input']) {
                return;
            }
            let draftText = this.$refs['input'].innerHTML.trim();
            let p = `" src="${Config.emojiBaseUrl()}72x72\\/[0-9a-z-]+\\.png">`
            let re = new RegExp(p, 'g');
            draftText = draftText
                .replace(/<br>/g, '\n')
                .replace(/<div>/g, '\n')
                .replace(/<\/div>/g, '')
                .replace(/<div><\/div>/g, ' ')
                .replace(/&nbsp;/g, ' ')
                .replace(/<img class="emoji" draggable="false" alt="/g, '')
                .replace(re, '')
                .replace(/<img src="local-resource:.*">/g, '')
                .trimStart()
                .replace(/\s+$/g, ' ');

            let mentions = [];
            this.mentions.forEach(e => {
                let mention;
                /**
                 *  e.key: "13866666666"
                 *  e.value: "@q0H7q7MM"
                 */
                let start = draftText.indexOf('@' + e.key);
                let end = start + 1 + e.key.length;
                if (start > -1) {
                    if (e.value === '@' + this.conversationInfo.conversation.target) {
                        mention = new Mention(start, end, this.conversationInfo.conversation.target, true)
                    } else {
                        mention = new Mention(start, end, e.value.substring(1), false)
                    }
                    mentions.push(mention);
                }
            });

            let mentionCount = this.mentions ? this.mentions.length : 0;
            if (mentionCount > 0
                && draftText.endsWith('@' + this.mentions[mentionCount - 1].key + ' ')) {
                // @的最后一个空格不能删除
                // do nothing
            } else {
                draftText = draftText.trimEnd();
            }

            let quoteInfo = null;
            if (conversationInfo._quotedMessage) {
                quoteInfo = QuoteInfo.initWithMessage(conversationInfo._quotedMessage);
            }

            if (draftText.length === 0 && !quoteInfo) {
                if (conversationInfo.draft !== '') {
                    Draft.setConversationDraft(conversationInfo.conversation, draftText, quoteInfo, mentions)
                }
            } else {
                if (draftText !== conversationInfo.draft || (!conversationInfo.draft && quoteInfo)) {
                    Draft.setConversationDraft(conversationInfo.conversation, draftText, quoteInfo, mentions)
                }
            }
        },

        onGroupMembersUpdate(groupId, groupMembers) {
            console.log('messageInput onGroupMembersUpdate', groupId)
            if (this.conversationInfo
                && this.conversationInfo.conversation.type === ConversationType.Group
                && this.conversationInfo.conversation.target === groupId) {
                this.initMention(this.conversationInfo.conversation);
                let groupMember = wfc.getGroupMember(groupId, wfc.getUserId());
                if (groupMember && groupMember.type === GroupMemberType.Muted) {
                    this.muted = true;
                }
            }
        },

        requestPttTalk(request) {
            if (request) {
                let talkingCallback = new TalkingCallback();
                talkingCallback.onStartTalking = (conversation) => {
                    this.isPttTalking = true;
                    console.log('onStartTalking', conversation)
                    this.$notify({
                        text: '请开始说话',
                        type: 'info'
                    });
                };
                talkingCallback.onRequestFail = (conversation, reason) => {
                    this.$notify({
                        text: '对讲请求失败: ' + reason,
                        type: 'error'
                    });
                }
                talkingCallback.onTalkingEnd = (conversation, reason) => {
                    if (conversation.equal(this.conversationInfo.conversation)) {
                        this.isPttTalking = false;
                    }
                }
                pttClient.requestTalk(this.conversationInfo.conversation, talkingCallback)
                window.addEventListener('mouseup', this.handleMouseUp)
            } else {
                this.isPttTalking = false;
                pttClient.releaseTalk(this.conversationInfo.conversation);
            }
        },

        recordAudio(start) {
            this.notifyTyping(TypingMessageContent.TYPING_VOICE);
            if (start) {
                if (!this.amrRecorder) {
                    this.amrRecorder = new BenzAMRRecorder();
                    this.amrRecorder.initWithRecord().then(() => {
                        this.isRecording = true;
                        this.amrRecorder.startRecord();
                        this.$notify({
                            text: '请开始说话',
                            type: 'info'
                        });
                    }).catch((e) => {
                        this.$notify({
                            text: '录音失败',
                            type: 'error'
                        });
                        console.log('录音失败', e);
                        this.amrRecorder = null;
                    });
                }
                window.addEventListener('mouseup', this.handleMouseUp)
            } else {
                this.isRecording = false;
                if (this.amrRecorder) {
                    this.amrRecorder.finishRecord().then(() => {
                        let duration = this.amrRecorder.getDuration();
                        if (duration > 1) {
                            let blob = this.amrRecorder.getBlob();
                            let file = new File([blob], new Date().getTime() + '.amr');
                            let content = new SoundMessageContent(file, null, Math.ceil(duration));
                            wfc.sendConversationMessage(this.conversationInfo.conversation, content);
                        } else {
                            this.$notify({
                                text: '录音时间太短',
                                type: 'warn'
                            });
                        }
                        this.amrRecorder = null;
                    });
                }
            }
        },
        handleMouseUp() {
            if (this.isPttTalking) {
                this.requestPttTalk(false);
            } else if (this.isRecording) {
                this.recordAudio(false);
            }
            window.removeEventListener('mouseup', this.handleMouseUp)
        },

        setupConversationInput() {
            this.$refs.input.innerHTML = '';
            this.restoreDraft();
            this.initMention(this.conversationInfo.conversation)
            this.focusInput();
            this.initEmojiPicker()
        }
    },

    activated() {
        if (!this.sharedConversationState.showChannelMenu) {
            this.restoreDraft();
            this.focusInput();
        }
    },

    deactivated() {
        if (!this.sharedConversationState.showChannelMenu) {
            this.storeDraft(this.lastConversationInfo);
            this.$refs['input'].innerHTML = '';
        }
    },

    mounted() {
        if (!this.sharedConversationState.showChannelMenu) {
            if (this.conversationInfo) {
                this.initMention(this.conversationInfo.conversation)
                this.initEmojiPicker()
                this.restoreDraft();
            }
            this.focusInput();
        }
        this.lastConversationInfo = this.conversationInfo;

        if (isElectron()) {
            ipcRenderer.on('screenshots-ok', (event, args) => {
                console.log('screenshots-ok', args)
                if (args.filePath) {
                    setTimeout(() => {
                        document.execCommand('insertImage', false, 'local-resource://' + args.filePath);
                        this.styleImageInEditor();
                    }, 100)
                }
            });
        }
        this.storeDraftIntervalId = setInterval(() => {
            this.storeDraft(this.conversationInfo);
        }, 5 * 1000)
    },

    created() {
        wfc.eventEmitter.on(EventType.GroupMembersUpdate, this.onGroupMembersUpdate)
    },

    destroyed() {
        if (isElectron()) {
            ipcRenderer.removeAllListeners('screenshots-ok');
        }
        if (this.storeDraftIntervalId) {
            clearInterval(this.storeDraftIntervalId)
        }
        wfc.eventEmitter.removeListener(EventType.GroupMembersUpdate, this.onGroupMembersUpdate)
    },

    watch: {
        conversationInfo() {
            if (this.lastConversationInfo && !this.conversationInfo.conversation.equal(this.lastConversationInfo.conversation)) {
                this.$nextTick(() => {
                    if (this.sharedConversationState.showChannelMenu) {
                        this.$parent.$refs['conversationMessageList'].style.flexGrow = 1;
                        return
                    }
                    if (this.$parent.messageInputViewResized) {
                        this.$parent.$refs['conversationMessageList'].style.flexGrow = 0;
                    }
                    if (this.lastConversationInfo && !this.conversationInfo.conversation.equal(this.lastConversationInfo.conversation)) {
                        this.storeDraft(this.lastConversationInfo);
                    }

                    if (!this.muted && this.conversationInfo && (!this.lastConversationInfo || !this.conversationInfo.conversation.equal(this.lastConversationInfo.conversation))) {
                        this.setupConversationInput();
                    }
                    this.lastConversationInfo = this.conversationInfo;
                })
            } else {
                // 其他端更新了草稿
                // fixme
                // this.restoreDraft();
                this.lastConversationInfo = this.conversationInfo;
            }
        },
        'muted': {
            handler(newValue) {
                if (!newValue) {
                    this.$nextTick(() => {
                        this.setupConversationInput();
                    })
                } else {
                    this.$parent.$refs['conversationMessageList'].style.flexGrow = 1;
                }
            }
        }
    },

    computed: {
        quotedMessage() {
            // side affect
            this.$refs.input && this.$refs.input.focus();
            return this.sharedConversationState.quotedMessage;
        },

        hasInputTextOrImage() {
            // TODO 监听input的输入情况
            return true;
        }
    },

    components: {
        ChannelMenuView,
        QuoteMessageView,
        VEmojiPicker
    },
    directives: {
        ClickOutside,
        focus,
    }
};
"},null]}