21 Commits

Author SHA1 Message Date
8fd11901c2 remove unnecessary volume mount 2025-12-19 09:19:28 +01:00
9f9641c32b extend .env example 2025-12-19 09:19:15 +01:00
fa94d4ba7a close burger menu on click 2025-12-19 09:18:39 +01:00
e2677b60a3 restyle TeamPanel and Calendar 2025-12-19 09:18:16 +01:00
1968c21c96 add Lucide icons ♡ 2025-12-19 07:23:41 +01:00
a43cb1cdc3 respect .python-version during uv sync 2025-12-18 19:56:33 +01:00
d5e8d0825f pin python 3.13 due to psycopg-binary only having wheels for 3.13 2025-12-18 19:54:46 +01:00
192edcea1f fix minor error, remove App.{tsx,css} 2025-12-18 19:49:21 +01:00
86f494f840 run with docker compose 2025-12-18 19:46:57 +01:00
c9f227c70c read database connection string from .env 2025-12-18 19:30:57 +01:00
25c1728c27 remove dev origin 2025-12-18 19:30:29 +01:00
7df09f580a move PyQt6 to dev deps 2025-12-18 19:29:38 +01:00
407b778131 move public dir to frontend 2025-12-18 19:29:24 +01:00
a38fd042ba build frontend in node docker container 2025-12-18 19:28:39 +01:00
45a842b6fe move frontend stuff to its own directory 2025-12-18 16:47:47 +01:00
4d07dde87a minor styling 2025-12-18 13:29:07 +01:00
5a4918330e add logo to login page (header not available then) 2025-12-18 13:28:18 +01:00
b6ce89b712 add more routes 2025-12-18 13:27:57 +01:00
2f68785a01 pull letters apart 2025-12-18 13:27:35 +01:00
d452809c44 new design 2025-12-18 12:29:49 +01:00
ed30bf6bb1 upgrade deps 2025-12-17 13:27:15 +01:00
53 changed files with 1234 additions and 1195 deletions

View File

@@ -1,3 +1,8 @@
VITE_BASE_URL=
SECRET_KEY="tg07Cp0daCpcbeeFw+lI4RG2lEuT8oq5xOwB5ETs9YaJTylG1euC4ogjTK4zym/d"
ACCESS_TOKEN_EXPIRE_MINUTES = 30
ACCESS_TOKEN_EXPIRE_MINUTES=30
DB_HOST=db
DB_NAME=cutt
DB_USER=postgres
DB_PASS=password
DB_PORT=5432

1
.python-version Normal file
View File

@@ -0,0 +1 @@
3.13

16
Dockerfile Normal file
View File

@@ -0,0 +1,16 @@
FROM ghcr.io/astral-sh/uv:alpine
EXPOSE 8000
ENV PYTHONUNBUFFERED=1 \
PYTHONDONTWRITEBYTECODE=1 \
PORT=8000 \
UV_NO_DEV=1
WORKDIR /app
COPY pyproject.toml .python-version /app
RUN uv sync
COPY . /app
CMD uv run fastapi run cutt/main.py

48
compose.yml Normal file
View File

@@ -0,0 +1,48 @@
services:
frontend-build:
build:
context: ./frontend
dockerfile: Dockerfile.frontend
container_name: cutt-frontend
environment:
VITE_BASE_URL: ${VITE_BASE_URL}
volumes:
- dist:/app/dist
backend:
build: .
container_name: cutt-backend
depends_on:
frontend-build:
condition: service_completed_successfully
db:
condition: service_healthy
restart: true
restart: unless-stopped
env_file:
- .env
volumes:
- dist:/app/dist
ports:
- 8000:8000
db:
image: postgres:17
container_name: cutt-db
restart: unless-stopped
volumes:
- db:/var/lib/postgresql
environment:
- POSTGRES_DB=${DB_NAME}
- POSTGRES_USER=${DB_USER}
- POSTGRES_PASSWORD=${DB_PASS}
healthcheck:
test: ["CMD-SHELL", "pg_isready -U $${POSTGRES_USER} -d $${POSTGRES_DB}"]
interval: 10s
retries: 5
start_period: 30s
timeout: 10s
volumes:
dist:
db:

