溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面

發(fā)布時(shí)間:2023-03-13 13:37:41 來(lái)源:億速云 閱讀:284 作者:iii 欄目:開(kāi)發(fā)技術(shù)

本文小編為大家詳細(xì)介紹“前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面”,內(nèi)容詳細(xì),步驟清晰,細(xì)節(jié)處理妥當(dāng),希望這篇“前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來(lái)學(xué)習(xí)新知識(shí)吧。

頁(yè)面布局

這一塊比較簡(jiǎn)單,分析過(guò)chatgpt的頁(yè)面的就會(huì)知道,他的頁(yè)面布局方式是采用flex布局的,flex布局確實(shí)好用,那么我也是基于Bootsrap+jquery+flex布局完成了簡(jiǎn)易版的對(duì)話功能!主要有兩個(gè)地方用到了flex布局!

flex布局一

前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面

這里的頭像和文字采用的就是flex布局,并且文字和圖片頂部對(duì)齊,防止文字較多依舊和圖片中間對(duì)齊的bug。
需要設(shè)置css:

display: flex;
align-items: flex-start;

其中align-items: flex-start;的作用就是讓文字與圖片頂部對(duì)齊!

前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面

flex布局二(重點(diǎn))

第二處用到flex布局的地方就是這個(gè)搜索框:

前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面

很多人覺(jué)得這個(gè)對(duì)話框很簡(jiǎn)單,flex布局實(shí)現(xiàn)輸入框和按鈕在同一行確實(shí)簡(jiǎn)單,但你要好好看看chatgpt的官網(wǎng),都是有小細(xì)節(jié)的,這里面還是有很多知識(shí)點(diǎn)的。

首先,我要說(shuō)的是這個(gè)輸入框用的textarea,而不是input,區(qū)別在于,input輸入的內(nèi)容是不能換行的,但textarea文本框可以,但使用textarea的問(wèn)題是,參數(shù)rows設(shè)置為一行,這個(gè)文本框的高度會(huì)很低,達(dá)不到chatgpt的那個(gè)頁(yè)面要求,rows設(shè)置大一點(diǎn)或者這個(gè)文本框的高度給高一點(diǎn)會(huì)有一個(gè)問(wèn)題就是輸入時(shí)他的光標(biāo)不會(huì)在文本框的高度中間,而是在第一行,我們是沒(méi)法通過(guò)其他方式讓輸入光標(biāo)垂直居中的,因此這也不符合chatgpt頁(yè)面的要求,所以這確實(shí)是個(gè)值的學(xué)習(xí)的一點(diǎn)!看了chatgpt頁(yè)面的做法后,我悟了,下面一張圖來(lái)說(shuō)明chatgpt是如何做的:

前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面

如圖,你只要將textarea邊框取消掉,然后focus偽類(lèi)將邊框效果也取消掉,外邊再套一個(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è)輸入框定位到頁(yè)面底部就好!

js部分

首先,頁(yè)面部分,我們添加消息到頁(yè)面,包括用戶的問(wèn)題以及ai的回復(fù),添加消息到頁(yè)面時(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);
}

這里消息添加帶頁(yè)面后,清空了輸入框的內(nèi)容,接下來(lái)還需要給輸入框添加加一個(gè)鍵盤(pán)事件,也就是點(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("請(qǐng)輸入內(nèi)容!")  // 彈窗  
    return  
  }
  addUserMessage(message);
  
  chatBtn.attr('disabled',true) // 消息發(fā)送后讓提交按鈕不可點(diǎn)擊

  // 發(fā)送信息到后臺(tái)
  $.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 >' + '出錯(cuò)啦!請(qǐng)稍后再試!' + '</span>');
      chatBtn.attr('disabled',false) 
    }
  });
});

這些邏輯都很簡(jiǎn)單,我不再總結(jié),需要注意的是,我在發(fā)送消息到后臺(tái)等待相應(yīng)的過(guò)程讓按鈕的狀態(tài)是不可點(diǎn)擊的,直到后臺(tái)返回消息才可以進(jìn)行下一次問(wèn)答!但這里我沒(méi)有處理鍵盤(pán)事件,也就是說(shuō)你可以點(diǎn)擊enter繼續(xù)向后臺(tái)發(fā)送消息,這也是一個(gè)bug,只不過(guò)我沒(méi)有處理,你們不需要的可以去掉這個(gè)鍵盤(pán)事件就好了,當(dāng)然也可以在發(fā)送消息到獲得回答的這個(gè)時(shí)間段像禁用發(fā)送按鈕一樣,禁止enter鍵盤(pán)事件或者解綁這個(gè)鍵盤(pán)事件,這個(gè)你們自己去完成,這里我不在多說(shuō)(總要留點(diǎn)東西讓你們自己去思考去感悟?。?/p>

完整代碼

<!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="external nofollow"  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>
              <h2 class="text-center m-b-lg">Chat with ChatGPT</h2>
            </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("請(qǐng)輸入內(nèi)容!")    
            return  
          }
          addUserMessage(message);

          // messages.push({"role": "user", "content": message})
          
          chatBtn.attr('disabled',true) // 消息發(fā)送后讓提交按鈕不可點(diǎn)擊

          // 發(fā)送信息到后臺(tái)
          $.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 >' + '出錯(cuò)啦!請(qǐng)稍后再試!' + '</span>');
              chatBtn.attr('disabled',false) 
            }
          });
        });

        // 處理 Enter 鍵按下
        chatInput.keypress(function(e) {
          if (e.which == 13) {
            chatBtn.click();
          }
        });
  });
</script>
</html>

這里面用到的layer.js就是一個(gè)彈窗組件,百度可以搜到,common.js是我自己對(duì)layer.js方法的封裝,這個(gè)頁(yè)面其實(shí)你不這兩個(gè)js文件也行,因?yàn)檎麄€(gè)頁(yè)面只有下面的代碼用到了彈窗:

if (message.length == 0){
 	common_ops.alert("請(qǐng)輸入內(nèi)容!")    
  	return  
}

其實(shí)簡(jiǎn)陋點(diǎn),一個(gè)alert就搞定了!

讀到這里,這篇“前端如何實(shí)現(xiàn)類(lèi)似chatgpt的對(duì)話頁(yè)面”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識(shí)點(diǎn)還需要大家自己動(dòng)手實(shí)踐使用過(guò)才能領(lǐng)會(huì),如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI