Compare commits
12 Commits
v1-beta
...
c9f227c70c
| Author | SHA1 | Date | |
|---|---|---|---|
|
c9f227c70c
|
|||
|
25c1728c27
|
|||
|
7df09f580a
|
|||
|
407b778131
|
|||
|
a38fd042ba
|
|||
|
45a842b6fe
|
|||
|
4d07dde87a
|
|||
|
5a4918330e
|
|||
|
b6ce89b712
|
|||
|
2f68785a01
|
|||
|
d452809c44
|
|||
|
ed30bf6bb1
|
@@ -1,3 +1,4 @@
|
|||||||
|
import os
|
||||||
from datetime import datetime, timezone
|
from datetime import datetime, timezone
|
||||||
from sqlmodel import (
|
from sqlmodel import (
|
||||||
ARRAY,
|
ARRAY,
|
||||||
@@ -10,8 +11,9 @@ from sqlmodel import (
|
|||||||
create_engine,
|
create_engine,
|
||||||
)
|
)
|
||||||
|
|
||||||
with open("db.secrets", "r") as f:
|
# with open("db.secrets", "r") as f:
|
||||||
db_secrets = f.readline().strip()
|
# db_secrets = f.readline().strip()
|
||||||
|
db_secrets = f"postgresql+psycopg://{os.environ['DB_USER']}:{os.environ['DB_PASS']}@{os.environ['DB_HOST']}:{os.environ['DB_PORT']}/{os.environ['DB_NAME']}"
|
||||||
|
|
||||||
engine = create_engine(
|
engine = create_engine(
|
||||||
db_secrets,
|
db_secrets,
|
||||||
|
|||||||
@@ -28,10 +28,7 @@ app = FastAPI(
|
|||||||
title="cutt", swagger_ui_parameters={"syntaxHighlight": {"theme": "monokai"}}
|
title="cutt", swagger_ui_parameters={"syntaxHighlight": {"theme": "monokai"}}
|
||||||
)
|
)
|
||||||
api_router = APIRouter(prefix="/api")
|
api_router = APIRouter(prefix="/api")
|
||||||
origins = [
|
origins = ["https://cutt.0124816.xyz"]
|
||||||
"https://cutt.0124816.xyz",
|
|
||||||
"http://localhost:5173",
|
|
||||||
]
|
|
||||||
|
|
||||||
app.add_middleware(
|
app.add_middleware(
|
||||||
CORSMiddleware,
|
CORSMiddleware,
|
||||||
|
|||||||
1
frontend/.env
Normal file
1
frontend/.env
Normal file
@@ -0,0 +1 @@
|
|||||||
|
VITE_BASE_URL=
|
||||||
10
frontend/Dockerfile.frontend
Normal file
10
frontend/Dockerfile.frontend
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
FROM node:alpine
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
COPY package.json ./
|
||||||
|
|
||||||
|
RUN npm install
|
||||||
|
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
RUN npm run build
|
||||||
42
frontend/package.json
Normal file
42
frontend/package.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"name": "cutt",
|
||||||
|
"private": true,
|
||||||
|
"version": "0.0.0",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "vite",
|
||||||
|
"build": "tsc -b && vite build",
|
||||||
|
"lint": "eslint .",
|
||||||
|
"preview": "vite preview"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"bulma": "^1.0.4",
|
||||||
|
"jwt-decode": "^4.0.0",
|
||||||
|
"react": "^19.2.3",
|
||||||
|
"react-dom": "^19.2.3",
|
||||||
|
"react-sortablejs": "^6.1.4",
|
||||||
|
"reagraph": "^4.30.7",
|
||||||
|
"sortablejs": "^1.15.6"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@eslint/js": "^9.39.2",
|
||||||
|
"@types/node": "^25.0.3",
|
||||||
|
"@types/react": "^19.2.7",
|
||||||
|
"@types/react-dom": "^19.2.3",
|
||||||
|
"@types/sortablejs": "^1.15.9",
|
||||||
|
"@vitejs/plugin-react": "^5.1.2",
|
||||||
|
"eslint": "^9.39.2",
|
||||||
|
"eslint-plugin-react-hooks": "^7.0.1",
|
||||||
|
"eslint-plugin-react-refresh": "^0.4.26",
|
||||||
|
"globals": "^16.5.0",
|
||||||
|
"react-router": "^7.10.1",
|
||||||
|
"typescript": "~5.9.3",
|
||||||
|
"typescript-eslint": "^8.50.0",
|
||||||
|
"vite": "^7.3.0"
|
||||||
|
},
|
||||||
|
"prettier": {
|
||||||
|
"trailingComma": "es5",
|
||||||
|
"tabWidth": 2,
|
||||||
|
"semi": true
|
||||||
|
}
|
||||||
|
}
|
||||||
58
frontend/public/cutt.svg
Normal file
58
frontend/public/cutt.svg
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<svg
|
||||||
|
width="80"
|
||||||
|
height="50"
|
||||||
|
viewBox="0 0 80 50"
|
||||||
|
version="1.1"
|
||||||
|
id="svg1"
|
||||||
|
sodipodi:docname="cutt.svg"
|
||||||
|
inkscape:version="1.4.2 (ebf0e940d0, 2025-05-08)"
|
||||||
|
inkscape:export-filename="cutt.svg"
|
||||||
|
inkscape:export-xdpi="362.84"
|
||||||
|
inkscape:export-ydpi="362.84"
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg">
|
||||||
|
<defs
|
||||||
|
id="defs1" />
|
||||||
|
<sodipodi:namedview
|
||||||
|
id="namedview1"
|
||||||
|
pagecolor="#ffffff"
|
||||||
|
bordercolor="#000000"
|
||||||
|
borderopacity="0.25"
|
||||||
|
inkscape:showpageshadow="2"
|
||||||
|
inkscape:pageopacity="0.0"
|
||||||
|
inkscape:pagecheckerboard="0"
|
||||||
|
inkscape:deskcolor="#d1d1d1"
|
||||||
|
inkscape:zoom="12.413459"
|
||||||
|
inkscape:cx="38.909381"
|
||||||
|
inkscape:cy="55.786224"
|
||||||
|
inkscape:window-width="1408"
|
||||||
|
inkscape:window-height="1727"
|
||||||
|
inkscape:window-x="0"
|
||||||
|
inkscape:window-y="0"
|
||||||
|
inkscape:window-maximized="0"
|
||||||
|
inkscape:current-layer="svg1" />
|
||||||
|
<ellipse
|
||||||
|
cx="40"
|
||||||
|
cy="25"
|
||||||
|
rx="20.262579"
|
||||||
|
ry="20.632982"
|
||||||
|
style="fill:#c7d6f1;fill-opacity:1;stroke:#3366cc;stroke-width:8.73336"
|
||||||
|
id="ellipse1" />
|
||||||
|
<path
|
||||||
|
d="m -3.4e-4,17.669765 h 80.00068 v 14.66047 H -3.4e-4 Z"
|
||||||
|
style="fill:#000000;stroke-width:3.56018;paint-order:stroke fill markers"
|
||||||
|
id="path1" />
|
||||||
|
<text
|
||||||
|
xml:space="preserve"
|
||||||
|
x="39.788086"
|
||||||
|
y="29.819336"
|
||||||
|
style="font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;font-size:14px;line-height:1;font-family:Sans;-inkscape-font-specification:'Sans Bold';text-align:center;letter-spacing:2.83px;writing-mode:lr-tb;direction:ltr;text-anchor:middle;fill:#ffffff;stroke:#ffffff;stroke-width:0.655;stroke-dasharray:none;stroke-opacity:1;paint-order:stroke fill markers"
|
||||||
|
id="text1"><tspan
|
||||||
|
x="39.788086"
|
||||||
|
y="29.819336"
|
||||||
|
style="font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;font-size:14px;font-family:Sans;-inkscape-font-specification:'Sans Bold';letter-spacing:2.83px;fill:#ffffff;stroke:#ffffff;stroke-width:0.655;stroke-dasharray:none;stroke-opacity:1"
|
||||||
|
id="tspan1">CUTT</tspan></text>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.2 KiB |
|
Before Width: | Height: | Size: 3.3 KiB After Width: | Height: | Size: 3.3 KiB |
|
Before Width: | Height: | Size: 214 B After Width: | Height: | Size: 214 B |
46
frontend/src/CUTT.tsx
Normal file
46
frontend/src/CUTT.tsx
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import "./main.css";
|
||||||
|
import Header from "./Header";
|
||||||
|
import Footer from "./Footer";
|
||||||
|
import { BrowserRouter, Route, Routes } from "react-router";
|
||||||
|
import { SessionProvider } from "./Session";
|
||||||
|
import Rankings from "./Form";
|
||||||
|
import TeamPanel from "./TeamPanel";
|
||||||
|
import { GraphComponent } from "./Network";
|
||||||
|
import MVPChart from "./MVPChart";
|
||||||
|
|
||||||
|
const Maintenance = () => {
|
||||||
|
return (
|
||||||
|
<section className="hero is-large">
|
||||||
|
<div className="hero-body has-text-centered">
|
||||||
|
<p className="title is-1">🚧</p>
|
||||||
|
<p className="subtitle">We are under maintenance.</p>
|
||||||
|
<p>Please check back later. Thank you for your patience.</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
function App() {
|
||||||
|
return (
|
||||||
|
<BrowserRouter>
|
||||||
|
<Routes>
|
||||||
|
<Route
|
||||||
|
path="/*"
|
||||||
|
element={
|
||||||
|
<SessionProvider>
|
||||||
|
<Header />
|
||||||
|
<Routes>
|
||||||
|
<Route index element={<Rankings />} />
|
||||||
|
<Route path="network" element={<GraphComponent />} />
|
||||||
|
<Route path="mvp" element={<MVPChart />} />
|
||||||
|
<Route path="team" element={<TeamPanel />} />
|
||||||
|
</Routes>
|
||||||
|
<Footer />
|
||||||
|
</SessionProvider>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</Routes>
|
||||||
|
</BrowserRouter>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
export default App;
|
||||||
25
frontend/src/Footer.tsx
Normal file
25
frontend/src/Footer.tsx
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import { useLocation } from "react-router";
|
||||||
|
import { Link } from "react-router";
|
||||||
|
import { useSession } from "./Session";
|
||||||
|
|
||||||
|
export default function Footer() {
|
||||||
|
const location = useLocation();
|
||||||
|
const { user, teams } = useSession();
|
||||||
|
return (
|
||||||
|
<footer className="footer">
|
||||||
|
<div className="content has-text-centered">
|
||||||
|
<p className="grey">
|
||||||
|
something not working? message <a href="https://t.me/x0124816">me </a>
|
||||||
|
or fix it here:
|
||||||
|
<a
|
||||||
|
className="icon is-small"
|
||||||
|
href="https://git.0124816.xyz/julius/cutt"
|
||||||
|
key="gitea"
|
||||||
|
>
|
||||||
|
<img src="gitea.svg" alt="gitea" />
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
);
|
||||||
|
}
|
||||||
528
frontend/src/Form.tsx
Normal file
528
frontend/src/Form.tsx
Normal file
@@ -0,0 +1,528 @@
|
|||||||
|
import { useEffect, useRef, useState } from "react";
|
||||||
|
import { ReactSortable, ReactSortableProps } from "react-sortablejs";
|
||||||
|
import { apiAuth, User } from "./api";
|
||||||
|
import { TeamState, useSession } from "./Session";
|
||||||
|
import TabController from "./TabController";
|
||||||
|
import { Chemistry, MVPRanking, PlayerType } from "./types";
|
||||||
|
|
||||||
|
type PlayerListProps = Partial<ReactSortableProps<any>> & {
|
||||||
|
orderedList?: boolean;
|
||||||
|
gender?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
function PlayerList(props: PlayerListProps) {
|
||||||
|
const fmps = props.list?.filter((item) => item.gender === "fmp").length;
|
||||||
|
return (
|
||||||
|
<ReactSortable
|
||||||
|
{...props}
|
||||||
|
className="buttons is-centered is-clearfix"
|
||||||
|
direction={"vertical"}
|
||||||
|
animation={200}
|
||||||
|
swapThreshold={1}
|
||||||
|
handle=".handle"
|
||||||
|
draggable=".handle"
|
||||||
|
>
|
||||||
|
{props.list &&
|
||||||
|
props.list.map((item, index) => (
|
||||||
|
<div className="handle" key={item.id}>
|
||||||
|
<button
|
||||||
|
className={
|
||||||
|
"button is-primary is-light " +
|
||||||
|
(props.gender ? item.gender : "")
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{props.orderedList
|
||||||
|
? props.gender
|
||||||
|
? index +
|
||||||
|
1 -
|
||||||
|
(item.gender !== "fmp" ? fmps! : 0) +
|
||||||
|
". " +
|
||||||
|
item.display_name
|
||||||
|
: index + 1 + ". " + item.display_name
|
||||||
|
: item.display_name}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</ReactSortable>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function PlayerMenuList(props: PlayerListProps) {
|
||||||
|
const fmps = props.list?.filter((item) => item.gender === "fmp").length;
|
||||||
|
return (
|
||||||
|
<ReactSortable
|
||||||
|
{...props}
|
||||||
|
className="menu-list"
|
||||||
|
animation={200}
|
||||||
|
swapThreshold={1}
|
||||||
|
>
|
||||||
|
{props.list &&
|
||||||
|
props.list.map((item, index) => (
|
||||||
|
<p
|
||||||
|
key={item.id}
|
||||||
|
className={
|
||||||
|
"menu-item is-primary is-light " +
|
||||||
|
(props.gender ? item.gender : "")
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{props.orderedList
|
||||||
|
? props.gender
|
||||||
|
? index +
|
||||||
|
1 -
|
||||||
|
(item.gender !== "fmp" ? fmps! : 0) +
|
||||||
|
". " +
|
||||||
|
item.display_name
|
||||||
|
: index + 1 + ". " + item.display_name
|
||||||
|
: item.display_name}
|
||||||
|
</p>
|
||||||
|
))}
|
||||||
|
</ReactSortable>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function filterSort(list: User[], ids: number[]): User[] {
|
||||||
|
const objectMap = new Map(list.map((obj) => [obj.id, obj]));
|
||||||
|
const filteredAndSortedObjects = ids
|
||||||
|
.map((id) => objectMap.get(id))
|
||||||
|
.filter((obj) => obj !== undefined);
|
||||||
|
return filteredAndSortedObjects;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface PlayerInfoProps {
|
||||||
|
user: User;
|
||||||
|
teams: TeamState;
|
||||||
|
players: User[];
|
||||||
|
}
|
||||||
|
|
||||||
|
function TypeDnD({ user, teams, players }: PlayerInfoProps) {
|
||||||
|
const [availablePlayers, setAvailablePlayers] = useState<User[]>(players);
|
||||||
|
const [handlers, setHandlers] = useState<User[]>([]);
|
||||||
|
const [combis, setCombis] = useState<User[]>([]);
|
||||||
|
const [cutters, setCutters] = useState<User[]>([]);
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
handleGet();
|
||||||
|
}, [players]);
|
||||||
|
|
||||||
|
const [dialog, setDialog] = useState("dialog");
|
||||||
|
const dialogRef = useRef<HTMLDialogElement>(null);
|
||||||
|
|
||||||
|
async function handleSubmit() {
|
||||||
|
if (dialogRef.current) dialogRef.current.showModal();
|
||||||
|
setDialog("sending...");
|
||||||
|
let handlerlist = handlers.map(({ id }) => id);
|
||||||
|
let combilist = combis.map(({ id }) => id);
|
||||||
|
let cutterlist = cutters.map(({ id }) => id);
|
||||||
|
const data = {
|
||||||
|
user: user.id,
|
||||||
|
handlers: handlerlist,
|
||||||
|
combis: combilist,
|
||||||
|
cutters: cutterlist,
|
||||||
|
team: teams.activeTeam,
|
||||||
|
};
|
||||||
|
const response = await apiAuth("playertype", data, "PUT");
|
||||||
|
setDialog(response || "try sending again");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleGet() {
|
||||||
|
setLoading(true);
|
||||||
|
const data = await apiAuth(`playertype/${teams.activeTeam}`, null, "GET");
|
||||||
|
if (data.detail) {
|
||||||
|
console.log(data.detail);
|
||||||
|
setAvailablePlayers(players);
|
||||||
|
setHandlers([]);
|
||||||
|
setCombis([]);
|
||||||
|
setCutters([]);
|
||||||
|
} else {
|
||||||
|
const playertype = data as PlayerType;
|
||||||
|
setAvailablePlayers(
|
||||||
|
players.filter(
|
||||||
|
(player) =>
|
||||||
|
!playertype.handlers.includes(player.id) &&
|
||||||
|
!playertype.combis.includes(player.id) &&
|
||||||
|
!playertype.cutters.includes(player.id)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
setHandlers(filterSort(players, playertype.handlers));
|
||||||
|
setCombis(filterSort(players, playertype.combis));
|
||||||
|
setCutters(filterSort(players, playertype.cutters));
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HeaderControl
|
||||||
|
onLoad={handleGet}
|
||||||
|
onClear={() => {
|
||||||
|
setAvailablePlayers(players);
|
||||||
|
setHandlers([]);
|
||||||
|
setCombis([]);
|
||||||
|
setCutters([]);
|
||||||
|
}}
|
||||||
|
onSubmit={handleSubmit}
|
||||||
|
/>
|
||||||
|
<div className="columns container is-multiline is-mobile is-1-mobile">
|
||||||
|
<div className="column is-full is-flex is-justify-content-center">
|
||||||
|
<div className="box" style={{ maxWidth: 800 }}>
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
available players
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={availablePlayers}
|
||||||
|
setList={setAvailablePlayers}
|
||||||
|
group={"type-shared"}
|
||||||
|
className="dragbox reservoir"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="column">
|
||||||
|
<div className="columns">
|
||||||
|
<div className="column">
|
||||||
|
<div className="box">
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
handler
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={handlers}
|
||||||
|
setList={setHandlers}
|
||||||
|
group={"type-shared"}
|
||||||
|
className="dragbox"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="column">
|
||||||
|
<div className="box">
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
combi
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={combis}
|
||||||
|
setList={setCombis}
|
||||||
|
group={"type-shared"}
|
||||||
|
className="middle dragbox"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="column">
|
||||||
|
<div className="box">
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
cutter
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={cutters}
|
||||||
|
setList={setCutters}
|
||||||
|
group={"type-shared"}
|
||||||
|
className="dragbox"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<dialog
|
||||||
|
ref={dialogRef}
|
||||||
|
id="PlayerTypeDialog"
|
||||||
|
onClick={(event) => {
|
||||||
|
event.currentTarget.close();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{dialog}
|
||||||
|
</dialog>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function MVPDnD({ user, teams, players }: PlayerInfoProps) {
|
||||||
|
const [availablePlayers, setAvailablePlayers] = useState<User[]>(players);
|
||||||
|
const [rankedPlayers, setRankedPlayers] = useState<User[]>([]);
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
const [mixed, setMixed] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const activeTeam = teams.teams.find((team) => team.id == teams.activeTeam);
|
||||||
|
activeTeam && setMixed(activeTeam.mixed);
|
||||||
|
handleGet();
|
||||||
|
}, [players]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
handleGet();
|
||||||
|
// setMixedList(rankedPlayers);
|
||||||
|
}, [mixed]);
|
||||||
|
|
||||||
|
const [dialog, setDialog] = useState("dialog");
|
||||||
|
const dialogRef = useRef<HTMLDialogElement>(null);
|
||||||
|
|
||||||
|
async function handleSubmit() {
|
||||||
|
if (dialogRef.current) dialogRef.current.showModal();
|
||||||
|
setDialog("sending...");
|
||||||
|
let mvps = rankedPlayers.map(({ id }) => id);
|
||||||
|
const data = { user: user.id, mvps: mvps, team: teams.activeTeam };
|
||||||
|
const response = await apiAuth("mvps", data, "PUT");
|
||||||
|
response ? setDialog(response) : setDialog("try sending again");
|
||||||
|
}
|
||||||
|
|
||||||
|
const setMixedList = (newList: User[]) =>
|
||||||
|
mixed
|
||||||
|
? setRankedPlayers(
|
||||||
|
newList.sort((a, b) =>
|
||||||
|
a.gender && b.gender ? a.gender.localeCompare(b.gender) : -1
|
||||||
|
)
|
||||||
|
)
|
||||||
|
: setRankedPlayers(newList);
|
||||||
|
|
||||||
|
async function handleGet() {
|
||||||
|
setLoading(true);
|
||||||
|
const data = await apiAuth(`mvps/${teams.activeTeam}`, null, "GET");
|
||||||
|
if (data.detail) {
|
||||||
|
console.log(data.detail);
|
||||||
|
setAvailablePlayers(players);
|
||||||
|
setRankedPlayers([]);
|
||||||
|
} else {
|
||||||
|
const mvps = data as MVPRanking;
|
||||||
|
setMixedList(filterSort(players, mvps.mvps));
|
||||||
|
setAvailablePlayers(
|
||||||
|
players.filter((user) => !mvps.mvps.includes(user.id))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HeaderControl
|
||||||
|
onLoad={handleGet}
|
||||||
|
onClear={() => {
|
||||||
|
setAvailablePlayers(players);
|
||||||
|
setRankedPlayers([]);
|
||||||
|
}}
|
||||||
|
onSubmit={handleSubmit}
|
||||||
|
/>
|
||||||
|
{loading ? (
|
||||||
|
<progress className="progress is-primary" max="100"></progress>
|
||||||
|
) : (
|
||||||
|
<div className="columns container is-mobile is-1-mobile">
|
||||||
|
<div className="column">
|
||||||
|
<div className="box">
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
available players
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={availablePlayers}
|
||||||
|
setList={setAvailablePlayers}
|
||||||
|
group={{
|
||||||
|
name: "mvp-shared",
|
||||||
|
}}
|
||||||
|
className="dragbox"
|
||||||
|
gender={mixed}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="column">
|
||||||
|
<div className="box">
|
||||||
|
<p className="subtitle is-2 has-text-centered is-uppercase has-text-weight-light">
|
||||||
|
🏆
|
||||||
|
</p>
|
||||||
|
<div className="menu">
|
||||||
|
<PlayerMenuList
|
||||||
|
list={rankedPlayers}
|
||||||
|
setList={setMixedList}
|
||||||
|
group={{
|
||||||
|
name: "mvp-shared",
|
||||||
|
}}
|
||||||
|
className="dragbox"
|
||||||
|
orderedList
|
||||||
|
gender={mixed}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<dialog
|
||||||
|
ref={dialogRef}
|
||||||
|
id="MVPDialog"
|
||||||
|
onClick={(event) => {
|
||||||
|
event.currentTarget.close();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{dialog}
|
||||||
|
</dialog>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ChemistryDnDMobile({ user, teams, players }: PlayerInfoProps) {
|
||||||
|
var otherPlayers = players.filter((player) => player.id !== user.id);
|
||||||
|
const [playersLeft, setPlayersLeft] = useState<User[]>([]);
|
||||||
|
const [playersMiddle, setPlayersMiddle] = useState<User[]>(otherPlayers);
|
||||||
|
const [playersRight, setPlayersRight] = useState<User[]>([]);
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
handleGet();
|
||||||
|
}, [players]);
|
||||||
|
|
||||||
|
const [dialog, setDialog] = useState("dialog");
|
||||||
|
const dialogRef = useRef<HTMLDialogElement>(null);
|
||||||
|
|
||||||
|
async function handleSubmit() {
|
||||||
|
if (dialogRef.current) dialogRef.current.showModal();
|
||||||
|
setDialog("sending...");
|
||||||
|
let left = playersLeft.map(({ id }) => id);
|
||||||
|
let middle = playersMiddle.map(({ id }) => id);
|
||||||
|
let right = playersRight.map(({ id }) => id);
|
||||||
|
const data = {
|
||||||
|
user: user.id,
|
||||||
|
hate: left,
|
||||||
|
undecided: middle,
|
||||||
|
love: right,
|
||||||
|
team: teams.activeTeam,
|
||||||
|
};
|
||||||
|
const response = await apiAuth("chemistry", data, "PUT");
|
||||||
|
setDialog(response || "try sending again");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleGet() {
|
||||||
|
setLoading(true);
|
||||||
|
const data = await apiAuth(`chemistry/${teams.activeTeam}`, null, "GET");
|
||||||
|
if (data.detail) {
|
||||||
|
console.log(data.detail);
|
||||||
|
setPlayersRight([]);
|
||||||
|
setPlayersMiddle(otherPlayers);
|
||||||
|
setPlayersLeft([]);
|
||||||
|
} else {
|
||||||
|
const chemistry = data as Chemistry;
|
||||||
|
setPlayersLeft(filterSort(otherPlayers, chemistry.hate));
|
||||||
|
setPlayersMiddle(
|
||||||
|
otherPlayers.filter(
|
||||||
|
(player) =>
|
||||||
|
!chemistry.hate.includes(player.id) &&
|
||||||
|
!chemistry.love.includes(player.id)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
setPlayersRight(filterSort(otherPlayers, chemistry.love));
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<HeaderControl
|
||||||
|
onLoad={handleGet}
|
||||||
|
onClear={() => {
|
||||||
|
setPlayersRight([]);
|
||||||
|
setPlayersMiddle(otherPlayers);
|
||||||
|
setPlayersLeft([]);
|
||||||
|
}}
|
||||||
|
onSubmit={handleSubmit}
|
||||||
|
/>
|
||||||
|
{loading ? (
|
||||||
|
<progress className="progress is-primary" max="100"></progress>
|
||||||
|
) : (
|
||||||
|
<div className="columns container is-multiline is-mobile is-1-mobile">
|
||||||
|
<div className="column is-full is-flex is-justify-content-center">
|
||||||
|
<div className="box" style={{ maxWidth: 800 }}>
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
neutral
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={playersMiddle}
|
||||||
|
setList={setPlayersMiddle}
|
||||||
|
group={"shared"}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="column">
|
||||||
|
<div className="columns is-mobile">
|
||||||
|
<div className="column">
|
||||||
|
<div className="box">
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
rather not
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={playersLeft}
|
||||||
|
setList={setPlayersLeft}
|
||||||
|
group={"shared"}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="column">
|
||||||
|
<div className="box">
|
||||||
|
<p className="subtitle is-6 is-uppercase has-text-weight-light">
|
||||||
|
yes, please ♥️
|
||||||
|
</p>
|
||||||
|
<PlayerList
|
||||||
|
list={playersRight}
|
||||||
|
setList={setPlayersRight}
|
||||||
|
group={"shared"}
|
||||||
|
orderedList
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<dialog
|
||||||
|
ref={dialogRef}
|
||||||
|
id="ChemistryDialog"
|
||||||
|
onClick={(event) => {
|
||||||
|
event.currentTarget.close();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{dialog}
|
||||||
|
</dialog>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
interface HeaderControlProps {
|
||||||
|
onLoad: () => void;
|
||||||
|
onClear: () => void;
|
||||||
|
onSubmit: () => void;
|
||||||
|
}
|
||||||
|
function HeaderControl({ onLoad, onClear, onSubmit }: HeaderControlProps) {
|
||||||
|
return (
|
||||||
|
<div className="buttons is-centered">
|
||||||
|
<button className="button is-small is-light" onClick={onLoad}>
|
||||||
|
🗃️ restore previous
|
||||||
|
</button>
|
||||||
|
<button className="button is-small is-light" onClick={onClear}>
|
||||||
|
🗑️ start over
|
||||||
|
</button>
|
||||||
|
<button className="button is-small is-light" onClick={onSubmit}>
|
||||||
|
💾 submit
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Rankings() {
|
||||||
|
const { user, teams, players } = useSession();
|
||||||
|
const tabs = [
|
||||||
|
{ id: "Chemistry", label: "🧪 Chemistry" },
|
||||||
|
{ id: "MVP", label: "🏆 MVP" },
|
||||||
|
{ id: "Type", label: "🃏 Type" },
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="container block">
|
||||||
|
<p className="notification is-warning is-light">
|
||||||
|
assign as many or as few players as you want and don't forget to 💾
|
||||||
|
<strong> submit</strong> when you're done :)
|
||||||
|
</p>
|
||||||
|
{user && teams && players ? (
|
||||||
|
<TabController tabs={tabs}>
|
||||||
|
<ChemistryDnDMobile {...{ user, teams, players }} />
|
||||||
|
<MVPDnD {...{ user, teams, players }} />
|
||||||
|
<TypeDnD {...{ user, teams, players }} />
|
||||||
|
</TabController>
|
||||||
|
) : (
|
||||||
|
<progress className="progress is-primary" max="100"></progress>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
85
frontend/src/Header.tsx
Normal file
85
frontend/src/Header.tsx
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
import { Link } from "react-router";
|
||||||
|
import { useSession } from "./Session";
|
||||||
|
import { useState } from "react";
|
||||||
|
|
||||||
|
export default function Header() {
|
||||||
|
const { user, teams, setTeams, players, onLogout } = useSession();
|
||||||
|
const [burgerActive, setBurgerActive] = useState(false);
|
||||||
|
return (
|
||||||
|
<nav className="navbar" role="navigation" aria-label="main navigation">
|
||||||
|
<div className="navbar-brand">
|
||||||
|
<Link className="navbar-item" to="/">
|
||||||
|
<img
|
||||||
|
style={{ maxHeight: "unset" }}
|
||||||
|
className="image"
|
||||||
|
alt="cool ultimate team tool"
|
||||||
|
src="cutt.svg"
|
||||||
|
/>
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<a
|
||||||
|
role="button"
|
||||||
|
className={"navbar-burger" + (burgerActive ? " is-active" : "")}
|
||||||
|
aria-label="menu"
|
||||||
|
aria-expanded="false"
|
||||||
|
data-target="navbar"
|
||||||
|
onClick={() => setBurgerActive(!burgerActive)}
|
||||||
|
>
|
||||||
|
<span aria-hidden="true"></span>
|
||||||
|
<span aria-hidden="true"></span>
|
||||||
|
<span aria-hidden="true"></span>
|
||||||
|
<span aria-hidden="true"></span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
className={"navbar-menu" + (burgerActive ? " is-active" : "")}
|
||||||
|
id="navbar"
|
||||||
|
>
|
||||||
|
{user?.scopes.includes(`team:${teams?.activeTeam}`) && (
|
||||||
|
<div className="navbar-start">
|
||||||
|
<Link className="navbar-item" to="/network">
|
||||||
|
<span>Sociogram</span>
|
||||||
|
</Link>
|
||||||
|
<Link className="navbar-item" to="/mvp">
|
||||||
|
<span>MVP</span>
|
||||||
|
</Link>
|
||||||
|
<Link className="navbar-item" to="/team">
|
||||||
|
<span>Team</span>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className="navbar-end">
|
||||||
|
{teams && (
|
||||||
|
<div className="navbar-item has-dropdown is-hoverable">
|
||||||
|
<a className="navbar-link">my teams</a>
|
||||||
|
<div className="navbar-dropdown">
|
||||||
|
{teams.teams.map((team, index) => (
|
||||||
|
<a
|
||||||
|
onClick={() => setTeams({ ...teams, activeTeam: team.id })}
|
||||||
|
className={
|
||||||
|
"navbar-item" +
|
||||||
|
(team.id === teams.activeTeam
|
||||||
|
? " is-bold has-text-weight-extrabold"
|
||||||
|
: "")
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{team.name}
|
||||||
|
</a>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className="navbar-item">
|
||||||
|
{user?.username}
|
||||||
|
<div className="buttons">
|
||||||
|
<button className="button is-light" onClick={onLogout}>
|
||||||
|
Log out
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
);
|
||||||
|
}
|
||||||
104
frontend/src/Login.tsx
Normal file
104
frontend/src/Login.tsx
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { currentUser, login, User } from "./api";
|
||||||
|
import Header from "./Header";
|
||||||
|
import { useLocation, useNavigate } from "react-router";
|
||||||
|
import { Eye, EyeSlash } from "./Icons";
|
||||||
|
|
||||||
|
export interface LoginProps {
|
||||||
|
onLogin: (user: User) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Login = ({ onLogin }: LoginProps) => {
|
||||||
|
const [username, setUsername] = useState("");
|
||||||
|
const [password, setPassword] = useState("");
|
||||||
|
const [error, setError] = useState("");
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
const [visible, setVisible] = useState(false);
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const location = useLocation();
|
||||||
|
|
||||||
|
async function doLogin() {
|
||||||
|
setLoading(true);
|
||||||
|
setError("");
|
||||||
|
const timeout = new Promise((r) => setTimeout(r, 1000));
|
||||||
|
let user: User;
|
||||||
|
try {
|
||||||
|
await login({ username, password });
|
||||||
|
user = await currentUser();
|
||||||
|
} catch (e) {
|
||||||
|
await timeout;
|
||||||
|
setError("failed");
|
||||||
|
setLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
await timeout;
|
||||||
|
onLogin(user);
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleSubmit(e: React.FormEvent) {
|
||||||
|
e.preventDefault();
|
||||||
|
doLogin();
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (location.state) {
|
||||||
|
const queryUsername = location.state.username;
|
||||||
|
const queryPassword = location.state.password;
|
||||||
|
if (queryUsername) setUsername(queryUsername);
|
||||||
|
if (queryPassword) setPassword(queryPassword);
|
||||||
|
navigate(location.pathname, { replace: true });
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<form onSubmit={handleSubmit}>
|
||||||
|
<div className="field">
|
||||||
|
<p className="control">
|
||||||
|
<input
|
||||||
|
className="input"
|
||||||
|
type="text"
|
||||||
|
id="username"
|
||||||
|
name="username"
|
||||||
|
placeholder="username"
|
||||||
|
required
|
||||||
|
value={username}
|
||||||
|
onChange={(evt) => {
|
||||||
|
setError("");
|
||||||
|
setUsername(evt.target.value);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="field">
|
||||||
|
<p className="control">
|
||||||
|
<input
|
||||||
|
className="input"
|
||||||
|
type={visible ? "text" : "password"}
|
||||||
|
id="password"
|
||||||
|
name="password"
|
||||||
|
placeholder="password"
|
||||||
|
minLength={8}
|
||||||
|
value={password}
|
||||||
|
required
|
||||||
|
onChange={(evt) => {
|
||||||
|
setError("");
|
||||||
|
setPassword(evt.target.value);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</p>
|
||||||
|
{error && <p className="help is-danger">{error}</p>}
|
||||||
|
</div>
|
||||||
|
<div className="control">
|
||||||
|
<button
|
||||||
|
className="button"
|
||||||
|
type="submit"
|
||||||
|
value="login"
|
||||||
|
style={{ fontSize: "small" }}
|
||||||
|
>
|
||||||
|
login
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{loading && <span className="loader" />}
|
||||||
|
</form>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -105,7 +105,23 @@ export function SessionProvider(props: SessionProviderProps) {
|
|||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
else if (err) {
|
else if (err) {
|
||||||
content = <Login onLogin={onLogin} />;
|
content = (
|
||||||
|
<section className="section is-medium">
|
||||||
|
<div className="container is-max-tablet">
|
||||||
|
<div className="block">
|
||||||
|
<p className="level-item has-text-centered">
|
||||||
|
<img
|
||||||
|
className="image"
|
||||||
|
alt="cool ultimate team tool"
|
||||||
|
src="cutt.svg"
|
||||||
|
style={{ width: 200 }}
|
||||||
|
/>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<Login onLogin={onLogin} />
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
);
|
||||||
} else
|
} else
|
||||||
content = (
|
content = (
|
||||||
<sessionContext.Provider
|
<sessionContext.Provider
|
||||||
41
frontend/src/TabController.tsx
Normal file
41
frontend/src/TabController.tsx
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import { Fragment, ReactNode, useState } from "react";
|
||||||
|
|
||||||
|
interface TabProps {
|
||||||
|
id: string;
|
||||||
|
label: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface TabControllerProps {
|
||||||
|
tabs: TabProps[];
|
||||||
|
children: ReactNode[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function TabController({ tabs, children }: TabControllerProps) {
|
||||||
|
const [currentIndex, setCurrentIndex] = useState(0);
|
||||||
|
const handleTabClick = (index: number) => {
|
||||||
|
setCurrentIndex(index);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="block">
|
||||||
|
<div className="tabs is-boxed is-centered">
|
||||||
|
<ul>
|
||||||
|
{tabs.map((tab, index) => (
|
||||||
|
<li className={currentIndex === index ? "is-active" : ""}>
|
||||||
|
<a onClick={() => handleTabClick(index)}>{tab.label}</a>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
{children.map((child, index) => (
|
||||||
|
<Fragment key={index}>
|
||||||
|
<div style={{ display: currentIndex === index ? "block" : "none" }}>
|
||||||
|
{child}
|
||||||
|
</div>
|
||||||
|
</Fragment>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { useSession } from "./Session";
|
import { useSession } from "./Session";
|
||||||
|
|
||||||
export const baseUrl = import.meta.env.VITE_BASE_URL as string;
|
export const baseUrl = "";
|
||||||
|
|
||||||
export async function apiAuth(
|
export async function apiAuth(
|
||||||
path: string,
|
path: string,
|
||||||
@@ -95,6 +95,7 @@ export type LoginRequest = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const login = async (req: LoginRequest): Promise<void> => {
|
export const login = async (req: LoginRequest): Promise<void> => {
|
||||||
|
console.log("baseUrl", baseUrl);
|
||||||
try {
|
try {
|
||||||
const response = await fetch(`${baseUrl}api/token`, {
|
const response = await fetch(`${baseUrl}api/token`, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
17
frontend/src/main.css
Normal file
17
frontend/src/main.css
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
@import "bulma/css/bulma.css";
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--bulma-primary-h: 220deg;
|
||||||
|
--bulma-primary-s: 60%;
|
||||||
|
--bulma-primary-l: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar {
|
||||||
|
--bulma-navbar-burger-color: var(--bulma-white);
|
||||||
|
--bulma-navbar-dropdown-border-color: var(--bulma-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.overflow-y {
|
||||||
|
overflow-y: auto;
|
||||||
|
max-height: 30vh;
|
||||||
|
}
|
||||||
9
frontend/src/main.tsx
Normal file
9
frontend/src/main.tsx
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
import { StrictMode } from "react";
|
||||||
|
import { createRoot } from "react-dom/client";
|
||||||
|
import App from "./CUTT.tsx";
|
||||||
|
|
||||||
|
createRoot(document.getElementById("root")!).render(
|
||||||
|
<StrictMode>
|
||||||
|
<App />
|
||||||
|
</StrictMode>
|
||||||
|
);
|
||||||
41
package.json
41
package.json
@@ -1,41 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "cutt",
|
|
||||||
"private": true,
|
|
||||||
"version": "0.0.0",
|
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
|
||||||
"dev": "vite",
|
|
||||||
"build": "tsc -b && vite build",
|
|
||||||
"lint": "eslint .",
|
|
||||||
"preview": "vite preview"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"jwt-decode": "^4.0.0",
|
|
||||||
"react": "18.3.1",
|
|
||||||
"react-dom": "18.3.1",
|
|
||||||
"react-sortablejs": "^6.1.4",
|
|
||||||
"reagraph": "^4.21.2",
|
|
||||||
"sortablejs": "^1.15.6"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@eslint/js": "^9.17.0",
|
|
||||||
"@types/node": "^22.13.10",
|
|
||||||
"@types/react": "18.3.18",
|
|
||||||
"@types/react-dom": "18.3.5",
|
|
||||||
"@types/sortablejs": "^1.15.8",
|
|
||||||
"@vitejs/plugin-react": "^4.3.4",
|
|
||||||
"eslint": "^9.17.0",
|
|
||||||
"eslint-plugin-react-hooks": "^5.0.0",
|
|
||||||
"eslint-plugin-react-refresh": "^0.4.16",
|
|
||||||
"globals": "^15.14.0",
|
|
||||||
"react-router": "^7.1.5",
|
|
||||||
"typescript": "~5.6.2",
|
|
||||||
"typescript-eslint": "^8.18.2",
|
|
||||||
"vite": "^6.0.5"
|
|
||||||
},
|
|
||||||
"prettier": {
|
|
||||||
"trailingComma": "es5",
|
|
||||||
"tabWidth": 2,
|
|
||||||
"semi": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -11,10 +11,14 @@ dependencies = [
|
|||||||
"matplotlib>=3.10.0",
|
"matplotlib>=3.10.0",
|
||||||
"networkx>=3.4.2",
|
"networkx>=3.4.2",
|
||||||
"passlib>=1.7.4",
|
"passlib>=1.7.4",
|
||||||
"psycopg>=3.2.4",
|
"psycopg[binary]>=3.2.4",
|
||||||
"pydantic-settings>=2.7.1",
|
"pydantic-settings>=2.7.1",
|
||||||
"pyjwt>=2.10.1",
|
"pyjwt>=2.10.1",
|
||||||
"pyqt6>=6.8.0",
|
|
||||||
"sqlmodel>=0.0.22",
|
"sqlmodel>=0.0.22",
|
||||||
"uvicorn>=0.34.0",
|
"uvicorn>=0.34.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[dependency-groups]
|
||||||
|
dev = [
|
||||||
|
"pyqt6>=6.10.1",
|
||||||
|
]
|
||||||
|
|||||||
@@ -1,42 +0,0 @@
|
|||||||
import { useLocation } from "react-router";
|
|
||||||
import { Link } from "react-router";
|
|
||||||
import { useSession } from "./Session";
|
|
||||||
|
|
||||||
export default function Footer() {
|
|
||||||
const location = useLocation();
|
|
||||||
const { user, teams } = useSession();
|
|
||||||
return (
|
|
||||||
<footer className={location.pathname === "/network" ? "fixed-footer" : ""}>
|
|
||||||
{(user?.scopes.split(" ").includes("analysis") ||
|
|
||||||
teams?.activeTeam === 42) && (
|
|
||||||
<div className="navbar">
|
|
||||||
<Link to="/">
|
|
||||||
<span>Form</span>
|
|
||||||
</Link>
|
|
||||||
<span>|</span>
|
|
||||||
<Link to="/network">
|
|
||||||
<span>Sociogram</span>
|
|
||||||
</Link>
|
|
||||||
<span>|</span>
|
|
||||||
<Link to="/mvp">
|
|
||||||
<span>MVP</span>
|
|
||||||
</Link>
|
|
||||||
<span>|</span>
|
|
||||||
<Link to="/team">
|
|
||||||
<span>Team</span>
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<p className="grey extra-margin">
|
|
||||||
something not working?
|
|
||||||
<br />
|
|
||||||
message <a href="https://t.me/x0124816">me</a>.
|
|
||||||
<br />
|
|
||||||
or fix it here:{" "}
|
|
||||||
<a href="https://git.0124816.xyz/julius/cutt" key="gitea">
|
|
||||||
<img src="gitea.svg" alt="gitea" height="16" />
|
|
||||||
</a>
|
|
||||||
</p>
|
|
||||||
</footer>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
import { Link, useLocation } from "react-router";
|
|
||||||
import Avatar from "./Avatar";
|
|
||||||
|
|
||||||
export default function Header() {
|
|
||||||
const location = useLocation();
|
|
||||||
return (
|
|
||||||
<div className={location.pathname === "/network" ? "networkroute" : ""}>
|
|
||||||
<div className="logo">
|
|
||||||
<Link to="/">
|
|
||||||
<img alt="logo" height="66%" src="logo.svg" />
|
|
||||||
<h3 className="centered">cutt</h3>
|
|
||||||
</Link>
|
|
||||||
<span className="grey">cool ultimate team tool</span>
|
|
||||||
</div>
|
|
||||||
<Avatar />
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
125
src/Login.tsx
125
src/Login.tsx
@@ -1,125 +0,0 @@
|
|||||||
import { useEffect, useState } from "react";
|
|
||||||
import { currentUser, login, User } from "./api";
|
|
||||||
import Header from "./Header";
|
|
||||||
import { useLocation, useNavigate } from "react-router";
|
|
||||||
import { Eye, EyeSlash } from "./Icons";
|
|
||||||
|
|
||||||
export interface LoginProps {
|
|
||||||
onLogin: (user: User) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const Login = ({ onLogin }: LoginProps) => {
|
|
||||||
const [username, setUsername] = useState("");
|
|
||||||
const [password, setPassword] = useState("");
|
|
||||||
const [error, setError] = useState("");
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [visible, setVisible] = useState(false);
|
|
||||||
const navigate = useNavigate();
|
|
||||||
const location = useLocation();
|
|
||||||
|
|
||||||
async function doLogin() {
|
|
||||||
setLoading(true);
|
|
||||||
setError("");
|
|
||||||
const timeout = new Promise((r) => setTimeout(r, 1000));
|
|
||||||
let user: User;
|
|
||||||
try {
|
|
||||||
await login({ username, password });
|
|
||||||
user = await currentUser();
|
|
||||||
} catch (e) {
|
|
||||||
await timeout;
|
|
||||||
setError("failed");
|
|
||||||
setLoading(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await timeout;
|
|
||||||
onLogin(user);
|
|
||||||
}
|
|
||||||
|
|
||||||
function handleSubmit(e: React.FormEvent) {
|
|
||||||
e.preventDefault();
|
|
||||||
doLogin();
|
|
||||||
}
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (location.state) {
|
|
||||||
const queryUsername = location.state.username;
|
|
||||||
const queryPassword = location.state.password;
|
|
||||||
if (queryUsername) setUsername(queryUsername);
|
|
||||||
if (queryPassword) setPassword(queryPassword);
|
|
||||||
navigate(location.pathname, { replace: true });
|
|
||||||
}
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Header />
|
|
||||||
<form onSubmit={handleSubmit}>
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
position: "relative",
|
|
||||||
display: "flex",
|
|
||||||
alignItems: "end",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
width: "100%",
|
|
||||||
marginRight: "8px",
|
|
||||||
display: "flex",
|
|
||||||
justifyContent: "center",
|
|
||||||
flexDirection: "column",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="username"
|
|
||||||
name="username"
|
|
||||||
placeholder="username"
|
|
||||||
required
|
|
||||||
value={username}
|
|
||||||
onChange={(evt) => {
|
|
||||||
setError("");
|
|
||||||
setUsername(evt.target.value);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<input
|
|
||||||
type={visible ? "text" : "password"}
|
|
||||||
id="password"
|
|
||||||
name="password"
|
|
||||||
placeholder="password"
|
|
||||||
minLength={8}
|
|
||||||
value={password}
|
|
||||||
required
|
|
||||||
onChange={(evt) => {
|
|
||||||
setError("");
|
|
||||||
setPassword(evt.target.value);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
position: "absolute",
|
|
||||||
right: "-1em",
|
|
||||||
margin: "auto 4px",
|
|
||||||
background: "unset",
|
|
||||||
fontSize: "x-large",
|
|
||||||
cursor: "pointer",
|
|
||||||
}}
|
|
||||||
onClick={() => setVisible(!visible)}
|
|
||||||
>
|
|
||||||
{visible ? <Eye /> : <EyeSlash />}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{error && <span style={{ color: "red" }}>{error}</span>}
|
|
||||||
<button type="submit" value="login" style={{ fontSize: "small" }}>
|
|
||||||
login
|
|
||||||
</button>
|
|
||||||
{loading && <span className="loader" />}
|
|
||||||
</form>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -1,45 +0,0 @@
|
|||||||
import { Fragment, ReactNode, useState } from "react";
|
|
||||||
|
|
||||||
interface TabProps {
|
|
||||||
id: string;
|
|
||||||
label: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface TabControllerProps {
|
|
||||||
tabs: TabProps[];
|
|
||||||
children: ReactNode[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function TabController({ tabs, children }: TabControllerProps) {
|
|
||||||
const [currentIndex, setCurrentIndex] = useState(0);
|
|
||||||
const handleTabClick = (index: number) => {
|
|
||||||
setCurrentIndex(index);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<div>
|
|
||||||
<div className="container navbar">
|
|
||||||
{tabs.map((tab, index) => (
|
|
||||||
<button
|
|
||||||
key={tab.id}
|
|
||||||
className={
|
|
||||||
currentIndex === index ? "tab-button active" : "tab-button"
|
|
||||||
}
|
|
||||||
onClick={() => handleTabClick(index)}
|
|
||||||
>
|
|
||||||
{tab.label}
|
|
||||||
</button>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{children.map((child, index) => (
|
|
||||||
<Fragment key={index}>
|
|
||||||
<div style={{ display: currentIndex === index ? "block" : "none" }}>
|
|
||||||
{child}
|
|
||||||
</div>
|
|
||||||
</Fragment>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
10
src/main.tsx
10
src/main.tsx
@@ -1,10 +0,0 @@
|
|||||||
import { StrictMode } from 'react'
|
|
||||||
import { createRoot } from 'react-dom/client'
|
|
||||||
import './index.css'
|
|
||||||
import App from './App.tsx'
|
|
||||||
|
|
||||||
createRoot(document.getElementById('root')!).render(
|
|
||||||
<StrictMode>
|
|
||||||
<App />
|
|
||||||
</StrictMode>,
|
|
||||||
)
|
|
||||||
Reference in New Issue
Block a user