View File

@@ -344,6 +344,7 @@ def last_submissions(
times[r.time.date()] = {}
times[r.time.date()][r.user] = (
times[r.time.date()].get(r.user, "")
+ " "
+ translate_tablename[survey.__tablename__]
)
return times

View File

@@ -1,3 +1,4 @@
import os
from datetime import datetime, timezone
from sqlmodel import (
ARRAY,
@@ -10,8 +11,9 @@ from sqlmodel import (
create_engine,
)
with open("db.secrets", "r") as f:
db_secrets = f.readline().strip()
# with open("db.secrets", "r") as f:
# 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(
db_secrets,

View File

@@ -28,10 +28,7 @@ app = FastAPI(
title="cutt", swagger_ui_parameters={"syntaxHighlight": {"theme": "monokai"}}
)
api_router = APIRouter(prefix="/api")
origins = [
"https://cutt.0124816.xyz",
"http://localhost:5173",
]
origins = ["https://cutt.0124816.xyz"]
app.add_middleware(
CORSMiddleware,

1
frontend/.env Normal file
View File

@@ -0,0 +1 @@
VITE_BASE_URL=

View File

@@ -0,0 +1,10 @@
FROM node:alpine
WORKDIR /app
COPY package.json ./
RUN npm install
COPY . .
RUN npm run build

43
frontend/package.json Normal file
View File

@@ -0,0 +1,43 @@
{
"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",
"lucide-react": "^0.562.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
View 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

View File

Before

Width:  |  Height:  |  Size: 3.3 KiB

After

Width:  |  Height:  |  Size: 3.3 KiB

View File

Before

Width:  |  Height:  |  Size: 214 B

After

Width:  |  Height:  |  Size: 214 B

46
frontend/src/CUTT.tsx Normal file
View 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;

View File

@@ -1,4 +1,4 @@
import { useEffect, useState } from "react";
import { JSX, useEffect, useState } from "react";
import { apiAuth } from "./api";
import { useSession } from "./Session";
@@ -55,16 +55,35 @@ const Calendar = ({ playerId }: { playerId: number }) => {
// Render month navigation
const renderMonthNavigation = () => {
return (
<div className="month-navigation">
<button onClick={handlePrevMonth}>&lt;</button>
<span>
<button onClick={() => setSelectedDate(new Date())}>📅</button>
<div className="field has-addons">
<p className="control">
<button
className="button is-light is-size-7-mobile"
onClick={handlePrevMonth}
>
&lt;
</button>
</p>
<p className="control">
<button
className="button is-light is-size-7-mobile"
onClick={() => setSelectedDate(new Date())}
>
📅{" "}
{selectedDate.toLocaleString("default", {
month: "long",
year: "numeric",
})}
</span>
<button onClick={handleNextMonth}>&gt;</button>
</button>
</p>
<p className="control">
<button
className="button is-light is-size-7-mobile"
onClick={handleNextMonth}
>
&gt;
</button>
</p>
</div>
);
};
@@ -89,11 +108,14 @@ const Calendar = ({ playerId }: { playerId: number }) => {
const date = new Date(0);
date.setDate(i + 5);
days.push(
<div key={"weekday_" + i} className="weekday">
<button
key={"weekday_" + i}
className="button is-size-7-mobile is-white is-static"
>
{date.toLocaleString("default", {
weekday: "narrow",
})}
</div>
</button>
);
}
@@ -109,34 +131,34 @@ const Calendar = ({ playerId }: { playerId: number }) => {
const todaysEvents = getEventsForDay(date);
days.push(
<div
<button
key={date.getDate()}
className={
"day" +
"cell button is-size-7-mobile" +
(date.toDateString() === selectedDate.toDateString()
? " selected-day"
? " is-focused is-active is-primary is-light"
: " is-white") +
(date.toDateString() === new Date().toDateString()
? " is-danger has-text-weight-extrabold"
: "") +
(todaysEvents ? " is-warning is-light" : "") +
(todaysEvents && playerId in todaysEvents
? " is-hovered has-text-weight-semibold"
: "")
}
onClick={() => handleDayClick(date)}
>
<div
className={
"day-circle" +
(date.toDateString() === new Date().toDateString()
? " today"
: "") +
(todaysEvents ? " has-event" : "") +
(todaysEvents && playerId in todaysEvents ? " active-player" : "")
}
>
{day}
</div>
</div>
</button>
);
day++;
}
return <div className="calendar">{days}</div>;
return (
<div className="fixed-grid has-7-cols">
<div className="grid is-gap-0.5">{days}</div>
</div>
);
};
// Render events for the selected day
@@ -144,29 +166,38 @@ const Calendar = ({ playerId }: { playerId: number }) => {
const eventsForDay = getEventsForDay(selectedDate);
return (
<div className="events">
{eventsForDay && (
<ul>
{Object.entries(eventsForDay).map(([id, sub]) => {
{eventsForDay &&
Object.entries(eventsForDay).map(([id, sub]) => {
const name = players?.find((p) => p.id === Number(id));
return (
<li key={id}>
{name !== undefined ? name.display_name : ""}:{" "}
<span style={{ letterSpacing: 8 }}>{sub}</span>
</li>
<p className="field">
<div className="control" key={id}>
<div className="tags are-medium has-addons">
<span className="tag is-warning is-size-7-mobile">
{name !== undefined ? name.display_name : ""}
</span>
<span className="tag is-primary is-light is-size-7-mobile">
{sub}
</span>
</div>
</div>
</p>
);
})}
</ul>
)}
</div>
);
};
return (
<div className="calendar-container">
<h2>Latest Submissions</h2>
<div className="block is-size-7-mobile">
<h2 className="title is-4">Latest Submissions</h2>
<div className="columns is-6">
<div className="column" style={{ maxWidth: 600 }}>
{renderMonthNavigation()}
{renderCalendar()}
{renderEvents()}
</div>
<div className="column is-narrow">{renderEvents()}</div>
</div>
</div>
);
};

25
frontend/src/Footer.tsx Normal file
View 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:&nbsp;
<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
View 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}>
🗃 &nbsp; restore previous
</button>
<button className="button is-small is-light" onClick={onClear}>
🗑&nbsp; start over
</button>
<button className="button is-small is-light" onClick={onSubmit}>
💾 &nbsp; 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>
);
}

97
frontend/src/Header.tsx Normal file
View File

@@ -0,0 +1,97 @@
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
onClick={() => setBurgerActive(false)}
className="navbar-item"
to="/network"
>
<span>Sociogram</span>
</Link>
<Link
onClick={() => setBurgerActive(false)}
className="navbar-item"
to="/mvp"
>
<span>MVP</span>
</Link>
<Link
onClick={() => setBurgerActive(false)}
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
View 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>
);
};

View File

@@ -105,7 +105,23 @@ export function SessionProvider(props: SessionProviderProps) {
</>
);
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
content = (
<sessionContext.Provider

View 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>
);
}

