前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >electron制作聊天界面(仿制qq)

electron制作聊天界面(仿制qq)

作者头像
李昊天
发布于 2019-05-26 00:43:14
发布于 2019-05-26 00:43:14
2.6K00
代码可运行
举报
运行总次数:0
代码可运行

效果图:

样式使用scss和flex布局

这也是制作IM系统的最后一个界面了! 在制作之前参考了qq和千牛

需要注意的点

qq将滚动条美化了 而且在无操作的情况下是不会显示的

滚动条美化

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
::-webkit-scrollbar { /*滚动条整体样式*/
    width: 5px; /*高宽分别对应横竖滚动条的尺寸*/
    height: 1px;
}

::-webkit-scrollbar-thumb { /*滚动条里面小方块*/
    border-radius: 10px;
    -webkit-box-shadow: inset 0 0 5px rgba(228, 57, 60, 0.2);
    background: rgba(20, 20, 50, 0.6);
    position: absolute;
}

::-webkit-scrollbar-track { /*滚动条里面轨道*/
    -webkit-box-shadow: inset 0 0 5px rgba(228, 57, 60, 0.2);
    border-radius: 10px;
    background: #EDEDED;
    position: absolute;
}

滚动条根据时机显示

其实这个也很简单 用的mouseentermouseleave事件

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<div
    :style="{overflowY:messageScroll? 'auto' : 'hidden',paddingRight: messageScroll ? '0': '5px' }"
    @mouseenter="showMessageScrolls" 
    @mouseleave="hideMessageScrolls">
</div>

# script
 showMessageScrolls(){
     this.messageScroll = true;
},
hideMessageScrolls(){
    this.messageScroll = false;
},

这里解释一下为什么有一个paddingRight 因为我们的滚动条是5px 如果不加 在滚动条显示的时候页面会抖动

简单写法

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    @mouseenter="messageScroll = true" 
    @mouseleave="messageScroll = false"

页面滚动

页面打开时消息列表滚动到底部

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
this.$nextTick(function () {
                this.$refs.msgBox.scrollTop = this.$refs.msgBox.scrollHeight
})

消息发送滚动到底部

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 this.$refs.msgBox.scrollTop = this.$refs.msgBox.scrollHeight;

内容编辑

没有使用表单元素 直接使用的 contenteditable 因为contenteditable 没法用双向数据绑定 不过 可以用数据侦听器 有很多办法 但是有很简单的 使用input事件就行了

代码

页面代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<template>
    <div class="friend_window">
        <header>
            <div class="nickname">Lee</div>
            <div class="buttons">
                <i class="iconfont">&#xe669;</i>
                <i class="iconfont">&#xe601;</i>
            </div>
        </header>
        <aside>
            <nav>
                <ul>
                    <li >
                        <div class="avatar"><img src="@/assets/img/1.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天-</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                    <li >
                        <div class="avatar"><img src="@/assets/img/2.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天-</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                    <li >
                        <div class="avatar"><img src="@/assets/img/3.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天-</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                    <li >
                        <div class="avatar"><img src="@/assets/img/4.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天-</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                    <li class="active">
                        <div class="avatar"><img src="@/assets/img/5.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天1-</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                    <li >
                        <div class="avatar"><img src="@/assets/img/6.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天-</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                    <li >
                        <div class="avatar"><img src="@/assets/img/7.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                    <li >
                        <div class="avatar"><img src="@/assets/img/8.jpg" alt=""></div>
                        <div class="msg_box">
                            <div class="nickname">李昊天-</div>
                            <div class="messages">最近还好吗</div>
                        </div>
                        <div class="push_right">
                            <div class="time">12:50</div>
                            <div class="number">1</div>
                        </div>
                    </li>
                </ul>
            </nav>
            <main>
                <div
                        class="message_main"
                        ref="ele"
                        :style="{overflowY:messageScroll? 'auto' : 'hidden',paddingRight: messageScroll ? '0': '5px' }"
                        @mouseenter="showMessageScrolls" @mouseleave="hideMessageScrolls"
                >
                    <div class="mes_box" v-for="(item,index) in list" :class="{'me' : index % 2 === 0}">
                        <div class="avatar">
                            <img src="@/assets/img/5.jpg" alt="">
                        </div>
                        <div class="message_box">
                            {{item.msg}}
                        </div>
                    </div>
                </div>
                <div class="input_box">
                    <div class="menubar">
                        <svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-biaoqing-weixiao"></use>
                        </svg>
                        <svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-folder"></use>
                        </svg>
                        <svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-tupian1"></use>
                        </svg>
                        <svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-shuangsechangyongtubiao-"></use>
                        </svg>
                    </div>
                    <div class="input" ref="input" contenteditable="true" @keydown.enter="sendMsg" @change="inputMsg"
                         @input="inputMsg"></div>
                    <div class="footerbar">
                        <Button>关闭</Button>
                        <Button type="primary">发送</Button>
                    </div>
                </div>
            </main>
        </aside>
    </div>
</template>

script代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<script>
    import '@/assets/css/scrool.css'
    import '@/assets/fonts/iconfont.js';

    export default {
        name: "friend",
        data() {
            return {
                list: [
                    {msg: '赵客缦胡缨,吴钩霜雪明'},
                    {msg: '银鞍照白马,飒沓如流星'},
                    {msg: '十步杀一人,千里不留行'},
                    {msg: '事了拂衣去,深藏身与名'},
                    {msg: '闲过信陵饮,脱剑膝前横。'},
                    {msg: '将炙啖朱亥,持觞劝侯嬴。'},
                    {msg: '三杯吐然诺,五岳倒为轻'},
                    {msg: '眼花耳热后,意气素霓生。'},
                    {msg: '救赵挥金槌,邯郸先震惊。'},
                    {msg: '千秋二壮士,烜赫大梁城。'},
                    {msg: '纵死侠骨香,不惭世上英。'},
                    {msg: '谁能书阁下,白首太玄经。'},
                    {msg: '是唐代大诗人李白借乐府古题创作的一首诗。此诗开头四句从侠客的装束、兵刃、坐骑刻画侠客的形象;第二个四句描写侠客高超的武术和淡泊名利的行藏;第三个四句引入信'},
                ],
                msg: '',
                number:8,
                messageScroll:false
            }
        },
        mounted() {
            this.$nextTick(function () {
                this.$refs.ele.scrollTop = this.$refs.ele.scrollHeight
            })
        },

        methods: {
            showMessageScrolls(){
                this.messageScroll = true;
            },
            hideMessageScrolls(){
                this.messageScroll = false;
            },
            inputMsg(e) {
                this.msg = e.target.innerHTML;
            },
            sendMsg(e) {
                this.list.push({msg: this.msg});
                this.msg = '';
                this.$refs.input.innerHTML = '';
                setTimeout(() => {
                    this.$refs.ele.scrollTop = this.$refs.ele.scrollHeight;
                }, 200);
                e.preventDefault();
            }
        }
    }
</script>

