| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421 | <template>    <div :data-id="info.id">        <!--文本、任务、报告、视频通话-->        <div v-if="info.type==='text' || info.type==='taskB' || info.type==='report' || info.type==='video' || info.type==='voice'">            <div v-if="info.self===true" class="list-right">                <div v-if="info.error" class="item-error" @click="clickError(info.error)">                    <Icon type="md-alert" />                </div>                <div class="item-right">                    <div class="item-username" @click="clickUser">                        <em class="item-name"><UserView :username="userName" :info="info" placement="left"/></em>                        <em v-if="info.indate" class="item-date">{{formatCDate(info.indate)}}</em>                    </div>                    <div class="item-text" :class="{'text-emoji':textIsEmoji(info.text), 'text-error':info.error}">                        <div class="item-text-view">{{textMsg(info.text)}}</div>                    </div>                    <template v-if="info.type==='taskB'">                        <div v-if="info.other.type==='task'" class="item-link" @click="taskDetail(info.other.id)"><span>{{$L('来自关注任务')}}:</span><a href="javascript:void(0)">{{info.other.title}}</a></div>                        <div v-if="info.other.type==='file'" class="item-link"><span>{{$L('来自关注任务')}}:</span><a target="_blank" :href="fileDownUrl(info.other.id)">{{info.other.name}}</a></div>                    </template>                    <div v-else-if="info.type==='report'" class="item-link" @click="reportDetail(info.other.id, info.other.title)"><span>{{$L('来自工作报告')}}:</span><a href="javascript:void(0)">{{info.other.title}}</a></div>                    <div v-else-if="info.type==='video' || info.type==='voice'" class="item-link">                        <Icon v-if="info.type==='voice'" type="ios-call-outline"/>                        <Icon v-else type="ios-videocam-outline"/>                        <span>{{$L('通话时长:%', formatSecond(info.other.second))}}</span>                    </div>                </div>                <UserImg :info="info" @click="clickUser" class="item-userimg"/>            </div>            <div v-else-if="info.self===false" class="list-item">                <UserImg :info="info" @click="clickUser" class="item-userimg"/>                <div class="item-left">                    <div class="item-username" @click="clickUser">                        <em class="item-name"><UserView :username="userName" :info="info" placement="right"/></em>                        <em v-if="info.__usertag" class="item-tag">{{info.__usertag}}</em>                        <em v-if="info.indate" class="item-date">{{formatCDate(info.indate)}}</em>                    </div>                    <div class="item-text" :class="{'text-emoji':textIsEmoji(info.text), 'text-error':info.error}">                        <div class="item-text-view">{{textMsg(info.text)}}</div>                    </div>                    <template v-if="info.type==='taskB'">                        <div v-if="info.other.type==='task'" class="item-link" @click="taskDetail(info.other.id)"><span>{{$L('来自关注任务')}}:</span><a href="javascript:void(0)">{{info.other.title}}</a></div>                        <div v-if="info.other.type==='file'" class="item-link"><span>{{$L('来自关注任务')}}:</span><a target="_blank" :href="fileDownUrl(info.other.id)">{{info.other.name}}</a></div>                    </template>                    <div v-else-if="info.type==='report'" class="item-link" @click="reportDetail(info.other.id, info.other.title)"><span>{{$L('来自工作报告')}}:</span><a href="javascript:void(0)">{{info.other.title}}</a></div>                    <div v-else-if="info.type==='video' || info.type==='voice'" class="item-link">                        <Icon v-if="info.type==='voice'" type="ios-call-outline"/>                        <Icon v-else type="ios-videocam-outline"/>                        <span>{{$L('通话时长:%', formatSecond(info.other.second))}}</span>                    </div>                </div>            </div>        </div>        <!--图片、文件-->        <div v-else-if="info.type==='image' || info.type==='file'">            <div v-if="info.self===true" class="list-right">                <div v-if="info.error" class="item-error" @click="clickError(info.error)">                    <Icon type="md-alert" />                </div>                <div class="item-right">                    <div class="item-username" @click="clickUser">                        <em class="item-name"><UserView :username="userName" :info="info" placement="left"/></em>                        <em v-if="info.indate" class="item-date">{{formatCDate(info.indate)}}</em>                    </div>                    <div v-if="info.url==='loading'" class="item-loading">                        <WLoading/>                    </div>                    <a v-else class="item-file" :href="info.url" target="_blank">                        <div v-if="info.type==='file'" class="item-file-box">                            <img class="item-file-thumb" :src="info.filethumb"/>                            <div class="item-file-info">                                <div class="item-file-name">{{info.filename}}</div>                                <div class="item-file-size">{{$A.bytesToSize($A.runNum(info.filesize) * 1024)}}</div>                            </div>                        </div>                        <img v-else class="item-file-img" :style="imageStyle(info)" :src="info.url"/>                    </a>                </div>                <UserImg :info="info" @click="clickUser" class="item-userimg"/>            </div>            <div v-else-if="info.self===false" class="list-item">                <UserImg :info="info" @click="clickUser" class="item-userimg"/>                <div class="item-left">                    <div class="item-username" @click="clickUser">                        <em class="item-name"><UserView :username="userName" :info="info" placement="right"/></em>                        <em v-if="info.__usertag" class="item-tag">{{info.__usertag}}</em>                        <em v-if="info.indate" class="item-date">{{formatCDate(info.indate)}}</em>                    </div>                    <div v-if="info.url==='loading'" class="item-loading">                        <WLoading/>                    </div>                    <a v-else class="item-file" :href="info.url" target="_blank">                        <div v-if="info.type==='file'" class="item-file-box">                            <img class="item-file-thumb" :src="info.filethumb"/>                            <div class="item-file-info">                                <div class="item-file-name">{{info.filename}}</div>                                <div class="item-file-size">{{$A.bytesToSize($A.runNum(info.filesize) * 1024)}}</div>                            </div>                        </div>                        <img v-else class="item-file-img" :style="imageStyle(info)" :src="info.url"/>                    </a>                </div>            </div>        </div>        <!--通知-->        <div v-else-if="info.type==='notice'">            <div class="item-notice">{{info.notice}}</div>        </div>    </div></template><style lang="scss" scoped>    /*通用*/    .list-item, .list-right {        display: flex;        width: 100%;        padding-top: 7px;        padding-bottom: 7px;        background-color: #E8EBF2;        .item-left, .item-right {            display: flex;            flex-direction: column;            max-width: 80%;            .item-username {                font-size: 12px;                padding-top: 1px;                padding-bottom: 4px;                display: flex;                flex-direction: row;                align-items: center;                em {                    display: inline-block;                    font-style: normal;                    &.item-name {                        color: #888888;                    }                    &.item-tag {                        color: #ffffff;                        background-color: #ff0000;                        line-height: 16px;                        padding: 2px 4px;                        margin-left: 3px;                        border-radius: 2px;                        font-size: 12px;                        transform: scale(0.8);                        font-weight: 600;                    }                    &.item-date {                        margin-left: 4px;                        color: #aaaaaa;                    }                }            }        }        .item-left {            align-items: flex-start;        }        .item-right {            align-items: flex-end;            .item-username {                text-align: right;            }            .item-link {                transform-origin: right center;            }        }        .item-userimg {            width: 38px;            height: 38px;            margin-left: 8px;            margin-right: 8px;            border-radius: 3px;            font-size: 20px;        }        .item-error {            cursor: pointer;            width: 48px;            position: relative;            > i {                color: #ff0000;                font-size: 18px;                position: absolute;                top: 50%;                left: 50%;                transform: translate(-50%, -50%);            }        }    }    .list-right {        justify-content: flex-end;    }    /*文本*/    .item-text {        display: inline-block;        border-radius: 6px;        padding: 8px;        background-color: #ffffff;        max-height: 580px;        overflow: auto;        &.text-emoji {            background-color: transparent;            .item-text-view {                font-size: 52px;                line-height: normal;            }        }        &.text-error {            box-shadow: 0 0 4px 0 #ffa1a1;        }        .item-text-view {            max-width: 520px;            color: #242424;            font-size: 14px;            line-height: 18px;            word-break: break-all;        }    }    /*信息底标*/    .item-link {        display: flex;        align-items: center;        max-width: 100%;        font-size: 12px;        color: #ffffff;        background-color: #cacaca;        margin-top: 6px;        margin-bottom: -2px;        height: 20px;        line-height: 20px;        padding: 0 5px;        border-radius: 4px;        transform: scale(0.96);        transform-origin: left center;        > i {            font-size: 14px;            padding-right: 2px;        }        > span {            white-space: nowrap;        }        > a {            color: #3D90E2;            padding-left: 3px;            overflow: hidden;            text-overflow: ellipsis;            white-space: nowrap;            max-width: 100%;        }    }    /*加载中*/    .item-loading {        display: flex;        justify-content: center;        align-items: center;        width: 28px;        height: 28px;        margin: 24px;    }    /*文件、图片*/    .item-file {        display: inline-block;        text-decoration: none;        .item-file-box {            background: #ffffff;            display: flex;            align-items: center;            padding: 10px 14px;            border-radius: 3px;            width: 220px;            .item-file-thumb {                width: 36px;            }            .item-file-info {                margin-left: 12px;                display: flex;                flex-direction: column;                justify-content: center;                .item-file-name {                    color: #333333;                    font-size: 14px;                    line-height: 18px;                    word-break: break-all;                    text-overflow: ellipsis;                    overflow: hidden;                    display: -webkit-box;                    -webkit-line-clamp: 2;                    -webkit-box-orient: vertical;                }                .item-file-size {                    padding-top: 4px;                    color: #666666;                    font-size: 14px;                }            }        }        .item-file-img {            max-width: 220px;            max-height: 220px;            border-radius: 6px;        }    }    /*通知*/    .item-notice {        color: #777777;        font-size: 12px;        text-align: center;        padding: 12px 24px;    }</style><script>    export default {        name: 'ChatMessage',        props: {            info: {                type: Object,                default: {},            },        },        mounted() {        },        computed: {            userName() {                return this.info.send_username || this.info.username;            },            userImg() {                return this.info.send_userimg || this.info.userimg;            },            imageStyle() {                return function (info) {                    const {width, height} = info;                    if (width && height) {                        let maxWidth = 220,                            maxHeight = 220,                            tempWidth = width,                            tempHeight = height;                        if (width > maxWidth || height > maxHeight) {                            if (width > height) {                                tempWidth = maxWidth;                                tempHeight = height * (maxWidth / width);                            } else {                                tempWidth = width * (maxHeight / height);                                tempHeight = maxHeight;                            }                        }                        return {                            width: tempWidth + 'px',                            height: tempHeight + 'px',                        };                    }                    return {};                }            }        },        methods: {            textMsg(text) {                return (text + "").replace(/\n/, '<br/>');            },            textIsEmoji(text) {                return text.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]/g, "_") === "_";            },            formatCDate(v) {                let string = '';                if ($A.runNum(v) > 0) {                    if ($A.formatDate('Ymd') === $A.formatDate('Ymd', v)) {                        string = $A.formatDate('H:i', v)                    } else if ($A.formatDate('Y') === $A.formatDate('Y', v)) {                        string = $A.formatDate('m-d', v)                    } else {                        string = $A.formatDate('Y-m-d', v)                    }                }                return string ? '(' + string + ')' : '';            },            clickError(err) {                this.$Modal.error({                    title: this.$L("错误详情"),                    content: err                });            },            clickUser(e) {                this.$emit('clickUser', this.info, e);            },            fileDownUrl(id) {                return $A.apiUrl('project/files/download?fileid=' + id);            },            formatSecond(d) {                if (d > 3600) {                    return Math.ceil(d / 3600) + '小时';                } else if (d > 60) {                    return Math.ceil(d / 60) + '分钟';                } else {                    return d + '秒';                }            }        }    }</script>
 |