View File

@@ -70,29 +70,21 @@ const TeamPanel = () => {
(team) => team.id == teams?.activeTeam
)[0];
return (
<div className="team-panel">
<h1>{activeTeam.name}</h1>
<div>
<input type="text" value={activeTeam.location || ""} disabled />
<br />
<input type="text" value={activeTeam.country || ""} disabled />
<hr style={{ width: "100%" }} />
<h2>players</h2>
<section className="section">
<h1 className="title">{activeTeam.name}</h1>
<h2 className="subtitle">
{activeTeam.location}, {activeTeam.country}
</h2>
<div className="box">
<h2 className="title is-4">Players</h2>
{players ? (
<div
style={{
display: "flex",
flexWrap: "wrap",
justifyContent: "center",
}}
>
<div className="buttons">
{players.map((p) => (
<button
className={
"team-player " +
"button is-primary is-light " +
p.gender +
(p.id === player.id ? " active-player" : "")
(p.id === player.id ? " is-focused is-active" : "")
}
key={p.id}
onClick={() => {
@@ -104,7 +96,7 @@ const TeamPanel = () => {
</button>
))}
<button
className="team-player new-player"
className="button is-success is-light new-player"
key="add-player"
onClick={() => {
setPlayer(newPlayerTemplate);
@@ -117,12 +109,14 @@ const TeamPanel = () => {
) : (
<span className="loader" />
)}
<hr style={{ width: "100%" }} />
</div>
<form className="new-player-inputs" onSubmit={handleSubmit}>
<div>
<label>name</label>
<form className="container block" onSubmit={handleSubmit}>
<div className="field">
<label className="label">name</label>
<div className="control">
<input
className="input"
type="text"
required
value={player.display_name}
@@ -138,9 +132,12 @@ const TeamPanel = () => {
}}
/>
</div>
<div>
<label>username</label>
</div>
<div className="field">
<label className="label">username</label>
<div className="control">
<input
className="input"
type="text"
required
disabled={player.id !== 0}
@@ -151,8 +148,11 @@ const TeamPanel = () => {
}}
/>
</div>
<div>
<label>gender</label>
</div>
<div className="field">
<label className="label">gender</label>
<div className="control">
<div className="select">
<select
name="gender"
value={player.gender}
@@ -166,9 +166,13 @@ const TeamPanel = () => {
<option value="mmp">MMP</option>
</select>
</div>
<div>
<label>number (optional)</label>
</div>
</div>
<div className="field">
<label className="label">number (optional)</label>
<div className="control">
<input
className="input"
type="text"
value={player.number || ""}
onChange={(e) => {
@@ -177,9 +181,12 @@ const TeamPanel = () => {
}}
/>
</div>
<div>
<label>email (optional)</label>
</div>
<div className="field">
<label className="label">email (optional)</label>
<div className="control">
<input
className="input"
type="email"
value={player.email || ""}
onChange={(e) => {
@@ -188,36 +195,33 @@ const TeamPanel = () => {
}}
/>
</div>
<div style={{ margin: "auto" }}>
{error?.message && (
<span
style={{
color: error.ok ? "green" : "red",
}}
>
<p className={"help" + (error.ok ? " is-success" : " is-danger")}>
{error.message}
</span>
</p>
)}
</div>
<div style={{ margin: "auto" }}>
<button className="team-player new-player">
<div className="field is-grouped">
<button
className={
"button is-light" +
(player.id === 0 ? " is-success" : " is-link")
}
>
{player.id === 0 ? "add player" : "modify player"}
</button>
</div>
{player.id !== 0 && (
<div style={{ margin: "auto" }}>
<button
className="team-player disable-player"
className="button is-danger is-light"
onClick={handleDisable}
>
remove player
</button>
</div>
)}
</div>
</form>
</div>
<Calendar playerId={player.id} />
</div>
</section>
);
} else <span className="loader" />;
};

View File

@@ -1,6 +1,6 @@
import { useSession } from "./Session";
export const baseUrl = import.meta.env.VITE_BASE_URL as string;
export const baseUrl = "";
export async function apiAuth(
path: string,
@@ -95,6 +95,7 @@ export type LoginRequest = {
};
export const login = async (req: LoginRequest): Promise<void> => {
console.log("baseUrl", baseUrl);
try {
const response = await fetch(`${baseUrl}api/token`, {
method: "POST",

16
frontend/src/main.css Normal file
View File

@@ -0,0 +1,16 @@
@import "bulma/css/bulma.css";
:root {
--bulma-primary-h: 220deg;
--bulma-primary-s: 60%;
--bulma-primary-l: 50%;
}
.navbar {
--bulma-navbar-dropdown-border-color: var(--bulma-primary);
}
.overflow-y {
overflow-y: auto;
max-height: 30vh;
}

9
frontend/src/main.tsx Normal file
View 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>
);

View File

@@ -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
}
}

View File

@@ -11,10 +11,14 @@ dependencies = [
"matplotlib>=3.10.0",
"networkx>=3.4.2",
"passlib>=1.7.4",
"psycopg>=3.2.4",
"psycopg[binary]>=3.2.4",
"pydantic-settings>=2.7.1",
"pyjwt>=2.10.1",
"pyqt6>=6.8.0",
"sqlmodel>=0.0.22",
"uvicorn>=0.34.0",
]
[dependency-groups]
dev = [
"pyqt6>=6.10.1",
]

View File

@@ -1,734 +0,0 @@
body {
background-color: aliceblue;
position: relative;
z-index: 0;
color: black;
text-align: center;
overflow-wrap: anywhere;
height: 100%;
}
#root {
max-width: 1280px;
margin: 0 auto;
padding: 8px;
}
footer {
margin-top: 24px;
font-size: x-small;
}
.fixed-footer {
position: absolute;
bottom: 4px;
left: 8px;
}
dialog {
border-radius: 1em;
}
/*=========Network Controls=========*/
.infobutton {
position: fixed;
right: 8px;
bottom: 8px;
padding: 0.4em;
border-radius: 1em;
background-color: rgba(0, 0, 0, 0.3);
font-size: medium;
margin-bottom: 16px;
margin-right: 16px;
}
.controls {
z-index: 9;
position: absolute;
color: black;
top: 1vh;
right: 0px;
padding: 8px;
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 8px;
.control {
display: flex;
flex-direction: row;
flex-wrap: wrap;
max-width: 240px;
margin: 0px;
background-color: #f0f8ffdd;
.slider,
span {
padding-left: 4px;
padding-right: 4px;
}
}
#three-slider {
display: flex;
flex-direction: row;
margin: auto;
justify-content: center;
align-items: center;
}
}
/* The switch - the box around the slider */
.switch {
position: relative;
width: 48px;
height: 24px;
}
/* Hide default HTML checkbox */
.switch input {
opacity: 0;
width: 0;
height: 0;
}
/* The slider */
.slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #ccc;
border-radius: 34px;
-webkit-transition: 0.4s;
transition: 0.4s;
}
.slider:before {
position: absolute;
content: "";
height: 18px;
width: 18px;
left: 3px;
bottom: 3px;
background-color: white;
border-radius: 50%;
-webkit-transition: 0.4s;
transition: 0.4s;
}
input:checked + .slider {
background-color: #2196f3;
}
input:focus + .slider {
box-shadow: 0 0 1px #2196f3;
}
input:checked + .slider:before {
-webkit-transform: translateX(24px);
-ms-transform: translateX(24px);
transform: translateX(24px);
}
.grey {
opacity: 66%;
}
.hint {
position: absolute;
font-size: 80%;
padding: 8px;
top: auto;
left: 4px;
bottom: auto;
right: 4px;
z-index: -1;
}
input,
select {
padding: 0.2em 16px;
margin-top: 0.25em;
margin-bottom: 0.25em;
border-radius: 1em;
color: black;
background-color: white;
}
h1,
h2,
h3 {
margin-top: 0px;
margin-bottom: 0px;
padding: 8px 16px;
}
.stack {
display: flex;
button,
img {
padding: 0px 1em 4px 1em;
margin: 3px auto;
}
}
.column {
flex-direction: column;
}
.container {
display: flex;
flex-wrap: nowrap;
width: min(96vw, 900px);
}
.dragbox {
display: flex;
flex-direction: column;
min-height: 32px;
height: 92%;
}
.box {
position: relative;
flex: 1;
border-width: 3px;
border-style: solid;
border-radius: 16px;
h4 {
margin: 4px;
}
&.one {
max-width: min(96%, 768px);
margin: 4px auto;
}
padding: 4px;
margin: 4px 0.5%;
}
.reservoir {
display: flex;
flex-direction: unset;
flex-wrap: wrap;
justify-content: space-around;
width: 100%;
}
.item {
cursor: pointer;
font-size: medium;
border: 2px solid;
border-radius: 1em;
margin: 3px auto;
padding: 5px 0.8em;
}
.extra-margin {
padding: 0px 8px;
margin: auto;
}
button {
margin: 4px;
font-weight: bold;
color: aliceblue;
background-color: black;
border-radius: 1.2em;
z-index: 1;
&:hover {
opacity: 80%;
}
}
#control-panel {
display: none;
overflow: hidden;
margin: auto;
gap: 16px;
grid-template-columns: repeat(3, 1fr);
transition: display 1s ease-out 0s;
}
#control-panel.opened {
display: grid;
}
.control {
display: flex;
border-radius: 16px;
flex-direction: column;
align-items: center;
justify-content: center;
border: 2px solid #404040;
padding: 8px 16px;
}
#three-slider input {
margin: 4px;
width: 50%;
}
@media only screen and (max-width: 1000px) {
#control-panel {
grid-template-columns: repeat(2, 1fr);
}
.control {
font-size: 80%;
margin: 0px;
}
}
@media only screen and (max-width: 768px) {
#control-panel {
grid-template-columns: 1fr;
}
.networkroute {
display: none;
}
.submit_text {
display: none;
}
.submit {
position: fixed;
right: 16px;
bottom: 16px;
padding: 0.4em;
border-radius: 1em;
background-color: #36c8;
font-size: xx-large;
margin-bottom: 16px;
margin-right: 16px;
}
.wavering {
animation: blink 40s infinite;
}
}
::backdrop {
background-image: linear-gradient(
45deg,
magenta,
rebeccapurple,
dodgerblue,
green
);
opacity: 0.75;
}
.tab-button {
color: black;
flex: 1;
background-color: #bfbfbf;
border: none;
margin: 4px auto;
cursor: pointer;
opacity: 80%;
}
.tab-button.active {
opacity: unset;
font-weight: bold;
background-color: black;
color: white;
}
.navbar {
span {
padding: 4px;
}
button {
font-size: medium;
margin: 4px 0.5%;
padding-top: 4px;
padding-bottom: 4px;
opacity: 50%;
&:hover {
opacity: 80%;
}
}
}
/* Style the tab content (and add height:100% for full page content) */
.tabcontent {
display: none;
height: 100%;
}
.renew {
cursor: pointer;
font-weight: bold;
position: absolute;
top: 0;
right: 8px;
font-size: 150%;
}
/*======LOGO=======*/
.logo {
position: relative;
text-align: center;
height: 140px;
span {
display: block;
margin: 2px;
}
img {
display: block;
margin: auto;
}
h3 {
position: absolute;
font-size: medium;
width: 140px;
top: 33%;
left: 50%;
transform: translate(-50%, -50%);
color: aliceblue;
background-color: black;
border-radius: unset;
letter-spacing: 8px;
padding: 0px 40px;
font-family: monospace;
}
}
.avatars {
margin: 16px auto;
}
.avatar {
background-color: #f0f8ff88;
font-weight: bold;
font-size: 110%;
padding: 3px 1em;
width: fit-content;
border: 3px solid;
border-radius: 1em;
margin: 4px auto;
}
.group-avatar {
background-color: #f0f8ff88;
color: inherit;
font-weight: bold;
font-size: 90%;
padding: 3px 1em;
width: fit-content;
border: 3px solid;
border-radius: 1em;
margin: 4px auto;
}
.user-info {
display: grid;
grid-template-columns: 8em 12em;
gap: 2px 16px;
div {
text-align: left;
}
}
/*=======CONTEXT MENU=======*/
.context-menu {
z-index: 3;
min-width: 8em;
position: absolute;
background: aliceblue;
box-shadow: 4px 4px black;
color: black;
border: 3px solid black;
border-radius: 16px;
padding: 0;
margin: 0;
list-style: none;
li {
padding: 4px 0.5em;
border-bottom: 2px solid #0008;
border-radius: 0;
cursor: pointer;
}
li:last-child {
border-bottom: none;
}
}
.networkroute {
z-index: 3;
position: absolute;
top: 24px;
left: 48px;
}
/*========TEAM PANEL========*/
.team-panel {
max-width: 800px;
padding: 1em;
border: 3px solid black;
box-shadow: 8px 8px black;
margin: 1em;
input {
max-width: 300px;
}
select {
max-width: 335px;
}
}
.team-player {
color: black;
background-color: #36c4;
border: 1px solid black;
border-radius: 1.5em;
margin: 4px;
padding: 0.2em 0.5em;
&:hover {
background-color: #36c8;
}
&.new-player {
background-color: #3838;
}
&.disable-player {
background-color: #e338;
}
}
.new-player-inputs {
display: flex;
flex-direction: column;
margin: auto;
div {
display: grid;
grid-template-columns: 20ch auto;
@media only screen and (max-width: 768px) {
grid-template-columns: auto;
place-items: center;
}
label {
text-align: left;
width: 20ch;
margin: auto 1em;
}
input,
select {
width: 90%;
margin: 4px 0;
}
}
}
.mmp {
background-color: lightskyblue;
}
.fmp {
background-color: salmon;
}
@keyframes blink {
0% {
background-color: #8888;
}
13% {
background-color: #8888;
}
15% {
background-color: #f00a;
}
17% {
background-color: #8888;
}
38% {
background-color: #8888;
}
40% {
background-color: #ff0a;
}
42% {
background-color: #8888;
}
63% {
background-color: #8888;
}
65% {
background-color: #248f24aa;
}
67% {
background-color: #8888;
}
88% {
background-color: #8888;
}
90% {
background-color: #4700b3aa;
}
92% {
background-color: #8888;
}
100% {
background-color: #8888;
}
}
/*======SPINNER=======*/
.loader {
display: block;
border-radius: 16px;
position: relative;
height: 12px;
width: 96%;
margin: auto;
border: 4px solid black;
overflow: hidden;
}
.loader::after {
content: "";
width: 32%;
height: 120%;
background: #36c;
position: absolute;
top: -2px;
left: 0;
box-sizing: border-box;
animation: animloader 2s linear infinite;
}
@keyframes animloader {
0% {
left: 0;
transform: translateX(-100%);
}
100% {
left: 100%;
transform: translateX(0%);
}
}
.calendar-container {
position: relative;
margin: 20px auto;
font-size: small;
}
.month-navigation {
display: flex;
align-items: center;
justify-content: space-between;
padding: 8px;
border-top: 2px solid grey;
border-bottom: 2px solid grey;
}
.month-navigation button {
cursor: pointer;
padding: 4px 8px;
border: none;
color: black;
background-color: transparent;
}
.month-navigation span {
font-weight: bold;
}
.calendar {
display: grid;
grid-template-columns: repeat(7, 1fr);
gap: 8px;
}
.day {
padding: 2px;
border: 1px solid grey;
cursor: pointer;
display: flex;
}
.selected-day {
border: 4px solid grey;
}
.weekday {
border-bottom: 3px solid black;
margin: 0 1em;
}
.day-circle {
text-align: center;
border-radius: 1.5em;
width: 1.5em;
height: 1.5em;
padding: 0;
margin: auto;
border: 2px solid transparent;
}
.today {
border-radius: 1.6em;
border: 4px solid red;
text-align: center;
}
.has-event {
border-radius: 1.5em;
background-color: lightskyblue;
}
.active-player {
border-radius: 1.5em;
border: 4px solid rebeccapurple;
}
.events {
font-size: large;
padding: 20px;
ul > li {
padding: 0;
margin: 0;
list-style-type: none;
}
}

