首次提交
This commit is contained in:
commit
99c8e04bb4
|
@ -0,0 +1,5 @@
|
|||
[*.{js,jsx,ts,tsx,vue}]
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
trim_trailing_whitespace = true
|
||||
insert_final_newline = true
|
|
@ -0,0 +1,23 @@
|
|||
.DS_Store
|
||||
node_modules
|
||||
/dist
|
||||
|
||||
|
||||
# local env files
|
||||
.env.local
|
||||
.env.*.local
|
||||
|
||||
# Log files
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
pnpm-debug.log*
|
||||
|
||||
# Editor directories and files
|
||||
.idea
|
||||
.vscode
|
||||
*.suo
|
||||
*.ntvs*
|
||||
*.njsproj
|
||||
*.sln
|
||||
*.sw?
|
|
@ -0,0 +1,24 @@
|
|||
# chat
|
||||
|
||||
## Project setup
|
||||
```
|
||||
npm install
|
||||
```
|
||||
|
||||
### Compiles and hot-reloads for development
|
||||
```
|
||||
npm run serve
|
||||
```
|
||||
|
||||
### Compiles and minifies for production
|
||||
```
|
||||
npm run build
|
||||
```
|
||||
|
||||
### Lints and fixes files
|
||||
```
|
||||
npm run lint
|
||||
```
|
||||
|
||||
### Customize configuration
|
||||
See [Configuration Reference](https://cli.vuejs.org/config/).
|
|
@ -0,0 +1,5 @@
|
|||
module.exports = {
|
||||
presets: [
|
||||
'@vue/cli-plugin-babel/preset'
|
||||
]
|
||||
}
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,58 @@
|
|||
{
|
||||
"name": "chat",
|
||||
"version": "0.1.0",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"serve": "vue-cli-service serve",
|
||||
"build": "vue-cli-service build",
|
||||
"lint": "vue-cli-service lint"
|
||||
},
|
||||
"dependencies": {
|
||||
"@kangc/v-md-editor": "^2.3.15",
|
||||
"axios": "^1.3.4",
|
||||
"core-js": "^3.6.5",
|
||||
"js-base64": "^3.7.5",
|
||||
"js-md5": "^0.7.3",
|
||||
"vue": "^3.0.0",
|
||||
"vue-axios": "^3.5.2",
|
||||
"vue-router": "^4.0.0-0",
|
||||
"vuex": "^4.0.0-0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@vue/cli-plugin-babel": "~4.5.1",
|
||||
"@vue/cli-plugin-eslint": "~4.5.1",
|
||||
"@vue/cli-plugin-router": "~4.5.1",
|
||||
"@vue/cli-plugin-vuex": "~4.5.1",
|
||||
"@vue/cli-service": "~4.5.1",
|
||||
"@vue/compiler-sfc": "^3.0.0",
|
||||
"@vue/eslint-config-standard": "^5.1.2",
|
||||
"babel-eslint": "^10.1.0",
|
||||
"eslint": "^6.7.2",
|
||||
"eslint-plugin-import": "^2.20.2",
|
||||
"eslint-plugin-node": "^11.1.0",
|
||||
"eslint-plugin-promise": "^4.2.1",
|
||||
"eslint-plugin-standard": "^4.0.0",
|
||||
"eslint-plugin-vue": "^7.0.0",
|
||||
"less": "^3.0.4",
|
||||
"less-loader": "^5.0.0"
|
||||
},
|
||||
"eslintConfig": {
|
||||
"root": true,
|
||||
"env": {
|
||||
"node": true
|
||||
},
|
||||
"extends": [
|
||||
"plugin:vue/vue3-essential",
|
||||
"@vue/standard"
|
||||
],
|
||||
"parserOptions": {
|
||||
"parser": "babel-eslint"
|
||||
},
|
||||
"rules": {}
|
||||
},
|
||||
"browserslist": [
|
||||
"> 1%",
|
||||
"last 2 versions",
|
||||
"not dead"
|
||||
]
|
||||
}
|
Binary file not shown.
After Width: | Height: | Size: 4.2 KiB |
|
@ -0,0 +1,19 @@
|
|||
<!DOCTYPE html>
|
||||
<html lang="">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||
<meta name="viewport"
|
||||
content="width=device-width,initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=no, viewport-fit=cover">
|
||||
<link rel="icon" href="<%= BASE_URL %>favicon.ico">
|
||||
<title><%= htmlWebpackPlugin.options.title %></title>
|
||||
</head>
|
||||
<body>
|
||||
<noscript>
|
||||
<strong>We're sorry but <%= htmlWebpackPlugin.options.title %> doesn't work properly without JavaScript enabled. Please enable it to continue.</strong>
|
||||
</noscript>
|
||||
<div id="app"></div>
|
||||
<!-- built files will be auto injected -->
|
||||
</body>
|
||||
</html>
|
|
@ -0,0 +1,18 @@
|
|||
<template>
|
||||
<router-view />
|
||||
</template>
|
||||
|
||||
<style lang="less">
|
||||
* {
|
||||
padding: 0;
|
||||
margin: 0;
|
||||
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
|
||||
}
|
||||
::-webkit-scrollbar {
|
||||
width: 4px;
|
||||
}
|
||||
::-webkit-scrollbar-thumb {
|
||||
border-radius: 10px;
|
||||
background: #b9b9b9;
|
||||
}
|
||||
</style>
|
|
@ -0,0 +1,25 @@
|
|||
import { createApp } from 'vue'
|
||||
import App from './App.vue'
|
||||
import router from './router'
|
||||
import store from './store'
|
||||
import axios from 'axios'
|
||||
import VueAxios from 'vue-axios'
|
||||
|
||||
import VMdPreview from '@kangc/v-md-editor/lib/preview';
|
||||
import '@kangc/v-md-editor/lib/style/preview.css';
|
||||
import githubTheme from '@kangc/v-md-editor/lib/theme/github.js';
|
||||
import '@kangc/v-md-editor/lib/theme/style/github.css';
|
||||
import hljs from 'highlight.js';
|
||||
VMdPreview.use(githubTheme, {
|
||||
Hljs: hljs,
|
||||
});
|
||||
|
||||
const app = createApp(App)
|
||||
|
||||
// 全局配置
|
||||
app.config.globalProperties.$config = {
|
||||
api: "",
|
||||
privateKey: "",
|
||||
}
|
||||
|
||||
app.use(store).use(router).use(VueAxios, axios).use(VMdPreview).mount('#app')
|
|
@ -0,0 +1,16 @@
|
|||
import { createRouter, createWebHashHistory } from 'vue-router'
|
||||
|
||||
const routes = [
|
||||
{
|
||||
path: '/',
|
||||
name: 'Home',
|
||||
component: () => import(/* webpackChunkName: "about" */ '../views/Home.vue')
|
||||
}
|
||||
]
|
||||
|
||||
const router = createRouter({
|
||||
history: createWebHashHistory(),
|
||||
routes
|
||||
})
|
||||
|
||||
export default router
|
|
@ -0,0 +1,12 @@
|
|||
import { createStore } from 'vuex'
|
||||
|
||||
export default createStore({
|
||||
state: {
|
||||
},
|
||||
mutations: {
|
||||
},
|
||||
actions: {
|
||||
},
|
||||
modules: {
|
||||
}
|
||||
})
|
|
@ -0,0 +1,14 @@
|
|||
import md5 from 'js-md5'
|
||||
export default {
|
||||
makeSign(data, key) {
|
||||
let keys = Object.keys(data).sort(), obj = {}, str = "";
|
||||
for (let i = 0; i < keys.length; i++) {
|
||||
if (keys[i].toLowerCase() == 'sign') {
|
||||
continue
|
||||
}
|
||||
str += keys[i] + "=" + data[keys[i]] + '&'
|
||||
}
|
||||
|
||||
return md5(str + 'key=' + key)
|
||||
}
|
||||
}
|
|
@ -0,0 +1,263 @@
|
|||
<template>
|
||||
<div class="chat-window">
|
||||
<div class="message-content">
|
||||
<div class="message-group">
|
||||
<div class="message-item" :class="[item.role]" v-for="(item, index) in lists" :key="index">
|
||||
<span class="name">{{ item.role == "user" ? "你" : pageTitle }}</span>
|
||||
<v-md-preview class="content" :text="item.content"></v-md-preview>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="user-input" :class="{ disable: !canUse }">
|
||||
<input
|
||||
class="inp"
|
||||
type="text"
|
||||
v-model="words"
|
||||
:disabled="!canUse"
|
||||
@keyup.enter="send"
|
||||
@input="scrollToBottom"
|
||||
@focus="scrollToBottom"
|
||||
placeholder="输入你的问题"
|
||||
/>
|
||||
<button class="btn" @click="send">{{ canUse ? "发送" : "正在回答" }}</button>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { Base64 } from "js-base64";
|
||||
import Sign from "../utils/sign";
|
||||
export default {
|
||||
name: "Home",
|
||||
data() {
|
||||
return {
|
||||
pageTitle: "GPT-3.5语言模型",
|
||||
canUse: true,
|
||||
words: "",
|
||||
lists: [
|
||||
{
|
||||
role: "system",
|
||||
content: "请问有什么可以帮你的吗?",
|
||||
},
|
||||
],
|
||||
};
|
||||
},
|
||||
components: {},
|
||||
created() {
|
||||
document.title = this.pageTitle;
|
||||
},
|
||||
mounted() {},
|
||||
unmounted() {},
|
||||
methods: {
|
||||
/**
|
||||
* 发送消息
|
||||
*/
|
||||
send() {
|
||||
if (!this.canUse || this.words == "") return;
|
||||
this.lists.push({
|
||||
role: "user",
|
||||
content: this.words,
|
||||
});
|
||||
let requestData = {
|
||||
words: Base64.encode(this.words),
|
||||
time: parseInt(Date.parse(new Date()) / 1000),
|
||||
};
|
||||
requestData.sign = Sign.makeSign(requestData, this.$config.privateKey);
|
||||
|
||||
this.lists.push({
|
||||
role: "system",
|
||||
content: "...",
|
||||
});
|
||||
let lastMessage = this.lists[this.lists.length - 1];
|
||||
|
||||
this.scrollToBottom();
|
||||
|
||||
this.canUse = false;
|
||||
this.axios({
|
||||
method: "POST",
|
||||
url: this.$config.api,
|
||||
data: JSON.stringify(requestData),
|
||||
}).then((r) => {
|
||||
this.canUse = true;
|
||||
if (r.data.code == 1) {
|
||||
this.words = "";
|
||||
let answer = JSON.parse(Base64.decode(r.data.data));
|
||||
lastMessage.content = answer.answer[0].message.content;
|
||||
} else {
|
||||
lastMessage.content = "[网络错误]";
|
||||
}
|
||||
this.scrollToBottom();
|
||||
});
|
||||
},
|
||||
/**
|
||||
* 控制滚动条
|
||||
*/
|
||||
scrollToBottom() {
|
||||
setTimeout(() => {
|
||||
var ele = document.getElementsByClassName("message-content")[0];
|
||||
if (ele.scrollHeight > ele.clientHeight) {
|
||||
ele.scrollTop = ele.scrollHeight;
|
||||
}
|
||||
}, 200);
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="less" scoped>
|
||||
.chat-window {
|
||||
width: 1200px;
|
||||
position: absolute;
|
||||
left: 50%;
|
||||
top: 50%;
|
||||
transform: translate(-50%, -50%);
|
||||
border-radius: 10px;
|
||||
box-shadow: 0px 0px 20px 0px #eaeaf9;
|
||||
background-color: #ededed;
|
||||
.message-content {
|
||||
z-index: 5;
|
||||
width: 100%;
|
||||
height: 700px;
|
||||
padding: 20px;
|
||||
box-sizing: border-box;
|
||||
overflow-y: scroll;
|
||||
}
|
||||
.message-group {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
.message-item {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
width: 100%;
|
||||
font-size: 14px;
|
||||
margin-bottom: 10px;
|
||||
.name {
|
||||
color: #666666;
|
||||
}
|
||||
.content {
|
||||
max-width: 650px;
|
||||
padding: 10px 20px;
|
||||
margin-top: 10px;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
}
|
||||
.message-item.system {
|
||||
align-items: flex-start;
|
||||
.content {
|
||||
color: #000000;
|
||||
background-color: #ffffff;
|
||||
border-radius: 0 10px 10px 10px;
|
||||
}
|
||||
}
|
||||
.message-item.user {
|
||||
align-items: flex-end;
|
||||
.content {
|
||||
color: #ffffff;
|
||||
background-color: #484aa1;
|
||||
border-radius: 10px 0 10px 10px;
|
||||
}
|
||||
}
|
||||
.user-input {
|
||||
z-index: 10;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 100%;
|
||||
height: 100px;
|
||||
border-top: 2px solid #f5f5f5;
|
||||
.inp,
|
||||
.btn {
|
||||
height: 50px;
|
||||
border: 0;
|
||||
outline: 0;
|
||||
box-sizing: border-box;
|
||||
font-size: 14px;
|
||||
}
|
||||
.inp {
|
||||
width: 600px;
|
||||
padding: 0 10px;
|
||||
border-radius: 10px 0 0 10px;
|
||||
background-color: #ffffff;
|
||||
}
|
||||
.btn {
|
||||
width: 100px;
|
||||
color: #ffffff;
|
||||
background-color: #484aa1;
|
||||
border-radius: 0 10px 10px 0;
|
||||
}
|
||||
}
|
||||
.user-input.disable {
|
||||
.btn {
|
||||
cursor: not-allowed;
|
||||
background-color: #b1b1b1;
|
||||
}
|
||||
}
|
||||
}
|
||||
.content:deep(.github-markdown-body) {
|
||||
padding: 0;
|
||||
font-size: unset;
|
||||
blockquote,
|
||||
details,
|
||||
dl,
|
||||
ol,
|
||||
p,
|
||||
pre,
|
||||
table,
|
||||
ul {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
}
|
||||
@media screen and (max-width: 768px) {
|
||||
.chat-window {
|
||||
position: relative;
|
||||
top: 0;
|
||||
left: 0;
|
||||
transform: translate(0, 0);
|
||||
width: 100%;
|
||||
height: 100vh;
|
||||
border-radius: 0;
|
||||
.message-content {
|
||||
height: 100%;
|
||||
padding: 10px;
|
||||
}
|
||||
.message-group {
|
||||
padding-bottom: 65px;
|
||||
}
|
||||
.message-item {
|
||||
.name {
|
||||
display: none;
|
||||
}
|
||||
.content {
|
||||
max-width: 100%;
|
||||
padding: 10px;
|
||||
}
|
||||
}
|
||||
.user-input {
|
||||
position: absolute;
|
||||
bottom: 20px;
|
||||
left: 0;
|
||||
right: 0;
|
||||
height: auto;
|
||||
border-top: 0;
|
||||
.inp,
|
||||
.btn {
|
||||
height: 45px;
|
||||
border: 2px solid #484aa1;
|
||||
}
|
||||
.inp {
|
||||
width: 70%;
|
||||
}
|
||||
.btn {
|
||||
width: 20%;
|
||||
}
|
||||
}
|
||||
.user-input.disable {
|
||||
.inp,
|
||||
.btn {
|
||||
border: 2px solid #b1b1b1;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</style>
|
|
@ -0,0 +1,4 @@
|
|||
module.exports = {
|
||||
publicPath: "/",
|
||||
lintOnSave: false
|
||||
}
|
Loading…
Reference in New Issue