fonchat/pages/index/index.vue
2023-10-25 10:02:04 +08:00

657 lines
15 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<template>
<view class="content">
<view class="top-bar">
<image
@click="changeMode"
class="version"
:src="
isGPT3 ? '../../static/image/3.5.png' : '../../static/image/4.0.png'
" />
<view class="title">FONCHAT</view>
<view></view>
</view>
<view class="bottom">
<div v-show="!acqStatus">
<div class="line"></div>
</div>
<view
class="chat-content"
id="chat-content"
ref="chatContent">
<view
class="chat-wrapper"
v-for="(item, index) in chatList"
:key="index">
<view
class="chat-friend"
v-if="item.uid !== 'admin'">
<span style="color: #175abd"
>客服AI<span
style="font-size: 20rpx"
v-show="!acqStatus && index === chatList.length - 1"
>(正在思索……)</span
></span
>
<image
style="
height: 40px;
width: 40px;
position: absolute;
left: -48px;
top: 5px;
"
src="../../static/image/avator.png" />
<view
class="chat-text"
v-if="item.chatType == 0">
<view v-if="item.msg" class="chat-word">{{ item.msg }}</view>
</view>
<view
class="chat-image"
v-if="item.chatType == 1">
<image
:src="item.msg"
alt="表情"
v-if="item.extend.imageType == 1"
style="width: 100px; height: 100px" />
<image
style="border-radius: 10px"
:src="item.msg"
v-else>
</image>
</view>
</view>
<view
class="chat-me"
v-else>
<image
:src="item.headimage"
alt=""
style="
height: 40px;
width: 40px;
position: absolute;
right: -48px;
top: 5px;
" />
<view
class="chat-text"
v-if="item.chatType == 0">
<span style="font-size: 16px">{{ item.msg }}</span>
</view>
<view
class="chat-image"
v-if="item.chatType == 1">
<image
:src="item.msg"
alt="表情"
v-if="item.extend.imageType == 1"
style="width: 100px; height: 100px" />
<a-image
style="max-width: 300px; border-radius: 10px"
:src="item.msg"
v-else>
</a-image>
</view>
</view>
</view>
</view>
</view>
<!-- 下面是两个是占位元素 -->
<view style="height: 12vh"></view>
<view class="text-area"> </view>
<view
class="text-area"
style="position: fixed">
<u--input
style="
width: 500rpx;
height: 60rpx;
border-radius: 40rpx;
background-color: #d8d8d8;
margin-right: 20rpx;
"
:disabled="!acqStatus"
@confirm="sendText"
v-model="inputMsg"
border="none"
placeholder="请输入内容"></u--input>
<u-button
style="
border-radius: 40rpx;
width: 120rpx;
height: 64rpx;
background-color: #175abd;
"
@click="sendText"
type="primary"
text="发送"></u-button>
</view>
</view>
</template>
<script>
import api from "@/http/";
import moment from "moment";
import {
OPENAI_API_KEY,
GPT_MODEL,
FREQUENCY_PENALTY,
MAX_TOKENS,
PRESENCE_PENALTY,
TEMPERATURE,
TOP_P,
} from "utils/openAiConfig";
export default {
data() {
return {
chatList: [],
inputMsg: "",
acqStatus: true,
gptMode: "gpt-3.5-turbo",
isGPT3: true,
};
},
onLoad(item) {},
methods: {
//发送文字信息
sendText() {
this.$nextTick(() => {
this.acqStatus = false;
});
const dateNow = moment().format("YYYY/M/DD/ HH:mm:ss");
let params = {};
console.log(this.inputMsg);
if (this.inputMsg) {
let chatMsg = {
headimage: "../../static/image/head.jpg",
name: GPT_MODEL,
time: dateNow,
msg: this.inputMsg,
chatType: 0, //信息类型0文字1图片
uid: "admin", //uid
};
console.log(chatMsg, 888);
this.sendMsg(chatMsg);
//如果是文字模式则进入
(params.model = this.gptMode),
(params.max_tokens = MAX_TOKENS),
(params.temperature = TEMPERATURE),
(params.top_p = TOP_P),
(params.presence_penalty = PRESENCE_PENALTY),
(params.frequency_penalty = FREQUENCY_PENALTY);
let chatBeforResMsg = {
headimage: "",
name: "",
time: moment().format("YYYY/M/DD/ HH:mm:ss"),
msg: "",
chatType: 0, //信息类型0文字1图片
uid: "ai", //uid
};
this.chatCompletion(params, chatBeforResMsg);
this.inputMsg = "";
} else {
this.$nextTick(() => {
this.acqStatus = true;
});
}
},
// 自动滚动到底部
scrollToBottom() {
this.$nextTick(() => {
uni
.createSelectorQuery()
.select(".chat-content")
.boundingClientRect((data) => {
uni.pageScrollTo({
duration: 100, //过渡时间
scrollTop: 1600, //到达目标class的top值
});
})
.exec();
});
},
// 切换模式
changeMode(e) {
this.acqStatus = true;
this.isGPT3 = !this.isGPT3;
this.gptMode = this.isGPT3 ? "gpt-3.5-turbo" : "gpt-4";
// 清空对话
this.chatList = [];
},
// 从接口获取对话结果
async chatCompletion(params, chatBeforResMsg) {
let conversation = this.contextualAssemblyData();
params.messages = conversation.map((item) => {
return {
role: item.speaker === "user" ? "user" : "assistant",
content: item.text,
};
});
params.stream = true;
//新增一个空的消息
this.sendMsg(chatBeforResMsg);
const currentResLocation = this.chatList.length - 1;
// 获取当前环境地址
const baseUrl = "http://114.218.158.24:9020/";
try {
uni.request({
url: baseUrl + "chat/app-completion", //你的接口地址
method: "POST",
data: JSON.stringify(params),
header: {
"Content-Type": "text/event-stream",
Accept: "application/json",
Authorization: "",
},
success: (response) => {
// 在这里处理你的响应数据
console.log(response.data, 123);
if (response.data) {
if (response.data.data.content) {
this.chatList[currentResLocation].msg =
response.data.data.content;
this.$nextTick(() => {
this.acqStatus = true;
});
this.scrollToBottom();
} else {
this.chatList[currentResLocation].msg =
"我还在学习中,等我学习完再来问我吧!";
this.$nextTick(() => {
this.acqStatus = true;
});
this.scrollToBottom();
}
}
},
fail: (err) => {
console.error(err);
},
});
} catch (error) {
console.error(error);
}
},
//组装上下文数据
contextualAssemblyData() {
const conversation = [];
for (var chat of this.chatList) {
if (chat.uid == "admin") {
let my = { speaker: "user", text: chat.msg };
conversation.push(my);
} else if (chat.uid == "ai") {
let ai = { speaker: "agent", text: chat.msg };
conversation.push(ai);
}
}
return conversation;
},
// 发送信息
sendMsg(msgList) {
this.chatList.push(msgList);
this.scrollToBottom();
},
// 当监听到向上滚动的时候
scrollToUpper() {
console.log("滚动到顶部");
},
},
};
</script>
<style lang="scss" scoped>
.uni-body {
height: 100vh;
background-color: #fff;
background-size: auto 100%;
background-attachment: fixed;
}
page {
height: 100vh;
background-color: #fff;
background-size: auto 100%;
background-attachment: fixed;
}
/deep/ .u-input__content {
color: #fff;
padding-left: 40rpx;
}
.text-area {
// position: fixed;
width: 95%;
bottom: 0;
height: 6vh;
background-size: 100% 100%;
background-color: #f2f2f2;
display: flex;
justify-content: space-between;
padding: 30rpx 40rpx 0rpx 20rpx;
}
.content {
width: 97%;
background-color: #fff;
background-size: auto 100%;
background-attachment: fixed;
height: 100%;
// overflow: hidden;
.top-bar {
width: 100%;
z-index: 100;
position: fixed;
height: 180rpx;
// 背景色渐变
background: linear-gradient(to bottom, #175abd 0%, #25a7f2 100%);
display: flex;
align-items: flex-end;
justify-content: space-between;
.version {
margin-bottom: 25rpx;
width: 100rpx;
height: 56rpx;
margin-left: 30rpx;
}
.title {
margin-bottom: 30rpx;
font-size: 36rpx;
color: #fff;
font-weight: bold;
margin-right: 120rpx;
}
}
.bottom {
width: 100%;
height: 80vh;
background-size: 100% 100%;
// background-color: rgb(50, 54, 68);
border-radius: 20px;
padding: 99px 20px 20px 20px;
box-sizing: border-box;
position: relative;
.chat-content {
width: 100%;
height: 100%;
// overflow-y: scroll;
// overflow-x: hidden;
padding: 20px;
box-sizing: border-box;
&::-webkit-scrollbar {
width: 3px;
/* 设置滚动条宽度 */
}
&::-webkit-scrollbar-thumb {
background-color: #8b67ef;
/* 设置滚动条滑块的背景色 */
}
.chat-friend {
width: 100%;
float: left;
margin-bottom: 20px;
position: relative;
display: flex;
margin-left: 30px;
flex-direction: column;
justify-content: flex-end;
align-items: flex-start;
.chat-text {
float: left;
max-width: 90%;
padding: 15px;
color: #535353;
border-radius: 20px 20px 20px 5px;
border: 1px solid #d8d8d8;
background-color: #fff;
}
.chat-image {
image {
max-width: 300px;
max-height: 200px;
border-radius: 10px;
}
}
.info-time {
margin: 10px 0;
font-size: 14px;
display: flex;
align-items: center;
justify-content: flex-start;
color: #8b67ef;
image {
width: 30px;
height: 30px;
border-radius: 50%;
vertical-align: middle;
margin-right: 10px;
}
span {
line-height: 30px;
}
span:last-child {
margin-left: 10px;
vertical-align: middle;
}
}
}
.chat-me {
width: 100%;
float: right;
margin-bottom: 20px;
position: relative;
margin-right: 28px;
display: flex;
flex-direction: column;
justify-content: flex-end;
align-items: flex-end;
.chat-text {
float: right;
max-width: 90%;
padding: 15px;
border-radius: 20px 20px 5px 20px;
background-color: #175abd;
color: #fff;
word-break: break-all;
// 让文字一个一个显示
}
.chat-image {
image {
max-width: 300px;
max-height: 200px;
border-radius: 10px;
}
}
.info-time {
margin: 10px 0;
color: #8b67ef;
font-size: 14px;
align-items: center;
display: flex;
justify-content: flex-end;
image {
width: 30px;
height: 30px;
border-radius: 50%;
vertical-align: middle;
margin-left: 10px;
}
span {
line-height: 30px;
}
span:first-child {
margin-right: 10px;
vertical-align: middle;
}
}
}
}
.chatInputs {
width: 90%;
position: absolute;
bottom: 0;
margin: 3%;
display: flex;
.boxinput {
width: 50px;
height: 50px;
background-color: rgb(50, 54, 68);
border-radius: 15px;
border: 1px solid rgb(80, 85, 103);
box-shadow: 0px 0px 5px 0px rgb(0, 136, 255);
position: relative;
cursor: pointer;
image {
width: 30px;
height: 30px;
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
}
}
.emoji {
transition: 0.3s;
width: 50px;
min-width: 50px;
}
.luyin {
color: #fff;
margin-left: 1.5%;
font-size: 30px;
text-align: center;
transition: 0.3s;
width: 50px;
min-width: 50px;
}
.inputs {
width: 95%;
height: 50px;
background-color: rgb(66, 70, 86);
border-radius: 15px;
border: 2px solid rgb(34, 135, 225);
padding: 10px;
box-sizing: border-box;
transition: 0.2s;
font-size: 20px;
color: #fff;
font-weight: 100;
margin: 0 20px;
&:focus {
outline: none;
}
}
}
}
.line {
position: relative;
width: 100%;
margin-left: 2%;
height: 2px;
background: linear-gradient(to right, #175abd 0%, #25a7f2 100%);
animation: shrink-and-expand 2s ease-in-out infinite;
}
.line::before,
.line::after {
content: "";
position: absolute;
top: 0;
width: 50%;
height: 100%;
background: inherit;
}
.line::before {
border-top-left-radius: 2px;
border-bottom-left-radius: 2px;
left: 0;
transform-origin: left;
animation: shrink-left 2s ease-in-out infinite;
}
.line::after {
border-top-left-radius: 2px;
border-bottom-left-radius: 2px;
right: 0;
transform-origin: right;
animation: shrink-right 2s ease-in-out infinite;
}
@keyframes shrink-and-expand {
0%,
100% {
transform: scaleX(1);
}
50% {
transform: scaleX(0);
}
}
@keyframes shrink-left {
0%,
50% {
transform: scaleX(1);
}
50.1%,
100% {
transform: scaleX(0);
}
}
@keyframes shrink-right {
0%,
50% {
transform: scaleX(1);
}
50.1%,
100% {
transform: scaleX(0);
}
}
.chat-word {
// 文字缓慢出现
animation: word 1s linear;
}
@keyframes word {
0% {
opacity: 0;
}
100% {
opacity: 1;
}
}
}
</style>