样式代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
.friend_window {
  position: absolute;
  width: 100%;
  height: 100%;
  background-image: url("../img/main_1.jpg");
  border-radius: 4px;
  -webkit-user-select: none;
  background-size: 100% 100%;

  header {
    height: 40px;
    background-color: rgba(0, 0, 0, 0.3);
    -webkit-app-region: drag;
    border-radius: 4px 4px 0 0;
    display: flex;
    justify-content: space-between;

    .nickname {
      color: #FFF;
      line-height: 40px;
      font-size: 20px;
      margin: auto;
      padding-left: 40px
    }

    .buttons {
      i {
        display: inline-block;
        color: #FFF;
        width: 40px;
        height: 40px;
        line-height: 40px;
        text-align: center;
        cursor: pointer;
        -webkit-app-region: no-drag;

        &:hover {
          background-color: rgba(255, 255, 255, 0.3);
        }
      }
    }
  }

  aside {
    height: calc(100% - 40px);
    border-radius: 0 0 4px 4px;
    display: flex;
  }

  nav {
    width: 240px;
    position: relative;

    background-size: 100% 100%;
    overflow-y: auto;

    &:after {
      display: inline-block;
      content: '';
      width: 5px;
      cursor: e-resize;
      position: absolute;
      right: -2px;
      top: 0;
      height: 100%;
    }

    ul {
      li.active {
        background-color: rgba(255, 255, 255, 0.2);
      }
      li {
        list-style: none;
        height: 60px;
        padding-left: 10px;
        cursor: pointer;
        display: flex;
        overflow: hidden;
        align-items: flex-start;

        &:hover {
          background-color: rgba(255, 255, 255, 0.2);
        }

        .push_right {
          padding-right: 10px;
          text-align: center;
          align-self: center;

          .time {
            font-size: 13px;
            color: #CFD3DA;
          }

          .number {
            display: inline-block;
            background-color: #e4393c;
            color: #fff;
            min-width: 15px;
            min-height: 15px;
            padding: 0 2px;
            line-height: 15px;
            border-radius: 50%;
            text-align: center;
            font-size: 12px;
          }
        }

        .msg_box {
          align-self: center;
          flex: 1;
          color: #EFF1F3;

          .messages {
            color: #CFD3DA;
          }
        }

        .avatar {
          width: 45px;
          height: 45px;
          align-self: center;
          margin-right: 10px;

          img {
            width: 100%;
            height: 100%;
            border-radius: 50%;
          }
        }
      }
    }
  }

  main {
    background-color: #fff;
    width: calc(100% - 240px);
    border-radius: 0 0 4px 0;

    .message_main {
      height: calc(100% - 35%);
      overflow-y: auto;

      &::-webkit-scrollbar {
        display: block !important;
      }

      .mes_box {
        display: flex;
        margin-bottom: 10px;
        margin-top: 10px;
        padding: 10px;

        .avatar {
          width: 40px;
          height: 40px;
          margin-right: 10px;

          img {
            width: 100%;
            height: 100%;
            border-radius: 50%;
          }
        }

        .message_box {
          background-color: #FFFFFF;
          color: #333;
          padding: 10px;
          border-radius: 5px;
          max-width: 72%;
          position: relative;
          border: 1px solid #D4D4D4;

          &::before {
            content: '';
            display: block;
            position: absolute;
            width: 10px;
            height: 10px;
            border: 1px solid #D4D4D4;
            border-right: none;
            border-top: none;
            background-color: #FFFFFF;
            border-radius: 3px;
            transform: rotate(44deg);
            left: -6px;
            top: 14px;
          }
        }
      }

      .me {
        display: flex;
        justify-content: flex-end;

        .message_box {
          background-color: #A0E759;
          color: #333;
          border: 1px solid #77BF41;

          &::before {
            display: none;
          }

          &::after {
            content: '';
            display: block;
            position: absolute;
            width: 10px;
            height: 10px;
            border: 1px solid #77BF41;
            border-bottom: none;
            border-left: none;
            border-radius: 3px;
            background-color: #A0E759;
            transform: rotate(45deg);
            right: -6px;
            top: 14px;
          }
        }

        .avatar {
          order: 2;
          margin-left: 10px;
        }
      }
    }

    .input_box {
      border-top: 1px solid #ccc;
      height: calc(100% - 65%);

      .menubar {
        height: 30px;
        width: 100%;
        display: flex;
        align-items: center;

        .icon {
          display: inline-block;
          padding: 2px;
          width: 25px;
          height: 25px;
          cursor: pointer;
          margin-right: 5px;
          margin-left: 5px;

          &:hover {
            background-color: rgba(0, 0, 0, 0.1);
          }
        }
      }

      .footerbar {
        display: flex;
        height: 70px;
        align-items: center;
        justify-content: flex-end;
        padding-right: 20px;

        button {
          margin: 0 10px;
          padding-left: 30px;
          padding-right: 30px;
        }
      }

      .input {
        font-size: 16px;
        padding: 4px 8px;
        overflow-y: auto;
        height: calc(100% - 70px - 30px);

        background-color: #fff;

        &::-webkit-scrollbar {
          display: block !important;
        }
      }
    }
  }
}

.icon {
  width: 1em;
  height: 1em;
  vertical-align: -0.15em;
  fill: currentColor;
  overflow: hidden;
}

声明

代码只为学习使用,如果有个人或者机构使用该代码带来的侵权行为,与本人无关 如果代码有不合理之处请大家提出

遗留问题

有一个问题就是左侧的列表是没法拉伸的 不过已经做了样式了 如果不想要的可以去掉这个css代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    &:after {
      display: inline-block;
      content: '';
      width: 5px;
      cursor: e-resize;
      position: absolute;
      right: -2px;
      top: 0;
      height: 100%;
    }

