Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cdd8a86de0 | ||
|
|
96902c9e14 | ||
|
|
b10fafd6a8 | ||
|
|
58e92bfe84 | ||
|
|
efd1c96852 | ||
|
|
1ee3469978 | ||
|
|
65629ca5a6 | ||
|
|
f64a45c0ee |
35
app.vue
35
app.vue
@@ -10,6 +10,9 @@ const themes = ref([
|
||||
const setTheme = (theme) => {
|
||||
colorMode.preference = theme
|
||||
}
|
||||
const feedback = () => {
|
||||
window.open('https://github.com/WongSaang/chatgpt-ui/issues', '_blank')
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
@@ -38,7 +41,9 @@ const setTheme = (theme) => {
|
||||
title="Theme mode"
|
||||
></v-list-item>
|
||||
</template>
|
||||
<v-list>
|
||||
<v-list
|
||||
bg-color="white"
|
||||
>
|
||||
<v-list-item
|
||||
v-for="(theme, idx) in themes"
|
||||
:key="idx"
|
||||
@@ -48,6 +53,13 @@ const setTheme = (theme) => {
|
||||
</v-list-item>
|
||||
</v-list>
|
||||
</v-menu>
|
||||
|
||||
<v-list-item
|
||||
rounded="xl"
|
||||
prepend-icon="help_outline"
|
||||
title="Feedback"
|
||||
@click="feedback"
|
||||
></v-list-item>
|
||||
</v-list>
|
||||
</template>
|
||||
</v-navigation-drawer>
|
||||
@@ -58,6 +70,27 @@ const setTheme = (theme) => {
|
||||
<v-app-bar-nav-icon @click="drawer = !drawer"></v-app-bar-nav-icon>
|
||||
|
||||
<v-toolbar-title>{{ runtimeConfig.public.appName }}</v-toolbar-title>
|
||||
|
||||
<v-spacer></v-spacer>
|
||||
|
||||
<v-menu
|
||||
>
|
||||
<template v-slot:activator="{ props }">
|
||||
<v-btn
|
||||
v-bind="props"
|
||||
icon="help_outline"
|
||||
title="Feedback"
|
||||
></v-btn>
|
||||
</template>
|
||||
<v-list
|
||||
>
|
||||
<v-list-item
|
||||
@click="feedback"
|
||||
>
|
||||
<v-list-item-title>Feedback</v-list-item-title>
|
||||
</v-list-item>
|
||||
</v-list>
|
||||
</v-menu>
|
||||
</v-app-bar>
|
||||
|
||||
<v-main>
|
||||
|
||||
@@ -1,21 +1,21 @@
|
||||
<template>
|
||||
<v-textarea
|
||||
v-model="message"
|
||||
clearable
|
||||
label="Message"
|
||||
placeholder="Type your message here"
|
||||
label="Write a message..."
|
||||
placeholder="Write a message..."
|
||||
rows="1"
|
||||
:auto-grow="autoGrow"
|
||||
:disabled="disabled"
|
||||
:loading="loading"
|
||||
hide-details
|
||||
:hint="hint"
|
||||
append-inner-icon="send"
|
||||
@keyup.enter="send"
|
||||
@click:append="send"
|
||||
@keyup.enter.exact="enterOnly"
|
||||
@click:appendInner="clickSendBtn"
|
||||
></v-textarea>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { isMobile } from 'is-mobile'
|
||||
export default {
|
||||
name: "MsgEditor",
|
||||
props: {
|
||||
@@ -30,6 +30,11 @@ export default {
|
||||
autoGrow: true,
|
||||
};
|
||||
},
|
||||
computed: {
|
||||
hint() {
|
||||
return isMobile() ? "" : "Press Enter to send your message or Shift+Enter to add a new line.";
|
||||
},
|
||||
},
|
||||
watch: {
|
||||
message(val) {
|
||||
const lines = val.split(/\r\n|\r|\n/).length;
|
||||
@@ -44,10 +49,24 @@ export default {
|
||||
},
|
||||
methods: {
|
||||
send() {
|
||||
const msg = this.message
|
||||
let msg = this.message
|
||||
// remove the last "\n"
|
||||
if (msg[msg.length - 1] === "\n") {
|
||||
msg = msg.slice(0, -1)
|
||||
}
|
||||
if (msg.length > 0) {
|
||||
this.sendMessage(msg)
|
||||
}
|
||||
this.message = ""
|
||||
this.sendMessage(msg);
|
||||
},
|
||||
clickSendBtn () {
|
||||
this.send()
|
||||
},
|
||||
enterOnly () {
|
||||
if (!isMobile()) {
|
||||
this.send()
|
||||
}
|
||||
}
|
||||
},
|
||||
}
|
||||
</script>
|
||||
|
||||
83
components/Welcome.vue
Normal file
83
components/Welcome.vue
Normal file
@@ -0,0 +1,83 @@
|
||||
<template>
|
||||
<v-container>
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<div class="text-center">
|
||||
<h2 class="text-h2">Welcome to <span class="text-primary">{{ runtimeConfig.public.appName }}</span></h2>
|
||||
<p class="text-caption mt-5">
|
||||
{{ runtimeConfig.public.appName }} is an unofficial client for ChatGPT, but uses the official OpenAI API.
|
||||
<br>
|
||||
You will need an OpenAI API Key before you can use this client.
|
||||
</p>
|
||||
</div>
|
||||
</v-col>
|
||||
</v-row>
|
||||
<v-row>
|
||||
<v-col cols="12" md="10" offset-md="1">
|
||||
<v-row>
|
||||
<v-col
|
||||
cols="12"
|
||||
md="4"
|
||||
>
|
||||
<v-row>
|
||||
<v-col>
|
||||
<div class="d-flex flex-column align-center">
|
||||
<v-icon icon="sunny"></v-icon>
|
||||
<h3 class="text-h6">Examples</h3>
|
||||
</div>
|
||||
</v-col>
|
||||
</v-row>
|
||||
<WelcomeCard v-for="example in examples" :content="example" />
|
||||
</v-col>
|
||||
<v-col
|
||||
cols="12"
|
||||
md="4"
|
||||
>
|
||||
<v-row>
|
||||
<v-col>
|
||||
<div class="d-flex flex-column align-center">
|
||||
<v-icon icon="bolt"></v-icon>
|
||||
<h3 class="text-h6">Capabilities</h3>
|
||||
</div>
|
||||
</v-col>
|
||||
</v-row>
|
||||
<WelcomeCard v-for="capabilitie in capabilities" :content="capabilitie" />
|
||||
</v-col>
|
||||
<v-col
|
||||
cols="12"
|
||||
md="4"
|
||||
>
|
||||
<v-row>
|
||||
<v-col>
|
||||
<div class="d-flex flex-column align-center">
|
||||
<v-icon icon="warning_amber"></v-icon>
|
||||
<h3 class="text-h6">Limitations</h3>
|
||||
</div>
|
||||
</v-col>
|
||||
</v-row>
|
||||
<WelcomeCard v-for="limitation in limitations" :content="limitation" />
|
||||
</v-col>
|
||||
</v-row>
|
||||
</v-col>
|
||||
</v-row>
|
||||
</v-container>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
const runtimeConfig = useRuntimeConfig()
|
||||
const examples = ref([
|
||||
'"Explain quantum computing in simple terms"',
|
||||
'"Got any creative ideas for a 10 year old’s birthday?"',
|
||||
'"How do I make an HTTP request in Javascript?"'
|
||||
])
|
||||
const capabilities = ref([
|
||||
'Remembers what user said earlier in the conversation',
|
||||
'Allows user to provide follow-up corrections',
|
||||
'Trained to decline inappropriate requests'
|
||||
])
|
||||
const limitations = ref([
|
||||
'May occasionally generate incorrect information',
|
||||
'May occasionally produce harmful instructions or biased content',
|
||||
'Limited knowledge of world and events after 2021'
|
||||
])
|
||||
</script>
|
||||
24
components/WelcomeCard.vue
Normal file
24
components/WelcomeCard.vue
Normal file
@@ -0,0 +1,24 @@
|
||||
<template>
|
||||
<v-row>
|
||||
<v-col>
|
||||
<v-hover
|
||||
v-slot="{ isHovering, props }"
|
||||
open-delay="100"
|
||||
>
|
||||
<v-card
|
||||
:elevation="isHovering ? 3 : 0"
|
||||
v-bind="props"
|
||||
variant="tonal"
|
||||
>
|
||||
<v-card-text class="text-center">
|
||||
{{ content }}
|
||||
</v-card-text>
|
||||
</v-card>
|
||||
</v-hover>
|
||||
</v-col>
|
||||
</v-row>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
const props = defineProps(['content'])
|
||||
</script>
|
||||
@@ -10,14 +10,15 @@
|
||||
"devDependencies": {
|
||||
"@nuxtjs/color-mode": "^3.2.0",
|
||||
"material-design-icons-iconfont": "^6.7.0",
|
||||
"nuxt": "^3.1.2"
|
||||
"nuxt": "^3.2.0"
|
||||
},
|
||||
"dependencies": {
|
||||
"@keyv/sqlite": "^3.6.4",
|
||||
"@microsoft/fetch-event-source": "^2.0.1",
|
||||
"@waylaidwanderer/chatgpt-api": "^1.12.2",
|
||||
"highlight.js": "^11.7.0",
|
||||
"is-mobile": "^3.1.1",
|
||||
"marked": "^4.2.12",
|
||||
"vuetify": "^3.0.6"
|
||||
}
|
||||
},
|
||||
"license": "MIT"
|
||||
}
|
||||
|
||||
@@ -106,13 +106,16 @@ createNewConversation()
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div ref="chatWindow">
|
||||
<div
|
||||
v-if="currentConversation.messages.length > 0"
|
||||
ref="chatWindow"
|
||||
>
|
||||
<v-card
|
||||
rounded="0"
|
||||
elevation="0"
|
||||
v-for="(conversation, index) in currentConversation.messages"
|
||||
:key="index"
|
||||
:variant="conversation.from === 'ai' ? 'tonal' : ''"
|
||||
:variant="conversation.from === 'ai' ? 'tonal' : 'text'"
|
||||
>
|
||||
<v-container>
|
||||
<v-card-text class="text-caption text-disabled">{{ conversation.from }}</v-card-text>
|
||||
@@ -124,6 +127,7 @@ createNewConversation()
|
||||
</v-card>
|
||||
<div ref="grab" class="w-100" style="height: 150px;"></div>
|
||||
</div>
|
||||
<Welcome v-else />
|
||||
<v-footer app class="d-flex flex-column">
|
||||
<div class="px-md-16 w-100 d-flex align-center">
|
||||
<v-btn
|
||||
@@ -137,12 +141,13 @@ createNewConversation()
|
||||
</div>
|
||||
|
||||
<div class="px-4 py-2 text-disabled text-caption font-weight-light text-center w-100">
|
||||
{{ new Date().getFullYear() }} — {{ runtimeConfig.public.appName }}
|
||||
© {{ new Date().getFullYear() }} — {{ runtimeConfig.public.appName }}
|
||||
</div>
|
||||
</v-footer>
|
||||
<v-snackbar
|
||||
v-model="snackbar"
|
||||
multi-line
|
||||
location="top"
|
||||
>
|
||||
{{ snackbarText }}
|
||||
|
||||
|
||||
@@ -69,7 +69,6 @@ export default defineEventHandler(async (event) => {
|
||||
// This is used for storing conversations, and supports additional drivers (conversations are stored in memory by default)
|
||||
// For example, to use a JSON file (`npm i keyv-file`) as a database:
|
||||
// store: new KeyvFile({ filename: 'cache.json' }),
|
||||
uri: 'sqlite://database.sqlite'
|
||||
};
|
||||
|
||||
const chatGptClient = new ChatGPTClient(body.openaiApiKey, clientOptions, cacheOptions);
|
||||
|
||||
@@ -1,19 +0,0 @@
|
||||
import {getSetting, setSetting} from "~/utils/keyv";
|
||||
import {apiError, apiSuccess} from "~/utils/api";
|
||||
|
||||
export default defineEventHandler(async (event) => {
|
||||
const runtimeConfig = useRuntimeConfig()
|
||||
const method = getMethod(event)
|
||||
if (method === 'GET') {
|
||||
const query = getQuery(event)
|
||||
let value = await getSetting(query.key)
|
||||
if (!value && query.key === 'modelName') {
|
||||
value = runtimeConfig.openaiModelName
|
||||
}
|
||||
return apiSuccess(value)
|
||||
} else if (method === 'POST') {
|
||||
const body = await readBody(event)
|
||||
await setSetting(body.key, body.value)
|
||||
return apiSuccess()
|
||||
}
|
||||
})
|
||||
@@ -1,18 +0,0 @@
|
||||
import Keyv from 'keyv'
|
||||
import KeyvSqlite from "@keyv/sqlite";
|
||||
|
||||
const sqlite = new KeyvSqlite()
|
||||
|
||||
const cacheOptions = {
|
||||
namespace: 'settings',
|
||||
uri: 'sqlite://database.sqlite',
|
||||
}
|
||||
const cache = new Keyv(cacheOptions);
|
||||
|
||||
export const getSetting = async (key) => {
|
||||
return await cache.get(key)
|
||||
}
|
||||
|
||||
export const setSetting = async (key, value) => {
|
||||
return await cache.set(key, value)
|
||||
}
|
||||
Reference in New Issue
Block a user