基于Next.js+React仿微信桌面端|next.js聊天室
2020/12/28 5:08:26
本文主要是介绍基于Next.js+React仿微信桌面端|next.js聊天室,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
项目介绍
Next.js
一款比较热门的React服务器端渲染框架。让你的网页拥有SEO能力。
Next-Webchat基于next+react+redux+antd+rlayer
等技术实现的仿微信PC网页端聊天项目。实现了消息表情混合发送、图片/视频预览、拖拽/粘贴截图发送、红包/朋友圈等功能。
技术框架
- 技术架构:next.js+react+Rredux
- UI组件库:Antd (蚂蚁金服react组件库)
- 字体图标:阿里iconfont图标库
- 弹窗组件:RLayer(react.js自定义对话框)
- 虚拟滚动:RScroll(react.js自定义滚动条)
效果预览
如果想让你的网页也能拥有SEO功能,那么Next.js是一个不错的选择。只要会React,上手就很容易了。
https://www.nextjs.cn/
https://github.com/vercel/nex...
Next|React自定义弹窗组件
项目中用到的所有弹窗均是自己开发的RLayer组件。
RLayer 基于react.js开发的PC端自定义弹框组件。支持超过30+参数自由配置,通过轻巧的布局设计、极简的调用方式来解决复杂的弹出层功能,为您呈上不一样的弹窗效果。
感兴趣的话,可以去看看之前的这篇分享文章。
https://segmentfault.com/a/11...
Next|React自定义滚动条组件
项目中用到的滚动条也是自己开发的一款PC桌面端虚拟美化滚动条组件RScroll。
Rscroll.js支持原生滚动条、是否自动隐藏、滚动条大小/层级/颜色等功能。
Next公共模板
新建layouts/index.js文件用于页面主入口模板。
function Layout(props) { const router = useRouter() // 拦截验证 useEffect(() => { // ... }, []) return ( <> {/* 配置公共head信息 */} <Head> <title>Next.js聊天室</title> <link rel="icon" href="/favicon.ico" /> <meta name="keywords" content="Next.js|React.js|Next.js聊天室|Next.js仿微信|React聊天实例"></meta> <meta name="description" content="Next-WebChat 基于Next.js+React+Redux构建的服务端渲染聊天应用程序"></meta> </Head> <div className="next__container flexbox flex-alignc flex-justifyc"> <div className={utils.classNames('next__wrapper')} style={{ backgroundImage: `url(${props.skin})` }}> <div className="next__board flexbox"> {/* 右上角按钮 */} <WinBar {...props} /> {/* 侧边栏 */} <Sidebar {...props} /> {/* 中间栏 */} <Middle /> {/* 主体布局 */} <div className="nt__mainbox flex1 flexbox flex-col"> {props.children} </div> </div> </div> </div> </> ) }
Head组件用于配置一些页面SEO信息,如:title、keyword、description及图标icon等信息。
聊天部分
编辑器模块单独抽离出来封装了一个editor组件,用于处理一些聊天输入、表情、光标处插入内容、粘贴截图等功能。
// react中实现div的contenteditable功能 return ( <div ref={editorRef} className="editor" contentEditable="true" dangerouslySetInnerHTML={{__html: state.editorText}} onClick={handleClicked} onInput={handleInput} onFocus={handleFocus} onBlur={handleBlur} style={{userSelect: 'text', WebkitUserSelect: 'text'}}> </div> )
如上图:利用RLayer弹窗实现视频播放预览功能。
handlePlayVideo = (item, e) => { rlayer({ content: ( <div className="flexbox flex-col" style={{height: '100%'}}> <div className="ntDrag__head"><i className="iconfont icon-bofang"></i> 视频预览</div> <div className="ntMain__cont flex1 flexbox flex-col"> {/* 视频video */} <video className="vplayer" src={item.videosrc} poster={item.imgsrc} autoPlay preload="auto" controls x5-video-player-fullscreen="true" webkit-playsinline="true" x-webkit-airplay="true" playsInline x5-playsinline="true" style={{height: '100%', width: '100%', objectFit: 'contain', outline: 'none'}} /> </div> </div> ), layerStyle: {background: '#f6f5ef'}, opacity: .2, area: ['550px', '450px'], drag: '.ntDrag__head', resize: true, maximize: true, }) }
编辑器支持拖拽发送图片。通过处理onDragEnter、onDragOver、onDrop
等事件。
handleDragEnter = (e) => { e.stopPropagation() e.preventDefault() } handleDragOver = (e) => { e.stopPropagation() e.preventDefault() } handleDrop = (e) => { e.stopPropagation() e.preventDefault() console.log(e.dataTransfer) this.handleFileList(e.dataTransfer) } // 获取拖拽文件列表 handleFileList = (filelist) => { let files = filelist.files if(files.length >= 2) { rlayer.message({icon: 'error', content: '暂时支持拖拽一张图片'}) return false } for(let i = 0; i < files.length; i++) { if(files[i].type != '') { this.handleFileAdd(files[i]) }else { rlayer.message({icon: 'error', content: '目前不支持文件夹拖拽功能'}) } } } handleFileAdd = (file) => { if(file.type.indexOf('image') == -1) { rlayer.message({icon: 'error', content: '目前不支持非图片拖拽功能'}) }else { let reader = new FileReader() reader.readAsDataURL(file) reader.onload = function() { let img = this.result console.log(img) } } }
编辑器还支持粘贴截图发送图片功能,监听paste
粘贴事件。
/** * 编辑器模块 */ import { useState, useRef, forwardRef, useEffect, useImperativeHandle } from 'react' const Editor = forwardRef(({value, onInput, onFocus, onBlur, onPaste}, ref) => { const [state, setState] = useState({ editorText: value, // 记录最后光标位置 lastRange: null }) const editorRef = useRef() useEffect(() => { // 编辑器粘贴事件 if(!editorRef.current) return editorRef.current.addEventListener('paste', function(e) { let cbd = e.clipboardData let ua = window.navigator.userAgent if(!(e.clipboardData && e.clipboardData.items)) return if(cbd.items && cbd.items.length === 2 && cbd.items[0].kind === "string" && cbd.items[1].kind === "file" && cbd.types && cbd.types.length === 2 && cbd.types[0] === "text/plain" && cbd.types[1] === "Files" && ua.match(/Macintosh/i) && Number(ua.match(/Chrome\/(\d{2})/i)[1]) < 49){ return; } for(var i = 0; i < cbd.items.length; i++) { var item = cbd.items[i] // console.log(item) // console.log(item.kind) if(item.kind == 'file') { var blob = item.getAsFile() if(blob.size === 0) return // 读取图片记录 var reader = new FileReader() reader.readAsDataURL(blob) reader.onload = function() { var _img = this.result // 返回图片给父组件 typeof onPaste == 'function' && onPaste(_img) } } } }) }, []) // ... }) export default Editor
Okey,基于Next.js+React开发聊天项目就分享到这里。希望大家能喜欢!✍✍
最后附上个Nuxt.js项目实例
Nuxt.js聊天室|nuxt+vue仿微信App实例聊天
这篇关于基于Next.js+React仿微信桌面端|next.js聊天室的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2024-11-23实现OSS直传,前端怎么实现?-icode9专业技术文章分享
- 2024-11-22在 HTML 中怎么实现当鼠标光标悬停在按钮上时显示提示文案?-icode9专业技术文章分享
- 2024-11-22html 自带属性有哪些?-icode9专业技术文章分享
- 2024-11-21Sass教程:新手入门及初级技巧
- 2024-11-21Sass学习:初学者必备的简单教程
- 2024-11-21Elmentplus入门:新手必看指南
- 2024-11-21Sass入门:初学者的简单教程
- 2024-11-21前端页面设计教程:新手入门指南
- 2024-11-21Elmentplus教程:初学者必备指南
- 2024-11-21SASS教程:从入门到实践的简单指南