我的博客即将同步至腾讯云+社区,邀请大家一同入驻:https://cloud.tencent.com/dev...

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
7b2主题评论气泡
这个功能来自于https://github.com/515184405/barrage,版权归他所有。
小狐狸说事
2022/11/17
4040
7b2主题评论气泡
ectron仿制qq(3) 主界面制作(2)
这里解释下 -顶部header的高度(140) 底部的高度(40) 写法很多 不过达到效果就行了 在窗口拉伸的时候可以自适应高度
李昊天
2019/05/26
7400
为在线客服系统接入chatGPT(四):chatGPT接口vue网页版,可以联系上下文语境,可以实现自己的chatGPT,附代码
如果想实现chatGPT的网页版,调用接口就可以了,但是如果需要联系上下文语境,就需要在传递的数据的时候进行下拼接
唯一Chat
2023/02/17
2.7K1
为在线客服系统接入chatGPT(四):chatGPT接口vue网页版,可以联系上下文语境,可以实现自己的chatGPT,附代码
vue.js客服系统实时聊天项目开发(三)实现对话框聊天界面
.chatRow 中的 align-items: flex-end; 让其子元素在底部对齐,实现头像垂直方向靠下对齐
唯一Chat
2023/01/15
4.9K0
新鲜出炉的个人主页开源 最好看的
github:https://github.com/yigehaozi/renhuang-home
用户1287596
2024/10/07
1260
electron仿制qq(2) 主界面制作
制作从头开始 最后会将写好的组件放到一起的! 之前写了好几天的纯css 有点累 本章中将使用sass 如果代码太长 会分两个或多个章节写 代码中会有详细的注释 以便于大家阅读and理解 界面可能会有部分偏差 比较是仿制的
李昊天
2019/05/26
1.5K0
7b2美化-首页添加导航会员区块
声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。
小狐狸说事
2022/11/17
7970
使用SpringBoot + WebSocket实现单人聊天
在做之前,不管在界面布局,还是功能实现方面都下了一点功夫,最终还是一点点实现了,现在就记录一下。
全栈开发Dream
2021/06/08
2.3K0
使用SpringBoot + WebSocket实现单人聊天
Vue 实现6位数密码(iOS WebView卡顿优化)
在ios系统,原生 webview 嵌套H5页面使用时,编写完成的6位数输入密码,输入密码卡顿问题的解决方案:
White feathe
2021/12/08
3840
Vue 实现6位数密码(iOS  WebView卡顿优化)
Spring Boot使用websocket实现聊天室
先看效果:这里面demo用的是原生js跟html,方便打包一体化,在效果上我更倾向于使用vue、react等进行页面开发
余生大大
2022/11/02
9020
Spring Boot使用websocket实现聊天室
Python项目47-前后端分离登录注册页(继续撸)
-多年互联网运维工作经验,曾负责过大规模集群架构自动化运维管理工作。 -擅长Web集群架构与自动化运维,曾负责国内某大型金融公司运维工作。 -devops项目经理兼DBA。 -开发过一套自动化运维平台(功能如下): 1)整合了各个公有云API,自主创建云主机。 2)ELK自动化收集日志功能。 3)Saltstack自动化运维统一配置管理工具。 4)Git、Jenkins自动化代码上线及自动化测试平台。 5)堡垒机,连接Linux、Windows平台及日志审计。 6)SQL执行及审批流程。 7)慢查询日志分析web界面。
DriverZeng
2022/09/26
4460
Python项目47-前后端分离登录注册页(继续撸)
electron 自动更新以及手动更新
从搭建开始 使用的是electron-vue 毕竟方便一点 如果只想安装electron 请参见我的另一个文章 https://segmentfault.com/a/11...
李昊天
2019/05/26
2.8K0
html+css+js+python(QtWebEngineWidgets) 实现微信聊天界面-包括时间,文件,纯文本等
PyQt5 和 html 双向通信 python负责网络通信和API(html没有python照样可以)
zmh-program
2023/02/06
1.5K0
Vue 2.x折腾记 - (20) JSX在业务中的具体实践以及跟React书写的差异化
Vue的jsx,能够支持部分vue独有的特性,比如拿到computed, 指令及自定义事件;
CRPER
2019/05/31
7370
移动web)开发(4)之携程案例
唉,光是最上面的搜索框就一大堆代码,有点麻烦,而且很多东西想不到,真的是哭了呜呜呜...
y191024
2022/09/20
3990
移动web)开发(4)之携程案例
7b2主题评论气泡
这个功能来自于https://github.com/515184405/barrage,版权归他所有。
小狐狸说事
2022/12/01
3060
【Vue3】什么是路由?Vue中的路由基本切换~
在介绍本节的内容之前,我们首先还是老样子,准备好需要的代码,准备好必要的html代码,方便后面的操作,这里我们写了三个a标签,学过前端的都知道,我这是要创建链接,这就引出了路由和路径的关联
小馒头学Python
2024/05/07
1620
【Vue3】什么是路由?Vue中的路由基本切换~
衔接-玩转AI新声态 | 玩转TTS/ASR/YuanQI 打造自己的AI助手
衔接上一篇 玩转AI新声态 | 玩转TTS/ASR/YuanQI 打造自己的AI助手 页面数据渲染篇
杨不易呀
2024/06/24
3831
衔接-玩转AI新声态 | 玩转TTS/ASR/YuanQI 打造自己的AI助手
如何制作登录界面
先在桌面建一个.html文件,.html前面名字可以以你们的风格来写,这里我就用index.html来演示。这次代码可以把CSS+JavaScript放在HTML文件里面。
海曈
2022/06/17
6.9K1
如何制作登录界面
Hexo修改友链界面
之前看到友链界面的修改需要动大量源码,就没有弄。后来逐渐开始摸索、修改。自己表示很满意,按照各路大神的教程来操作也并不难。效果展示可以点击我的友链页面查看。
花猪
2022/02/16
8910
相关推荐
7b2主题评论气泡
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验