init + login
This commit is contained in:
1
.env.example
Normal file
1
.env.example
Normal file
@@ -0,0 +1 @@
|
||||
API=
|
||||
50
.eslintrc.json
Normal file
50
.eslintrc.json
Normal file
@@ -0,0 +1,50 @@
|
||||
{
|
||||
"root": true,
|
||||
"extends": ["eslint:recommended", "plugin:@typescript-eslint/recommended"],
|
||||
"parser": "@typescript-eslint/parser",
|
||||
"plugins": ["@typescript-eslint"],
|
||||
"rules": {
|
||||
"@typescript-eslint/strict-boolean-expressions": [
|
||||
2,
|
||||
{
|
||||
"allowString": false,
|
||||
"allowNumber": false
|
||||
}
|
||||
],
|
||||
|
||||
/* important */
|
||||
"prefer-const": "error",
|
||||
"quotes": ["error", "single"],
|
||||
|
||||
"block-scoped-var": "error",
|
||||
"camelcase": "error",
|
||||
"consistent-this": ["error", "that"],
|
||||
"no-else-return": "error",
|
||||
"no-eq-null": "error",
|
||||
"no-floating-decimal": "error",
|
||||
"no-implicit-coercion": "error",
|
||||
"no-implied-eval": "error",
|
||||
"no-invalid-this": "error",
|
||||
"require-await": "error",
|
||||
"yoda": "error",
|
||||
|
||||
"semi": ["error", "always"],
|
||||
"semi-style": ["error", "last"],
|
||||
|
||||
/* less important */
|
||||
"no-unreachable-loop": "error",
|
||||
"no-unused-private-class-members": "error",
|
||||
"no-use-before-define": "error",
|
||||
"no-unmodified-loop-condition": "error",
|
||||
"no-duplicate-imports": "error",
|
||||
"no-promise-executor-return": "error",
|
||||
"no-self-compare": "error",
|
||||
"no-constructor-return": "error",
|
||||
"no-template-curly-in-string": "error",
|
||||
"array-callback-return": "error",
|
||||
"no-eval": "error",
|
||||
"no-extend-native": "error",
|
||||
"no-extra-bind": "error"
|
||||
},
|
||||
"ignorePatterns": ["src/**/*.test.ts", "src/frontend/generated/*"]
|
||||
}
|
||||
8
.gitignore
vendored
Normal file
8
.gitignore
vendored
Normal file
@@ -0,0 +1,8 @@
|
||||
.DS_Store
|
||||
.vscode
|
||||
.env
|
||||
dist
|
||||
node_modules
|
||||
package-lock.json
|
||||
pnpm-lock.yaml
|
||||
yarn.lock
|
||||
6
.prettierignore
Normal file
6
.prettierignore
Normal file
@@ -0,0 +1,6 @@
|
||||
dist
|
||||
log
|
||||
storage
|
||||
package.lock.json
|
||||
pnpm-lock.yaml
|
||||
yarn.lock
|
||||
10
.prettierrc
Normal file
10
.prettierrc
Normal file
@@ -0,0 +1,10 @@
|
||||
{
|
||||
"printWidth": 80,
|
||||
"tabWidth": 4,
|
||||
"useTabs": false,
|
||||
"semi": false,
|
||||
"singleQuote": true,
|
||||
"trailingComma": "es5",
|
||||
"bracketSpacing": true,
|
||||
"arrowParens": "avoid"
|
||||
}
|
||||
30
package.json
Normal file
30
package.json
Normal file
@@ -0,0 +1,30 @@
|
||||
{
|
||||
"name": "discord-retokenizer-web",
|
||||
"version": "1.0.0",
|
||||
"description": "",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"env-ts": "import-meta-env-typescript -x .env.example -o src"
|
||||
},
|
||||
"keywords": [],
|
||||
"author": "",
|
||||
"license": "ISC",
|
||||
"dependencies": {
|
||||
"@import-meta-env/typescript": "^0.2.0",
|
||||
"@types/react": "^18.0.26",
|
||||
"@types/react-dom": "^18.0.9",
|
||||
"@vitejs/plugin-react": "^3.0.0",
|
||||
"dotenv": "^16.0.3",
|
||||
"eslint": "^8.29.0",
|
||||
"prettier": "^2.8.1",
|
||||
"react": "^18.2.0",
|
||||
"react-dom": "^18.2.0",
|
||||
"react-router-dom": "^6.4.5",
|
||||
"sass": "^1.56.2",
|
||||
"typescript": "^4.9.4",
|
||||
"vite": "^4.0.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@import-meta-env/unplugin": "^0.4.0"
|
||||
}
|
||||
}
|
||||
25
src/App.tsx
Normal file
25
src/App.tsx
Normal file
@@ -0,0 +1,25 @@
|
||||
import { BrowserRouter, Routes, Route } from "react-router-dom";
|
||||
import { Layout } from "./pages/Layout";
|
||||
import { Login } from "./pages/Login";
|
||||
|
||||
function App() {
|
||||
return (
|
||||
<div>
|
||||
<BrowserRouter>
|
||||
<Routes>
|
||||
<Route path="/login" element={<Login />} />
|
||||
<Route path="/" element={<Layout />}> {/* collapsible sidebar with links */}
|
||||
{/* <Route index element={<Dashboard />} />
|
||||
<Route path="bots" element={<Bots />} />
|
||||
<Route path="bot/:id" element={<Bot />} />
|
||||
<Route path="tokens" element={<Contact />} />
|
||||
<Route path="token/:id" element={<Contact />} />
|
||||
<Route path="*" element={<MissingPage />} /> */}
|
||||
</Route>
|
||||
</Routes>
|
||||
</BrowserRouter>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
export default App
|
||||
48
src/authorization.ts
Normal file
48
src/authorization.ts
Normal file
@@ -0,0 +1,48 @@
|
||||
export function getTokenFromStorage() {
|
||||
return localStorage.getItem('token')
|
||||
}
|
||||
|
||||
export enum LoggedIn {
|
||||
No,
|
||||
Yes,
|
||||
Invalid,
|
||||
}
|
||||
|
||||
export function isLoggedIn(): LoggedIn {
|
||||
const token = getTokenFromStorage()
|
||||
if (token) {
|
||||
return LoggedIn.Yes
|
||||
}
|
||||
return LoggedIn.No
|
||||
}
|
||||
|
||||
export function authorizeLogin(
|
||||
username: string,
|
||||
password: string,
|
||||
setLoggedIn: (loggedIn: LoggedIn) => void
|
||||
) {
|
||||
// TODO why is ENV undefined?
|
||||
console.log(import.meta.env.API)
|
||||
|
||||
fetch('http://localhost:8080' + '/login', {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
body: JSON.stringify({
|
||||
username: username,
|
||||
password: password,
|
||||
}),
|
||||
})
|
||||
.then(response => {
|
||||
return response.json()
|
||||
})
|
||||
.then(data => {
|
||||
if (data.token) {
|
||||
localStorage.setItem('token', data.token)
|
||||
setLoggedIn(LoggedIn.Yes)
|
||||
} else {
|
||||
setLoggedIn(LoggedIn.Invalid)
|
||||
}
|
||||
})
|
||||
}
|
||||
9
src/import-meta-env.d.ts
vendored
Normal file
9
src/import-meta-env.d.ts
vendored
Normal file
@@ -0,0 +1,9 @@
|
||||
// Generated by '@import-meta-env/typescript'
|
||||
|
||||
interface ImportMetaEnv {
|
||||
readonly API: string;
|
||||
}
|
||||
|
||||
interface ImportMeta {
|
||||
readonly env: ImportMetaEnv;
|
||||
}
|
||||
7
src/index.html
Normal file
7
src/index.html
Normal file
@@ -0,0 +1,7 @@
|
||||
<html>
|
||||
<head> </head>
|
||||
<body>
|
||||
<div id="root"></div>
|
||||
<script type="module" src="./main.tsx"></script>
|
||||
</body>
|
||||
</html>
|
||||
95
src/index.scss
Normal file
95
src/index.scss
Normal file
@@ -0,0 +1,95 @@
|
||||
body {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
#root {
|
||||
--bg-color-1: #262626;
|
||||
--bg-color-2: #363636;
|
||||
--bg-color-3: #464646;
|
||||
--text-color-1: #d6d6d6;
|
||||
--text-color-2: #a6a6a6;
|
||||
--color-error: #860000;
|
||||
|
||||
background-color: var(--bg-color-1);
|
||||
color: var(--text-color-1);
|
||||
font-family: Helvetica, Arial, sans-serif;
|
||||
font-size: 16px;
|
||||
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
|
||||
input {
|
||||
border: none;
|
||||
outline: none;
|
||||
background-color: var(--bg-color-2);
|
||||
color: var(--text-color-2);
|
||||
font-size: 16px;
|
||||
|
||||
&[type="text"],
|
||||
&[type="password"] {
|
||||
width: 200px;
|
||||
}
|
||||
|
||||
padding: 7px 10px;
|
||||
margin: 0 5px;
|
||||
|
||||
border-radius: 200px;
|
||||
|
||||
&:focus {
|
||||
background-color: var(--bg-color-3);
|
||||
color: var(--text-color-1);
|
||||
}
|
||||
|
||||
border-bottom: 1px solid transparent;
|
||||
&.error {
|
||||
border-bottom: 1px solid var(--color-error);
|
||||
}
|
||||
}
|
||||
|
||||
button {
|
||||
border: none;
|
||||
outline: none;
|
||||
background-color: var(--bg-color-2);
|
||||
color: var(--text-color-2);
|
||||
font-size: 16px;
|
||||
|
||||
&.svg svg {
|
||||
fill: var(--text-color-2);
|
||||
height: 16px;
|
||||
}
|
||||
|
||||
padding: 7px 10px;
|
||||
margin: 0 5px;
|
||||
|
||||
&.pad {
|
||||
padding: 10px 15px;
|
||||
margin: 10px;
|
||||
}
|
||||
|
||||
border-radius: 200px;
|
||||
|
||||
&:hover,
|
||||
&:focus {
|
||||
background-color: var(--bg-color-3);
|
||||
color: var(--text-color-1);
|
||||
|
||||
&.svg svg {
|
||||
fill: var(--text-color-1);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#login {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
|
||||
.form {
|
||||
margin-bottom: 5px;
|
||||
}
|
||||
}
|
||||
}
|
||||
10
src/main.tsx
Normal file
10
src/main.tsx
Normal file
@@ -0,0 +1,10 @@
|
||||
import React from 'react'
|
||||
import ReactDOM from 'react-dom/client'
|
||||
import App from './App'
|
||||
import './index.scss'
|
||||
|
||||
ReactDOM.createRoot(document.getElementById('root') as HTMLElement).render(
|
||||
<React.StrictMode>
|
||||
<App />
|
||||
</React.StrictMode>
|
||||
)
|
||||
14
src/pages/Layout.tsx
Normal file
14
src/pages/Layout.tsx
Normal file
@@ -0,0 +1,14 @@
|
||||
import { Outlet, Navigate } from 'react-router-dom'
|
||||
import { isLoggedIn, LoggedIn } from '../authorization'
|
||||
|
||||
export function Layout() {
|
||||
return (
|
||||
<div>
|
||||
<h1>Layout</h1>
|
||||
{
|
||||
isLoggedIn() === LoggedIn.No && <Navigate to="/login" replace={true} />
|
||||
}
|
||||
<Outlet />
|
||||
</div>
|
||||
)
|
||||
}
|
||||
42
src/pages/Login.tsx
Normal file
42
src/pages/Login.tsx
Normal file
@@ -0,0 +1,42 @@
|
||||
import { useMemo, useState } from "react"
|
||||
import { Navigate } from "react-router-dom"
|
||||
import { authorizeLogin, isLoggedIn, LoggedIn } from "../authorization"
|
||||
|
||||
export function Login() {
|
||||
var [username, setUsername] = useState("")
|
||||
var [password, setPassword] = useState("")
|
||||
var [loggedIn, setLoggedIn] = useState(isLoggedIn())
|
||||
|
||||
var login = () => {
|
||||
if (username.length === 0 || password.length === 0) {
|
||||
setLoggedIn(LoggedIn.Invalid)
|
||||
return
|
||||
}
|
||||
authorizeLogin(username, password, setLoggedIn)
|
||||
}
|
||||
|
||||
useMemo(() => {
|
||||
if (loggedIn === LoggedIn.Invalid) {
|
||||
setLoggedIn(LoggedIn.No)
|
||||
}
|
||||
}, [username, password])
|
||||
|
||||
return (
|
||||
<div id="login">
|
||||
<div className="form">
|
||||
<input type="text" className={loggedIn === LoggedIn.Invalid ? "error" : ""} onChange={e => setUsername(e.target.value)} />
|
||||
<input type="password" className={loggedIn === LoggedIn.Invalid ? "error" : ""} onChange={e => setPassword(e.target.value)} />
|
||||
</div>
|
||||
<div>
|
||||
<button className="pad svg" onClick={login}>
|
||||
<svg viewBox="0 0 1000 1000">
|
||||
<path d="M619.5,899.5l350-350c27.3-27.3,27.3-71.7,0-99l-350-350c-27.3-27.3-71.7-27.3-99,0c-27.3,27.3-27.3,71.7,0,99L751,430H80c-38.7,0-70,31.3-70,70c0,38.7,31.3,70,70,70h671L520.5,800.5C506.8,814.2,500,832.1,500,850c0,17.9,6.8,35.8,20.5,49.5C547.8,926.8,592.2,926.8,619.5,899.5z" />
|
||||
</svg>
|
||||
</button>
|
||||
</div>
|
||||
{
|
||||
loggedIn === LoggedIn.Yes && <Navigate to="/" replace={true} />
|
||||
}
|
||||
</div>
|
||||
)
|
||||
}
|
||||
23
tsconfig.json
Normal file
23
tsconfig.json
Normal file
@@ -0,0 +1,23 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"target": "ESNext",
|
||||
"module": "ESNext",
|
||||
"rootDir": "./src",
|
||||
"outDir": "./dist",
|
||||
"esModuleInterop": true,
|
||||
"forceConsistentCasingInFileNames": true,
|
||||
"strict": true,
|
||||
"noImplicitReturns": true,
|
||||
"noUnusedLocals": true,
|
||||
"noUnusedParameters": true,
|
||||
"jsx": "react-jsx",
|
||||
|
||||
"useDefineForClassFields": true,
|
||||
"lib": ["ESNext", "DOM"],
|
||||
"moduleResolution": "Node",
|
||||
"sourceMap": true,
|
||||
"resolveJsonModule": true,
|
||||
"noEmit": true,
|
||||
},
|
||||
"include": ["src/**/*"],
|
||||
}
|
||||
23
vite.config.js
Normal file
23
vite.config.js
Normal file
@@ -0,0 +1,23 @@
|
||||
import { defineConfig } from "vite";
|
||||
import { resolve } from "path";
|
||||
import react from '@vitejs/plugin-react'
|
||||
import importMetaEnv from "@import-meta-env/unplugin";
|
||||
|
||||
export default defineConfig({
|
||||
root: resolve(__dirname, 'src'),
|
||||
plugins: [
|
||||
react(),
|
||||
importMetaEnv.vite({
|
||||
example: ".env.example",
|
||||
env: ".env",
|
||||
transformMode: "development",
|
||||
})
|
||||
],
|
||||
build: {
|
||||
rollupOptions: {
|
||||
input: {
|
||||
main: resolve(__dirname, 'index.html')
|
||||
},
|
||||
}
|
||||
}
|
||||
});
|
||||
Reference in New Issue
Block a user