?? 個(gè)人簡介
- ?? 作者簡介:大家好,我是阿牛,全棧領(lǐng)域優(yōu)質(zhì)創(chuàng)作者??
- ?? 個(gè)人主頁:館主阿牛??
- ?? 支持我:點(diǎn)贊??+收藏??+留言??
- ?? 系列專欄:前端實(shí)用小demo??
- ??格言:迄今所有人生都大寫著失敗,但不妨礙我繼續(xù)向前!??
前言
自從去年11月份chatgpt出圈之后,他的熱度就居高不減,也出現(xiàn)了很多人借助接口開發(fā)的國內(nèi)版本,那么本篇博客就從前端的角度來看看前端如何實(shí)現(xiàn)類似chatgpt的對話功能!
最終效果
源碼在文末獲取!
因?yàn)檫@是我寫在一個(gè)項(xiàng)目中的,單獨(dú)提出來可能配色效果不同,但功能和做法只要我們掌握了,那么自己想怎么寫就怎么寫!
頁面布局
這一塊比較簡單,分析過chatgpt的頁面的就會知道,他的頁面布局方式是采用flex布局的,flex布局確實(shí)好用,那么我也是基于Bootsrap+jquery+flex布局完成了簡易版的對話功能!主要有兩個(gè)地方用到了flex布局!
flex布局一
這里的頭像和文字采用的就是flex布局,并且文字和圖片頂部對齊,防止文字較多依舊和圖片中間對齊的bug。
需要設(shè)置css:
display: flex;
align-items: flex-start;
其中align-items: flex-start;的作用就是讓文字與圖片頂部對齊!
flex布局二(重點(diǎn))
第二處用到flex布局的地方就是這個(gè)搜索框:
很多人覺得這個(gè)對話框很簡單,flex布局實(shí)現(xiàn)輸入框和按鈕在同一行確實(shí)簡單,但你要好好看看chatgpt的官網(wǎng),都是有小細(xì)節(jié)的,這里面還是有很多知識點(diǎn)的。
首先,我要說的是這個(gè)輸入框用的textarea,而不是input,區(qū)別在于,input輸入的內(nèi)容是不能換行的,但textarea文本框可以,但使用textarea的問題是,參數(shù)rows設(shè)置為一行,這個(gè)文本框的高度會很低,達(dá)不到chatgpt的那個(gè)頁面要求,rows設(shè)置大一點(diǎn)或者這個(gè)文本框的高度給高一點(diǎn)會有一個(gè)問題就是輸入時(shí)他的光標(biāo)不會在文本框的高度中間,而是在第一行,我們是沒法通過其他方式讓輸入光標(biāo)垂直居中的,因此這也不符合chatgpt頁面的要求,所以這確實(shí)是個(gè)值的學(xué)習(xí)的一點(diǎn)!看了chatgpt頁面的做法后,我悟了,下面一張圖來說明chatgpt是如何做的:
如圖,你只要將textarea邊框取消掉,然后focus偽類將邊框效果也取消掉,外邊再套一個(gè)div邊框?qū)extarea文本框和按鈕套在里面就好了!
.ipt{
display:flex;
align-items: center;
position: absolute;
bottom: 60px;
margin: 0 15px;
padding-right: 15px;
border-radius: 15px;
width: calc(100% - 30px);
height: 50px;
border: 1px solid #e7eaec;
}
.ipt textarea {
resize: none;
overflow-y: auto;
border: none;
box-shadow: none;
}
.ipt textarea:focus{
border: none !important;
box-shadow: none !important;
}
最后,將這個(gè)輸入框定位到頁面底部就好!
js部分
首先,頁面部分,我們添加消息到頁面,包括用戶的問題以及ai的回復(fù),添加消息到頁面時(shí)需要向上滾動(dòng):
// 添加用戶消息到窗口
function addUserMessage(message) {
var messageElement = $('<div class="row message-bubble"><img class="chat-icon" src="' + userIcon + '"><p class="message-text">' + message + '</p></div>');
chatWindow.append(messageElement);
chatInput.val('');
chatWindow.animate({ scrollTop: chatWindow.prop('scrollHeight') }, 500);
}
// 添加回復(fù)消息到窗口
function addBotMessage(message) {
var messageElement = $('<div class="row message-bubble"><img class="chat-icon" src="' + botIcon + '"><p class="message-text">' + message + '</p></div>');
chatWindow.append(messageElement);
chatInput.val('');
chatWindow.animate({ scrollTop: chatWindow.prop('scrollHeight') }, 500);
}
這里消息添加帶頁面后,清空了輸入框的內(nèi)容,接下來還需要給輸入框添加加一個(gè)鍵盤事件,也就是點(diǎn)擊enter鍵也可以發(fā)送消息!
// 處理 Enter 鍵按下
chatInput.keypress(function(e) {
if (e.which == 13) {
chatBtn.click();
}
});
最后就是發(fā)送消息與獲得消息的一部分了:
// 處理用戶輸入
chatBtn.click(function() {
var message = chatInput.val();
if (message.length == 0){
common_ops.alert("請輸入內(nèi)容!") // 彈窗
return
}
addUserMessage(message);
chatBtn.attr('disabled',true) // 消息發(fā)送后讓提交按鈕不可點(diǎn)擊
// 發(fā)送信息到后臺
$.ajax({
url: '/chat',
method: 'POST',
data: {
"prompt": JSON.stringify(message)
},
success: function(res) {
res = JSON.parse(res);
addBotMessage(res.content);
chatBtn.attr('disabled',false) // 成功接受消息后讓提交按鈕再次可以點(diǎn)擊
},
error: function(jqXHR, textStatus, errorThrown) {
addBotMessage('<span style="color:red;">' + '出錯(cuò)啦!請稍后再試!' + '</span>');
chatBtn.attr('disabled',false)
}
});
});
這些邏輯都很簡單,我不再總結(jié),需要注意的是,我在發(fā)送消息到后臺等待相應(yīng)的過程讓按鈕的狀態(tài)是不可點(diǎn)擊的,直到后臺返回消息才可以進(jìn)行下一次問答!但這里我沒有處理鍵盤事件,也就是說你可以點(diǎn)擊enter繼續(xù)向后臺發(fā)送消息,這也是一個(gè)bug,只不過我沒有處理,你們不需要的可以去掉這個(gè)鍵盤事件就好了,當(dāng)然也可以在發(fā)送消息到獲得回答的這個(gè)時(shí)間段像禁用發(fā)送按鈕一樣,禁止enter鍵盤事件或者解綁這個(gè)鍵盤事件,這個(gè)你們自己去完成,這里我不在多說(總要留點(diǎn)東西讓你們自己去思考去感悟!)
完整代碼
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link href="../../static/css/bootstrap.min.css" rel="stylesheet">
<title>chat</title>
<style>
.answer{
width: 100%;
position: relative;
height: 70vh;
}
.ipt{
display:flex;
align-items: center;
position: absolute;
bottom: 60px;
margin: 0 15px;
padding-right: 15px;
border-radius: 15px;
width: calc(100% - 30px);
height: 50px;
border: 1px solid #e7eaec;
}
.ipt textarea {
resize: none;
overflow-y: auto;
border: none;
box-shadow: none;
}
.ipt textarea:focus{
border: none !important;
box-shadow: none !important;
}
#chatWindow {
max-height: calc(70vh - 120px);
height:auto;
overflow-y: auto;
}
.message-bubble {
padding: 10px;
margin: 5px;
display: flex;
align-items: flex-start;
border-bottom: 1px dashed #e7eaec;
}
.message-bubble p {
font-size: 18px;
margin-left:15px;
}
.chat-icon {
width: 30px;
height: 30px;
border-radius: 3px;
}
</style>
</head>
<body>
<div>
<div class="row">
<div class="col-xs-12">
<div>
<h1 class="text-center m-b-lg">Chat with ChatGPT</h1>
</div>
<div class="answer">
<div id="chatWindow" class="mb-3"></div>
<div class="input-group ipt">
<div class="col-xs-12">
<textarea id="chatInput" class="form-control" rows="1"></textarea>
</div>
<button id="chatBtn" class="btn btn-primary" type="button">Go !</button>
</div>
</div>
</div>
</div>
</div>
</div>
</body>
<script src="../../static/plugins/jquery-2.1.1.js"></script>
<script src="../../static/js/bootstrap.min.js"></script>
<script src="../../static/plugins/layer/layer.js"></script>
<script src="../../static/js/common.js"></script>
<script>
$(document).ready(function() {
var chatBtn = $('#chatBtn');
var chatInput = $('#chatInput');
var chatWindow = $('#chatWindow');
var userIcon = '/static/images/user/{{ current_user.avatar }}'
var botIcon = '/static/images/aichat/chatgpt.png';
// 添加用戶消息到窗口
function addUserMessage(message) {
var messageElement = $('<div class="row message-bubble"><img class="chat-icon" src="' + userIcon + '"><p class="message-text">' + message + '</p></div>');
chatWindow.append(messageElement);
chatInput.val('');
chatWindow.animate({ scrollTop: chatWindow.prop('scrollHeight') }, 500);
}
// 添加回復(fù)消息到窗口
function addBotMessage(message) {
var messageElement = $('<div class="row message-bubble"><img class="chat-icon" src="' + botIcon + '"><p class="message-text">' + message + '</p></div>');
chatWindow.append(messageElement);
chatInput.val('');
chatWindow.animate({ scrollTop: chatWindow.prop('scrollHeight') }, 500);
}
// 處理用戶輸入
chatBtn.click(function() {
var message = chatInput.val();
if (message.length == 0){
common_ops.alert("請輸入內(nèi)容!")
return
}
addUserMessage(message);
// messages.push({"role": "user", "content": message})
chatBtn.attr('disabled',true) // 消息發(fā)送后讓提交按鈕不可點(diǎn)擊
// 發(fā)送信息到后臺
$.ajax({
url: '/chat',
method: 'POST',
data: {
"prompt": JSON.stringify(message)
},
success: function(res) {
res = JSON.parse(res);
addBotMessage(res.content);
chatBtn.attr('disabled',false) // 成功接受消息后讓提交按鈕再次可以點(diǎn)擊
},
error: function(jqXHR, textStatus, errorThrown) {
addBotMessage('<span style="color:red;">' + '出錯(cuò)啦!請稍后再試!' + '</span>');
chatBtn.attr('disabled',false)
}
});
});
// 處理 Enter 鍵按下
chatInput.keypress(function(e) {
if (e.which == 13) {
chatBtn.click();
}
});
});
</script>
</html>
這里面用到的layer.js就是一個(gè)彈窗組件,百度可以搜到,common.js是我自己對layer.js方法的封裝,這個(gè)頁面其實(shí)你不這兩個(gè)js文件也行,因?yàn)檎麄€(gè)頁面只有下面的代碼用到了彈窗:
if (message.length == 0){
common_ops.alert("請輸入內(nèi)容!")
return
}
其實(shí)簡陋點(diǎn),一個(gè)alert就搞定了!
結(jié)語
如果你覺得博主寫的還不錯(cuò)的話,可以訂閱下面的這個(gè)flask專欄,這是博主唯一的付費(fèi)專欄,我做的這個(gè)頁面也是最近項(xiàng)目的一部分,這個(gè)項(xiàng)目也是用flask做的,我會將他全部總結(jié)開源到這個(gè)flask專欄中。
【flask從入門到實(shí)戰(zhàn)】專欄9.9火熱訂閱中,已包含兩個(gè)項(xiàng)目,全站獨(dú)一無二的腳手架搭建,直接復(fù)制簡單無腦操作,項(xiàng)目結(jié)構(gòu)類似Django,感興趣的可以看看哦!
??系列專欄
??flask框架快速入門文章來源:http://www.zghlxwxcb.cn/news/detail-778357.html
其他專欄請前往博主主頁查看!文章來源地址http://www.zghlxwxcb.cn/news/detail-778357.html
到了這里,關(guān)于chatgpt這么火?前端如何實(shí)現(xiàn)類似chatgpt的對話頁面的文章就介紹完了。如果您還想了解更多內(nèi)容,請?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!