mirror of
https://github.com/arc53/DocsGPT
synced 2024-11-19 21:25:39 +00:00
commit
12e8ee3088
63
frontend/package-lock.json
generated
63
frontend/package-lock.json
generated
@ -9,7 +9,8 @@
|
|||||||
"version": "0.0.0",
|
"version": "0.0.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0"
|
"react-dom": "^18.2.0",
|
||||||
|
"react-router-dom": "^6.8.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/react": "^18.0.27",
|
"@types/react": "^18.0.27",
|
||||||
@ -896,6 +897,14 @@
|
|||||||
"node": ">= 8"
|
"node": ">= 8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@remix-run/router": {
|
||||||
|
"version": "1.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.3.2.tgz",
|
||||||
|
"integrity": "sha512-t54ONhl/h75X94SWsHGQ4G/ZrCEguKSRQr7DrjTciJXW0YU1QhlwYeycvK5JgkzlxmvrK7wq1NB/PLtHxoiDcA==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=14"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@types/json-schema": {
|
"node_modules/@types/json-schema": {
|
||||||
"version": "7.0.11",
|
"version": "7.0.11",
|
||||||
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.11.tgz",
|
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.11.tgz",
|
||||||
@ -4757,6 +4766,36 @@
|
|||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/react-router": {
|
||||||
|
"version": "6.8.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/react-router/-/react-router-6.8.1.tgz",
|
||||||
|
"integrity": "sha512-Jgi8BzAJQ8MkPt8ipXnR73rnD7EmZ0HFFb7jdQU24TynGW1Ooqin2KVDN9voSC+7xhqbbCd2cjGUepb6RObnyg==",
|
||||||
|
"dependencies": {
|
||||||
|
"@remix-run/router": "1.3.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=14"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": ">=16.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/react-router-dom": {
|
||||||
|
"version": "6.8.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.8.1.tgz",
|
||||||
|
"integrity": "sha512-67EXNfkQgf34P7+PSb6VlBuaacGhkKn3kpE51+P6zYSG2kiRoumXEL6e27zTa9+PGF2MNXbgIUHTVlleLbIcHQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"@remix-run/router": "1.3.2",
|
||||||
|
"react-router": "6.8.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=14"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": ">=16.8",
|
||||||
|
"react-dom": ">=16.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/read-cache": {
|
"node_modules/read-cache": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
|
||||||
@ -6226,6 +6265,11 @@
|
|||||||
"fastq": "^1.6.0"
|
"fastq": "^1.6.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"@remix-run/router": {
|
||||||
|
"version": "1.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.3.2.tgz",
|
||||||
|
"integrity": "sha512-t54ONhl/h75X94SWsHGQ4G/ZrCEguKSRQr7DrjTciJXW0YU1QhlwYeycvK5JgkzlxmvrK7wq1NB/PLtHxoiDcA=="
|
||||||
|
},
|
||||||
"@types/json-schema": {
|
"@types/json-schema": {
|
||||||
"version": "7.0.11",
|
"version": "7.0.11",
|
||||||
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.11.tgz",
|
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.11.tgz",
|
||||||
@ -8885,6 +8929,23 @@
|
|||||||
"integrity": "sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ==",
|
"integrity": "sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"react-router": {
|
||||||
|
"version": "6.8.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/react-router/-/react-router-6.8.1.tgz",
|
||||||
|
"integrity": "sha512-Jgi8BzAJQ8MkPt8ipXnR73rnD7EmZ0HFFb7jdQU24TynGW1Ooqin2KVDN9voSC+7xhqbbCd2cjGUepb6RObnyg==",
|
||||||
|
"requires": {
|
||||||
|
"@remix-run/router": "1.3.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"react-router-dom": {
|
||||||
|
"version": "6.8.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.8.1.tgz",
|
||||||
|
"integrity": "sha512-67EXNfkQgf34P7+PSb6VlBuaacGhkKn3kpE51+P6zYSG2kiRoumXEL6e27zTa9+PGF2MNXbgIUHTVlleLbIcHQ==",
|
||||||
|
"requires": {
|
||||||
|
"@remix-run/router": "1.3.2",
|
||||||
|
"react-router": "6.8.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"read-cache": {
|
"read-cache": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
|
||||||
|
@ -20,7 +20,8 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0"
|
"react-dom": "^18.2.0",
|
||||||
|
"react-router-dom": "^6.8.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/react": "^18.0.27",
|
"@types/react": "^18.0.27",
|
||||||
|
@ -1,10 +1,16 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
import { Routes, Route } from 'react-router-dom';
|
||||||
import Navigation from './components/Navigation/Navigation';
|
import Navigation from './components/Navigation/Navigation';
|
||||||
import DocsGPT from './components/DocsGPT';
|
import DocsGPT from './components/DocsGPT';
|
||||||
|
import APIKeyModal from './components/APIKeyModal';
|
||||||
import './App.css';
|
import './App.css';
|
||||||
|
|
||||||
function App() {
|
export default function App() {
|
||||||
|
//Currently using primitive state management. Will most likely be replaced with Redux.
|
||||||
const [isMobile, setIsMobile] = useState(true);
|
const [isMobile, setIsMobile] = useState(true);
|
||||||
|
const [isMenuOpen, setIsMenuOpen] = useState(true);
|
||||||
|
const [isApiModalOpen, setIsApiModalOpen] = useState(true);
|
||||||
|
const [apiKey, setApiKey] = useState('');
|
||||||
|
|
||||||
const handleResize = () => {
|
const handleResize = () => {
|
||||||
if (window.innerWidth > 768 && isMobile) {
|
if (window.innerWidth > 768 && isMobile) {
|
||||||
@ -24,11 +30,26 @@ function App() {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={`${isMobile ? 'flex-col' : 'flex-row'} flex`}>
|
<div
|
||||||
<Navigation isMobile={isMobile} />
|
className={`${
|
||||||
<DocsGPT />
|
isMobile ? 'flex-col' : 'flex-row'
|
||||||
|
} relative flex transition-all`}
|
||||||
|
>
|
||||||
|
<APIKeyModal
|
||||||
|
apiKey={apiKey}
|
||||||
|
setApiKey={setApiKey}
|
||||||
|
isApiModalOpen={isApiModalOpen}
|
||||||
|
setIsApiModalOpen={setIsApiModalOpen}
|
||||||
|
/>
|
||||||
|
<Navigation
|
||||||
|
isMobile={isMobile}
|
||||||
|
isMenuOpen={isMenuOpen}
|
||||||
|
setIsMenuOpen={setIsMenuOpen}
|
||||||
|
setIsApiModalOpen={setIsApiModalOpen}
|
||||||
|
/>
|
||||||
|
<Routes>
|
||||||
|
<Route path="/" element={<DocsGPT isMenuOpen={isMenuOpen} />} />
|
||||||
|
</Routes>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default App;
|
|
||||||
|
60
frontend/src/components/APIKeyModal.tsx
Normal file
60
frontend/src/components/APIKeyModal.tsx
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
export default function APIKeyModal({
|
||||||
|
isApiModalOpen,
|
||||||
|
setIsApiModalOpen,
|
||||||
|
apiKey,
|
||||||
|
setApiKey,
|
||||||
|
}: {
|
||||||
|
isApiModalOpen: boolean;
|
||||||
|
setIsApiModalOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
||||||
|
apiKey: string;
|
||||||
|
setApiKey: React.Dispatch<React.SetStateAction<string>>;
|
||||||
|
}) {
|
||||||
|
const [formError, setFormError] = useState(false);
|
||||||
|
|
||||||
|
const handleResetKey = () => {
|
||||||
|
if (!apiKey) {
|
||||||
|
setFormError(true);
|
||||||
|
} else {
|
||||||
|
setFormError(false);
|
||||||
|
setIsApiModalOpen(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={`${
|
||||||
|
isApiModalOpen ? 'visible' : 'hidden'
|
||||||
|
} absolute z-30 h-screen w-screen bg-gray-alpha`}
|
||||||
|
>
|
||||||
|
<div className="mx-auto mt-24 flex w-128 flex-col gap-4 rounded-lg bg-white p-6 shadow-lg">
|
||||||
|
<p className="text-xl text-jet">OpenAI API Key</p>
|
||||||
|
<p className="text-lg leading-5 text-gray-500">
|
||||||
|
Before you can start using DocsGPT we need you to provide an API key
|
||||||
|
for llm. Currently, we support only OpenAI but soon many more. You can
|
||||||
|
find it here.
|
||||||
|
</p>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
className="h-10 w-full border-b-2 border-jet focus:outline-none"
|
||||||
|
value={apiKey}
|
||||||
|
maxLength={100}
|
||||||
|
placeholder="API Key"
|
||||||
|
onChange={(e) => setApiKey(e.target.value)}
|
||||||
|
/>
|
||||||
|
<div className="flex justify-between">
|
||||||
|
{formError && (
|
||||||
|
<p className="text-sm text-red-500">Please enter a valid API key</p>
|
||||||
|
)}
|
||||||
|
<button
|
||||||
|
onClick={handleResetKey}
|
||||||
|
className="ml-auto h-10 w-20 rounded-lg bg-violet-800 text-white transition-all hover:bg-violet-700"
|
||||||
|
>
|
||||||
|
Save
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
@ -1,3 +1,7 @@
|
|||||||
export default function DocsGPT() {
|
export default function DocsGPT({ isMenuOpen }: { isMenuOpen: boolean }) {
|
||||||
return <div className="md:ml-72 lg:ml-96">Docs GPT Chat Placeholder</div>;
|
return (
|
||||||
|
<div className={`${isMenuOpen ? 'md:ml-72 lg:ml-96' : 'ml-16'}`}>
|
||||||
|
Docs GPT Chat Placeholder
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,26 +1,103 @@
|
|||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import Arrow1 from './imgs/arrow.svg';
|
import Arrow1 from './imgs/arrow.svg';
|
||||||
|
import Key from './imgs/key.svg';
|
||||||
|
import Info from './imgs/info.svg';
|
||||||
|
import Link from './imgs/link.svg';
|
||||||
|
|
||||||
function MobileNavigation() {
|
function MobileNavigation() {
|
||||||
return <div>Mobile Navigation</div>;
|
return <div>Mobile Navigation</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
function DesktopNavigation() {
|
function DesktopNavigation({
|
||||||
|
isMenuOpen,
|
||||||
|
setIsMenuOpen,
|
||||||
|
setIsApiModalOpen,
|
||||||
|
}: {
|
||||||
|
isMenuOpen: boolean;
|
||||||
|
setIsMenuOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
||||||
|
setIsApiModalOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
||||||
|
}) {
|
||||||
return (
|
return (
|
||||||
<div className="fixed h-screen w-72 border-r-2 border-gray-100 bg-gray-50 lg:w-96">
|
<div
|
||||||
<div className="h-16 border-b-2 border-gray-100">
|
className={`${
|
||||||
<button className="float-right mr-4 mt-5 h-5 w-5">
|
isMenuOpen ? 'w-72 lg:w-96' : 'w-16'
|
||||||
<img src={Arrow1} alt="menu toggle" className="m-auto w-3" />
|
} fixed flex h-screen flex-col border-r-2 border-gray-100 bg-gray-50 transition-all`}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className={`${
|
||||||
|
isMenuOpen ? 'w-full' : 'w-16'
|
||||||
|
} ml-auto h-16 border-b-2 border-gray-100`}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
className="float-right mr-5 mt-5 h-5 w-5"
|
||||||
|
onClick={() => setIsMenuOpen(!isMenuOpen)}
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={Arrow1}
|
||||||
|
alt="menu toggle"
|
||||||
|
className={`${
|
||||||
|
isMenuOpen ? 'rotate-0' : 'rotate-180'
|
||||||
|
} m-auto w-3 transition-all`}
|
||||||
|
/>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{isMenuOpen && (
|
||||||
|
<>
|
||||||
|
<div className="flex-grow border-b-2 border-gray-100"></div>
|
||||||
|
|
||||||
|
<div className="flex h-16 flex-col border-b-2 border-gray-100">
|
||||||
|
<div
|
||||||
|
className="my-auto mx-4 flex h-12 cursor-pointer gap-4 rounded-md hover:bg-gray-100"
|
||||||
|
onClick={() => setIsApiModalOpen(true)}
|
||||||
|
>
|
||||||
|
<img src={Key} alt="key" className="ml-2 w-6" />
|
||||||
|
<p className="my-auto text-eerie-black">Reset Key</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex h-48 flex-col border-b-2 border-gray-100">
|
||||||
|
<div className="my-auto mx-4 flex h-12 cursor-pointer gap-4 rounded-md hover:bg-gray-100">
|
||||||
|
<img src={Info} alt="info" className="ml-2 w-5" />
|
||||||
|
<p className="my-auto text-eerie-black">About</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="my-auto mx-4 flex h-12 cursor-pointer gap-4 rounded-md hover:bg-gray-100">
|
||||||
|
<img src={Link} alt="link" className="ml-2 w-5" />
|
||||||
|
<p className="my-auto text-eerie-black">Discord</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="my-auto mx-4 flex h-12 cursor-pointer gap-4 rounded-md hover:bg-gray-100">
|
||||||
|
<img src={Link} alt="link" className="ml-2 w-5" />
|
||||||
|
<p className="my-auto text-eerie-black">Github</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Navigation({ isMobile }: { isMobile: boolean }) {
|
export default function Navigation({
|
||||||
|
isMobile,
|
||||||
|
isMenuOpen,
|
||||||
|
setIsMenuOpen,
|
||||||
|
setIsApiModalOpen,
|
||||||
|
}: {
|
||||||
|
isMobile: boolean;
|
||||||
|
isMenuOpen: boolean;
|
||||||
|
setIsMenuOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
||||||
|
setIsApiModalOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
||||||
|
}) {
|
||||||
if (isMobile) {
|
if (isMobile) {
|
||||||
return <MobileNavigation />;
|
return <MobileNavigation />;
|
||||||
} else {
|
} else {
|
||||||
return <DesktopNavigation />;
|
return (
|
||||||
|
<DesktopNavigation
|
||||||
|
isMenuOpen={isMenuOpen}
|
||||||
|
setIsMenuOpen={setIsMenuOpen}
|
||||||
|
setIsApiModalOpen={setIsApiModalOpen}
|
||||||
|
/>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
1
frontend/src/components/Navigation/PastChat.tsx
Normal file
1
frontend/src/components/Navigation/PastChat.tsx
Normal file
@ -0,0 +1 @@
|
|||||||
|
export default function PastChat() {}
|
3
frontend/src/components/Navigation/imgs/info.svg
Normal file
3
frontend/src/components/Navigation/imgs/info.svg
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M10 0C4.48 0 0 4.48 0 10C0 15.52 4.48 20 10 20C15.52 20 20 15.52 20 10C20 4.48 15.52 0 10 0ZM11 15H9V9H11V15ZM11 7H9V5H11V7Z" fill="black" fill-opacity="0.54"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 273 B |
3
frontend/src/components/Navigation/imgs/key.svg
Normal file
3
frontend/src/components/Navigation/imgs/key.svg
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<svg width="22" height="12" viewBox="0 0 22 12" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M11.65 4C10.83 1.67 8.61 0 6 0C2.69 0 0 2.69 0 6C0 9.31 2.69 12 6 12C8.61 12 10.83 10.33 11.65 8H16V12H20V8H22V4H11.65ZM6 8C4.9 8 4 7.1 4 6C4 4.9 4.9 4 6 4C7.1 4 8 4.9 8 6C8 7.1 7.1 8 6 8Z" fill="black" fill-opacity="0.54"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 337 B |
3
frontend/src/components/Navigation/imgs/link.svg
Normal file
3
frontend/src/components/Navigation/imgs/link.svg
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<svg width="18" height="18" viewBox="0 0 18 18" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M16 16H2V2H9V0H2C0.89 0 0 0.9 0 2V16C0 17.1 0.89 18 2 18H16C17.1 18 18 17.1 18 16V9H16V16ZM11 0V2H14.59L4.76 11.83L6.17 13.24L16 3.41V7H18V0H11Z" fill="black" fill-opacity="0.54"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 293 B |
@ -1,10 +1,13 @@
|
|||||||
import React from 'react'
|
import React from 'react';
|
||||||
import ReactDOM from 'react-dom/client'
|
import ReactDOM from 'react-dom/client';
|
||||||
import App from './App'
|
import App from './App';
|
||||||
import './index.css'
|
import { BrowserRouter } from 'react-router-dom';
|
||||||
|
import './index.css';
|
||||||
|
|
||||||
ReactDOM.createRoot(document.getElementById('root') as HTMLElement).render(
|
ReactDOM.createRoot(document.getElementById('root') as HTMLElement).render(
|
||||||
<React.StrictMode>
|
<React.StrictMode>
|
||||||
|
<BrowserRouter>
|
||||||
<App />
|
<App />
|
||||||
|
</BrowserRouter>
|
||||||
</React.StrictMode>,
|
</React.StrictMode>,
|
||||||
)
|
);
|
||||||
|
@ -7,7 +7,12 @@ module.exports = {
|
|||||||
112: '28rem',
|
112: '28rem',
|
||||||
128: '32rem',
|
128: '32rem',
|
||||||
},
|
},
|
||||||
|
colors: {
|
||||||
|
'eerie-black': '#212121',
|
||||||
|
jet: '#343541',
|
||||||
|
'gray-alpha': 'rgba(0,0,0, .1)',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
plugins: [],
|
plugins: [],
|
||||||
}
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user