|
@@ -0,0 +1,446 @@
|
|
|
+<template>
|
|
|
+ <u-navbar :titleStyle="{ color: '#000' }" :autoBack="true" title="小天AI" :placeholder="true" :safeAreaInsetTop="true" bgColor="#fff">
|
|
|
+ <template #left>
|
|
|
+ <view class="u-navbar__content__left__item">
|
|
|
+ <u-icon name="arrow-left" size="20" color="#000"></u-icon>
|
|
|
+ </view>
|
|
|
+ </template>
|
|
|
+ </u-navbar>
|
|
|
+
|
|
|
+ <oa-scroll
|
|
|
+ customClass="doorList-container scroll-height"
|
|
|
+ :isSticky="true"
|
|
|
+ :customStyle="{
|
|
|
+ //#ifdef APP-PLUS || MP-WEIXIN
|
|
|
+ height: `calc(100vh - (44px + ${footerHeight}px + ${proxy.$settingStore.StatusBarHeight}))`,
|
|
|
+ //#endif
|
|
|
+ //#ifdef H5
|
|
|
+ height: `calc(100vh - (44px + ${footerHeight}px))`,
|
|
|
+ //#endif
|
|
|
+ }"
|
|
|
+ :refresherLoad="false"
|
|
|
+ :refresherEnabled="false"
|
|
|
+ :scrollTop="scrollTop"
|
|
|
+ :scrollIntoView="scrollIntoView"
|
|
|
+ :refresherDefaultStyle="'none'"
|
|
|
+ :refresherBackground="'#fffff'"
|
|
|
+ :data-theme="'theme-' + proxy.$settingStore.themeColor.name"
|
|
|
+ >
|
|
|
+ <template #default>
|
|
|
+ <u-loading-page :loading="state.loading" fontSize="16" style="z-index: 99"></u-loading-page>
|
|
|
+ <view class="mainArea">
|
|
|
+ <view class="center-area">
|
|
|
+ <view class="center-area-item">
|
|
|
+ <viwe class="center-area-item__avatar">
|
|
|
+ <image src="@/static/ai-avatar.png" mode="widthFix"></image>
|
|
|
+ </viwe>
|
|
|
+ <view class="center-area-item__content">
|
|
|
+ <span class="defaultQuestion-title">您好,我是智能助手小天!</span>
|
|
|
+ 作为您的智能助手,我能帮助您解决各种问题。 除此之外,小天还可以尝试帮助解决其他方面的问题或在您需要的时候陪您聊天!
|
|
|
+ <!-- 您可以试着问我:
|
|
|
+ <view class="defaultQuestion">
|
|
|
+ <view class="defaultQuestion-item" @click="onClickDefaultQuestion($event, item)" v-for="item in defaultQuestion.list" :key="item.id" :span="12">
|
|
|
+ {{ item.value }}
|
|
|
+ </view>
|
|
|
+ </view> -->
|
|
|
+ </view>
|
|
|
+ </view>
|
|
|
+
|
|
|
+ <viwe v-for="(item, index) in answerList" :key="index" :class="['center-area-item', { roleUser: item.role === 'user' }]" :id="index == answerList.length - 1 ? 'bottomInfo' : ''">
|
|
|
+ <viwe class="center-area-item__avatar">
|
|
|
+ <image v-show="item.role === 'assistant'" src="@/static/ai-avatar.png" mode="widthFix"></image>
|
|
|
+ <image v-show="item.role === 'user'" src="@/static/ai-question-avatar.png" mode="widthFix"></image>
|
|
|
+ </viwe>
|
|
|
+
|
|
|
+ <viwe v-if="item.role === 'user'" class="center-area-item__content">{{ item.content }}</viwe>
|
|
|
+ <viwe v-else-if="item.role === 'assistant'" class="center-area-item__content">
|
|
|
+ <viwe class="center-area-item__outputReasonContent">{{ item.reasoningContent }}</viwe>
|
|
|
+ <oa-chatSSEClient ref="chatSSEClientRef" @onOpen="openCore" @onError="errorCore" @onMessage="messageCore" @onFinish="finishCore" />
|
|
|
+ </viwe>
|
|
|
+ </viwe>
|
|
|
+ </view>
|
|
|
+ </view>
|
|
|
+ </template>
|
|
|
+ </oa-scroll>
|
|
|
+
|
|
|
+ <view class="footer-area">
|
|
|
+ <view class="footer-area-top">
|
|
|
+ <view class="footer-area-top__record" @click="handle('Record')">对话历史</view>
|
|
|
+ <view class="footer-area-top__new" @click="handle('CreateChat')">新话题</view>
|
|
|
+ </view>
|
|
|
+ <view class="footer-area-center">
|
|
|
+ <textarea
|
|
|
+ class="footer-area-center-textarea"
|
|
|
+ v-model="answerKeyword"
|
|
|
+ placeholder="请输入问题,可通过回车换行"
|
|
|
+ maxlength="500"
|
|
|
+ @confirm="onSubmit"
|
|
|
+ @linechange="linechange"
|
|
|
+ auto-height
|
|
|
+ ></textarea>
|
|
|
+ <button class="footer-area-center-button" type="primary" :loading="submitLoading" @click="onSubmit">发送</button>
|
|
|
+ </view>
|
|
|
+ <view class="footer-area-prompt">以上内容均由AI大模型生成,仅供参考</view>
|
|
|
+ </view>
|
|
|
+
|
|
|
+ <uni-popup ref="popup" type="left" safeArea backgroundColor="#fff">
|
|
|
+ <view class="chatArea">
|
|
|
+ <view class="content">
|
|
|
+ <view class="content-title">近30天</view>
|
|
|
+ <view class="content-item" v-for="item in state.chatHistory.list" :key="item" @click="onClickChatItem(item)"> {{ item.question }}</view>
|
|
|
+ </view>
|
|
|
+ </view>
|
|
|
+ </uni-popup>
|
|
|
+</template>
|
|
|
+
|
|
|
+<script setup>
|
|
|
+/*----------------------------------依赖引入-----------------------------------*/
|
|
|
+import config from "@/config";
|
|
|
+import { getToken } from "@/utils/auth";
|
|
|
+import { onLoad, onShow, onReady, onHide, onLaunch, onUnload, onNavigationBarButtonTap, onPageScroll } from "@dcloudio/uni-app";
|
|
|
+import { ref, reactive, computed, getCurrentInstance, toRefs, inject, nextTick } from "vue";
|
|
|
+/*----------------------------------接口引入-----------------------------------*/
|
|
|
+import { aiApi } from "@/api/business/ai.js";
|
|
|
+/*----------------------------------组件引入-----------------------------------*/
|
|
|
+/*----------------------------------store引入-----------------------------------*/
|
|
|
+import { useStores, commonStores, controlStores } from "@/store/modules/index";
|
|
|
+/*----------------------------------公共方法引入-----------------------------------*/
|
|
|
+/*----------------------------------公共变量-----------------------------------*/
|
|
|
+const { proxy } = getCurrentInstance();
|
|
|
+const useStore = useStores();
|
|
|
+const controlStore = controlStores();
|
|
|
+/*----------------------------------变量声明-----------------------------------*/
|
|
|
+const state = reactive({
|
|
|
+ loading: false,
|
|
|
+ dataList: [],
|
|
|
+ pageSize: 20,
|
|
|
+ current: 1,
|
|
|
+ total: 0,
|
|
|
+
|
|
|
+ buffer: "", // 用于存储流式数据不完整的行
|
|
|
+ defaultQuestion: {
|
|
|
+ list: [
|
|
|
+ { id: 2, value: "如何看待中国锂矿储量从全球占比6%升至16.5%,从世界第六跃至第二?" },
|
|
|
+ { id: 3, value: "为什么特斯拉和理想,都不想承认自己是「汽车公司」?" },
|
|
|
+ { id: 4, value: "全美第二大的城市洛杉矶大火,为什么能蔓延到无法控制的地步?" },
|
|
|
+ { id: 4, value: "为什么主流都不再力推英特尔 CPU?" },
|
|
|
+ ],
|
|
|
+ },
|
|
|
+ submitLoading: false,
|
|
|
+ answerKeyword: undefined,
|
|
|
+ answerList: [],
|
|
|
+ // 会话记录
|
|
|
+ chatHistory: {
|
|
|
+ list: [],
|
|
|
+ sessionId: undefined,
|
|
|
+ },
|
|
|
+
|
|
|
+ footerHeight: 0, //底部区域高度
|
|
|
+ scrollIntoView: "", //滚动位置
|
|
|
+});
|
|
|
+
|
|
|
+const { tabsList, tabsCurrent, dataList, buffer, defaultQuestion, submitLoading, answerKeyword, answerList, chatHistory, footerHeight, scrollIntoView } = toRefs(state);
|
|
|
+
|
|
|
+/** 操作 */
|
|
|
+function handle(type) {
|
|
|
+ if (type == "Record") {
|
|
|
+ getChatList();
|
|
|
+ proxy.$refs["popup"].open();
|
|
|
+ } else if (type == "CreateChat") {
|
|
|
+ if (state.chatHistory.list.length >= 10) {
|
|
|
+ return uni.showToast({ title: "您最多可同时建立10个对话,如需新建对话,请先删除会话!", icon: "none" });
|
|
|
+ }
|
|
|
+ state.chatHistory.sessionId = undefined;
|
|
|
+ state.answerList = [];
|
|
|
+ state.answerKeyword = undefined;
|
|
|
+ } else if ("Delete") {
|
|
|
+ uni.showModal({ title: "提示", content: "确定删除吗" });
|
|
|
+ } else if ("Update") {
|
|
|
+ uni.showToast({ title: "触发修改", icon: "none" });
|
|
|
+ getChatList();
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+function onClickEditTitle(item) {
|
|
|
+ state.chatHistory.list.map((item) => (item._edit = false));
|
|
|
+ item._edit = true;
|
|
|
+ nextTick(() => {});
|
|
|
+}
|
|
|
+
|
|
|
+function onClickDefaultQuestion(event, item) {
|
|
|
+ state.answerKeyword = item.value;
|
|
|
+ onSubmit(event);
|
|
|
+}
|
|
|
+
|
|
|
+/** 获取历史对话列表 */
|
|
|
+function getChatList() {
|
|
|
+ aiApi()
|
|
|
+ .recordSelect()
|
|
|
+ .then((response) => {
|
|
|
+ state.chatHistory.list = response.data;
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+/** 点击历史对话事件 */
|
|
|
+function onClickChatItem(item) {
|
|
|
+ state.scrollIntoView = "";
|
|
|
+ state.chatHistory.sessionId = item.sessionId;
|
|
|
+ state.answerList = item.itemList;
|
|
|
+
|
|
|
+ nextTick(() => {
|
|
|
+ state.scrollIntoView = "bottomInfo";
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+/** 当输入框行数发生变化 */
|
|
|
+function linechange(e) {
|
|
|
+ nextTick(() => {
|
|
|
+ const query = uni.createSelectorQuery();
|
|
|
+ query
|
|
|
+ .select(".footer-area")
|
|
|
+ .boundingClientRect((rect) => {
|
|
|
+ state.footerHeight = rect.height;
|
|
|
+ })
|
|
|
+ .exec();
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+async function onSubmit(event) {
|
|
|
+ if (event.keyCode === 13 || event.key === "Enter") {
|
|
|
+ event.preventDefault(); // 阻止默认行为,即回车换行
|
|
|
+ if (event.shiftKey) {
|
|
|
+ state.answerKeyword += "\n";
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (!state.answerKeyword) return uni.showToast({ title: "请输入内容", icon: "none" });
|
|
|
+ if (state.submitLoading) return;
|
|
|
+
|
|
|
+ state.submitLoading = true;
|
|
|
+ state.answerList.push({
|
|
|
+ role: "user",
|
|
|
+ content: state.answerKeyword,
|
|
|
+ });
|
|
|
+
|
|
|
+ proxy.$refs["chatSSEClientRef"].startChat({
|
|
|
+ // 将它换成你的地址
|
|
|
+ url: config.baseUrl + "/service-ai/ai/aliTyqw",
|
|
|
+ // 请求头
|
|
|
+ headers: {
|
|
|
+ Authorization: getToken(),
|
|
|
+ "Content-Type": "application/json; charset=utf-8",
|
|
|
+ },
|
|
|
+ // 默认为 post
|
|
|
+ method: "post",
|
|
|
+ body: {
|
|
|
+ sessionId: state.chatHistory.sessionId,
|
|
|
+ content: state.answerKeyword,
|
|
|
+ },
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+//数据处理
|
|
|
+function processSSEChunk(chunk) {
|
|
|
+ state.scrollIntoView = "";
|
|
|
+ try {
|
|
|
+ const jsonData = JSON.parse(chunk);
|
|
|
+
|
|
|
+ if (jsonData.sessionId && !state.chatHistory.sessionId) {
|
|
|
+ state.chatHistory.sessionId = jsonData.sessionId;
|
|
|
+ }
|
|
|
+ if (jsonData.reasoningContent) {
|
|
|
+ state.answerList[state.answerList.length - 1].reasoningContent += jsonData.reasoningContent;
|
|
|
+ } else if (jsonData.content) {
|
|
|
+ state.answerList[state.answerList.length - 1].content += jsonData.content;
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.log("parsing JSON错误", jsonString, index);
|
|
|
+ console.log("parsing JSON错误", error);
|
|
|
+ }
|
|
|
+
|
|
|
+ nextTick(() => {
|
|
|
+ state.scrollIntoView = "bottomInfo";
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+//开始回答回调事件
|
|
|
+function openCore() {
|
|
|
+ console.log("open sse");
|
|
|
+
|
|
|
+ state.answerKeyword = undefined;
|
|
|
+
|
|
|
+ // 定义一个函数来读取流数据
|
|
|
+ state.answerList.push({
|
|
|
+ role: "assistant",
|
|
|
+ content: "",
|
|
|
+ reasoningContent: "",
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+//回答异常回调事件
|
|
|
+function errorCore(err) {
|
|
|
+ console.log("error sse:", err);
|
|
|
+ state.answerKeyword = undefined;
|
|
|
+}
|
|
|
+
|
|
|
+//回答中回调事件
|
|
|
+function messageCore(response) {
|
|
|
+ processSSEChunk(response);
|
|
|
+}
|
|
|
+
|
|
|
+//回答完毕回调事件
|
|
|
+function finishCore() {
|
|
|
+ console.log("finish sse");
|
|
|
+ state.submitLoading = false;
|
|
|
+}
|
|
|
+
|
|
|
+//停止回答
|
|
|
+function stop() {
|
|
|
+ proxy.$refs["chatSSEClientRef"].stopChat();
|
|
|
+ console.log("stop");
|
|
|
+}
|
|
|
+
|
|
|
+onReady(() => {});
|
|
|
+
|
|
|
+onShow(() => {});
|
|
|
+
|
|
|
+onLoad((options) => {});
|
|
|
+
|
|
|
+onUnload(() => {});
|
|
|
+</script>
|
|
|
+
|
|
|
+<style lang="scss" scoped>
|
|
|
+.doorList-container {
|
|
|
+ background-color: #ffffff;
|
|
|
+}
|
|
|
+
|
|
|
+.center-area {
|
|
|
+ height: auto;
|
|
|
+ overflow: auto;
|
|
|
+
|
|
|
+ &-item {
|
|
|
+ display: flex;
|
|
|
+ color: #05073b;
|
|
|
+ padding: 15px;
|
|
|
+
|
|
|
+ &__avatar {
|
|
|
+ width: 26px;
|
|
|
+ margin-top: 2px;
|
|
|
+ margin-right: 10px;
|
|
|
+
|
|
|
+ uni-image {
|
|
|
+ border-radius: 50%;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ &__content {
|
|
|
+ background: #f4f5f9;
|
|
|
+ border-radius: 8px;
|
|
|
+ box-shadow: 0 16px 20px 0 #f4f5f9;
|
|
|
+ display: flex;
|
|
|
+ flex-direction: column;
|
|
|
+ padding: 10px 15px;
|
|
|
+ position: relative;
|
|
|
+ font-size: 14px;
|
|
|
+ line-height: 1.75;
|
|
|
+ min-height: 44.5px;
|
|
|
+ box-sizing: border-box;
|
|
|
+ word-break: break-all; //文字默认换行
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ .roleUser {
|
|
|
+ justify-content: end;
|
|
|
+ margin-left: auto;
|
|
|
+
|
|
|
+ .center-area-item__avatar {
|
|
|
+ margin-right: 0;
|
|
|
+ margin-left: 10px;
|
|
|
+ order: 2;
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+.footer-area {
|
|
|
+ position: fixed;
|
|
|
+ bottom: 0;
|
|
|
+ width: 100%;
|
|
|
+ // height: 137px;
|
|
|
+ background-color: #ffffff;
|
|
|
+
|
|
|
+ &-top {
|
|
|
+ display: flex;
|
|
|
+ margin: 10px 20px 0 20px;
|
|
|
+ color: #7f7f7f;
|
|
|
+
|
|
|
+ &__record {
|
|
|
+ margin-left: auto;
|
|
|
+ }
|
|
|
+
|
|
|
+ &__new {
|
|
|
+ margin-left: 15px;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ &-center {
|
|
|
+ display: flex;
|
|
|
+ width: calc(100% - 30px);
|
|
|
+ background-color: #f4f5f9;
|
|
|
+ margin: 10px 15px 10px 15px;
|
|
|
+ padding: 15px;
|
|
|
+ border-radius: 8px;
|
|
|
+
|
|
|
+ &-textarea {
|
|
|
+ width: 100%;
|
|
|
+ margin-right: 20px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &-button {
|
|
|
+ width: 54px;
|
|
|
+ height: 33px;
|
|
|
+ display: flex;
|
|
|
+ justify-content: flex-end;
|
|
|
+ font-size: 13px;
|
|
|
+ white-space: nowrap;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ &-prompt {
|
|
|
+ margin: 10px 0;
|
|
|
+ color: #adadad;
|
|
|
+ font-size: 20rpx;
|
|
|
+ text-align: center;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+.chatArea {
|
|
|
+ width: 70vw;
|
|
|
+ overflow-y: auto;
|
|
|
+ height: 100vh;
|
|
|
+ padding: 15px;
|
|
|
+
|
|
|
+ .content {
|
|
|
+ &-title {
|
|
|
+ color: #8b8b8b;
|
|
|
+ margin-bottom: 5px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &-item {
|
|
|
+ padding: 10px 5px;
|
|
|
+ border-radius: 5px;
|
|
|
+ letter-spacing: 1px;
|
|
|
+ white-space: nowrap;
|
|
|
+ overflow: hidden;
|
|
|
+ text-overflow: ellipsis; /* 超出部分显示省略号 */
|
|
|
+ position: relative; /* 为伪元素定位 */
|
|
|
+
|
|
|
+ &:active {
|
|
|
+ background-color: #e5e5e5;
|
|
|
+ }
|
|
|
+
|
|
|
+ &:hover {
|
|
|
+ background-color: #e5e5e5;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+</style>
|