Compare commits
2 Commits
9614403d5e
...
4a7f2945f8
Author | SHA1 | Date |
---|---|---|
weipengfei | 4a7f2945f8 | |
weipengfei | ee7b9a341e |
|
@ -23,9 +23,9 @@
|
||||||
watch: { // 监视latex的变化
|
watch: { // 监视latex的变化
|
||||||
latex: {
|
latex: {
|
||||||
handler(newval, oldval) {
|
handler(newval, oldval) {
|
||||||
console.log("--------- 监视latex的变化 ----------")
|
// console.log("--------- 监视latex的变化 ----------")
|
||||||
console.log("oldval", oldval);
|
// console.log("oldval", oldval);
|
||||||
console.log("newval", newval);
|
// console.log("newval", newval);
|
||||||
this.mathlab = newval;
|
this.mathlab = newval;
|
||||||
this.$forceUpdate();
|
this.$forceUpdate();
|
||||||
},
|
},
|
||||||
|
@ -44,9 +44,9 @@
|
||||||
methods: {
|
methods: {
|
||||||
// 接收逻辑层发送的数据
|
// 接收逻辑层发送的数据
|
||||||
receiveLab(newValue, oldValue, ownerVm, vm) {
|
receiveLab(newValue, oldValue, ownerVm, vm) {
|
||||||
console.log("--------- 接收逻辑层发送的数据 ----------")
|
// console.log("--------- 接收逻辑层发送的数据 ----------")
|
||||||
console.log("newValue", newValue);
|
// console.log("newValue", newValue);
|
||||||
console.log("oldValue", oldValue);
|
// console.log("oldValue", oldValue);
|
||||||
|
|
||||||
this.mathlab = newValue;
|
this.mathlab = newValue;
|
||||||
|
|
||||||
|
@ -87,16 +87,16 @@
|
||||||
},
|
},
|
||||||
|
|
||||||
svg: {
|
svg: {
|
||||||
scale: 1.2
|
scale: 1.0
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
(function() {
|
(function() {
|
||||||
var script = document.createElement('script')
|
var script = document.createElement('script')
|
||||||
script.id = 'MathJax-script'
|
script.id = 'MathJax-script'
|
||||||
// cdn引入比较慢
|
// cdn引入比较慢
|
||||||
script.src = 'https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-chtml.js'
|
// script.src = 'https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-chtml.js'
|
||||||
// script.src = 'static/tex-svg.js'; // 下载源码导入比较快,svg比chtml视觉上比较好
|
// script.src = 'static/tex-svg.js'; // 下载源码导入比较快,svg比chtml视觉上比较好
|
||||||
// script.src = 'static/tex-chtml.js';
|
script.src = 'static/tex-chtml.js';
|
||||||
document.head.appendChild(script)
|
document.head.appendChild(script)
|
||||||
})();
|
})();
|
||||||
console.log("initMathJax() window.MathJax", window.MathJax);
|
console.log("initMathJax() window.MathJax", window.MathJax);
|
||||||
|
|
48
pages.json
48
pages.json
|
@ -1,23 +1,33 @@
|
||||||
{
|
{
|
||||||
"pages": [ //pages数组中第一项表示应用启动页,参考:https://uniapp.dcloud.io/collocation/pages
|
"pages": [ //pages数组中第一项表示应用启动页,参考:https://uniapp.dcloud.io/collocation/pages
|
||||||
{
|
|
||||||
"path": "pages/index/index",
|
|
||||||
"style": {
|
|
||||||
"navigationBarTitleText": "讯飞AI"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"path": "pages/index/chat",
|
"path": "pages/index/index",
|
||||||
"style": {
|
"style": {
|
||||||
"navigationBarTitleText": "讯飞星火"
|
"navigationBarTitleText": "讯飞AI",
|
||||||
}
|
"bounce": "none",
|
||||||
|
"softinputMode": "adjustResize"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"path": "pages/index/chat",
|
||||||
|
"style": {
|
||||||
|
"navigationBarTitleText": "讯飞星火",
|
||||||
|
"bounce": "none"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"globalStyle": {
|
"globalStyle": {
|
||||||
"navigationBarTextStyle": "black",
|
"navigationBarTextStyle": "black",
|
||||||
"navigationBarTitleText": "人工智能",
|
"navigationBarTitleText": "人工智能",
|
||||||
"navigationBarBackgroundColor": "#F8F8F8",
|
"navigationBarBackgroundColor": "#F8F8F8",
|
||||||
"backgroundColor": "#F8F8F8"
|
"backgroundColor": "#F8F8F8"
|
||||||
},
|
},
|
||||||
"uniIdRouter": {}
|
"uniIdRouter": {},
|
||||||
|
"condition": { //模式配置,仅开发期间生效
|
||||||
|
"current": 0, //当前激活的模式(list 的索引项)
|
||||||
|
"list": [{
|
||||||
|
"name": "", //模式名称
|
||||||
|
"path": "", //启动页面,必选
|
||||||
|
"query": "" //启动参数,在页面的onLoad函数里面得到
|
||||||
|
}]
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -1,547 +1,165 @@
|
||||||
<template>
|
<template>
|
||||||
<view class="wrapper">
|
<view>
|
||||||
<view class="tips color_fff size_12 align_c" :class="{ 'show':ajax.loading }" @tap="getHistoryMsg">{{ajax.loadText}}
|
带上下文问题(可修改):<input v-model="TEXT" style="border: 1px solid gainsboro;" />
|
||||||
</view>
|
<button style="background-color: blue;color: white;" @click="sendToSpark()">发送给大模型</button>
|
||||||
<view class="box-1" id="list-box">
|
<text style="width: 90%;flex-wrap:wrap">{{sparkResult}}</text>
|
||||||
<view class="talk-list">
|
</view>
|
||||||
<view v-for="(item,index) in talkList" :key="index" :id="`msg-${item.id}`">
|
|
||||||
<view class="item flex_col" :class=" item.type == 1 ? 'push':'pull' ">
|
|
||||||
<image :src="item.pic" mode="aspectFill" class="pic"></image>
|
|
||||||
<!-- <view v-if="talkList.length-1==index" class="content multiline-text">{{c_content}}</view> -->
|
|
||||||
<view class="content multiline-text">
|
|
||||||
<!-- <rich-text :nodes="item.content"></rich-text> -->
|
|
||||||
<bing-math class="bing-math" :latex="item.content"></bing-math>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
<view class="box-2">
|
|
||||||
<view class="flex_col">
|
|
||||||
<view class="flex_grow">
|
|
||||||
<input type="text" class="content" v-model="content" placeholder="请输入聊天内容" @confirm="send"
|
|
||||||
placeholder-style="color:#DDD;" :cursor-spacing="6">
|
|
||||||
</view>
|
|
||||||
<button class="send" @tap="send">发送</button>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import * as base64 from "base-64"
|
import * as base64 from "base-64"
|
||||||
import CryptoJS from '../../static/crypto-js/crypto-js.js'
|
import CryptoJS from '../../static/crypto-js/crypto-js.js'
|
||||||
import parser from '../../static/fast-xml-parser/src/parser'
|
import parser from '../../static/fast-xml-parser/src/parser'
|
||||||
import * as utf8 from "utf8"
|
import * as utf8 from "utf8"
|
||||||
import fetch from 'miniprogram-fetch';
|
export default {
|
||||||
import axios from 'axios';
|
// https://spark-api.xf-yun.com/v1.1/chat V1.5 domain general
|
||||||
import BingMath from "@/components/bing-math/bing-math.vue"
|
// https://spark-api.xf-yun.com/v2.1/chat V2.0 domain generalv2
|
||||||
export default {
|
data() {
|
||||||
components: {
|
return {
|
||||||
'bing-math': BingMath
|
TEXT: '你好,我的名字叫大王',
|
||||||
},
|
APPID: '2eda6c2e', // 控制台获取填写
|
||||||
data() {
|
APISecret: 'MDEyMzE5YTc5YmQ5NjMwOTU1MWY4N2Y2',
|
||||||
return {
|
APIKey: '12ec1f9d113932575fc4b114a2f60ffd',
|
||||||
talkList: [],
|
sparkResult: '',
|
||||||
ajax: {
|
historyTextList: [], // 历史会话信息,由于最大token12000,可以结合实际使用,进行移出
|
||||||
rows: 20, //每页数量
|
tempRes: '' // 临时答复保存
|
||||||
page: 1, //页码
|
}
|
||||||
flag: false, // 请求开关
|
},
|
||||||
loading: false, // 加载中
|
methods: {
|
||||||
loadText: '正在获取消息'
|
async sendToSpark() {
|
||||||
},
|
let myUrl = await this.getWebSocketUrl();
|
||||||
content: '',
|
this.tempRes = "";
|
||||||
c_content: '',
|
// this.sparkResult = "";
|
||||||
n_content: '',
|
let realThis = this;
|
||||||
TEXT: '你好,我的名字叫大王',
|
this.socketTask = uni.connectSocket({
|
||||||
APPID: '2eda6c2e', // 控制台获取填写
|
//url: encodeURI(encodeURI(myUrl).replace(/\+/g, '%2B')),
|
||||||
APISecret: 'MDEyMzE5YTc5YmQ5NjMwOTU1MWY4N2Y2',
|
url: myUrl,
|
||||||
APIKey: '12ec1f9d113932575fc4b114a2f60ffd',
|
method: 'GET',
|
||||||
sparkResult: '',
|
success: res => {
|
||||||
historyTextList: [], // 历史会话信息,由于最大token12000,可以结合实际使用,进行移出
|
console.log(res, "ws成功连接...", myUrl)
|
||||||
tempRes: '', // 临时答复保存
|
realThis.wsLiveFlag = true;
|
||||||
}
|
}
|
||||||
},
|
})
|
||||||
mounted() {
|
realThis.socketTask.onError((res) => {
|
||||||
this.$nextTick(() => {
|
console.log("连接发生错误,请检查appid是否填写", res)
|
||||||
// this.getHistoryMsg();
|
})
|
||||||
});
|
realThis.socketTask.onOpen((res) => {
|
||||||
},
|
this.historyTextList.push({
|
||||||
onPageScroll(e) {
|
"role": "user",
|
||||||
if (e.scrollTop < 5) {
|
"content": this.TEXT
|
||||||
// this.getHistoryMsg();
|
})
|
||||||
}
|
console.info("wss的onOpen成功执行...", res)
|
||||||
},
|
// 第一帧..........................................
|
||||||
watch: {
|
console.log('open成功...')
|
||||||
n_content(n, o) {
|
let params = {
|
||||||
this.c_content = n;
|
"header": {
|
||||||
}
|
"app_id": this.APPID,
|
||||||
},
|
"uid": "aef9f963-7"
|
||||||
methods: {
|
},
|
||||||
copyText(str) {
|
"parameter": {
|
||||||
uni.setClipboardData({
|
"chat": {
|
||||||
data: str,
|
"domain": "generalv2",
|
||||||
success: function () {
|
"temperature": 0.5,
|
||||||
uni.showToast({
|
"max_tokens": 1024
|
||||||
icon: 'none',
|
}
|
||||||
title: '复制成功'
|
},
|
||||||
});
|
"payload": {
|
||||||
}
|
"message": {
|
||||||
});
|
"text": this.historyTextList
|
||||||
},
|
}
|
||||||
// 获取历史消息
|
}
|
||||||
getHistoryMsg() {
|
};
|
||||||
if (!this.ajax.flag) {
|
console.log("请求的params:" + JSON.stringify(params))
|
||||||
return; //
|
this.sparkResult = this.sparkResult + "\r\n我:" + this.TEXT + "\r\n"
|
||||||
}
|
this.sparkResult = this.sparkResult + "大模型:"
|
||||||
|
console.log("发送第一帧...", params)
|
||||||
|
realThis.socketTask.send({ // 发送消息,,都用uni的官方版本
|
||||||
|
data: JSON.stringify(params),
|
||||||
|
success() {
|
||||||
|
console.log('第一帧发送成功')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// 此处用到 ES7 的 async/await 知识,为使代码更加优美。不懂的请自行学习。
|
// 接受到消息时
|
||||||
let get = async () => {
|
realThis.socketTask.onMessage((res) => {
|
||||||
this.hideLoadTips();
|
console.log('收到API返回的内容:', res.data);
|
||||||
this.ajax.flag = false;
|
let obj = JSON.parse(res.data)
|
||||||
let data = await this.joinHistoryMsg();
|
// console.log("我打印的"+obj.payload);
|
||||||
|
let dataArray = obj.payload.choices.text;
|
||||||
|
for (let i = 0; i < dataArray.length; i++) {
|
||||||
|
realThis.sparkResult = realThis.sparkResult + dataArray[i].content
|
||||||
|
realThis.tempRes = realThis.tempRes + dataArray[i].content
|
||||||
|
}
|
||||||
|
// realThis.sparkResult =realThis.sparkResult+
|
||||||
|
let temp = JSON.parse(res.data)
|
||||||
|
// console.log("0726",temp.header.code)
|
||||||
|
if (temp.header.code !== 0) {
|
||||||
|
console.log(`${temp.header.code}:${temp.message}`);
|
||||||
|
realThis.socketTask.close({
|
||||||
|
success(res) {
|
||||||
|
console.log('关闭成功', res)
|
||||||
|
realThis.wsLiveFlag = false;
|
||||||
|
},
|
||||||
|
fail(err) {
|
||||||
|
console.log('关闭失败', err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
if (temp.header.code === 0) {
|
||||||
|
if (res.data && temp.header.status === 2) {
|
||||||
|
realThis.sparkResult = realThis.sparkResult +
|
||||||
|
"\r\n**********************************************"
|
||||||
|
this.historyTextList.push({
|
||||||
|
"role": "assistant",
|
||||||
|
"content": this.tempRes
|
||||||
|
})
|
||||||
|
/* let dataArray= obj.payload.choices.text;
|
||||||
|
for(let i=0;i<dataArray.length;i++){
|
||||||
|
realThis.sparkResult =realThis.sparkResult+ dataArray[i].content
|
||||||
|
} */
|
||||||
|
setTimeout(() => {
|
||||||
|
realThis.socketTask.close({
|
||||||
|
success(res) {
|
||||||
|
console.log('关闭成功', res)
|
||||||
|
},
|
||||||
|
fail(err) {
|
||||||
|
// console.log('关闭失败', err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, 1000)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
// 鉴权
|
||||||
|
getWebSocketUrl() {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
// https://spark-api.xf-yun.com/v1.1/chat V1.5 domain general
|
||||||
|
// https://spark-api.xf-yun.com/v2.1/chat V2.0 domain generalv2
|
||||||
|
var url = "wss://spark-api.xf-yun.com/v2.1/chat";
|
||||||
|
var host = "spark-api.xf-yun.com";
|
||||||
|
var apiKeyName = "api_key";
|
||||||
|
console.log(new Date().toGMTString());
|
||||||
|
var date = new Date().toGMTString();
|
||||||
|
var algorithm = "hmac-sha256";
|
||||||
|
var headers = "host date request-line";
|
||||||
|
var signatureOrigin = `host: ${host}\ndate: ${date}\nGET /v2.1/chat HTTP/1.1`;
|
||||||
|
var signatureSha = CryptoJS.HmacSHA256(signatureOrigin, this.APISecret);
|
||||||
|
var signature = CryptoJS.enc.Base64.stringify(signatureSha);
|
||||||
|
var authorizationOrigin =
|
||||||
|
`${apiKeyName}="${this.APIKey}", algorithm="${algorithm}", headers="${headers}", signature="${signature}"`;
|
||||||
|
var authorization = base64.encode(authorizationOrigin);
|
||||||
|
url = `${url}?authorization=${authorization}&date=${encodeURI(date)}&host=${host}`;
|
||||||
|
|
||||||
console.log('----- 模拟数据格式,供参考 -----');
|
// console.log(url)
|
||||||
console.log(data); // 查看请求返回的数据结构
|
|
||||||
|
|
||||||
// 获取待滚动元素选择器,解决插入数据后,滚动条定位时使用
|
|
||||||
let selector = '';
|
|
||||||
|
|
||||||
if (this.ajax.page > 1) {
|
|
||||||
// 非第一页,则取历史消息数据的第一条信息元素
|
|
||||||
selector = `#msg-${this.talkList[0].id}`;
|
|
||||||
} else {
|
|
||||||
// 第一页,则取当前消息数据的最后一条信息元素
|
|
||||||
selector = `#msg-${data[data.length-1].id}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 将获取到的消息数据合并到消息数组中
|
|
||||||
this.talkList = [...data, ...this.talkList];
|
|
||||||
|
|
||||||
// 数据挂载后执行,不懂的请自行阅读 Vue.js 文档对 Vue.nextTick 函数说明。
|
|
||||||
this.$nextTick(() => {
|
|
||||||
// 设置当前滚动的位置
|
|
||||||
this.setPageScrollTo(selector);
|
|
||||||
|
|
||||||
this.hideLoadTips(true);
|
|
||||||
|
|
||||||
if (data.length < this.ajax.rows) {
|
|
||||||
// 当前消息数据条数小于请求要求条数时,则无更多消息,不再允许请求。
|
|
||||||
// 可在此处编写无更多消息数据时的逻辑
|
|
||||||
} else {
|
|
||||||
this.ajax.page++;
|
|
||||||
|
|
||||||
// 延迟 200ms ,以保证设置窗口滚动已完成
|
|
||||||
setTimeout(() => {
|
|
||||||
this.ajax.flag = true;
|
|
||||||
}, 200)
|
|
||||||
}
|
|
||||||
|
|
||||||
})
|
|
||||||
}
|
|
||||||
get();
|
|
||||||
},
|
|
||||||
// 拼接历史记录消息,正式项目可替换为请求历史记录接口
|
|
||||||
joinHistoryMsg() {
|
|
||||||
let join = () => {
|
|
||||||
let arr = [];
|
|
||||||
|
|
||||||
//通过当前页码及页数,模拟数据内容
|
|
||||||
let startIndex = (this.ajax.page - 1) * this.ajax.rows;
|
|
||||||
let endIndex = startIndex + this.ajax.rows;
|
|
||||||
for (let i = startIndex; i < endIndex; i++) {
|
|
||||||
arr.push({
|
|
||||||
"id": i, // 消息的ID
|
|
||||||
"content": `这是历史记录的第${i+1}条消息`, // 消息内容
|
|
||||||
"type": Math.random() > 0.5 ? 1 : 0, // 此为消息类别,设 1 为发出去的消息,0 为收到对方的消息,
|
|
||||||
"pic": "/static/avatar.png" // 头像
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
颠倒数组中元素的顺序。将最新的数据排在本次接口返回数据的最后面。
|
|
||||||
后端接口按 消息的时间降序查找出当前页的数据后,再将本页数据按消息时间降序排序返回。
|
|
||||||
这是数据的重点,因为页面滚动条和上拉加载历史的问题。
|
|
||||||
*/
|
|
||||||
arr.reverse();
|
|
||||||
|
|
||||||
return arr;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 此处用到 ES6 的 Promise 知识,不懂的请自行学习。
|
|
||||||
return new Promise((done, fail) => {
|
|
||||||
// 无数据请求接口,由 setTimeout 模拟,正式项目替换为 ajax 即可。
|
|
||||||
setTimeout(() => {
|
|
||||||
let data = join();
|
|
||||||
done(data);
|
|
||||||
}, 1500);
|
|
||||||
})
|
|
||||||
},
|
|
||||||
// 设置页面滚动位置
|
|
||||||
setPageScrollTo(selector) {
|
|
||||||
let view = uni.createSelectorQuery().in(this).select(selector);
|
|
||||||
view.boundingClientRect((res) => {
|
|
||||||
uni.pageScrollTo({
|
|
||||||
scrollTop: res.top - 30, // -30 为多显示出大半个消息的高度,示意上面还有信息。
|
|
||||||
duration: 0
|
|
||||||
});
|
|
||||||
}).exec();
|
|
||||||
},
|
|
||||||
// 隐藏加载提示
|
|
||||||
hideLoadTips(flag) {
|
|
||||||
if (flag) {
|
|
||||||
this.ajax.loadText = '消息获取成功';
|
|
||||||
setTimeout(() => {
|
|
||||||
this.ajax.loading = false;
|
|
||||||
}, 300);
|
|
||||||
} else {
|
|
||||||
this.ajax.loading = true;
|
|
||||||
this.ajax.loadText = '正在获取消息';
|
|
||||||
}
|
|
||||||
},
|
|
||||||
// 发送信息
|
|
||||||
send() {
|
|
||||||
if (!this.content) {
|
|
||||||
uni.showToast({
|
|
||||||
title: '请输入有效的内容',
|
|
||||||
icon: 'none'
|
|
||||||
})
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 将当前发送信息 添加到消息列表。
|
|
||||||
let data = {
|
|
||||||
"id": new Date().getTime(),
|
|
||||||
"content": this.content,
|
|
||||||
"type": 1,
|
|
||||||
"pic": "/static/avatar.png"
|
|
||||||
}
|
|
||||||
this.TEXT = this.content;
|
|
||||||
this.sendToSpark();
|
|
||||||
this.talkList.push(data);
|
|
||||||
this.talkList.push({
|
|
||||||
"id": new Date().getTime(),
|
|
||||||
"content": '',
|
|
||||||
"type": 2,
|
|
||||||
"pic": "/static/avatar.png"
|
|
||||||
});
|
|
||||||
|
|
||||||
this.$nextTick(() => {
|
|
||||||
// 清空内容框中的内容
|
|
||||||
this.content = '';
|
|
||||||
uni.pageScrollTo({
|
|
||||||
scrollTop: 999999, // 设置一个超大值,以保证滚动条滚动到底部
|
|
||||||
duration: 0
|
|
||||||
});
|
|
||||||
})
|
|
||||||
},
|
|
||||||
async sendToSpark() {
|
|
||||||
let myUrl = await this.getWebSocketUrl();
|
|
||||||
this.tempRes = "";
|
|
||||||
// this.sparkResult = "";
|
|
||||||
let realThis = this;
|
|
||||||
this.socketTask = uni.connectSocket({
|
|
||||||
//url: encodeURI(encodeURI(myUrl).replace(/\+/g, '%2B')),
|
|
||||||
url: myUrl,
|
|
||||||
method: 'GET',
|
|
||||||
success: res => {
|
|
||||||
console.log(res, "ws成功连接...", myUrl)
|
|
||||||
realThis.wsLiveFlag = true;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
realThis.socketTask.onError((res) => {
|
|
||||||
console.log("连接发生错误,请检查appid是否填写", res)
|
|
||||||
})
|
|
||||||
realThis.socketTask.onOpen((res) => {
|
|
||||||
this.historyTextList.push({
|
|
||||||
"role": "user",
|
|
||||||
"content": this.TEXT
|
|
||||||
})
|
|
||||||
console.info("wss的onOpen成功执行...", res)
|
|
||||||
// 第一帧..........................................
|
|
||||||
console.log('open成功...')
|
|
||||||
let params = {
|
|
||||||
"header": {
|
|
||||||
"app_id": this.APPID,
|
|
||||||
"uid": "aef9f963-7"
|
|
||||||
},
|
|
||||||
"parameter": {
|
|
||||||
"chat": {
|
|
||||||
"domain": "generalv2",
|
|
||||||
"temperature": 0.5,
|
|
||||||
"max_tokens": 1024
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"payload": {
|
|
||||||
"message": {
|
|
||||||
"text": this.historyTextList
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
console.log("请求的params:" + JSON.stringify(params))
|
|
||||||
this.sparkResult = this.sparkResult + "\r\n我:" + this.TEXT + "\r\n"
|
|
||||||
this.sparkResult = this.sparkResult + "大模型:"
|
|
||||||
console.log("发送第一帧...", params)
|
|
||||||
realThis.socketTask.send({ // 发送消息,,都用uni的官方版本
|
|
||||||
data: JSON.stringify(params),
|
|
||||||
success() {
|
|
||||||
console.log('第一帧发送成功')
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// 接受到消息时
|
|
||||||
realThis.socketTask.onMessage((res) => {
|
|
||||||
console.log('收到API返回的内容:', res.data);
|
|
||||||
let obj = JSON.parse(res.data)
|
|
||||||
// console.log("我打印的"+obj.payload);
|
|
||||||
let dataArray = obj.payload.choices.text;
|
|
||||||
for (let i = 0; i < dataArray.length; i++) {
|
|
||||||
realThis.sparkResult = realThis.sparkResult + dataArray[i].content;
|
|
||||||
this.talkList[this.talkList.length - 1].content += dataArray[i].content;
|
|
||||||
this.n_content = this.talkList[this.talkList.length - 1].content;
|
|
||||||
realThis.tempRes = realThis.tempRes + dataArray[i].content
|
|
||||||
}
|
|
||||||
this.$nextTick(() => {
|
|
||||||
uni.pageScrollTo({
|
|
||||||
scrollTop: 9999999,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
// realThis.sparkResult =realThis.sparkResult+
|
|
||||||
let temp = JSON.parse(res.data)
|
|
||||||
// console.log("0726",temp.header.code)
|
|
||||||
if (temp.header.code !== 0) {
|
|
||||||
console.log(`${temp.header.code}:${temp.message}`);
|
|
||||||
realThis.socketTask.close({
|
|
||||||
success(res) {
|
|
||||||
console.log('关闭成功', res)
|
|
||||||
realThis.wsLiveFlag = false;
|
|
||||||
},
|
|
||||||
fail(err) {
|
|
||||||
console.log('关闭失败', err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if (temp.header.code === 0) {
|
|
||||||
if (res.data && temp.header.status === 2) {
|
|
||||||
realThis.sparkResult = realThis.sparkResult +
|
|
||||||
"\r\n**********************************************"
|
|
||||||
this.historyTextList.push({
|
|
||||||
"role": "assistant",
|
|
||||||
"content": this.tempRes
|
|
||||||
})
|
|
||||||
/* let dataArray= obj.payload.choices.text;
|
|
||||||
for(let i=0;i<dataArray.length;i++){
|
|
||||||
realThis.sparkResult =realThis.sparkResult+ dataArray[i].content
|
|
||||||
} */
|
|
||||||
setTimeout(() => {
|
|
||||||
realThis.socketTask.close({
|
|
||||||
success(res) {
|
|
||||||
console.log('关闭成功', res)
|
|
||||||
},
|
|
||||||
fail(err) {
|
|
||||||
// console.log('关闭失败', err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}, 1000)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
},
|
|
||||||
// 鉴权
|
|
||||||
getWebSocketUrl() {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
// https://spark-api.xf-yun.com/v1.1/chat V1.5 domain general
|
|
||||||
// https://spark-api.xf-yun.com/v2.1/chat V2.0 domain generalv2
|
|
||||||
var url = "wss://spark-api.xf-yun.com/v2.1/chat";
|
|
||||||
var host = "spark-api.xf-yun.com";
|
|
||||||
var apiKeyName = "api_key";
|
|
||||||
var date = new Date().toGMTString();
|
|
||||||
var algorithm = "hmac-sha256";
|
|
||||||
var headers = "host date request-line";
|
|
||||||
var signatureOrigin = `host: ${host}\ndate: ${date}\nGET /v2.1/chat HTTP/1.1`;
|
|
||||||
var signatureSha = CryptoJS.HmacSHA256(signatureOrigin, this.APISecret);
|
|
||||||
var signature = CryptoJS.enc.Base64.stringify(signatureSha);
|
|
||||||
var authorizationOrigin =
|
|
||||||
`${apiKeyName}="${this.APIKey}", algorithm="${algorithm}", headers="${headers}", signature="${signature}"`;
|
|
||||||
var authorization = base64.encode(authorizationOrigin);
|
|
||||||
url = `${url}?authorization=${authorization}&date=${encodeURI(date)}&host=${host}`;
|
|
||||||
|
|
||||||
// console.log(url)
|
|
||||||
resolve(url);
|
resolve(url);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style>
|
||||||
@import "../../lib/global.scss";
|
|
||||||
|
|
||||||
page {
|
|
||||||
background-color: #F3F3F3;
|
|
||||||
font-size: 28rpx;
|
|
||||||
}
|
|
||||||
|
|
||||||
.wrapper {
|
|
||||||
height: auto !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 加载数据提示 */
|
|
||||||
.tips {
|
|
||||||
position: fixed;
|
|
||||||
left: 0;
|
|
||||||
top: var(--window-top);
|
|
||||||
width: 100%;
|
|
||||||
z-index: 9;
|
|
||||||
background-color: rgba(0, 0, 0, 0.15);
|
|
||||||
height: 72rpx;
|
|
||||||
line-height: 72rpx;
|
|
||||||
transform: translateY(-80rpx);
|
|
||||||
transition: transform 0.3s ease-in-out 0s;
|
|
||||||
|
|
||||||
&.show {
|
|
||||||
transform: translateY(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.box-1 {
|
|
||||||
width: 100%;
|
|
||||||
height: auto;
|
|
||||||
padding-bottom: 100rpx;
|
|
||||||
box-sizing: content-box;
|
|
||||||
|
|
||||||
/* 兼容iPhoneX */
|
|
||||||
margin-bottom: 0;
|
|
||||||
margin-bottom: constant(safe-area-inset-bottom);
|
|
||||||
margin-bottom: env(safe-area-inset-bottom);
|
|
||||||
}
|
|
||||||
|
|
||||||
.multiline-text {
|
|
||||||
white-space: pre-line;
|
|
||||||
/* 或 white-space: pre-wrap; */
|
|
||||||
}
|
|
||||||
|
|
||||||
.box-2 {
|
|
||||||
position: fixed;
|
|
||||||
left: 0;
|
|
||||||
width: 100%;
|
|
||||||
bottom: 0;
|
|
||||||
height: auto;
|
|
||||||
z-index: 2;
|
|
||||||
border-top: #e5e5e5 solid 1px;
|
|
||||||
box-sizing: content-box;
|
|
||||||
background-color: #F3F3F3;
|
|
||||||
|
|
||||||
/* 兼容iPhoneX */
|
|
||||||
padding-bottom: 0;
|
|
||||||
padding-bottom: constant(safe-area-inset-bottom);
|
|
||||||
padding-bottom: env(safe-area-inset-bottom);
|
|
||||||
|
|
||||||
>view {
|
|
||||||
padding: 0 20rpx;
|
|
||||||
height: 100rpx;
|
|
||||||
}
|
|
||||||
|
|
||||||
.content {
|
|
||||||
background-color: #fff;
|
|
||||||
height: 64rpx;
|
|
||||||
padding: 0 20rpx;
|
|
||||||
border-radius: 6rpx;
|
|
||||||
font-size: 28rpx;
|
|
||||||
}
|
|
||||||
|
|
||||||
.send {
|
|
||||||
background-color: #2573fb;
|
|
||||||
color: #fff;
|
|
||||||
height: 64rpx;
|
|
||||||
margin-left: 20rpx;
|
|
||||||
border-radius: 6rpx;
|
|
||||||
padding: 0;
|
|
||||||
width: 120rpx;
|
|
||||||
line-height: 62rpx;
|
|
||||||
|
|
||||||
&:active {
|
|
||||||
background-color: #1573fb;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.talk-list {
|
|
||||||
padding-bottom: 20rpx;
|
|
||||||
|
|
||||||
/* 消息项,基础类 */
|
|
||||||
.item {
|
|
||||||
padding: 20rpx 20rpx 0 20rpx;
|
|
||||||
align-items: flex-start;
|
|
||||||
align-content: flex-start;
|
|
||||||
color: #333;
|
|
||||||
|
|
||||||
.pic {
|
|
||||||
width: 92rpx;
|
|
||||||
height: 92rpx;
|
|
||||||
border-radius: 50%;
|
|
||||||
border: #fff solid 1px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.content {
|
|
||||||
padding: 20rpx;
|
|
||||||
border-radius: 4px;
|
|
||||||
max-width: 500rpx;
|
|
||||||
word-break: break-all;
|
|
||||||
line-height: 52rpx;
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 收到的消息 */
|
|
||||||
&.pull {
|
|
||||||
.content {
|
|
||||||
margin-left: 32rpx;
|
|
||||||
background-color: #fff;
|
|
||||||
|
|
||||||
&::after {
|
|
||||||
content: '';
|
|
||||||
display: block;
|
|
||||||
width: 0;
|
|
||||||
height: 0;
|
|
||||||
border-top: 16rpx solid transparent;
|
|
||||||
border-bottom: 16rpx solid transparent;
|
|
||||||
border-right: 20rpx solid #fff;
|
|
||||||
position: absolute;
|
|
||||||
top: 30rpx;
|
|
||||||
left: -18rpx;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 发出的消息 */
|
|
||||||
&.push {
|
|
||||||
/* 主轴为水平方向,起点在右端。使不修改DOM结构,也能改变元素排列顺序 */
|
|
||||||
flex-direction: row-reverse;
|
|
||||||
|
|
||||||
.content {
|
|
||||||
margin-right: 32rpx;
|
|
||||||
background-color: #2573fb;
|
|
||||||
color: #fff;
|
|
||||||
|
|
||||||
&::after {
|
|
||||||
content: '';
|
|
||||||
display: block;
|
|
||||||
width: 0;
|
|
||||||
height: 0;
|
|
||||||
border-top: 16rpx solid transparent;
|
|
||||||
border-bottom: 16rpx solid transparent;
|
|
||||||
border-left: 20rpx solid #2573fb;
|
|
||||||
position: absolute;
|
|
||||||
top: 30rpx;
|
|
||||||
right: -18rpx;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.bing-math{
|
|
||||||
margin: 0 !important;
|
|
||||||
padding: 0 !important;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
|
@ -1,165 +1,646 @@
|
||||||
<template>
|
<template>
|
||||||
<view>
|
<view class="wrapper">
|
||||||
带上下文问题(可修改):<input v-model="TEXT" style="border: 1px solid gainsboro;" />
|
<view class="tips color_fff size_12 align_c" :class="{ 'show':ajax.loading }" @tap="getHistoryMsg">{{ajax.loadText}}
|
||||||
<button style="background-color: blue;color: white;" @click="sendToSpark()">发送给大模型</button>
|
</view>
|
||||||
<text style="width: 90%;flex-wrap:wrap">{{sparkResult}}</text>
|
<view class="placeholder"></view>
|
||||||
</view>
|
<view class="box-1" id="list-box" ref="box">
|
||||||
|
<view class="talk-list">
|
||||||
|
<scroll-view :scroll-y="true" style="height: 100%;overflow: hidden;" :scroll-top="scrollTop"
|
||||||
|
:scroll-with-animation="true" @scrolltoupper="bindScroll">
|
||||||
|
<view v-for="(item,index) in talkList" :key="index" :id="`msg-${item.id}`">
|
||||||
|
<view class="item flex_col" :class=" item.type == 1 ? 'push':'pull' ">
|
||||||
|
<image :src="item.pic" mode="aspectFill" class="pic"></image>
|
||||||
|
<view v-if="talkList.length-1==index" class="content multiline-text">
|
||||||
|
<!-- <rich-text :nodes="item.content"></rich-text> -->
|
||||||
|
<bing-math v-if="c_content!=''" :key="`math-${item.id}`" class="bing-math" :latex="c_content"></bing-math>
|
||||||
|
<view v-else class="blinking-box">|</view>
|
||||||
|
</view>
|
||||||
|
<view v-else class="content multiline-text">
|
||||||
|
<!-- <rich-text :nodes="item.content"></rich-text> -->
|
||||||
|
<bing-math :key="`math-${item.id}`" class="bing-math" :latex="item.content"></bing-math>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</scroll-view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="box-2">
|
||||||
|
<view class="flex_col">
|
||||||
|
<view class="flex_grow">
|
||||||
|
<input type="text" class="content" v-model="content" placeholder="请输入聊天内容" @focus="focus" @confirm="send"
|
||||||
|
placeholder-style="color:#DDD;" :cursor-spacing="6">
|
||||||
|
</view>
|
||||||
|
<button class="send" @tap="send">发送</button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- <view v-show="showplc" :style="{'min-height': (keyboardHeight+200)+'px'}" class="placeholder">显示</view> -->
|
||||||
|
</view>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import * as base64 from "base-64"
|
import * as base64 from "base-64"
|
||||||
import CryptoJS from '../../static/crypto-js/crypto-js.js'
|
import CryptoJS from '../../static/crypto-js/crypto-js.js'
|
||||||
import parser from '../../static/fast-xml-parser/src/parser'
|
import parser from '../../static/fast-xml-parser/src/parser'
|
||||||
import * as utf8 from "utf8"
|
import * as utf8 from "utf8"
|
||||||
export default {
|
import fetch from 'miniprogram-fetch';
|
||||||
// https://spark-api.xf-yun.com/v1.1/chat V1.5 domain general
|
import axios from 'axios';
|
||||||
// https://spark-api.xf-yun.com/v2.1/chat V2.0 domain generalv2
|
import BingMath from "@/components/bing-math/bing-math.vue"
|
||||||
data() {
|
export default {
|
||||||
return {
|
components: {
|
||||||
TEXT: '你好,我的名字叫大王',
|
'bing-math': BingMath
|
||||||
APPID: '2eda6c2e', // 控制台获取填写
|
},
|
||||||
APISecret: 'MDEyMzE5YTc5YmQ5NjMwOTU1MWY4N2Y2',
|
data() {
|
||||||
APIKey: '12ec1f9d113932575fc4b114a2f60ffd',
|
return {
|
||||||
sparkResult: '',
|
talkList: [],
|
||||||
historyTextList: [], // 历史会话信息,由于最大token12000,可以结合实际使用,进行移出
|
ajax: {
|
||||||
tempRes: '' // 临时答复保存
|
rows: 20, //每页数量
|
||||||
}
|
page: 1, //页码
|
||||||
},
|
flag: false, // 请求开关
|
||||||
methods: {
|
loading: false, // 加载中
|
||||||
async sendToSpark() {
|
loadText: '正在获取消息'
|
||||||
let myUrl = await this.getWebSocketUrl();
|
},
|
||||||
this.tempRes = "";
|
keyboardHeight: 0,
|
||||||
// this.sparkResult = "";
|
showplc: true,
|
||||||
let realThis = this;
|
content: '',
|
||||||
this.socketTask = uni.connectSocket({
|
c_content: '',
|
||||||
//url: encodeURI(encodeURI(myUrl).replace(/\+/g, '%2B')),
|
n_content: '',
|
||||||
url: myUrl,
|
timer: '',
|
||||||
method: 'GET',
|
socketTask: {},
|
||||||
success: res => {
|
TEXT: '',
|
||||||
console.log(res, "ws成功连接...", myUrl)
|
historyTextList: [], // 历史会话信息,由于最大token12000,可以结合实际使用,进行移出
|
||||||
realThis.wsLiveFlag = true;
|
tempRes: '', // 临时答复保存
|
||||||
}
|
scrollTop: 0
|
||||||
})
|
}
|
||||||
realThis.socketTask.onError((res) => {
|
},
|
||||||
console.log("连接发生错误,请检查appid是否填写", res)
|
mounted() {
|
||||||
})
|
this.$nextTick(() => {
|
||||||
realThis.socketTask.onOpen((res) => {
|
this.getHistoryMsg();
|
||||||
this.historyTextList.push({
|
});
|
||||||
"role": "user",
|
uni.onKeyboardHeightChange(e => {
|
||||||
"content": this.TEXT
|
let h = this.keyboardHeight;
|
||||||
})
|
this.keyboardHeight = e.height;
|
||||||
console.info("wss的onOpen成功执行...", res)
|
if(e.height==0)h *= -1;
|
||||||
// 第一帧..........................................
|
else h = e.height;
|
||||||
console.log('open成功...')
|
uni.createSelectorQuery().selectViewport().scrollOffset(function(res) {
|
||||||
let params = {
|
let scrollTop = res.scrollTop; // 页面滚动距离
|
||||||
"header": {
|
// let os = uni.getSystemInfoSync();
|
||||||
"app_id": this.APPID,
|
// // 可视窗口高度 = 设备可使用高度 - 状态栏高度 - 导航栏高度
|
||||||
"uid": "aef9f963-7"
|
// const viewportHeight = os.windowHeight - os.statusBarHeight - (os.platform == "android" ? 50 : 45)-10;
|
||||||
},
|
// if(scrollTop<viewportHeight)scrollTop=viewportHeight;
|
||||||
"parameter": {
|
scrollTop+=h;
|
||||||
"chat": {
|
uni.pageScrollTo({
|
||||||
"domain": "generalv2",
|
scrollTop: scrollTop, // 当前位置向下滚动
|
||||||
"temperature": 0.5,
|
duration: 0 // 滚动过渡时间为300ms,默认值为300ms
|
||||||
"max_tokens": 1024
|
});
|
||||||
}
|
}).exec();
|
||||||
},
|
})
|
||||||
"payload": {
|
},
|
||||||
"message": {
|
beforeDestroy() {
|
||||||
"text": this.historyTextList
|
// #ifdef APP-PLUS
|
||||||
}
|
uni.offKeyboardHeightChange();
|
||||||
}
|
// #endif
|
||||||
};
|
},
|
||||||
console.log("请求的params:" + JSON.stringify(params))
|
onPageScroll(e) {
|
||||||
this.sparkResult = this.sparkResult + "\r\n我:" + this.TEXT + "\r\n"
|
if (e.scrollTop < 5) {
|
||||||
this.sparkResult = this.sparkResult + "大模型:"
|
this.getHistoryMsg();
|
||||||
console.log("发送第一帧...", params)
|
}
|
||||||
realThis.socketTask.send({ // 发送消息,,都用uni的官方版本
|
},
|
||||||
data: JSON.stringify(params),
|
watch: {
|
||||||
success() {
|
n_content(n, o) {
|
||||||
console.log('第一帧发送成功')
|
// this.c_content = n;
|
||||||
}
|
if (this.timer) clearInterval(this.timer);
|
||||||
});
|
let cl = this.c_content.length;
|
||||||
});
|
let nc = this.n_content.split('')
|
||||||
|
this.timer = setInterval(() => {
|
||||||
|
if (cl < nc.length) {
|
||||||
|
this.c_content += nc[cl];
|
||||||
|
cl++;
|
||||||
|
if (cl % 6 == 0) this.$nextTick(()=>{
|
||||||
|
uni.pageScrollTo({
|
||||||
|
scrollTop: 999999,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
clearInterval(this.timer);
|
||||||
|
this.$nextTick(() => {
|
||||||
|
uni.pageScrollTo({
|
||||||
|
scrollTop: 9999999,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}, 60)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
copyText(str) {
|
||||||
|
uni.setClipboardData({
|
||||||
|
data: str,
|
||||||
|
success: function() {
|
||||||
|
uni.showToast({
|
||||||
|
icon: 'none',
|
||||||
|
title: '复制成功'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
// 获取历史消息
|
||||||
|
getHistoryMsg() {
|
||||||
|
if (!this.ajax.flag) {
|
||||||
|
return; //
|
||||||
|
}
|
||||||
|
|
||||||
// 接受到消息时
|
let get = async () => {
|
||||||
realThis.socketTask.onMessage((res) => {
|
this.hideLoadTips();
|
||||||
console.log('收到API返回的内容:', res.data);
|
this.ajax.flag = false;
|
||||||
let obj = JSON.parse(res.data)
|
let data = await this.joinHistoryMsg();
|
||||||
// console.log("我打印的"+obj.payload);
|
|
||||||
let dataArray = obj.payload.choices.text;
|
|
||||||
for (let i = 0; i < dataArray.length; i++) {
|
|
||||||
realThis.sparkResult = realThis.sparkResult + dataArray[i].content
|
|
||||||
realThis.tempRes = realThis.tempRes + dataArray[i].content
|
|
||||||
}
|
|
||||||
// realThis.sparkResult =realThis.sparkResult+
|
|
||||||
let temp = JSON.parse(res.data)
|
|
||||||
// console.log("0726",temp.header.code)
|
|
||||||
if (temp.header.code !== 0) {
|
|
||||||
console.log(`${temp.header.code}:${temp.message}`);
|
|
||||||
realThis.socketTask.close({
|
|
||||||
success(res) {
|
|
||||||
console.log('关闭成功', res)
|
|
||||||
realThis.wsLiveFlag = false;
|
|
||||||
},
|
|
||||||
fail(err) {
|
|
||||||
console.log('关闭失败', err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if (temp.header.code === 0) {
|
|
||||||
if (res.data && temp.header.status === 2) {
|
|
||||||
realThis.sparkResult = realThis.sparkResult +
|
|
||||||
"\r\n**********************************************"
|
|
||||||
this.historyTextList.push({
|
|
||||||
"role": "assistant",
|
|
||||||
"content": this.tempRes
|
|
||||||
})
|
|
||||||
/* let dataArray= obj.payload.choices.text;
|
|
||||||
for(let i=0;i<dataArray.length;i++){
|
|
||||||
realThis.sparkResult =realThis.sparkResult+ dataArray[i].content
|
|
||||||
} */
|
|
||||||
setTimeout(() => {
|
|
||||||
realThis.socketTask.close({
|
|
||||||
success(res) {
|
|
||||||
console.log('关闭成功', res)
|
|
||||||
},
|
|
||||||
fail(err) {
|
|
||||||
// console.log('关闭失败', err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}, 1000)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
},
|
|
||||||
// 鉴权
|
|
||||||
getWebSocketUrl() {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
// https://spark-api.xf-yun.com/v1.1/chat V1.5 domain general
|
|
||||||
// https://spark-api.xf-yun.com/v2.1/chat V2.0 domain generalv2
|
|
||||||
var url = "wss://spark-api.xf-yun.com/v2.1/chat";
|
|
||||||
var host = "spark-api.xf-yun.com";
|
|
||||||
var apiKeyName = "api_key";
|
|
||||||
console.log(new Date().toGMTString());
|
|
||||||
var date = new Date().toGMTString();
|
|
||||||
var algorithm = "hmac-sha256";
|
|
||||||
var headers = "host date request-line";
|
|
||||||
var signatureOrigin = `host: ${host}\ndate: ${date}\nGET /v2.1/chat HTTP/1.1`;
|
|
||||||
var signatureSha = CryptoJS.HmacSHA256(signatureOrigin, this.APISecret);
|
|
||||||
var signature = CryptoJS.enc.Base64.stringify(signatureSha);
|
|
||||||
var authorizationOrigin =
|
|
||||||
`${apiKeyName}="${this.APIKey}", algorithm="${algorithm}", headers="${headers}", signature="${signature}"`;
|
|
||||||
var authorization = base64.encode(authorizationOrigin);
|
|
||||||
url = `${url}?authorization=${authorization}&date=${encodeURI(date)}&host=${host}`;
|
|
||||||
|
|
||||||
// console.log(url)
|
console.log('----- 模拟数据格式,供参考 -----');
|
||||||
|
console.log(data); // 查看请求返回的数据结构
|
||||||
|
|
||||||
|
// 获取待滚动元素选择器,解决插入数据后,滚动条定位时使用
|
||||||
|
let selector = '';
|
||||||
|
|
||||||
|
if (this.ajax.page > 1) {
|
||||||
|
// 非第一页,则取历史消息数据的第一条信息元素
|
||||||
|
selector = `#msg-${this.talkList[0].id}`;
|
||||||
|
} else {
|
||||||
|
// 第一页,则取当前消息数据的最后一条信息元素
|
||||||
|
selector = `#msg-${data[data.length-1].id}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将获取到的消息数据合并到消息数组中
|
||||||
|
this.talkList = [...data, ...this.talkList];
|
||||||
|
|
||||||
|
// 数据挂载后执行,不懂的请自行阅读 Vue.js 文档对 Vue.nextTick 函数说明。
|
||||||
|
this.$nextTick(() => {
|
||||||
|
// 设置当前滚动的位置
|
||||||
|
this.setPageScrollTo(selector);
|
||||||
|
|
||||||
|
this.hideLoadTips(true);
|
||||||
|
|
||||||
|
if (data.length < this.ajax.rows) {
|
||||||
|
// 当前消息数据条数小于请求要求条数时,则无更多消息,不再允许请求。
|
||||||
|
// 可在此处编写无更多消息数据时的逻辑
|
||||||
|
} else {
|
||||||
|
this.ajax.page++;
|
||||||
|
|
||||||
|
// 延迟 200ms ,以保证设置窗口滚动已完成
|
||||||
|
setTimeout(() => {
|
||||||
|
this.ajax.flag = true;
|
||||||
|
}, 200)
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
get();
|
||||||
|
},
|
||||||
|
// 拼接历史记录消息
|
||||||
|
joinHistoryMsg() {
|
||||||
|
let join = () => {
|
||||||
|
let arr = [];
|
||||||
|
|
||||||
|
//通过当前页码及页数,模拟数据内容
|
||||||
|
let startIndex = (this.ajax.page - 1) * this.ajax.rows;
|
||||||
|
let endIndex = startIndex + this.ajax.rows;
|
||||||
|
for (let i = startIndex; i < endIndex; i++) {
|
||||||
|
arr.push({
|
||||||
|
"id": i, // 消息的ID
|
||||||
|
"content": `这是历史记录的第${i+1}条消息`, // 消息内容
|
||||||
|
"type": Math.random() > 0.5 ? 1 : 0, // 此为消息类别,设 1 为发出去的消息,0 为收到对方的消息,
|
||||||
|
"pic": "/static/avatar.png" // 头像
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
颠倒数组中元素的顺序。将最新的数据排在本次接口返回数据的最后面。
|
||||||
|
后端接口按 消息的时间降序查找出当前页的数据后,再将本页数据按消息时间降序排序返回。
|
||||||
|
这是数据的重点,因为页面滚动条和上拉加载历史的问题。
|
||||||
|
*/
|
||||||
|
arr.reverse();
|
||||||
|
|
||||||
|
return arr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 此处用到 ES6 的 Promise 知识,不懂的请自行学习。
|
||||||
|
return new Promise((done, fail) => {
|
||||||
|
// 无数据请求接口,由 setTimeout 模拟,正式项目替换为 ajax 即可。
|
||||||
|
setTimeout(() => {
|
||||||
|
let data = join();
|
||||||
|
done(data);
|
||||||
|
}, 1500);
|
||||||
|
})
|
||||||
|
},
|
||||||
|
focus(){
|
||||||
|
uni.pageScrollTo({
|
||||||
|
top: this.keyboardHeight
|
||||||
|
})
|
||||||
|
},
|
||||||
|
// 设置页面滚动位置
|
||||||
|
setPageScrollTo(selector) {
|
||||||
|
let view = uni.createSelectorQuery().in(this).select(selector);
|
||||||
|
view.boundingClientRect((res) => {
|
||||||
|
uni.pageScrollTo({
|
||||||
|
scrollTop: res.top - 30, // -30 为多显示出大半个消息的高度,示意上面还有信息。
|
||||||
|
duration: 0
|
||||||
|
});
|
||||||
|
}).exec();
|
||||||
|
},
|
||||||
|
// 隐藏加载提示
|
||||||
|
hideLoadTips(flag) {
|
||||||
|
if (flag) {
|
||||||
|
this.ajax.loadText = '消息获取成功';
|
||||||
|
setTimeout(() => {
|
||||||
|
this.ajax.loading = false;
|
||||||
|
}, 300);
|
||||||
|
} else {
|
||||||
|
this.ajax.loading = true;
|
||||||
|
this.ajax.loadText = '正在获取消息';
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// 发送信息
|
||||||
|
send() {
|
||||||
|
if (!this.content) {
|
||||||
|
uni.showToast({
|
||||||
|
title: '请输入有效的内容',
|
||||||
|
icon: 'none'
|
||||||
|
})
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try{
|
||||||
|
clearInterval(this.timer);
|
||||||
|
this.talkList[this.talkList.length-1].content = this.c_content+'';
|
||||||
|
this.c_content = '';
|
||||||
|
this.n_content = '';
|
||||||
|
this.socketTask.close({
|
||||||
|
success:(res)=> {
|
||||||
|
console.log('关闭成功', res);
|
||||||
|
this.wsLiveFlag = false;
|
||||||
|
},
|
||||||
|
fail(err) {
|
||||||
|
console.log('关闭失败', err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}catch(e){
|
||||||
|
//TODO handle the exception
|
||||||
|
}
|
||||||
|
// 将当前发送信息 添加到消息列表。
|
||||||
|
let data = {
|
||||||
|
"id": new Date().getTime(),
|
||||||
|
"content": this.content,
|
||||||
|
"type": 1,
|
||||||
|
"pic": "/static/avatar.png"
|
||||||
|
}
|
||||||
|
this.TEXT = this.content;
|
||||||
|
this.n_content = '';
|
||||||
|
this.c_content = '';
|
||||||
|
this.talkList.push(data);
|
||||||
|
this.talkList.push({
|
||||||
|
"id": new Date().getTime(),
|
||||||
|
"content": '',
|
||||||
|
"type": 2,
|
||||||
|
"pic": "/static/avatar.png"
|
||||||
|
});
|
||||||
|
// return ;
|
||||||
|
this.$nextTick(() => {
|
||||||
|
// 清空内容框中的内容
|
||||||
|
this.content = '';
|
||||||
|
uni.pageScrollTo({
|
||||||
|
scrollTop: 999999, // 设置一个超大值,以保证滚动条滚动到底部
|
||||||
|
duration: 0
|
||||||
|
});
|
||||||
|
})
|
||||||
|
this.sendToSpark();
|
||||||
|
|
||||||
|
},
|
||||||
|
async sendToSpark() {
|
||||||
|
// let myUrl = await this.getWebSocketUrl();
|
||||||
|
this.tempRes = "";
|
||||||
|
let realThis = this;
|
||||||
|
this.socketTask = uni.connectSocket({
|
||||||
|
//url: encodeURI(encodeURI(myUrl).replace(/\+/g, '%2B')),
|
||||||
|
url: 'wss://chat.lihaink.cn/chat',
|
||||||
|
method: 'GET',
|
||||||
|
success: res => {
|
||||||
|
console.log(res, "ws成功连接...")
|
||||||
|
realThis.wsLiveFlag = true;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
realThis.socketTask.onError((res) => {
|
||||||
|
console.log("连接发生错误,请检查appid是否填写", res)
|
||||||
|
})
|
||||||
|
realThis.socketTask.onOpen((res) => {
|
||||||
|
this.historyTextList.push({
|
||||||
|
"role": "user",
|
||||||
|
"content": this.TEXT
|
||||||
|
})
|
||||||
|
// 第一帧..........................................
|
||||||
|
console.log('连接成功...')
|
||||||
|
// let params = {
|
||||||
|
// "header": {
|
||||||
|
// "app_id": this.APPID,
|
||||||
|
// "uid": "aef9f963-7"
|
||||||
|
// },
|
||||||
|
// "parameter": {
|
||||||
|
// "chat": {
|
||||||
|
// "domain": "generalv2",
|
||||||
|
// "temperature": 0.5,
|
||||||
|
// "max_tokens": 1024
|
||||||
|
// }
|
||||||
|
// },
|
||||||
|
// "payload": {
|
||||||
|
// "message": {
|
||||||
|
// "text": this.historyTextList
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// };
|
||||||
|
let params = this.TEXT
|
||||||
|
realThis.socketTask.send({ // 发送消息,,都用uni的官方版本
|
||||||
|
data: JSON.stringify(params),
|
||||||
|
success() {
|
||||||
|
console.log('第一帧发送成功')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 接受到消息时
|
||||||
|
realThis.socketTask.onMessage((res) => {
|
||||||
|
console.log('收到API返回的内容:', res.data);
|
||||||
|
let obj = JSON.parse(res.data)
|
||||||
|
// console.log("我打印的"+obj.payload);
|
||||||
|
let dataArray = obj.payload.choices.text;
|
||||||
|
for (let i = 0; i < dataArray.length; i++) {
|
||||||
|
this.talkList[this.talkList.length - 1].content += dataArray[i].content;
|
||||||
|
this.n_content = this.talkList[this.talkList.length - 1].content;
|
||||||
|
realThis.tempRes = realThis.tempRes + dataArray[i].content
|
||||||
|
}
|
||||||
|
let temp = JSON.parse(res.data)
|
||||||
|
// console.log("0726",temp.header.code)
|
||||||
|
if (temp.header.code !== 0) {
|
||||||
|
console.log(`${temp.header.code}:${temp.message}`);
|
||||||
|
realThis.socketTask.close({
|
||||||
|
success(res) {
|
||||||
|
console.log('关闭成功', res)
|
||||||
|
realThis.wsLiveFlag = false;
|
||||||
|
},
|
||||||
|
fail(err) {
|
||||||
|
console.log('关闭失败', err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
if (temp.header.code === 0) {
|
||||||
|
if (res.data && temp.header.status === 2) {
|
||||||
|
this.historyTextList.push({
|
||||||
|
"role": "assistant",
|
||||||
|
"content": this.tempRes
|
||||||
|
})
|
||||||
|
setTimeout(() => {
|
||||||
|
realThis.socketTask.close({
|
||||||
|
success(res) {
|
||||||
|
console.log('关闭成功', res)
|
||||||
|
},
|
||||||
|
fail(err) {
|
||||||
|
// console.log('关闭失败', err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, 1000)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
// 鉴权
|
||||||
|
getWebSocketUrl() {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
// https://spark-api.xf-yun.com/v1.1/chat V1.5 domain general
|
||||||
|
// https://spark-api.xf-yun.com/v2.1/chat V2.0 domain generalv2
|
||||||
|
var url = "wss://spark-api.xf-yun.com/v2.1/chat";
|
||||||
|
var host = "spark-api.xf-yun.com";
|
||||||
|
var apiKeyName = "api_key";
|
||||||
|
var date = new Date().toGMTString();
|
||||||
|
var algorithm = "hmac-sha256";
|
||||||
|
var headers = "host date request-line";
|
||||||
|
var signatureOrigin = `host: ${host}\ndate: ${date}\nGET /v2.1/chat HTTP/1.1`;
|
||||||
|
var signatureSha = CryptoJS.HmacSHA256(signatureOrigin, this.APISecret);
|
||||||
|
var signature = CryptoJS.enc.Base64.stringify(signatureSha);
|
||||||
|
var authorizationOrigin =
|
||||||
|
`${apiKeyName}="${this.APIKey}", algorithm="${algorithm}", headers="${headers}", signature="${signature}"`;
|
||||||
|
var authorization = base64.encode(authorizationOrigin);
|
||||||
|
url = `${url}?authorization=${authorization}&date=${encodeURI(date)}&host=${host}`;
|
||||||
|
|
||||||
|
// console.log(url)
|
||||||
resolve(url);
|
resolve(url);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
}
|
// 滚动到头部
|
||||||
}
|
bindScroll() {
|
||||||
|
if (this.userId == 0) {
|
||||||
|
this.getHistory();
|
||||||
|
this.getproductInfo();
|
||||||
|
this.getOrderInfo();
|
||||||
|
this.getRefundDetail();
|
||||||
|
this.getStoreDetail();
|
||||||
|
} else {
|
||||||
|
this.getMerHistory();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style>
|
<style lang="scss">
|
||||||
|
@import "../../lib/global.scss";
|
||||||
|
|
||||||
|
page {
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
font-size: 28rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes blink {
|
||||||
|
0% { opacity: 1; }
|
||||||
|
50% { opacity: 0; }
|
||||||
|
100% { opacity: 1; }
|
||||||
|
}
|
||||||
|
|
||||||
|
.blinking-box {
|
||||||
|
background-color: #333;
|
||||||
|
color: #333;
|
||||||
|
animation: blink 1s infinite;
|
||||||
|
width: 2px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wrapper {
|
||||||
|
height: auto !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 加载数据提示 */
|
||||||
|
.tips {
|
||||||
|
position: fixed;
|
||||||
|
left: 0;
|
||||||
|
top: var(--window-top);
|
||||||
|
width: 100%;
|
||||||
|
z-index: 9;
|
||||||
|
background-color: rgba(0, 0, 0, 0.15);
|
||||||
|
height: 72rpx;
|
||||||
|
line-height: 72rpx;
|
||||||
|
transform: translateY(-80rpx);
|
||||||
|
transition: transform 0.3s ease-in-out 0s;
|
||||||
|
|
||||||
|
&.show {
|
||||||
|
transform: translateY(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.box-1 {
|
||||||
|
width: 100%;
|
||||||
|
height: auto;
|
||||||
|
padding-bottom: 100rpx;
|
||||||
|
box-sizing: content-box;
|
||||||
|
|
||||||
|
/* 兼容iPhoneX */
|
||||||
|
margin-bottom: 0;
|
||||||
|
margin-bottom: constant(safe-area-inset-bottom);
|
||||||
|
margin-bottom: env(safe-area-inset-bottom);
|
||||||
|
}
|
||||||
|
|
||||||
|
.multiline-text {
|
||||||
|
white-space: pre-line;
|
||||||
|
/* 或 white-space: pre-wrap; */
|
||||||
|
}
|
||||||
|
|
||||||
|
.box-2 {
|
||||||
|
position: fixed;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
bottom: 0;
|
||||||
|
height: auto;
|
||||||
|
z-index: 2;
|
||||||
|
border-top: #e5e5e5 solid 1px;
|
||||||
|
box-sizing: content-box;
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
transform: translateY(0); /* 初始化 transform 属性 */
|
||||||
|
transition: transform 0.3s ease; /* 添加过渡效果 */
|
||||||
|
|
||||||
|
/* 兼容iPhoneX */
|
||||||
|
padding-bottom: 0;
|
||||||
|
padding-bottom: constant(safe-area-inset-bottom);
|
||||||
|
padding-bottom: env(safe-area-inset-bottom);
|
||||||
|
|
||||||
|
>view {
|
||||||
|
padding: 0 20rpx;
|
||||||
|
height: 100rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
background-color: #fff;
|
||||||
|
height: 64rpx;
|
||||||
|
padding: 0 20rpx;
|
||||||
|
border-radius: 6rpx;
|
||||||
|
font-size: 28rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.send {
|
||||||
|
background-color: #2573fb;
|
||||||
|
color: #fff;
|
||||||
|
height: 64rpx;
|
||||||
|
margin-left: 20rpx;
|
||||||
|
border-radius: 6rpx;
|
||||||
|
padding: 0;
|
||||||
|
width: 120rpx;
|
||||||
|
line-height: 62rpx;
|
||||||
|
|
||||||
|
&:active {
|
||||||
|
background-color: #1573fb;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.talk-list {
|
||||||
|
padding-bottom: 20rpx;
|
||||||
|
|
||||||
|
/* 消息项,基础类 */
|
||||||
|
.item {
|
||||||
|
padding: 20rpx 20rpx 0 20rpx;
|
||||||
|
align-items: flex-start;
|
||||||
|
align-content: flex-start;
|
||||||
|
color: #333;
|
||||||
|
|
||||||
|
.pic {
|
||||||
|
width: 92rpx;
|
||||||
|
height: 92rpx;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: #fff solid 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
padding: 20rpx;
|
||||||
|
border-radius: 4px;
|
||||||
|
max-width: 500rpx;
|
||||||
|
word-break: break-all;
|
||||||
|
line-height: 52rpx;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 收到的消息 */
|
||||||
|
&.pull {
|
||||||
|
.content {
|
||||||
|
min-width: 20rpx;
|
||||||
|
min-height: 52rpx;
|
||||||
|
margin-left: 32rpx;
|
||||||
|
background-color: #fff;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
width: 0;
|
||||||
|
height: 0;
|
||||||
|
border-top: 16rpx solid transparent;
|
||||||
|
border-bottom: 16rpx solid transparent;
|
||||||
|
border-right: 20rpx solid #fff;
|
||||||
|
position: absolute;
|
||||||
|
top: 30rpx;
|
||||||
|
left: -18rpx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 发出的消息 */
|
||||||
|
&.push {
|
||||||
|
/* 主轴为水平方向,起点在右端。使不修改DOM结构,也能改变元素排列顺序 */
|
||||||
|
flex-direction: row-reverse;
|
||||||
|
|
||||||
|
.content {
|
||||||
|
min-width: 20rpx;
|
||||||
|
min-height: 52rpx;
|
||||||
|
margin-right: 32rpx;
|
||||||
|
background-color: #2573fb;
|
||||||
|
color: #fff;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
width: 0;
|
||||||
|
height: 0;
|
||||||
|
border-top: 16rpx solid transparent;
|
||||||
|
border-bottom: 16rpx solid transparent;
|
||||||
|
border-left: 20rpx solid #2573fb;
|
||||||
|
position: absolute;
|
||||||
|
top: 30rpx;
|
||||||
|
right: -18rpx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.bing-math {
|
||||||
|
margin: 0 !important;
|
||||||
|
padding: 0 !important;
|
||||||
|
}
|
||||||
|
.placeholder {
|
||||||
|
width: 100vw;
|
||||||
|
background-color: #1573fb;
|
||||||
|
// background-color: transparent;
|
||||||
|
// transform: translateY(0); /* 初始化 transform 属性 */
|
||||||
|
// transition: transform 0.3s ease; /* 添加过渡效果 */
|
||||||
|
}
|
||||||
</style>
|
</style>
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -1,8 +1,8 @@
|
||||||
|
|
||||||
var isReady=false;var onReadyCallbacks=[];
|
var isReady=false;var onReadyCallbacks=[];
|
||||||
var isServiceReady=false;var onServiceReadyCallbacks=[];
|
var isServiceReady=false;var onServiceReadyCallbacks=[];
|
||||||
var __uniConfig = {"pages":["pages/index/index","pages/index/chat"],"window":{"navigationBarTextStyle":"black","navigationBarTitleText":"人工智能","navigationBarBackgroundColor":"#F8F8F8","backgroundColor":"#F8F8F8"},"darkmode":false,"nvueCompiler":"uni-app","nvueStyleCompiler":"uni-app","renderer":"auto","splashscreen":{"alwaysShowBeforeRender":true,"autoclose":false},"appname":"xunfeiAi","compilerVersion":"3.8.12","entryPagePath":"pages/index/chat","entryPageQuery":"","realEntryPagePath":"pages/index/index","networkTimeout":{"request":60000,"connectSocket":60000,"uploadFile":60000,"downloadFile":60000}};
|
var __uniConfig = {"pages":["pages/index/index","pages/index/chat"],"window":{"navigationBarTextStyle":"black","navigationBarTitleText":"人工智能","navigationBarBackgroundColor":"#F8F8F8","backgroundColor":"#F8F8F8"},"darkmode":false,"nvueCompiler":"uni-app","nvueStyleCompiler":"uni-app","renderer":"auto","splashscreen":{"alwaysShowBeforeRender":true,"autoclose":false},"appname":"xunfeiAi","compilerVersion":"3.8.12","entryPagePath":"pages/index/index","networkTimeout":{"request":60000,"connectSocket":60000,"uploadFile":60000,"downloadFile":60000}};
|
||||||
var __uniRoutes = [{"path":"/pages/index/index","meta":{"isQuit":true},"window":{"navigationBarTitleText":"讯飞AI"}},{"path":"/pages/index/chat","meta":{},"window":{"navigationBarTitleText":"讯飞星火"}}];
|
var __uniRoutes = [{"path":"/pages/index/index","meta":{"isQuit":true},"window":{"navigationBarTitleText":"讯飞AI","bounce":"none","softinputMode":"adjustResize"}},{"path":"/pages/index/chat","meta":{},"window":{"navigationBarTitleText":"讯飞星火","bounce":"none"}}];
|
||||||
__uniConfig.onReady=function(callback){if(__uniConfig.ready){callback()}else{onReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"ready",{get:function(){return isReady},set:function(val){isReady=val;if(!isReady){return}const callbacks=onReadyCallbacks.slice(0);onReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
__uniConfig.onReady=function(callback){if(__uniConfig.ready){callback()}else{onReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"ready",{get:function(){return isReady},set:function(val){isReady=val;if(!isReady){return}const callbacks=onReadyCallbacks.slice(0);onReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
||||||
__uniConfig.onServiceReady=function(callback){if(__uniConfig.serviceReady){callback()}else{onServiceReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"serviceReady",{get:function(){return isServiceReady},set:function(val){isServiceReady=val;if(!isServiceReady){return}const callbacks=onServiceReadyCallbacks.slice(0);onServiceReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
__uniConfig.onServiceReady=function(callback){if(__uniConfig.serviceReady){callback()}else{onServiceReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"serviceReady",{get:function(){return isServiceReady},set:function(val){isServiceReady=val;if(!isServiceReady){return}const callbacks=onServiceReadyCallbacks.slice(0);onServiceReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
||||||
service.register("uni-app-config",{create(a,b,c){if(!__uniConfig.viewport){var d=b.weex.config.env.scale,e=b.weex.config.env.deviceWidth,f=Math.ceil(e/d);Object.assign(__uniConfig,{viewport:f,defaultFontSize:Math.round(f/20)})}return{instance:{__uniConfig:__uniConfig,__uniRoutes:__uniRoutes,global:void 0,window:void 0,document:void 0,frames:void 0,self:void 0,location:void 0,navigator:void 0,localStorage:void 0,history:void 0,Caches:void 0,screen:void 0,alert:void 0,confirm:void 0,prompt:void 0,fetch:void 0,XMLHttpRequest:void 0,WebSocket:void 0,webkit:void 0,print:void 0}}}});
|
service.register("uni-app-config",{create(a,b,c){if(!__uniConfig.viewport){var d=b.weex.config.env.scale,e=b.weex.config.env.deviceWidth,f=Math.ceil(e/d);Object.assign(__uniConfig,{viewport:f,defaultFontSize:Math.round(f/20)})}return{instance:{__uniConfig:__uniConfig,__uniRoutes:__uniRoutes,global:void 0,window:void 0,document:void 0,frames:void 0,self:void 0,location:void 0,navigator:void 0,localStorage:void 0,history:void 0,Caches:void 0,screen:void 0,alert:void 0,confirm:void 0,prompt:void 0,fetch:void 0,XMLHttpRequest:void 0,WebSocket:void 0,webkit:void 0,print:void 0}}}});
|
||||||
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -1 +1 @@
|
||||||
{"@platforms":["android","iPhone","iPad"],"id":"__UNI__9620511","name":"xunfeiAi","version":{"name":"1.0.0","code":"100"},"description":"","launch_path":"__uniappview.html","developer":{"name":"","email":"","url":""},"permissions":{"UniNView":{"description":"UniNView原生渲染"}},"plus":{"useragent":{"value":"uni-app","concatenate":true},"splashscreen":{"target":"id:1","autoclose":true,"waiting":true,"delay":0},"popGesture":"close","launchwebview":{"render":"always","id":"1","kernel":"WKWebview"},"statusbar":{"immersed":"supportedDevice","style":"dark","background":"#F8F8F8"},"usingComponents":true,"nvueStyleCompiler":"uni-app","compilerVersion":3,"distribute":{"google":{"permissions":["<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>","<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>","<uses-permission android:name=\"android.permission.VIBRATE\"/>","<uses-permission android:name=\"android.permission.READ_LOGS\"/>","<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>","<uses-feature android:name=\"android.hardware.camera.autofocus\"/>","<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>","<uses-permission android:name=\"android.permission.CAMERA\"/>","<uses-permission android:name=\"android.permission.GET_ACCOUNTS\"/>","<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>","<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>","<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>","<uses-permission android:name=\"android.permission.FLASHLIGHT\"/>","<uses-feature android:name=\"android.hardware.camera\"/>","<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>"]},"apple":{},"plugins":{"audio":{"mp3":{"description":"Android平台录音支持MP3格式文件"}}}},"uniStatistics":{"enable":false},"arguments":"{\"id\":0,\"name\":\"pages/index/chat\",\"pathName\":\"pages/index/chat\",\"query\":\"\"}","allowsInlineMediaPlayback":true,"uni-app":{"compilerVersion":"3.8.12","control":"uni-v3","nvueCompiler":"uni-app","renderer":"auto","nvue":{"flex-direction":"column"},"nvueLaunchMode":"normal"},"launch_path":"__uniappview.html"},"arguments":{"pathName":"pages/index/chat","query":""}}
|
{"@platforms":["android","iPhone","iPad"],"id":"__UNI__9620511","name":"xunfeiAi","version":{"name":"1.0.0","code":"100"},"description":"","launch_path":"__uniappview.html","developer":{"name":"","email":"","url":""},"permissions":{"UniNView":{"description":"UniNView原生渲染"}},"plus":{"useragent":{"value":"uni-app","concatenate":true},"splashscreen":{"target":"id:1","autoclose":true,"waiting":true,"delay":0},"popGesture":"close","launchwebview":{"render":"always","id":"1","kernel":"WKWebview"},"statusbar":{"immersed":"supportedDevice","style":"dark","background":"#F8F8F8"},"usingComponents":true,"nvueStyleCompiler":"uni-app","compilerVersion":3,"distribute":{"google":{"permissions":["<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>","<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>","<uses-permission android:name=\"android.permission.VIBRATE\"/>","<uses-permission android:name=\"android.permission.READ_LOGS\"/>","<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>","<uses-feature android:name=\"android.hardware.camera.autofocus\"/>","<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>","<uses-permission android:name=\"android.permission.CAMERA\"/>","<uses-permission android:name=\"android.permission.GET_ACCOUNTS\"/>","<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>","<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>","<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>","<uses-permission android:name=\"android.permission.FLASHLIGHT\"/>","<uses-feature android:name=\"android.hardware.camera\"/>","<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>"]},"apple":{},"plugins":{"audio":{"mp3":{"description":"Android平台录音支持MP3格式文件"}}}},"uniStatistics":{"enable":false},"arguments":"{\"name\":\"\",\"path\":\"\",\"query\":\"\",\"id\":0}","allowsInlineMediaPlayback":true,"uni-app":{"compilerVersion":"3.8.12","control":"uni-v3","nvueCompiler":"uni-app","renderer":"auto","nvue":{"flex-direction":"column"},"nvueLaunchMode":"normal"},"launch_path":"__uniappview.html"}}
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
Loading…
Reference in New Issue