View File

@@ -1,50 +0,0 @@
import "./App.css";
import Footer from "./Footer";
import Header from "./Header";
import Rankings from "./Rankings";
import { BrowserRouter, Routes, Route } from "react-router";
import { SessionProvider } from "./Session";
import { GraphComponent } from "./Network";
import MVPChart from "./MVPChart";
import { SetPassword } from "./SetPassword";
import { ThemeProvider } from "./ThemeProvider";
import TeamPanel from "./TeamPanel";
const Maintenance = () => {
return (
<div style={{ textAlign: "center", padding: "20px" }}>
<h2>We are under maintenance.</h2>
<p>Please check back later. Thank you for your patience.</p>
<span style={{ fontSize: "xx-large" }}>🚧</span>
</div>
);
};
function App() {
return (
<ThemeProvider>
<BrowserRouter>
<Routes>
<Route path="/password" element={<SetPassword />} />
<Route
path="/*"
element={
<SessionProvider>
<Header />
<Routes>
<Route index element={<Rankings />} />
<Route path="network" element={<GraphComponent />} />
<Route path="mvp" element={<MVPChart />} />
<Route path="changepassword" element={<SetPassword />} />
<Route path="team" element={<TeamPanel />} />
</Routes>
<Footer />
</SessionProvider>
}
/>
</Routes>
</BrowserRouter>
</ThemeProvider>
);
}
export default App;

View File

@@ -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>
);
}

View File

@@ -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>
);
}

View File

@@ -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>
</>
);
};

View File

@@ -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>
);
}

View File

@@ -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>,
)