Compare commits
	
		
			49 Commits
		
	
	
		
			feat/inter
			...
			1067b12be8
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1067b12be8 | |||
| c42231907d | |||
| 95e66e5d73 | |||
| 6d2bf057a5 | |||
| b07c2fd8ab | |||
| 82ffa06a00 | |||
| 00442be4b5 | |||
| 26ee4b84a9 | |||
| aa3c3df5da | |||
| 401ac316c1 | |||
| 53fc8bb6e3 | |||
| 92a98064e5 | |||
| 1773a9885a | |||
| 9996752d94 | |||
| b386ee365f | |||
| 045c26d258 | |||
| a37971ed86 | |||
| f3e6382101 | |||
| 59e2fc4502 | |||
| 33c505fee4 | |||
| cfe2df01f7 | |||
| 7580a4f1e6 | |||
| 7bf35b65fb | |||
| d3f5c3cb82 | |||
| 8b092fed51 | |||
| 99e80c8077 | |||
| 854bd03c40 | |||
| bc6c2a4a98 | |||
| b7c8136b1e | |||
| b8c4190072 | |||
| d61bea3c86 | |||
| 70a4ece5bc | |||
| 406ea9ffdd | |||
| 104ec70695 | |||
| 9d65c1d1df | |||
| de79970987 | |||
| a52dae5605 | |||
| a46427c6b8 | |||
| fd323db6d0 | |||
| c2d94c0400 | |||
| f94c3402c2 | |||
| 5c21cf1fc3 | |||
| 5cd793b278 | |||
| de8688133f | |||
| d6e5d0334c | |||
| 5fef47f692 | |||
| 978aafc204 | |||
| 47fd9bd859 | |||
| 13bb965b28 | 
							
								
								
									
										106
									
								
								analysis.py
									
									
									
									
									
								
							
							
						
						
									
										106
									
								
								analysis.py
									
									
									
									
									
								
							| @@ -1,4 +1,3 @@ | |||||||
| from datetime import datetime |  | ||||||
| import io | import io | ||||||
| import base64 | import base64 | ||||||
| from fastapi import APIRouter | from fastapi import APIRouter | ||||||
| @@ -6,8 +5,9 @@ from fastapi.responses import JSONResponse | |||||||
| from pydantic import BaseModel, Field | from pydantic import BaseModel, Field | ||||||
| from sqlmodel import Session, func, select | from sqlmodel import Session, func, select | ||||||
| from sqlmodel.sql.expression import SelectOfScalar | from sqlmodel.sql.expression import SelectOfScalar | ||||||
| from db import Chemistry, Player, engine | from db import Chemistry, MVPRanking, Player, engine | ||||||
| import networkx as nx | import networkx as nx | ||||||
|  | import numpy as np | ||||||
| import matplotlib | import matplotlib | ||||||
|  |  | ||||||
| matplotlib.use("agg") | matplotlib.use("agg") | ||||||
| @@ -18,6 +18,7 @@ analysis_router = APIRouter(prefix="/analysis") | |||||||
|  |  | ||||||
|  |  | ||||||
| C = Chemistry | C = Chemistry | ||||||
|  | R = MVPRanking | ||||||
| P = Player | P = Player | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -25,14 +26,13 @@ def sociogram_json(): | |||||||
|     nodes = [] |     nodes = [] | ||||||
|     necessary_nodes = set() |     necessary_nodes = set() | ||||||
|     edges = [] |     edges = [] | ||||||
|  |     players = {} | ||||||
|     with Session(engine) as session: |     with Session(engine) as session: | ||||||
|         for p in session.exec(select(P)).fetchall(): |         for p in session.exec(select(P)).fetchall(): | ||||||
|             nodes.append({"id": p.name, "label": p.name}) |             nodes.append({"id": p.display_name, "label": p.display_name}) | ||||||
|  |             players[p.id] = p.display_name | ||||||
|         subquery = ( |         subquery = ( | ||||||
|             select(C.user, func.max(C.time).label("latest")) |             select(C.user, func.max(C.time).label("latest")).group_by(C.user).subquery() | ||||||
|             .where(C.time > datetime(2025, 2, 1, 10)) |  | ||||||
|             .group_by(C.user) |  | ||||||
|             .subquery() |  | ||||||
|         ) |         ) | ||||||
|         statement2 = select(C).join( |         statement2 = select(C).join( | ||||||
|             subquery, (C.user == subquery.c.user) & (C.time == subquery.c.latest) |             subquery, (C.user == subquery.c.user) & (C.time == subquery.c.latest) | ||||||
| @@ -40,13 +40,13 @@ def sociogram_json(): | |||||||
|         for c in session.exec(statement2): |         for c in session.exec(statement2): | ||||||
|             # G.add_node(c.user) |             # G.add_node(c.user) | ||||||
|             necessary_nodes.add(c.user) |             necessary_nodes.add(c.user) | ||||||
|             for p in c.love: |             for p in [players[p_id] for p_id in c.love]: | ||||||
|                 # G.add_edge(c.user, p) |                 # G.add_edge(c.user, p) | ||||||
|                 # p_id = session.exec(select(P.id).where(P.name == p)).one() |                 # p_id = session.exec(select(P.id).where(P.name == p)).one() | ||||||
|                 necessary_nodes.add(p) |                 necessary_nodes.add(p) | ||||||
|                 edges.append({"from": c.user, "to": p, "relation": "likes"}) |                 edges.append({"from": players[c.user], "to": p, "relation": "likes"}) | ||||||
|             for p in c.hate: |             for p in [players[p_id] for p_id in c.hate]: | ||||||
|                 edges.append({"from": c.user, "to": p, "relation": "dislikes"}) |                 edges.append({"from": players[c.user], "to": p, "relation": "dislikes"}) | ||||||
|     # nodes = [n for n in nodes if n["name"] in necessary_nodes] |     # nodes = [n for n in nodes if n["name"] in necessary_nodes] | ||||||
|     return JSONResponse({"nodes": nodes, "edges": edges}) |     return JSONResponse({"nodes": nodes, "edges": edges}) | ||||||
|  |  | ||||||
| @@ -54,51 +54,65 @@ def sociogram_json(): | |||||||
| def graph_json(): | def graph_json(): | ||||||
|     nodes = [] |     nodes = [] | ||||||
|     edges = [] |     edges = [] | ||||||
|  |     players = {} | ||||||
|     with Session(engine) as session: |     with Session(engine) as session: | ||||||
|         for p in session.exec(select(P)).fetchall(): |         for p in session.exec(select(P)).fetchall(): | ||||||
|             nodes.append({"id": p.name, "label": p.name}) |             players[p.id] = p.display_name | ||||||
|  |             nodes.append({"id": p.display_name, "label": p.display_name}) | ||||||
|         subquery = ( |         subquery = ( | ||||||
|             select(C.user, func.max(C.time).label("latest")) |             select(C.user, func.max(C.time).label("latest")).group_by(C.user).subquery() | ||||||
|             .where(C.time > datetime(2025, 2, 1, 10)) |  | ||||||
|             .group_by(C.user) |  | ||||||
|             .subquery() |  | ||||||
|         ) |         ) | ||||||
|         statement2 = select(C).join( |         statement2 = select(C).join( | ||||||
|             subquery, (C.user == subquery.c.user) & (C.time == subquery.c.latest) |             subquery, (C.user == subquery.c.user) & (C.time == subquery.c.latest) | ||||||
|         ) |         ) | ||||||
|         for c in session.exec(statement2): |         for c in session.exec(statement2): | ||||||
|             for p in c.love: |             user = players[c.user] | ||||||
|  |             for i, p_id in enumerate(c.love): | ||||||
|  |                 p = players[p_id] | ||||||
|                 edges.append( |                 edges.append( | ||||||
|                     { |                     { | ||||||
|                         "id": f"{c.user}->{p}", |                         "id": f"{user}->{p}", | ||||||
|                         "source": c.user, |                         "source": user, | ||||||
|                         "target": p, |                         "target": p, | ||||||
|                         "relation": "likes", |                         "size": max(1.0 - 0.1 * i, 0.3), | ||||||
|  |                         "data": { | ||||||
|  |                             "relation": 2, | ||||||
|  |                             "origSize": max(1.0 - 0.1 * i, 0.3), | ||||||
|  |                             "origFill": "#bed4ff", | ||||||
|  |                         }, | ||||||
|                     } |                     } | ||||||
|                 ) |                 ) | ||||||
|             continue |             for p_id in c.hate: | ||||||
|             for p in c.hate: |                 p = players[p_id] | ||||||
|                 edges.append( |                 edges.append( | ||||||
|                     { |                     { | ||||||
|                         id: f"{c.user}-x>{p}", |                         "id": f"{user}-x>{p}", | ||||||
|                         "source": c.user, |                         "source": user, | ||||||
|                         "target": p, |                         "target": p, | ||||||
|                         "relation": "dislikes", |                         "size": 0.3, | ||||||
|  |                         "data": {"relation": 0, "origSize": 0.3, "origFill": "#ff7c7c"}, | ||||||
|  |                         "fill": "#ff7c7c", | ||||||
|                     } |                     } | ||||||
|                 ) |                 ) | ||||||
|  |  | ||||||
|  |     G = nx.DiGraph() | ||||||
|  |     G.add_weighted_edges_from([(e["source"], e["target"], e["size"]) for e in edges]) | ||||||
|  |     in_degrees = G.in_degree(weight="weight") | ||||||
|  |     nodes = [ | ||||||
|  |         dict(node, **{"data": {"inDegree": in_degrees[node["id"]]}}) for node in nodes | ||||||
|  |     ] | ||||||
|     return JSONResponse({"nodes": nodes, "edges": edges}) |     return JSONResponse({"nodes": nodes, "edges": edges}) | ||||||
|  |  | ||||||
|  |  | ||||||
| def sociogram_data(show: int | None = 2): | def sociogram_data(show: int | None = 2): | ||||||
|     G = nx.DiGraph() |     G = nx.DiGraph() | ||||||
|     with Session(engine) as session: |     with Session(engine) as session: | ||||||
|  |         players = {} | ||||||
|         for p in session.exec(select(P)).fetchall(): |         for p in session.exec(select(P)).fetchall(): | ||||||
|             G.add_node(p.name) |             G.add_node(p.display_name) | ||||||
|  |             players[p.id] = p.display_name | ||||||
|         subquery = ( |         subquery = ( | ||||||
|             select(C.user, func.max(C.time).label("latest")) |             select(C.user, func.max(C.time).label("latest")).group_by(C.user).subquery() | ||||||
|             .where(C.time > datetime(2025, 2, 1, 10)) |  | ||||||
|             .group_by(C.user) |  | ||||||
|             .subquery() |  | ||||||
|         ) |         ) | ||||||
|         statement2 = ( |         statement2 = ( | ||||||
|             select(C) |             select(C) | ||||||
| @@ -107,10 +121,12 @@ def sociogram_data(show: int | None = 2): | |||||||
|         ) |         ) | ||||||
|         for c in session.exec(statement2): |         for c in session.exec(statement2): | ||||||
|             if show >= 1: |             if show >= 1: | ||||||
|                 for i, p in enumerate(c.love): |                 for i, p_id in enumerate(c.love): | ||||||
|  |                     p = players[p_id] | ||||||
|                     G.add_edge(c.user, p, group="love", rank=i, popularity=1 - 0.08 * i) |                     G.add_edge(c.user, p, group="love", rank=i, popularity=1 - 0.08 * i) | ||||||
|             if show <= 1: |             if show <= 1: | ||||||
|                 for i, p in enumerate(c.hate): |                 for i, p_id in enumerate(c.hate): | ||||||
|  |                     p = players[p_id] | ||||||
|                     G.add_edge(c.user, p, group="hate", rank=8, popularity=-0.16) |                     G.add_edge(c.user, p, group="hate", rank=8, popularity=-0.16) | ||||||
|     return G |     return G | ||||||
|  |  | ||||||
| @@ -183,9 +199,35 @@ async def render_sociogram(params: Params): | |||||||
|     return {"image": encoded_image} |     return {"image": encoded_image} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def mvp(): | ||||||
|  |     ranks = dict() | ||||||
|  |     with Session(engine) as session: | ||||||
|  |         players = {p.id: p.display_name for p in session.exec(select(P)).fetchall()} | ||||||
|  |         subquery = ( | ||||||
|  |             select(R.user, func.max(R.time).label("latest")).group_by(R.user).subquery() | ||||||
|  |         ) | ||||||
|  |         statement2 = select(R).join( | ||||||
|  |             subquery, (R.user == subquery.c.user) & (R.time == subquery.c.latest) | ||||||
|  |         ) | ||||||
|  |         for r in session.exec(statement2): | ||||||
|  |             for i, p_id in enumerate(r.mvps): | ||||||
|  |                 p = players[p_id] | ||||||
|  |                 ranks[p] = ranks.get(p, []) + [i + 1] | ||||||
|  |     return [ | ||||||
|  |         { | ||||||
|  |             "name": p, | ||||||
|  |             "rank": f"{np.mean(v):.02f}", | ||||||
|  |             "std": f"{np.std(v):.02f}", | ||||||
|  |             "n": len(v), | ||||||
|  |         } | ||||||
|  |         for p, v in ranks.items() | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |  | ||||||
| analysis_router.add_api_route("/json", endpoint=sociogram_json, methods=["GET"]) | analysis_router.add_api_route("/json", endpoint=sociogram_json, methods=["GET"]) | ||||||
| analysis_router.add_api_route("/graph_json", endpoint=graph_json, methods=["GET"]) | analysis_router.add_api_route("/graph_json", endpoint=graph_json, methods=["GET"]) | ||||||
| analysis_router.add_api_route("/image", endpoint=render_sociogram, methods=["POST"]) | analysis_router.add_api_route("/image", endpoint=render_sociogram, methods=["POST"]) | ||||||
|  | analysis_router.add_api_route("/mvp", endpoint=mvp, methods=["GET"]) | ||||||
|  |  | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|     with Session(engine) as session: |     with Session(engine) as session: | ||||||
|   | |||||||
							
								
								
									
										38
									
								
								db.py
									
									
									
									
									
								
							
							
						
						
									
										38
									
								
								db.py
									
									
									
									
									
								
							| @@ -2,17 +2,22 @@ from datetime import datetime, timezone | |||||||
| from sqlmodel import ( | from sqlmodel import ( | ||||||
|     ARRAY, |     ARRAY, | ||||||
|     Column, |     Column, | ||||||
|  |     Integer, | ||||||
|     Relationship, |     Relationship, | ||||||
|     SQLModel, |     SQLModel, | ||||||
|     Field, |     Field, | ||||||
|     create_engine, |     create_engine, | ||||||
|     String, |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| with open("db.secrets", "r") as f: | with open("db.secrets", "r") as f: | ||||||
|     db_secrets = f.readline().strip() |     db_secrets = f.readline().strip() | ||||||
|  |  | ||||||
| engine = create_engine(db_secrets, connect_args={"connect_timeout": 8}) | engine = create_engine( | ||||||
|  |     db_secrets, | ||||||
|  |     pool_timeout=20, | ||||||
|  |     pool_size=2, | ||||||
|  |     connect_args={"connect_timeout": 8}, | ||||||
|  | ) | ||||||
| del db_secrets | del db_secrets | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -39,36 +44,33 @@ class Team(SQLModel, table=True): | |||||||
|  |  | ||||||
| class Player(SQLModel, table=True): | class Player(SQLModel, table=True): | ||||||
|     id: int | None = Field(default=None, primary_key=True) |     id: int | None = Field(default=None, primary_key=True) | ||||||
|     name: str |     username: str = Field(default=None, unique=True) | ||||||
|  |     display_name: str | ||||||
|  |     email: str | None = None | ||||||
|  |     full_name: str | None = None | ||||||
|  |     disabled: bool | None = None | ||||||
|  |     hashed_password: str | None = None | ||||||
|     number: str | None = None |     number: str | None = None | ||||||
|     teams: list[Team] | None = Relationship( |     teams: list[Team] | None = Relationship( | ||||||
|         back_populates="players", link_model=PlayerTeamLink |         back_populates="players", link_model=PlayerTeamLink | ||||||
|     ) |     ) | ||||||
|  |     scopes: str = "" | ||||||
|  |  | ||||||
|  |  | ||||||
| class Chemistry(SQLModel, table=True): | class Chemistry(SQLModel, table=True): | ||||||
|     id: int | None = Field(default=None, primary_key=True) |     id: int | None = Field(default=None, primary_key=True) | ||||||
|     time: datetime | None = Field(default_factory=utctime) |     time: datetime | None = Field(default_factory=utctime) | ||||||
|     user: str |     user: int | None = Field(default=None, foreign_key="player.id") | ||||||
|     love: list[str] = Field(sa_column=Column(ARRAY(String))) |     hate: list[int] = Field(sa_column=Column(ARRAY(Integer))) | ||||||
|     hate: list[str] = Field(sa_column=Column(ARRAY(String))) |     undecided: list[int] = Field(sa_column=Column(ARRAY(Integer))) | ||||||
|     undecided: list[str] = Field(sa_column=Column(ARRAY(String))) |     love: list[int] = Field(sa_column=Column(ARRAY(Integer))) | ||||||
|  |  | ||||||
|  |  | ||||||
| class MVPRanking(SQLModel, table=True): | class MVPRanking(SQLModel, table=True): | ||||||
|     id: int | None = Field(default=None, primary_key=True) |     id: int | None = Field(default=None, primary_key=True) | ||||||
|     time: datetime | None = Field(default_factory=utctime) |     time: datetime | None = Field(default_factory=utctime) | ||||||
|     user: str |     user: int | None = Field(default=None, foreign_key="player.id") | ||||||
|     mvps: list[str] = Field(sa_column=Column(ARRAY(String))) |     mvps: list[int] = Field(sa_column=Column(ARRAY(Integer))) | ||||||
|  |  | ||||||
|  |  | ||||||
| class User(SQLModel, table=True): |  | ||||||
|     username: str = Field(default=None, primary_key=True) |  | ||||||
|     email: str | None = None |  | ||||||
|     full_name: str | None = None |  | ||||||
|     disabled: bool | None = None |  | ||||||
|     hashed_password: str | None = None |  | ||||||
|     player_id: int | None = Field(default=None, foreign_key="player.id") |  | ||||||
|  |  | ||||||
|  |  | ||||||
| SQLModel.metadata.create_all(engine) | SQLModel.metadata.create_all(engine) | ||||||
|   | |||||||
							
								
								
									
										33
									
								
								main.py
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								main.py
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| from fastapi import APIRouter, Depends, FastAPI, status | from fastapi import APIRouter, Depends, FastAPI, Security, status | ||||||
| from fastapi.staticfiles import StaticFiles | from fastapi.staticfiles import StaticFiles | ||||||
| from db import Player, Team, Chemistry, MVPRanking, engine | from db import Player, Team, Chemistry, MVPRanking, engine | ||||||
| from sqlmodel import ( | from sqlmodel import ( | ||||||
| @@ -8,20 +8,21 @@ from sqlmodel import ( | |||||||
| from fastapi.middleware.cors import CORSMiddleware | from fastapi.middleware.cors import CORSMiddleware | ||||||
| from analysis import analysis_router | from analysis import analysis_router | ||||||
| from security import ( | from security import ( | ||||||
|  |     change_password, | ||||||
|     get_current_active_user, |     get_current_active_user, | ||||||
|     login_for_access_token, |     login_for_access_token, | ||||||
|     read_users_me, |     logout, | ||||||
|  |     read_player_me, | ||||||
|     read_own_items, |     read_own_items, | ||||||
|  |     set_first_password, | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| app = FastAPI(title="cutt") | app = FastAPI(title="cutt") | ||||||
| api_router = APIRouter(prefix="/api") | api_router = APIRouter(prefix="/api") | ||||||
| origins = [ | origins = [ | ||||||
|     "*", |     "https://cutt.0124816.xyz", | ||||||
|     "http://localhost", |     "http://localhost:5173", | ||||||
|     "http://localhost:3000", |  | ||||||
|     "http://localhost:8000", |  | ||||||
| ] | ] | ||||||
|  |  | ||||||
| app.add_middleware( | app.add_middleware( | ||||||
| @@ -54,7 +55,7 @@ def add_players(players: list[Player]): | |||||||
|  |  | ||||||
| def list_players(): | def list_players(): | ||||||
|     with Session(engine) as session: |     with Session(engine) as session: | ||||||
|         statement = select(Player).order_by(Player.name) |         statement = select(Player).order_by(Player.display_name) | ||||||
|         return session.exec(statement).fetchall() |         return session.exec(statement).fetchall() | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -67,6 +68,11 @@ def list_teams(): | |||||||
| player_router = APIRouter(prefix="/player") | player_router = APIRouter(prefix="/player") | ||||||
| player_router.add_api_route("/list", endpoint=list_players, methods=["GET"]) | player_router.add_api_route("/list", endpoint=list_players, methods=["GET"]) | ||||||
| player_router.add_api_route("/add", endpoint=add_player, methods=["POST"]) | player_router.add_api_route("/add", endpoint=add_player, methods=["POST"]) | ||||||
|  | player_router.add_api_route("/me", endpoint=read_player_me, methods=["GET"]) | ||||||
|  | player_router.add_api_route("/me/items", endpoint=read_own_items, methods=["GET"]) | ||||||
|  | player_router.add_api_route( | ||||||
|  |     "/change_password", endpoint=change_password, methods=["POST"] | ||||||
|  | ) | ||||||
|  |  | ||||||
| team_router = APIRouter(prefix="/team") | team_router = APIRouter(prefix="/team") | ||||||
| team_router.add_api_route("/list", endpoint=list_teams, methods=["GET"]) | team_router.add_api_route("/list", endpoint=list_teams, methods=["GET"]) | ||||||
| @@ -95,13 +101,16 @@ class SPAStaticFiles(StaticFiles): | |||||||
|         return response |         return response | ||||||
|  |  | ||||||
|  |  | ||||||
| api_router.include_router(player_router) |  | ||||||
| api_router.include_router(team_router) |  | ||||||
| api_router.include_router( | api_router.include_router( | ||||||
|     analysis_router, dependencies=[Depends(get_current_active_user)] |     player_router, dependencies=[Depends(get_current_active_user)] | ||||||
|  | ) | ||||||
|  | api_router.include_router(team_router, dependencies=[Depends(get_current_active_user)]) | ||||||
|  | api_router.include_router( | ||||||
|  |     analysis_router, | ||||||
|  |     dependencies=[Security(get_current_active_user, scopes=["analysis"])], | ||||||
| ) | ) | ||||||
| api_router.add_api_route("/token", endpoint=login_for_access_token, methods=["POST"]) | api_router.add_api_route("/token", endpoint=login_for_access_token, methods=["POST"]) | ||||||
| api_router.add_api_route("/users/me/", endpoint=read_users_me, methods=["GET"]) | api_router.add_api_route("/set_password", endpoint=set_first_password, methods=["POST"]) | ||||||
| api_router.add_api_route("/users/me/items/", endpoint=read_own_items, methods=["GET"]) | api_router.add_api_route("/logout", endpoint=logout, methods=["POST"]) | ||||||
| app.include_router(api_router) | app.include_router(api_router) | ||||||
| app.mount("/", SPAStaticFiles(directory="dist", html=True), name="site") | app.mount("/", SPAStaticFiles(directory="dist", html=True), name="site") | ||||||
|   | |||||||
| @@ -10,16 +10,17 @@ | |||||||
|     "preview": "vite preview" |     "preview": "vite preview" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "react": "^18.3.1", |     "jwt-decode": "^4.0.0", | ||||||
|     "react-dom": "^18.3.1", |     "react": "18.3.1", | ||||||
|  |     "react-dom": "18.3.1", | ||||||
|     "react-sortablejs": "^6.1.4", |     "react-sortablejs": "^6.1.4", | ||||||
|     "reagraph": "^4.21.2", |     "reagraph": "^4.21.2", | ||||||
|     "sortablejs": "^1.15.6" |     "sortablejs": "^1.15.6" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@eslint/js": "^9.17.0", |     "@eslint/js": "^9.17.0", | ||||||
|     "@types/react": "^18.3.18", |     "@types/react": "18.3.18", | ||||||
|     "@types/react-dom": "^18.3.5", |     "@types/react-dom": "18.3.5", | ||||||
|     "@types/sortablejs": "^1.15.8", |     "@types/sortablejs": "^1.15.8", | ||||||
|     "@vitejs/plugin-react": "^4.3.4", |     "@vitejs/plugin-react": "^4.3.4", | ||||||
|     "eslint": "^9.17.0", |     "eslint": "^9.17.0", | ||||||
|   | |||||||
							
								
								
									
										175
									
								
								security.py
									
									
									
									
									
								
							
							
						
						
									
										175
									
								
								security.py
									
									
									
									
									
								
							| @@ -1,12 +1,17 @@ | |||||||
| from datetime import timedelta, timezone, datetime | from datetime import timedelta, timezone, datetime | ||||||
| from typing import Annotated | from typing import Annotated | ||||||
| from fastapi import Depends, HTTPException, Response, status | from fastapi import Depends, HTTPException, Request, Response, status | ||||||
| from pydantic import BaseModel | from fastapi.responses import PlainTextResponse | ||||||
|  | from pydantic import BaseModel, ValidationError | ||||||
| import jwt | import jwt | ||||||
| from jwt.exceptions import InvalidTokenError | from jwt.exceptions import ExpiredSignatureError, InvalidTokenError | ||||||
| from sqlmodel import Session, select | from sqlmodel import Session, select | ||||||
| from db import engine, User | from db import engine, Player | ||||||
| from fastapi.security import OAuth2PasswordBearer, OAuth2PasswordRequestForm | from fastapi.security import ( | ||||||
|  |     OAuth2PasswordBearer, | ||||||
|  |     OAuth2PasswordRequestForm, | ||||||
|  |     SecurityScopes, | ||||||
|  | ) | ||||||
| from pydantic_settings import BaseSettings, SettingsConfigDict | from pydantic_settings import BaseSettings, SettingsConfigDict | ||||||
| from passlib.context import CryptContext | from passlib.context import CryptContext | ||||||
| from sqlalchemy.exc import OperationalError | from sqlalchemy.exc import OperationalError | ||||||
| @@ -14,7 +19,7 @@ from sqlalchemy.exc import OperationalError | |||||||
|  |  | ||||||
| class Config(BaseSettings): | class Config(BaseSettings): | ||||||
|     secret_key: str = "" |     secret_key: str = "" | ||||||
|     access_token_expire_minutes: int = 30 |     access_token_expire_minutes: int = 15 | ||||||
|     model_config = SettingsConfigDict( |     model_config = SettingsConfigDict( | ||||||
|         env_file=".env", env_file_encoding="utf-8", extra="ignore" |         env_file=".env", env_file_encoding="utf-8", extra="ignore" | ||||||
|     ) |     ) | ||||||
| @@ -25,17 +30,38 @@ config = Config() | |||||||
|  |  | ||||||
| class Token(BaseModel): | class Token(BaseModel): | ||||||
|     access_token: str |     access_token: str | ||||||
|     token_type: str |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TokenData(BaseModel): | class TokenData(BaseModel): | ||||||
|     username: str | None = None |     username: str | None = None | ||||||
|  |     scopes: list[str] = [] | ||||||
|  |  | ||||||
|  |  | ||||||
| pwd_context = CryptContext(schemes=["argon2"], deprecated="auto") | pwd_context = CryptContext(schemes=["argon2"], deprecated="auto") | ||||||
|  |  | ||||||
|  |  | ||||||
| oauth2_scheme = OAuth2PasswordBearer(tokenUrl="api/token") | class CookieOAuth2(OAuth2PasswordBearer): | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     async def __call__(self, request: Request): | ||||||
|  |         cookie_token = request.cookies.get("access_token") | ||||||
|  |         if cookie_token: | ||||||
|  |             return cookie_token | ||||||
|  |         else: | ||||||
|  |             header_token = await super().__call__(request) | ||||||
|  |             if header_token: | ||||||
|  |                 return header_token | ||||||
|  |             else: | ||||||
|  |                 raise HTTPException(status_code=401) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | oauth2_scheme = CookieOAuth2( | ||||||
|  |     tokenUrl="api/token", | ||||||
|  |     scopes={ | ||||||
|  |         "analysis": "Access the results.", | ||||||
|  |     }, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def verify_password(plain_password, hashed_password): | def verify_password(plain_password, hashed_password): | ||||||
| @@ -51,7 +77,7 @@ def get_user(username: str | None): | |||||||
|         try: |         try: | ||||||
|             with Session(engine) as session: |             with Session(engine) as session: | ||||||
|                 return session.exec( |                 return session.exec( | ||||||
|                     select(User).where(User.username == username) |                     select(Player).where(Player.username == username) | ||||||
|                 ).one_or_none() |                 ).one_or_none() | ||||||
|         except OperationalError: |         except OperationalError: | ||||||
|             return |             return | ||||||
| @@ -71,34 +97,61 @@ def create_access_token(data: dict, expires_delta: timedelta | None = None): | |||||||
|     if expires_delta: |     if expires_delta: | ||||||
|         expire = datetime.now(timezone.utc) + expires_delta |         expire = datetime.now(timezone.utc) + expires_delta | ||||||
|     else: |     else: | ||||||
|         expire = datetime.now(timezone.utc) + timedelta(minutes=15) |         expire = datetime.now(timezone.utc) + timedelta( | ||||||
|  |             minutes=config.access_token_expire_minutes | ||||||
|  |         ) | ||||||
|     to_encode.update({"exp": expire}) |     to_encode.update({"exp": expire}) | ||||||
|     encoded_jwt = jwt.encode(to_encode, config.secret_key, algorithm="HS256") |     encoded_jwt = jwt.encode(to_encode, config.secret_key, algorithm="HS256") | ||||||
|     return encoded_jwt |     return encoded_jwt | ||||||
|  |  | ||||||
|  |  | ||||||
| async def get_current_user(token: Annotated[str, Depends(oauth2_scheme)]): | async def get_current_user( | ||||||
|  |     token: Annotated[str, Depends(oauth2_scheme)], | ||||||
|  |     security_scopes: SecurityScopes, | ||||||
|  | ): | ||||||
|  |     if security_scopes.scopes: | ||||||
|  |         authenticate_value = f'Bearer scope="{security_scopes.scope_str}"' | ||||||
|  |     else: | ||||||
|  |         authenticate_value = "Bearer" | ||||||
|     credentials_exception = HTTPException( |     credentials_exception = HTTPException( | ||||||
|         status_code=status.HTTP_401_UNAUTHORIZED, |         status_code=status.HTTP_401_UNAUTHORIZED, | ||||||
|         detail="Could not validate credentials", |         detail="Could not validate credentials", | ||||||
|         headers={"WWW-Authenticate": "Bearer"}, |         headers={"WWW-Authenticate": authenticate_value}, | ||||||
|     ) |     ) | ||||||
|  |     # access_token = request.cookies.get("access_token") | ||||||
|  |     access_token = token | ||||||
|  |     if not access_token: | ||||||
|  |         raise credentials_exception | ||||||
|     try: |     try: | ||||||
|         payload = jwt.decode(token, config.secret_key, algorithms=["HS256"]) |         payload = jwt.decode(access_token, config.secret_key, algorithms=["HS256"]) | ||||||
|         username: str = payload.get("sub") |         username: str = payload.get("sub") | ||||||
|         if username is None: |         if username is None: | ||||||
|             raise credentials_exception |             raise credentials_exception | ||||||
|         token_data = TokenData(username=username) |         token_scopes = payload.get("scopes", []) | ||||||
|     except InvalidTokenError: |         token_data = TokenData(username=username, scopes=token_scopes) | ||||||
|  |     except ExpiredSignatureError: | ||||||
|  |         raise HTTPException( | ||||||
|  |             status_code=status.HTTP_401_UNAUTHORIZED, | ||||||
|  |             detail="Access token expired", | ||||||
|  |             headers={"WWW-Authenticate": authenticate_value}, | ||||||
|  |         ) | ||||||
|  |     except (InvalidTokenError, ValidationError): | ||||||
|         raise credentials_exception |         raise credentials_exception | ||||||
|     user = get_user(username=token_data.username) |     user = get_user(username=token_data.username) | ||||||
|     if user is None: |     if user is None: | ||||||
|         raise credentials_exception |         raise credentials_exception | ||||||
|  |     for scope in security_scopes.scopes: | ||||||
|  |         if scope not in token_data.scopes: | ||||||
|  |             raise HTTPException( | ||||||
|  |                 status_code=status.HTTP_401_UNAUTHORIZED, | ||||||
|  |                 detail="Not enough permissions", | ||||||
|  |                 headers={"WWW-Authenticate": authenticate_value}, | ||||||
|  |             ) | ||||||
|     return user |     return user | ||||||
|  |  | ||||||
|  |  | ||||||
| async def get_current_active_user( | async def get_current_active_user( | ||||||
|     current_user: Annotated[User, Depends(get_current_user)], |     current_user: Annotated[Player, Depends(get_current_user)], | ||||||
| ): | ): | ||||||
|     if current_user.disabled: |     if current_user.disabled: | ||||||
|         raise HTTPException(status_code=400, detail="Inactive user") |         raise HTTPException(status_code=400, detail="Inactive user") | ||||||
| @@ -115,23 +168,99 @@ async def login_for_access_token( | |||||||
|             detail="Incorrect username or password", |             detail="Incorrect username or password", | ||||||
|             headers={"WWW-Authenticate": "Bearer"}, |             headers={"WWW-Authenticate": "Bearer"}, | ||||||
|         ) |         ) | ||||||
|     access_token_expires = timedelta(minutes=config.access_token_expire_minutes) |     allowed_scopes = set(user.scopes.split()) | ||||||
|  |     requested_scopes = set(form_data.scopes) | ||||||
|     access_token = create_access_token( |     access_token = create_access_token( | ||||||
|         data={"sub": user.username}, expires_delta=access_token_expires |         data={"sub": user.username, "scopes": list(allowed_scopes)} | ||||||
|     ) |     ) | ||||||
|     response.set_cookie( |     response.set_cookie( | ||||||
|         "Authorization", value=f"Bearer {access_token}", httponly=True, samesite="none" |         "access_token", | ||||||
|  |         value=access_token, | ||||||
|  |         httponly=True, | ||||||
|  |         samesite="strict", | ||||||
|     ) |     ) | ||||||
|     return Token(access_token=access_token, token_type="bearer") |     return Token(access_token=access_token) | ||||||
|  |  | ||||||
|  |  | ||||||
| async def read_users_me( | async def logout(response: Response): | ||||||
|     current_user: Annotated[User, Depends(get_current_active_user)], |     response.set_cookie("access_token", "", expires=0, httponly=True, samesite="strict") | ||||||
|  |     return {"message": "Successfully logged out"} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def generate_one_time_token(username): | ||||||
|  |     user = get_user(username) | ||||||
|  |     if user: | ||||||
|  |         expire = timedelta(days=7) | ||||||
|  |         token = create_access_token( | ||||||
|  |             data={"sub": username, "name": user.display_name}, | ||||||
|  |             expires_delta=expire, | ||||||
|  |         ) | ||||||
|  |         return token | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FirstPassword(BaseModel): | ||||||
|  |     token: str | ||||||
|  |     password: str | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def set_first_password(req: FirstPassword): | ||||||
|  |     credentials_exception = HTTPException( | ||||||
|  |         status_code=status.HTTP_401_UNAUTHORIZED, | ||||||
|  |         detail="Could not validate token", | ||||||
|  |     ) | ||||||
|  |     try: | ||||||
|  |         payload = jwt.decode(req.token, config.secret_key, algorithms=["HS256"]) | ||||||
|  |         username: str = payload.get("sub") | ||||||
|  |         if username is None: | ||||||
|  |             raise credentials_exception | ||||||
|  |     except ExpiredSignatureError: | ||||||
|  |         raise HTTPException( | ||||||
|  |             status_code=status.HTTP_401_UNAUTHORIZED, | ||||||
|  |             detail="Access token expired", | ||||||
|  |         ) | ||||||
|  |     except (InvalidTokenError, ValidationError): | ||||||
|  |         raise credentials_exception | ||||||
|  |  | ||||||
|  |     user = get_user(username) | ||||||
|  |     if user: | ||||||
|  |         with Session(engine) as session: | ||||||
|  |             user.hashed_password = get_password_hash(req.password) | ||||||
|  |             session.add(user) | ||||||
|  |             session.commit() | ||||||
|  |             return Response("Password set successfully", status_code=status.HTTP_200_OK) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def change_password( | ||||||
|  |     current_password: str, | ||||||
|  |     new_password: str, | ||||||
|  |     user: Annotated[Player, Depends(get_current_active_user)], | ||||||
|  | ): | ||||||
|  |     if ( | ||||||
|  |         new_password | ||||||
|  |         and user.hashed_password | ||||||
|  |         and verify_password(current_password, user.hashed_password) | ||||||
|  |     ): | ||||||
|  |         with Session(engine) as session: | ||||||
|  |             user.hashed_password = get_password_hash(new_password) | ||||||
|  |             session.add(user) | ||||||
|  |             session.commit() | ||||||
|  |             return PlainTextResponse( | ||||||
|  |                 "Password changed successfully", status_code=status.HTTP_200_OK | ||||||
|  |             ) | ||||||
|  |     else: | ||||||
|  |         raise HTTPException( | ||||||
|  |             status_code=status.HTTP_400_BAD_REQUEST, | ||||||
|  |             detail="Wrong password", | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def read_player_me( | ||||||
|  |     current_user: Annotated[Player, Depends(get_current_active_user)], | ||||||
| ): | ): | ||||||
|     return current_user |     return current_user | ||||||
|  |  | ||||||
|  |  | ||||||
| async def read_own_items( | async def read_own_items( | ||||||
|     current_user: Annotated[User, Depends(get_current_active_user)], |     current_user: Annotated[Player, Depends(get_current_active_user)], | ||||||
| ): | ): | ||||||
|     return [{"item_id": "Foo", "owner": current_user.username}] |     return [{"item_id": "Foo", "owner": current_user.username}] | ||||||
|   | |||||||
| @@ -42,7 +42,7 @@ interface DeferredProps { | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| let timeoutID: number | null = null; | let timeoutID: NodeJS.Timeout | null = null; | ||||||
| export default function Analysis() { | export default function Analysis() { | ||||||
|   const [image, setImage] = useState(""); |   const [image, setImage] = useState(""); | ||||||
|   const [params, setParams] = useState<Params>({ |   const [params, setParams] = useState<Params>({ | ||||||
|   | |||||||
							
								
								
									
										160
									
								
								src/App.css
									
									
									
									
									
								
							
							
						
						
									
										160
									
								
								src/App.css
									
									
									
									
									
								
							| @@ -23,6 +23,114 @@ footer { | |||||||
|   font-size: x-small; |   font-size: x-small; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .fixed-footer { | ||||||
|  |   position: absolute; | ||||||
|  |   bottom: 4px; | ||||||
|  |   left: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /*=========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; | ||||||
|  |   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: .4s; | ||||||
|  |   transition: .4s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .slider:before { | ||||||
|  |   position: absolute; | ||||||
|  |   content: ""; | ||||||
|  |   height: 18px; | ||||||
|  |   width: 18px; | ||||||
|  |   left: 3px; | ||||||
|  |   bottom: 3px; | ||||||
|  |   background-color: white; | ||||||
|  |   border-radius: 50%; | ||||||
|  |   -webkit-transition: .4s; | ||||||
|  |   transition: .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 { | .grey { | ||||||
|   color: #444; |   color: #444; | ||||||
| @@ -129,6 +237,7 @@ h3 { | |||||||
|  |  | ||||||
| button, | button, | ||||||
| .button { | .button { | ||||||
|  |   margin: 4px; | ||||||
|   font-weight: bold; |   font-weight: bold; | ||||||
|   font-size: large; |   font-size: large; | ||||||
|   color: aliceblue; |   color: aliceblue; | ||||||
| @@ -168,13 +277,24 @@ button, | |||||||
|   #control-panel { |   #control-panel { | ||||||
|     grid-template-columns: repeat(2, 1fr); |     grid-template-columns: repeat(2, 1fr); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   .control { | ||||||
|  |     font-size: 80%; | ||||||
|  |     margin: 0px; | ||||||
|  |   } | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @media only screen and (max-width: 768px) { | @media only screen and (max-width: 768px) { | ||||||
|   #control-panel { |   #control-panel { | ||||||
|     grid-template-columns: 1fr; |     grid-template-columns: 1fr; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   .networkroute { | ||||||
|  |     display: none; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   .submit_text { |   .submit_text { | ||||||
|     display: none; |     display: none; | ||||||
|   } |   } | ||||||
| @@ -241,11 +361,17 @@ button, | |||||||
|   font-size: 150%; |   font-size: 150%; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /*======LOGO=======*/ | ||||||
|  |  | ||||||
| .logo { | .logo { | ||||||
|   position: relative; |   position: relative; | ||||||
|   text-align: center; |   text-align: center; | ||||||
|   height: 140px; |   height: 140px; | ||||||
|   margin-bottom: 20px; |  | ||||||
|  |   span { | ||||||
|  |     display: block; | ||||||
|  |     margin: 2px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   img { |   img { | ||||||
|     display: block; |     display: block; | ||||||
| @@ -268,6 +394,38 @@ button, | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .avatar { | ||||||
|  |   background-color: lightsteelblue; | ||||||
|  |   padding: 3px 8px; | ||||||
|  |   width: fit-content; | ||||||
|  |   border: 3px solid black; | ||||||
|  |   margin: 0 auto 16px auto; | ||||||
|  |  | ||||||
|  |   ul { | ||||||
|  |     min-width: 100px; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-info { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: 1fr 1fr; | ||||||
|  |   gap: 2px 16px; | ||||||
|  |  | ||||||
|  |   div { | ||||||
|  |     text-align: left; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .networkroute { | ||||||
|  |   z-index: 3; | ||||||
|  |   position: absolute; | ||||||
|  |   top: 24px; | ||||||
|  |   left: 48px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /*======SPINNER=======*/ | ||||||
|  |  | ||||||
| .loader { | .loader { | ||||||
|   display: block; |   display: block; | ||||||
|   position: relative; |   position: relative; | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								src/App.tsx
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								src/App.tsx
									
									
									
									
									
								
							| @@ -6,25 +6,30 @@ import Rankings from "./Rankings"; | |||||||
| import { BrowserRouter, Routes, Route } from "react-router"; | import { BrowserRouter, Routes, Route } from "react-router"; | ||||||
| import { SessionProvider } from "./Session"; | import { SessionProvider } from "./Session"; | ||||||
| import { GraphComponent } from "./Network"; | import { GraphComponent } from "./Network"; | ||||||
|  | import MVPChart from "./MVPChart"; | ||||||
|  | import { SetPassword } from "./SetPassword"; | ||||||
|  |  | ||||||
| function App() { | function App() { | ||||||
|   return ( |   return ( | ||||||
|     <BrowserRouter> |     <BrowserRouter> | ||||||
|  |       <Routes> | ||||||
|  |         <Route path="/password" element={<SetPassword />} /> | ||||||
|  |         <Route | ||||||
|  |           path="/*" | ||||||
|  |           element={ | ||||||
|  |             <SessionProvider> | ||||||
|               <Header /> |               <Header /> | ||||||
|               <Routes> |               <Routes> | ||||||
|                 <Route index element={<Rankings />} /> |                 <Route index element={<Rankings />} /> | ||||||
|         <Route path="/network" element={ |                 <Route path="/network" element={<GraphComponent />} /> | ||||||
|           <SessionProvider> |                 <Route path="/analysis" element={<Analysis />} /> | ||||||
|             <GraphComponent /> |                 <Route path="/mvp" element={<MVPChart />} /> | ||||||
|           </SessionProvider> |  | ||||||
|         } /> |  | ||||||
|         <Route path="/analysis" element={ |  | ||||||
|           <SessionProvider> |  | ||||||
|             <Analysis /> |  | ||||||
|           </SessionProvider> |  | ||||||
|         } /> |  | ||||||
|               </Routes> |               </Routes> | ||||||
|               <Footer /> |               <Footer /> | ||||||
|  |             </SessionProvider> | ||||||
|  |           } | ||||||
|  |         /> | ||||||
|  |       </Routes> | ||||||
|     </BrowserRouter> |     </BrowserRouter> | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										165
									
								
								src/Avatar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										165
									
								
								src/Avatar.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,165 @@ | |||||||
|  | import { createRef, MouseEventHandler, useEffect, useState } from "react"; | ||||||
|  | import { useSession } from "./Session"; | ||||||
|  | import { User } from "./api"; | ||||||
|  |  | ||||||
|  | interface ContextMenuItem { | ||||||
|  |   label: string; | ||||||
|  |   onClick: () => void; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const UserInfo = (user: User) => { | ||||||
|  |   return ( | ||||||
|  |     <div className="user-info"> | ||||||
|  |       <div> | ||||||
|  |         <b>username: </b> | ||||||
|  |       </div> | ||||||
|  |       <div>{user?.username}</div> | ||||||
|  |       <div> | ||||||
|  |         <b>display name: </b> | ||||||
|  |       </div> | ||||||
|  |       <div>{user?.display_name}</div> | ||||||
|  |       <div> | ||||||
|  |         <b>number: </b> | ||||||
|  |       </div> | ||||||
|  |       <div>{user?.number ? user?.number : "-"}</div> | ||||||
|  |       <div> | ||||||
|  |         <b>email: </b> | ||||||
|  |       </div> | ||||||
|  |       <div>{user?.email ? user?.email : "-"}</div> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default function Avatar() { | ||||||
|  |   const { user, onLogout } = useSession(); | ||||||
|  |   const [contextMenu, setContextMenu] = useState<{ | ||||||
|  |     open: boolean; | ||||||
|  |     allowOpen: boolean; | ||||||
|  |     mouseX: number; | ||||||
|  |     mouseY: number; | ||||||
|  |   }>({ open: false, allowOpen: true, mouseX: 0, mouseY: 0 }); | ||||||
|  |   const contextMenuRef = createRef<HTMLUListElement>(); | ||||||
|  |   const avatarRef = createRef<HTMLDivElement>(); | ||||||
|  |  | ||||||
|  |   const contextMenuItems: ContextMenuItem[] = [ | ||||||
|  |     { label: "View Profile", onClick: handleViewProfile }, | ||||||
|  |     { label: "Logout", onClick: onLogout }, | ||||||
|  |   ]; | ||||||
|  |  | ||||||
|  |   const handleMenuClick: MouseEventHandler<HTMLDivElement> = (event) => { | ||||||
|  |     if (!contextMenu.allowOpen) return; | ||||||
|  |     event.preventDefault(); | ||||||
|  |     setContextMenu({ | ||||||
|  |       open: !contextMenu.open, | ||||||
|  |       allowOpen: contextMenu.allowOpen, | ||||||
|  |       mouseX: event.clientX + 4, | ||||||
|  |       mouseY: event.clientY + 2, | ||||||
|  |     }); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (contextMenu.open) { | ||||||
|  |       document.addEventListener("click", handleCloseContextMenuOutside); | ||||||
|  |     } | ||||||
|  |     return () => { | ||||||
|  |       document.removeEventListener("click", handleCloseContextMenuOutside); | ||||||
|  |     }; | ||||||
|  |   }, [contextMenu.open]); | ||||||
|  |  | ||||||
|  |   const handleMenuClose = () => { | ||||||
|  |     setContextMenu({ ...contextMenu, open: false }); | ||||||
|  |     setContextMenu((prevContextMenu) => ({ | ||||||
|  |       ...prevContextMenu, | ||||||
|  |       allowOpen: false, | ||||||
|  |     })); | ||||||
|  |     setTimeout(() => { | ||||||
|  |       setContextMenu((prevContextMenu) => ({ | ||||||
|  |         ...prevContextMenu, | ||||||
|  |         allowOpen: true, | ||||||
|  |       })); | ||||||
|  |     }, 100); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleCloseContextMenuOutside: (event: MouseEvent) => void = (ev) => { | ||||||
|  |     if ( | ||||||
|  |       !( | ||||||
|  |         contextMenuRef.current?.contains(ev.target as Node) || | ||||||
|  |         avatarRef.current?.contains(ev.target as Node) | ||||||
|  |       ) | ||||||
|  |     ) | ||||||
|  |       handleMenuClose(); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const [dialog, setDialog] = useState(<></>); | ||||||
|  |   const dialogRef = createRef<HTMLDialogElement>(); | ||||||
|  |  | ||||||
|  |   function handleViewProfile() { | ||||||
|  |     handleMenuClose(); | ||||||
|  |     if (user) { | ||||||
|  |       dialogRef.current?.showModal(); | ||||||
|  |       setDialog(UserInfo(user)); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       className="avatar" | ||||||
|  |       onContextMenu={handleMenuClick} | ||||||
|  |       style={{ display: user ? "block" : "none" }} | ||||||
|  |       onClick={(event) => { | ||||||
|  |         if (contextMenu.open && event.target === avatarRef.current) { | ||||||
|  |           handleMenuClose(); | ||||||
|  |         } else { | ||||||
|  |           handleMenuClick(event); | ||||||
|  |         } | ||||||
|  |       }} | ||||||
|  |       ref={avatarRef} | ||||||
|  |     > | ||||||
|  |       👤 {user?.username} | ||||||
|  |       {contextMenu.open && ( | ||||||
|  |         <ul | ||||||
|  |           className="context-menu" | ||||||
|  |           ref={contextMenuRef} | ||||||
|  |           style={{ | ||||||
|  |             zIndex: 3, | ||||||
|  |             position: "absolute", | ||||||
|  |             top: contextMenu.mouseY, | ||||||
|  |             left: contextMenu.mouseX, | ||||||
|  |             background: "white", | ||||||
|  |             border: "1px solid #ddd", | ||||||
|  |             padding: 0, | ||||||
|  |             margin: 0, | ||||||
|  |             listStyle: "none", | ||||||
|  |           }} | ||||||
|  |         > | ||||||
|  |           {contextMenuItems.map((item, index) => ( | ||||||
|  |             <li | ||||||
|  |               key={index} | ||||||
|  |               style={{ | ||||||
|  |                 padding: "10px", | ||||||
|  |                 borderBottom: "1px solid #ddd", | ||||||
|  |                 cursor: "pointer", | ||||||
|  |               }} | ||||||
|  |               onClick={() => { | ||||||
|  |                 item.onClick(); | ||||||
|  |                 handleMenuClose(); | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               {item.label} | ||||||
|  |             </li> | ||||||
|  |           ))} | ||||||
|  |         </ul> | ||||||
|  |       )} | ||||||
|  |       <dialog | ||||||
|  |         id="AvatarDialog" | ||||||
|  |         ref={dialogRef} | ||||||
|  |         onClick={(event) => { | ||||||
|  |           event.stopPropagation(); | ||||||
|  |           event.currentTarget.close(); | ||||||
|  |         }} | ||||||
|  |       > | ||||||
|  |         {dialog} | ||||||
|  |       </dialog> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | } | ||||||
							
								
								
									
										95
									
								
								src/BarChart.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/BarChart.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | |||||||
|  | import { FC } from 'react'; | ||||||
|  | import { PlayerRanking } from './types'; | ||||||
|  |  | ||||||
|  | interface BarChartProps { | ||||||
|  |   players: PlayerRanking[]; | ||||||
|  |   width: number; | ||||||
|  |   height: number; | ||||||
|  |   std: boolean; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const BarChart: FC<BarChartProps> = ({ players, width, height, std }) => { | ||||||
|  |   const padding = 24; | ||||||
|  |   const maxValue = Math.max(...players.map((player) => player.rank)) + 1; | ||||||
|  |   const barWidth = (width - 2 * padding) / players.length; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <svg width={width} height={height}> | ||||||
|  |  | ||||||
|  |       {players.map((player, index) => ( | ||||||
|  |         <rect | ||||||
|  |           key={index} | ||||||
|  |           x={index * barWidth + padding} | ||||||
|  |           y={height - (1 - player.rank / maxValue) * height} | ||||||
|  |           width={barWidth - 8} // subtract 2 for some spacing between bars | ||||||
|  |           height={(1 - player.rank / maxValue) * height} | ||||||
|  |           fill="#69f" | ||||||
|  |         /> | ||||||
|  |       ))} | ||||||
|  |  | ||||||
|  |       {players.map((player, index) => ( | ||||||
|  |         <text | ||||||
|  |           key={index} | ||||||
|  |           x={index * barWidth + barWidth / 2 - 4 + padding} | ||||||
|  |           y={height - (1 - player.rank / maxValue) * height - 5} | ||||||
|  |           textAnchor="middle" | ||||||
|  |           //transform='rotate(-27)' | ||||||
|  |           //style={{ transformOrigin: "center", transformBox: "fill-box" }} | ||||||
|  |           fontSize="16px" | ||||||
|  |           fill="#404040" | ||||||
|  |         > | ||||||
|  |           {player.name} | ||||||
|  |         </text> | ||||||
|  |       ))} | ||||||
|  |  | ||||||
|  |       {players.map((player, index) => ( | ||||||
|  |         <text | ||||||
|  |           key={index} | ||||||
|  |           x={index * barWidth + barWidth / 2 + padding - 4} | ||||||
|  |           y={height - 8} | ||||||
|  |           textAnchor="middle" | ||||||
|  |           fontSize="12px" | ||||||
|  |           fill="#404040" | ||||||
|  |         > | ||||||
|  |           {player.rank} | ||||||
|  |         </text> | ||||||
|  |       ))} | ||||||
|  |  | ||||||
|  |       {std && players.map((player, index) => ( | ||||||
|  |         <line | ||||||
|  |           key={`error-${index}`} | ||||||
|  |           x1={index * barWidth + barWidth / 2 + padding} | ||||||
|  |           y1={height - (1 - player.rank / maxValue) * height - (player.std / maxValue) * height} | ||||||
|  |           x2={index * barWidth + barWidth / 2 + padding} | ||||||
|  |           y2={height - (1 - player.rank / maxValue) * height + (player.std / maxValue) * height} | ||||||
|  |           stroke="#ff0000" | ||||||
|  |           strokeWidth="1" | ||||||
|  |         /> | ||||||
|  |       ))} | ||||||
|  |       {std && players.map((player, index) => ( | ||||||
|  |         <line | ||||||
|  |           key={`cap-${index}-top`} | ||||||
|  |           x1={index * barWidth + barWidth / 2 - 2 + padding} | ||||||
|  |           y1={height - (1 - player.rank / maxValue) * height - (player.std / maxValue) * height} | ||||||
|  |           x2={index * barWidth + barWidth / 2 + 2 + padding} | ||||||
|  |           y2={height - (1 - player.rank / maxValue) * height - (player.std / maxValue) * height} | ||||||
|  |           stroke="#ff0000" | ||||||
|  |           strokeWidth="1" | ||||||
|  |         /> | ||||||
|  |       ))} | ||||||
|  |       {std && players.map((player, index) => ( | ||||||
|  |         <line | ||||||
|  |           key={`cap-${index}-bottom`} | ||||||
|  |           x1={index * barWidth + barWidth / 2 - 2 + padding} | ||||||
|  |           y1={height - (1 - player.rank / maxValue) * height + (player.std / maxValue) * height} | ||||||
|  |           x2={index * barWidth + barWidth / 2 + 2 + padding} | ||||||
|  |           y2={height - (1 - player.rank / maxValue) * height + (player.std / maxValue) * height} | ||||||
|  |           stroke="#ff0000" | ||||||
|  |           strokeWidth="1" | ||||||
|  |         /> | ||||||
|  |       ))} | ||||||
|  |     </svg> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default BarChart; | ||||||
| @@ -1,12 +1,27 @@ | |||||||
|  | import { useLocation } from "react-router"; | ||||||
| import { Link } from "react-router"; | import { Link } from "react-router"; | ||||||
|  | import { useSession } from "./Session"; | ||||||
|  |  | ||||||
| export default function Footer() { | export default function Footer() { | ||||||
|         return <footer> |   const location = useLocation(); | ||||||
|  |   const { user } = useSession(); | ||||||
|  |   return ( | ||||||
|  |     <footer className={location.pathname === "/network" ? "fixed-footer" : ""}> | ||||||
|  |       {user?.scopes.split(" ").includes("analysis") && ( | ||||||
|         <div className="navbar"> |         <div className="navbar"> | ||||||
|                         <Link to="/" ><span>Form</span></Link> |           <Link to="/"> | ||||||
|  |             <span>Form</span> | ||||||
|  |           </Link> | ||||||
|           <span>|</span> |           <span>|</span> | ||||||
|                         <Link to="/network" ><span>Trainer Analysis</span></Link> |           <Link to="/network"> | ||||||
|  |             <span>Trainer Analysis</span> | ||||||
|  |           </Link> | ||||||
|  |           <span>|</span> | ||||||
|  |           <Link to="/mvp"> | ||||||
|  |             <span>MVP</span> | ||||||
|  |           </Link> | ||||||
|         </div> |         </div> | ||||||
|  |       )} | ||||||
|       <p className="grey extra-margin"> |       <p className="grey extra-margin"> | ||||||
|         something not working? |         something not working? | ||||||
|         <br /> |         <br /> | ||||||
| @@ -18,4 +33,5 @@ export default function Footer() { | |||||||
|         </a> |         </a> | ||||||
|       </p> |       </p> | ||||||
|     </footer> |     </footer> | ||||||
|  |   ); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,11 +1,18 @@ | |||||||
| import { baseUrl } from "./api"; | import { Link, useLocation } from "react-router"; | ||||||
|  | import Avatar from "./Avatar"; | ||||||
|  |  | ||||||
| export default function Header() { | export default function Header() { | ||||||
|   return <div className="logo"> |   const location = useLocation(); | ||||||
|     <a href={"/"}> |   return ( | ||||||
|  |     <div className={location.pathname === "/network" ? "networkroute" : ""}> | ||||||
|  |       <div className="logo"> | ||||||
|  |         <Link to="/"> | ||||||
|           <img alt="logo" height="66%" src="logo.svg" /> |           <img alt="logo" height="66%" src="logo.svg" /> | ||||||
|           <h3 className="centered">cutt</h3> |           <h3 className="centered">cutt</h3> | ||||||
|     </a> |         </Link> | ||||||
|         <span className="grey">cool ultimate team tool</span> |         <span className="grey">cool ultimate team tool</span> | ||||||
|       </div> |       </div> | ||||||
|  |       <Avatar /> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										106
									
								
								src/Login.tsx
									
									
									
									
									
								
							
							
						
						
									
										106
									
								
								src/Login.tsx
									
									
									
									
									
								
							| @@ -1,6 +1,7 @@ | |||||||
| import { FormEvent, useContext, useState } from "react"; | import { useEffect, useState } from "react"; | ||||||
| import { useNavigate } from "react-router"; | import { currentUser, login, User } from "./api"; | ||||||
| import { currentUser, login, LoginRequest, User } from "./api"; | import Header from "./Header"; | ||||||
|  | import { useLocation, useNavigate } from "react-router"; | ||||||
|  |  | ||||||
| export interface LoginProps { | export interface LoginProps { | ||||||
|   onLogin: (user: User) => void; |   onLogin: (user: User) => void; | ||||||
| @@ -9,78 +10,83 @@ export interface LoginProps { | |||||||
| export const Login = ({ onLogin }: LoginProps) => { | export const Login = ({ onLogin }: LoginProps) => { | ||||||
|   const [username, setUsername] = useState(""); |   const [username, setUsername] = useState(""); | ||||||
|   const [password, setPassword] = useState(""); |   const [password, setPassword] = useState(""); | ||||||
|   const [error, setError] = useState<unknown>(null); |   const [error, setError] = useState(""); | ||||||
|   const [loading, setLoading] = useState(false); |   const [loading, setLoading] = useState(false); | ||||||
|  |   const navigate = useNavigate(); | ||||||
|  |   const location = useLocation(); | ||||||
|  |  | ||||||
|   async function doLogin() { |   async function doLogin() { | ||||||
|     setLoading(true); |     setLoading(true); | ||||||
|     setError(null); |     setError(""); | ||||||
|     const timeout = new Promise((r) => setTimeout(r, 1500)); |     const timeout = new Promise((r) => setTimeout(r, 1000)); | ||||||
|     let user: User; |     let user: User; | ||||||
|     try { |     try { | ||||||
|       login({ username, password }); |       await login({ username, password }); | ||||||
|       user = await currentUser(); |       user = await currentUser(); | ||||||
|     } catch (e) { |     } catch (e) { | ||||||
|       await timeout; |       await timeout; | ||||||
|       setError(e); |       setError("failed"); | ||||||
|       setLoading(false); |       setLoading(false); | ||||||
|       return |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     await timeout; |     await timeout; | ||||||
|     onLogin(user); |     onLogin(user); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   function handleClick() { |  | ||||||
|     doLogin(); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   function handleSubmit(e: React.FormEvent) { |   function handleSubmit(e: React.FormEvent) { | ||||||
|     e.preventDefault(); |     e.preventDefault(); | ||||||
|     doLogin(); |     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 ( |   return ( | ||||||
|  |     <> | ||||||
|  |       <Header /> | ||||||
|       <form onSubmit={handleSubmit}> |       <form onSubmit={handleSubmit}> | ||||||
|         <div> |         <div> | ||||||
|         <input type="text" id="username" name="username" placeholder="username" required value={username} onChange={evt => setUsername(evt.target.value)} /> |           <input | ||||||
|  |             type="text" | ||||||
|  |             id="username" | ||||||
|  |             name="username" | ||||||
|  |             placeholder="username" | ||||||
|  |             required | ||||||
|  |             value={username} | ||||||
|  |             onChange={(evt) => { | ||||||
|  |               setError(""); | ||||||
|  |               setUsername(evt.target.value); | ||||||
|  |             }} | ||||||
|  |           /> | ||||||
|         </div> |         </div> | ||||||
|         <div> |         <div> | ||||||
|         <input type="password" id="password" name="password" placeholder="password" minLength={8} value={password} required onChange={evt => setPassword(evt.target.value)} /> |           <input | ||||||
|  |             type="password" | ||||||
|  |             id="password" | ||||||
|  |             name="password" | ||||||
|  |             placeholder="password" | ||||||
|  |             minLength={8} | ||||||
|  |             value={password} | ||||||
|  |             required | ||||||
|  |             onChange={(evt) => { | ||||||
|  |               setError(""); | ||||||
|  |               setPassword(evt.target.value); | ||||||
|  |             }} | ||||||
|  |           /> | ||||||
|         </div> |         </div> | ||||||
|       <button type="submit" value="login" style={{ fontSize: "small" }} onClick={handleClick} >login</button> |         <div>{error && <span style={{ color: "red" }}>{error}</span>}</div> | ||||||
|  |         <button type="submit" value="login" style={{ fontSize: "small" }}> | ||||||
|  |           login | ||||||
|  |         </button> | ||||||
|         {loading && <span className="loader" />} |         {loading && <span className="loader" />} | ||||||
|       </form> |       </form> | ||||||
|   ) |     </> | ||||||
| } |   ); | ||||||
|  | }; | ||||||
| /* |  | ||||||
| export default function Login(props: { onLogin: (user: User) => void }) { |  | ||||||
|   const { onLogin } = props; |  | ||||||
|   const [username, setUsername] = useState(""); |  | ||||||
|   const [password, setPassword] = useState(""); |  | ||||||
|  |  | ||||||
|   async function handleLogin(e: FormEvent) { |  | ||||||
|     e.preventDefault() |  | ||||||
|     const timeout = new Promise((r) => setTimeout(r, 1500)); |  | ||||||
|     let user: User; |  | ||||||
|     try { |  | ||||||
|       login({ username, password }) |  | ||||||
|       user = await currentUser() |  | ||||||
|     } catch (e) { await timeout; return } |  | ||||||
|     await timeout; |  | ||||||
|     onLogin(user); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return <div> |  | ||||||
|     <form onSubmit={handleLogin}> |  | ||||||
|       <div> |  | ||||||
|         <input type="text" id="username" name="username" placeholder="username" required value={username} onChange={evt => setUsername(evt.target.value)} /> |  | ||||||
|       </div> |  | ||||||
|       <div> |  | ||||||
|         <input type="password" id="password" name="password" placeholder="password" minLength={8} value={password} required onChange={evt => setPassword(evt.target.value)} /> |  | ||||||
|       </div> |  | ||||||
|       <input className="button" type="submit" value="login" onSubmit={handleLogin} /> |  | ||||||
|     </form> |  | ||||||
|   </div> |  | ||||||
| } */ |  | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								src/MVPChart.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								src/MVPChart.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | import { useEffect, useState } from "react"; | ||||||
|  | import { apiAuth } from "./api"; | ||||||
|  | import BarChart from "./BarChart"; | ||||||
|  | import { PlayerRanking } from "./types"; | ||||||
|  | import RaceChart from "./RaceChart"; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const MVPChart = () => { | ||||||
|  |   const [data, setData] = useState({} as PlayerRanking[]); | ||||||
|  |   const [loading, setLoading] = useState(true); | ||||||
|  |   const [showStd, setShowStd] = useState(false); | ||||||
|  |  | ||||||
|  |   async function loadData() { | ||||||
|  |     setLoading(true); | ||||||
|  |     await apiAuth("analysis/mvp", null) | ||||||
|  |       .then(json => json as Promise<PlayerRanking[]>).then(json => { setData(json.sort((a, b) => a.rank - b.rank)) }) | ||||||
|  |     setLoading(false); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   useEffect(() => { loadData() }, []) | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <> | ||||||
|  |       {loading ? <span className="loader" /> : <RaceChart std={showStd} players={data} /> | ||||||
|  |       } | ||||||
|  |     </>) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export default MVPChart; | ||||||
							
								
								
									
										254
									
								
								src/Network.tsx
									
									
									
									
									
								
							
							
						
						
									
										254
									
								
								src/Network.tsx
									
									
									
									
									
								
							| @@ -1,46 +1,276 @@ | |||||||
| import { useEffect, useRef, useState } from "react"; | import { useEffect, useRef, useState } from "react"; | ||||||
| import { apiAuth } from "./api"; | import { apiAuth } from "./api"; | ||||||
| import { GraphCanvas, GraphCanvasRef, GraphEdge, GraphNode, useSelection } from "reagraph"; | import { | ||||||
|  |   GraphCanvas, | ||||||
|  |   GraphCanvasRef, | ||||||
|  |   GraphEdge, | ||||||
|  |   GraphNode, | ||||||
|  |   SelectionProps, | ||||||
|  |   SelectionResult, | ||||||
|  |   useSelection, | ||||||
|  | } from "reagraph"; | ||||||
|  | import { customTheme } from "./NetworkTheme"; | ||||||
|  |  | ||||||
| interface NetworkData { | interface NetworkData { | ||||||
|   nodes: GraphNode[], |   nodes: GraphNode[]; | ||||||
|   edges: GraphEdge[], |   edges: GraphEdge[]; | ||||||
| } | } | ||||||
|  | interface CustomSelectionProps extends SelectionProps { | ||||||
|  |   ignore: (GraphEdge | undefined)[]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const useCustomSelection = (props: CustomSelectionProps): SelectionResult => { | ||||||
|  |   var result = useSelection(props); | ||||||
|  |   result.actives = result.actives.filter( | ||||||
|  |     (s) => !props.ignore.map((edge) => edge?.id).includes(s) | ||||||
|  |   ); | ||||||
|  |   const ignored_nodes = props.ignore.map((edge) => | ||||||
|  |     edge && | ||||||
|  |     result.selections?.includes(edge.source) && | ||||||
|  |     !result.selections?.includes(edge.target) | ||||||
|  |       ? edge.target | ||||||
|  |       : "" | ||||||
|  |   ); | ||||||
|  |   result.actives = result.actives.filter((s) => !ignored_nodes.includes(s)); | ||||||
|  |   return result; | ||||||
|  | }; | ||||||
|  |  | ||||||
| export const GraphComponent = () => { | export const GraphComponent = () => { | ||||||
|   const [data, setData] = useState({ nodes: [], edges: [] } as NetworkData); |   const [data, setData] = useState({ nodes: [], edges: [] } as NetworkData); | ||||||
|   const [loading, setLoading] = useState(true); |   const [loading, setLoading] = useState(true); | ||||||
|  |   const [threed, setThreed] = useState(false); | ||||||
|  |   const [likes, setLikes] = useState(2); | ||||||
|  |   const [popularity, setPopularity] = useState(false); | ||||||
|  |   const [mutuality, setMutuality] = useState(false); | ||||||
|  |  | ||||||
|   async function loadData() { |   async function loadData() { | ||||||
|     setLoading(true); |     setLoading(true); | ||||||
|     await apiAuth("analysis/graph_json", null) |     await apiAuth("analysis/graph_json", null) | ||||||
|       .then(json => json as Promise<NetworkData>).then(json => { setData(json) }) |       .then((json) => json as Promise<NetworkData>) | ||||||
|  |       .then((json) => { | ||||||
|  |         setData(json); | ||||||
|  |       }); | ||||||
|     setLoading(false); |     setLoading(false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   useEffect(() => { loadData() }, []) |   useEffect(() => { | ||||||
|  |     loadData(); | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|   const graphRef = useRef<GraphCanvasRef | null>(null); |   const graphRef = useRef<GraphCanvasRef | null>(null); | ||||||
|  |  | ||||||
|   const { selections, actives, onNodeClick, onCanvasClick } = useSelection({ |   function handleThreed() { | ||||||
|  |     setThreed(!threed); | ||||||
|  |     graphRef.current?.fitNodesInView(); | ||||||
|  |     graphRef.current?.centerGraph(); | ||||||
|  |     graphRef.current?.resetControls(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function handlePopularity() { | ||||||
|  |     setPopularity(!popularity); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function handleMutuality() { | ||||||
|  |     colorMatches(!mutuality); | ||||||
|  |     setMutuality(!mutuality); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function showLabel() { | ||||||
|  |     switch (likes) { | ||||||
|  |       case 0: | ||||||
|  |         return "dislike"; | ||||||
|  |       case 1: | ||||||
|  |         return "both"; | ||||||
|  |       case 2: | ||||||
|  |         return "like"; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function findMatches(edges: GraphEdge[]) { | ||||||
|  |     const adjacencyList = edges.map( | ||||||
|  |       (edge) => edge.source + edge.target + edge.data.relation | ||||||
|  |     ); | ||||||
|  |     return edges.filter((edge) => | ||||||
|  |       adjacencyList.includes(edge.target + edge.source + edge.data.relation) | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   //const matches = useMemo(() => findMatches(data.edges), []) | ||||||
|  |  | ||||||
|  |   function colorMatches(mutuality: boolean) { | ||||||
|  |     const matches = findMatches(data.edges); | ||||||
|  |     const newEdges = data.edges; | ||||||
|  |     if (mutuality) { | ||||||
|  |       newEdges.forEach((edge) => { | ||||||
|  |         if ( | ||||||
|  |           (likes === 1 || edge.data.relation === likes) && | ||||||
|  |           matches.map((edge) => edge.id).includes(edge.id) | ||||||
|  |         ) { | ||||||
|  |           edge.fill = "#9c3"; | ||||||
|  |           if (edge.size) edge.size = edge.size * 1.5; | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } else { | ||||||
|  |       newEdges.forEach((edge) => { | ||||||
|  |         if ( | ||||||
|  |           (likes === 1 || edge.data.relation === likes) && | ||||||
|  |           matches.map((edge) => edge.id).includes(edge.id) | ||||||
|  |         ) { | ||||||
|  |           edge.fill = edge.data.origFill; | ||||||
|  |           edge.size = edge.data.origSize; | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |     setData({ nodes: data.nodes, edges: newEdges }); | ||||||
|  |   } | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (mutuality) colorMatches(false); | ||||||
|  |     colorMatches(mutuality); | ||||||
|  |   }, [likes]); | ||||||
|  |  | ||||||
|  |   const { | ||||||
|  |     selections, | ||||||
|  |     actives, | ||||||
|  |     onNodeClick, | ||||||
|  |     onCanvasClick, | ||||||
|  |     onNodePointerOver, | ||||||
|  |     onNodePointerOut, | ||||||
|  |   } = useCustomSelection({ | ||||||
|     ref: graphRef, |     ref: graphRef, | ||||||
|     nodes: data.nodes, |     nodes: data.nodes, | ||||||
|     edges: data.edges, |     edges: data.edges.filter((edge) => edge.data.relation === likes), | ||||||
|     pathSelectionType: 'out' |     ignore: data.edges.map((edge) => { | ||||||
|  |       if (likes === 1 && edge.data.relation !== 2) return edge; | ||||||
|  |     }), | ||||||
|  |     pathSelectionType: "out", | ||||||
|  |     pathHoverType: "in", | ||||||
|  |     type: "multiModifier", | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|  |     <div style={{ position: "absolute", top: 0, bottom: 0, left: 0, right: 0 }}> | ||||||
|  |       <div className="controls"> | ||||||
|  |         <div className="control" onClick={handleMutuality}> | ||||||
|  |           <div className="switch"> | ||||||
|  |             <input type="checkbox" checked={mutuality} onChange={() => {}} /> | ||||||
|  |             <span className="slider round"></span> | ||||||
|  |           </div> | ||||||
|  |           <span>mutuality</span> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <div className="control" onClick={handleThreed}> | ||||||
|  |           <span>2D</span> | ||||||
|  |           <div className="switch"> | ||||||
|  |             <input type="checkbox" checked={threed} onChange={() => {}} /> | ||||||
|  |             <span className="slider round"></span> | ||||||
|  |           </div> | ||||||
|  |           <span>3D</span> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <div className="control"> | ||||||
|  |           <div className="stack column"> | ||||||
|  |             <datalist id="markers"> | ||||||
|  |               <option value="0"></option> | ||||||
|  |               <option value="1"></option> | ||||||
|  |               <option value="2"></option> | ||||||
|  |             </datalist> | ||||||
|  |             <div id="three-slider"> | ||||||
|  |               <label>😬</label> | ||||||
|  |               <input | ||||||
|  |                 type="range" | ||||||
|  |                 list="markers" | ||||||
|  |                 min="0" | ||||||
|  |                 max="2" | ||||||
|  |                 step="1" | ||||||
|  |                 width="16px" | ||||||
|  |                 onChange={(evt) => setLikes(Number(evt.target.value))} | ||||||
|  |               /> | ||||||
|  |               <label>😍</label> | ||||||
|  |             </div> | ||||||
|  |             {showLabel()} | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <div className="control" onClick={handlePopularity}> | ||||||
|  |           <div className="switch"> | ||||||
|  |             <input type="checkbox" checked={popularity} onChange={() => {}} /> | ||||||
|  |             <span className="slider round"></span> | ||||||
|  |           </div> | ||||||
|  |           <span> | ||||||
|  |             popularity<sup>*</sup> | ||||||
|  |           </span> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       {popularity && ( | ||||||
|  |         <div | ||||||
|  |           style={{ position: "absolute", bottom: 0, right: "10px", zIndex: 10 }} | ||||||
|  |         > | ||||||
|  |           <span className="grey" style={{ fontSize: "70%" }}> | ||||||
|  |             <sup>*</sup>popularity meassured by rank-weighted in-degree | ||||||
|  |           </span> | ||||||
|  |         </div> | ||||||
|  |       )} | ||||||
|  |  | ||||||
|  |       {loading ? ( | ||||||
|  |         <span className="loader" /> | ||||||
|  |       ) : ( | ||||||
|         <GraphCanvas |         <GraphCanvas | ||||||
|           draggable |           draggable | ||||||
|  |           cameraMode={threed ? "rotate" : "pan"} | ||||||
|  |           layoutType={threed ? "forceDirected3d" : "forceDirected2d"} | ||||||
|  |           layoutOverrides={{ | ||||||
|  |             nodeStrength: -200, | ||||||
|  |             linkDistance: 100, | ||||||
|  |           }} | ||||||
|  |           labelType="nodes" | ||||||
|  |           sizingType="attribute" | ||||||
|  |           sizingAttribute={popularity ? "inDegree" : undefined} | ||||||
|           ref={graphRef} |           ref={graphRef} | ||||||
|  |           theme={customTheme} | ||||||
|           nodes={data.nodes} |           nodes={data.nodes} | ||||||
|       edges={data.edges} |           edges={data.edges.filter( | ||||||
|  |             (edge) => edge.data.relation === likes || likes === 1 | ||||||
|  |           )} | ||||||
|           selections={selections} |           selections={selections} | ||||||
|           actives={actives} |           actives={actives} | ||||||
|           onCanvasClick={onCanvasClick} |           onCanvasClick={onCanvasClick} | ||||||
|           onNodeClick={onNodeClick} |           onNodeClick={onNodeClick} | ||||||
|  |           onNodePointerOut={onNodePointerOut} | ||||||
|  |           onNodePointerOver={onNodePointerOver} | ||||||
|         /> |         /> | ||||||
|  |       )} | ||||||
|  |       <button | ||||||
|  |         className="infobutton" | ||||||
|  |         onClick={() => { | ||||||
|  |           const dialog = document.querySelector("dialog[id='InfoDialog']"); | ||||||
|  |           (dialog as HTMLDialogElement).showModal(); | ||||||
|  |         }} | ||||||
|  |       > | ||||||
|  |         info | ||||||
|  |       </button> | ||||||
|  |  | ||||||
|  |       <dialog | ||||||
|  |         id="InfoDialog" | ||||||
|  |         style={{ textAlign: "left" }} | ||||||
|  |         onClick={(event) => { | ||||||
|  |           event.currentTarget.close(); | ||||||
|  |         }} | ||||||
|  |       > | ||||||
|  |         scroll to zoom | ||||||
|  |         <br /> | ||||||
|  |         <br /> | ||||||
|  |         <b>hover</b>: show inbound links | ||||||
|  |         <br /> | ||||||
|  |         <b>click</b>: show outward links | ||||||
|  |         <br /> | ||||||
|  |         <br /> | ||||||
|  |         multi-selection possible | ||||||
|  |         <br /> | ||||||
|  |         with <i>Ctrl</i> or <i>Shift</i> | ||||||
|  |         <br /> | ||||||
|  |         <br /> | ||||||
|  |         drag to pan/rotate | ||||||
|  |       </dialog> | ||||||
|  |     </div> | ||||||
|   ); |   ); | ||||||
|  | }; | ||||||
| } |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										59
									
								
								src/NetworkTheme.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								src/NetworkTheme.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | import { Theme } from "reagraph"; | ||||||
|  |  | ||||||
|  | export const customTheme: Theme = { | ||||||
|  |   canvas: { | ||||||
|  |     background: 'aliceblue', | ||||||
|  |   }, | ||||||
|  |   node: { | ||||||
|  |     fill: '#69F', | ||||||
|  |     activeFill: '#36C', | ||||||
|  |     opacity: 1, | ||||||
|  |     selectedOpacity: 1, | ||||||
|  |     inactiveOpacity: 0.333, | ||||||
|  |     label: { | ||||||
|  |       color: '#404040', | ||||||
|  |       stroke: 'white', | ||||||
|  |       activeColor: 'black' | ||||||
|  |     }, | ||||||
|  |     subLabel: { | ||||||
|  |       color: '#ddd', | ||||||
|  |       stroke: 'transparent', | ||||||
|  |       activeColor: '#1DE9AC' | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   lasso: { | ||||||
|  |     border: '1px solid #55aaff', | ||||||
|  |     background: 'rgba(75, 160, 255, 0.1)' | ||||||
|  |   }, | ||||||
|  |   ring: { | ||||||
|  |     fill: '#69F', | ||||||
|  |     activeFill: '#36C' | ||||||
|  |   }, | ||||||
|  |   edge: { | ||||||
|  |     fill: '#bed4ff', | ||||||
|  |     activeFill: '#36C', | ||||||
|  |     opacity: 1, | ||||||
|  |     selectedOpacity: 1, | ||||||
|  |     inactiveOpacity: 0.333, | ||||||
|  |     label: { | ||||||
|  |       stroke: '#fff', | ||||||
|  |       color: '#2A6475', | ||||||
|  |       activeColor: '#1DE9AC', | ||||||
|  |       fontSize: 6 | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   arrow: { | ||||||
|  |     fill: '#bed4ff', | ||||||
|  |     activeFill: '#36C' | ||||||
|  |   }, | ||||||
|  |   cluster: { | ||||||
|  |     stroke: '#D8E6EA', | ||||||
|  |     opacity: 1, | ||||||
|  |     selectedOpacity: 1, | ||||||
|  |     inactiveOpacity: 0.1, | ||||||
|  |     label: { | ||||||
|  |       stroke: '#fff', | ||||||
|  |       color: '#2A6475' | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }; | ||||||
							
								
								
									
										99
									
								
								src/RaceChart.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								src/RaceChart.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | |||||||
|  | import { FC, useEffect, useState } from "react"; | ||||||
|  | import { PlayerRanking } from "./types"; | ||||||
|  | import { useSession } from "./Session"; | ||||||
|  |  | ||||||
|  | interface RaceChartProps { | ||||||
|  |   players: PlayerRanking[]; | ||||||
|  |   std: boolean; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const determineNiceWidth = (width: number) => { | ||||||
|  |   const max = 1080; | ||||||
|  |   if (width >= max) return max; | ||||||
|  |   else if (width > 768) return width * 0.8; | ||||||
|  |   else return width * 0.96; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const RaceChart: FC<RaceChartProps> = ({ players, std }) => { | ||||||
|  |   // State to store window's width and height | ||||||
|  |   const [width, setWidth] = useState(determineNiceWidth(window.innerWidth)); | ||||||
|  |   //const [height, setHeight] = useState(window.innerHeight); | ||||||
|  |   const height = players.length * 40; | ||||||
|  |  | ||||||
|  |   // Update state on  resize | ||||||
|  |   useEffect(() => { | ||||||
|  |     const handleResize = () => { | ||||||
|  |       setWidth(determineNiceWidth(window.innerWidth)); | ||||||
|  |       //setHeight(window.innerHeight); | ||||||
|  |     }; | ||||||
|  |     window.addEventListener("resize", handleResize); | ||||||
|  |     return () => { | ||||||
|  |       window.removeEventListener("resize", handleResize); | ||||||
|  |     }; | ||||||
|  |   }, []); | ||||||
|  |   const padding = 24; | ||||||
|  |   const gap = 8; | ||||||
|  |   const maxValue = Math.max(...players.map((player) => player.rank)) + 1; | ||||||
|  |   const barHeight = (height - 2 * padding) / players.length; | ||||||
|  |   const fontSize = Math.min(barHeight - 1.5 * gap, width / 22); | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <svg width={width} height={height} id="RaceChartSVG"> | ||||||
|  |       {players.map((player, index) => ( | ||||||
|  |         <rect | ||||||
|  |           key={String(index)} | ||||||
|  |           x={0} | ||||||
|  |           y={index * barHeight + padding} | ||||||
|  |           width={(1 - player.rank / maxValue) * width} | ||||||
|  |           height={barHeight - gap} // subtract 2 for some spacing between bars | ||||||
|  |           fill="#36c" | ||||||
|  |         /> | ||||||
|  |       ))} | ||||||
|  |  | ||||||
|  |       {players.map((player, index) => ( | ||||||
|  |         <g key={"group" + index}> | ||||||
|  |           <text | ||||||
|  |             key={index + "_name"} | ||||||
|  |             x={4} | ||||||
|  |             y={index * barHeight + barHeight / 2 + padding + gap / 2} | ||||||
|  |             width={(1 - player.rank / maxValue) * width} | ||||||
|  |             height={barHeight - 8} // subtract 2 for some spacing between bars | ||||||
|  |             fontSize={fontSize} | ||||||
|  |             fill="aliceblue" | ||||||
|  |             stroke="#36c" | ||||||
|  |             strokeWidth={4} | ||||||
|  |             fontWeight={"bold"} | ||||||
|  |             paintOrder={"stroke fill"} | ||||||
|  |             fontFamily="monospace" | ||||||
|  |             style={{ whiteSpace: "pre" }} | ||||||
|  |           > | ||||||
|  |             {`${String(index + 1).padStart(2)}. ${player.name}`} | ||||||
|  |           </text> | ||||||
|  |           <text | ||||||
|  |             key={index + "_value"} | ||||||
|  |             x={ | ||||||
|  |               4 + | ||||||
|  |               (4 + Math.max(...players.map((p, _) => p.name.length))) * | ||||||
|  |                 fontSize * | ||||||
|  |                 0.66 | ||||||
|  |             } | ||||||
|  |             y={index * barHeight + barHeight / 2 + padding + gap / 2} | ||||||
|  |             width={(1 - player.rank / maxValue) * width} | ||||||
|  |             height={barHeight - 8} // subtract 2 for some spacing between bars | ||||||
|  |             fontSize={0.8 * fontSize} | ||||||
|  |             fill="aliceblue" | ||||||
|  |             stroke="#36c" | ||||||
|  |             fontWeight={"bold"} | ||||||
|  |             fontFamily="monospace" | ||||||
|  |             strokeWidth={4} | ||||||
|  |             paintOrder={"stroke fill"} | ||||||
|  |             style={{ whiteSpace: "pre" }} | ||||||
|  |           > | ||||||
|  |             {`${String(player.rank).padStart(5)} ± ${player.std}   N = ${player.n}`} | ||||||
|  |           </text> | ||||||
|  |         </g> | ||||||
|  |       ))} | ||||||
|  |     </svg> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  | export default RaceChart; | ||||||
							
								
								
									
										173
									
								
								src/Rankings.tsx
									
									
									
									
									
								
							
							
						
						
									
										173
									
								
								src/Rankings.tsx
									
									
									
									
									
								
							| @@ -1,12 +1,7 @@ | |||||||
| import { Dispatch, SetStateAction, useEffect, useMemo, useState } from "react"; | import { useEffect, useState } from "react"; | ||||||
| import { ReactSortable, ReactSortableProps } from "react-sortablejs"; | import { ReactSortable, ReactSortableProps } from "react-sortablejs"; | ||||||
| import api, { baseUrl } from "./api"; | import { apiAuth, User } from "./api"; | ||||||
|  | import { useSession } from "./Session"; | ||||||
| interface Player { |  | ||||||
|   id: number; |  | ||||||
|   name: string; |  | ||||||
|   number: string | null; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type PlayerListProps = Partial<ReactSortableProps<any>> & { | type PlayerListProps = Partial<ReactSortableProps<any>> & { | ||||||
|   orderedList?: boolean; |   orderedList?: boolean; | ||||||
| @@ -17,117 +12,45 @@ function PlayerList(props: PlayerListProps) { | |||||||
|     <ReactSortable {...props} animation={200}> |     <ReactSortable {...props} animation={200}> | ||||||
|       {props.list?.map((item, index) => ( |       {props.list?.map((item, index) => ( | ||||||
|         <div key={item.id} className="item"> |         <div key={item.id} className="item"> | ||||||
|           {props.orderedList ? index + 1 + ". " + item.name : item.name} |           {props.orderedList | ||||||
|  |             ? index + 1 + ". " + item.display_name | ||||||
|  |             : item.display_name} | ||||||
|         </div> |         </div> | ||||||
|       ))} |       ))} | ||||||
|     </ReactSortable> |     </ReactSortable> | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| interface SelectUserProps { |  | ||||||
|   user: Player[]; |  | ||||||
|   setUser: Dispatch<SetStateAction<Player[]>>; |  | ||||||
|   players: Player[]; |  | ||||||
|   setPlayers: Dispatch<SetStateAction<Player[]>>; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function SelectUser({ |  | ||||||
|   user, |  | ||||||
|   setUser, |  | ||||||
|   players, |  | ||||||
|   setPlayers, |  | ||||||
| }: SelectUserProps) { |  | ||||||
|   return ( |  | ||||||
|     <> |  | ||||||
|       <div className="box user"> |  | ||||||
|         {user.length < 1 ? ( |  | ||||||
|           <> |  | ||||||
|             <span>your name?</span> |  | ||||||
|             <br /> <span className="grey hint">drag your name here</span> |  | ||||||
|           </> |  | ||||||
|         ) : ( |  | ||||||
|           <> |  | ||||||
|             <span |  | ||||||
|               className="renew" |  | ||||||
|               onClick={() => { |  | ||||||
|                 setUser([]); |  | ||||||
|               }} |  | ||||||
|             > |  | ||||||
|               {" ✕"} |  | ||||||
|             </span> |  | ||||||
|           </> |  | ||||||
|         )} |  | ||||||
|         <PlayerList |  | ||||||
|           list={user} |  | ||||||
|           setList={setUser} |  | ||||||
|           group={{ |  | ||||||
|             name: "user-shared", |  | ||||||
|             put: function (to) { |  | ||||||
|               return to.el.children.length < 1; |  | ||||||
|             }, |  | ||||||
|           }} |  | ||||||
|           className="dragbox" |  | ||||||
|         /> |  | ||||||
|       </div> |  | ||||||
|       {user.length < 1 && ( |  | ||||||
|         <div className="box one"> |  | ||||||
|           <h2>🥏🏃</h2> |  | ||||||
|           <ReactSortable |  | ||||||
|             list={players} |  | ||||||
|             setList={setPlayers} |  | ||||||
|             group={{ name: "user-shared", pull: "clone" }} |  | ||||||
|             className="dragbox reservoir" |  | ||||||
|             animation={200} |  | ||||||
|           > |  | ||||||
|             {players.length < 1 ? ( |  | ||||||
|               <span className="loader"></span> |  | ||||||
|             ) : ( |  | ||||||
|               players.map((item, _) => ( |  | ||||||
|                 <div key={"extra" + item.id} className="extra-margin"> |  | ||||||
|                   <div key={item.id} className="item"> |  | ||||||
|                     {item.name} |  | ||||||
|                   </div> |  | ||||||
|                 </div> |  | ||||||
|               )) |  | ||||||
|             )} |  | ||||||
|           </ReactSortable> |  | ||||||
|         </div> |  | ||||||
|       )} |  | ||||||
|     </> |  | ||||||
|   ); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| interface PlayerInfoProps { | interface PlayerInfoProps { | ||||||
|   user: Player[]; |   user: User; | ||||||
|   players: Player[]; |   players: User[]; | ||||||
| } | } | ||||||
|  |  | ||||||
| export function Chemistry({ user, players }: PlayerInfoProps) { | export function Chemistry({ user, players }: PlayerInfoProps) { | ||||||
|   const index = players.indexOf(user[0]); |   const index = players.indexOf(user); | ||||||
|   var otherPlayers = players.slice(); |   var otherPlayers = players.slice(); | ||||||
|   otherPlayers.splice(index, 1); |   otherPlayers.splice(index, 1); | ||||||
|   const [playersLeft, setPlayersLeft] = useState<Player[]>([]); |   const [playersLeft, setPlayersLeft] = useState<User[]>([]); | ||||||
|   const [playersMiddle, setPlayersMiddle] = useState<Player[]>(otherPlayers); |   const [playersMiddle, setPlayersMiddle] = useState<User[]>(otherPlayers); | ||||||
|   const [playersRight, setPlayersRight] = useState<Player[]>([]); |   const [playersRight, setPlayersRight] = useState<User[]>([]); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     setPlayersMiddle(otherPlayers); | ||||||
|  |   }, [players]); | ||||||
|  |  | ||||||
|   const [dialog, setDialog] = useState("dialog"); |   const [dialog, setDialog] = useState("dialog"); | ||||||
|  |  | ||||||
|   async function handleSubmit() { |   async function handleSubmit() { | ||||||
|     const dialog = document.querySelector("dialog[id='ChemistryDialog']"); |     const dialog = document.querySelector("dialog[id='ChemistryDialog']"); | ||||||
|     (dialog as HTMLDialogElement).showModal(); |     (dialog as HTMLDialogElement).showModal(); | ||||||
|     if (user.length < 1) { |  | ||||||
|       setDialog("who are you?"); |  | ||||||
|     } else { |  | ||||||
|     setDialog("sending..."); |     setDialog("sending..."); | ||||||
|       let _user = user.map(({ name }) => name)[0]; |     let left = playersLeft.map(({ display_name }) => display_name); | ||||||
|       let left = playersLeft.map(({ name }) => name); |     let middle = playersMiddle.map(({ display_name }) => display_name); | ||||||
|       let middle = playersMiddle.map(({ name }) => name); |     let right = playersRight.map(({ display_name }) => display_name); | ||||||
|       let right = playersRight.map(({ name }) => name); |     const data = { user: user, hate: left, undecided: middle, love: right }; | ||||||
|       const data = { user: _user, hate: left, undecided: middle, love: right }; |     const response = await apiAuth("chemistry", data); | ||||||
|       const response = await api("chemistry", data); |  | ||||||
|     response.ok ? setDialog("success!") : setDialog("try sending again"); |     response.ok ? setDialog("success!") : setDialog("try sending again"); | ||||||
|   } |   } | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
| @@ -188,25 +111,24 @@ export function Chemistry({ user, players }: PlayerInfoProps) { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function MVP({ user, players }: PlayerInfoProps) { | export function MVP({ user, players }: PlayerInfoProps) { | ||||||
|   const [availablePlayers, setAvailablePlayers] = useState<Player[]>(players); |   const [availablePlayers, setAvailablePlayers] = useState<User[]>(players); | ||||||
|   const [rankedPlayers, setRankedPlayers] = useState<Player[]>([]); |   const [rankedPlayers, setRankedPlayers] = useState<User[]>([]); | ||||||
|  |  | ||||||
|   const [dialog, setDialog] = useState("dialog"); |   const [dialog, setDialog] = useState("dialog"); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     setAvailablePlayers(players); | ||||||
|  |   }, [players]); | ||||||
|  |  | ||||||
|   async function handleSubmit() { |   async function handleSubmit() { | ||||||
|     const dialog = document.querySelector("dialog[id='MVPDialog']"); |     const dialog = document.querySelector("dialog[id='MVPDialog']"); | ||||||
|     (dialog as HTMLDialogElement).showModal(); |     (dialog as HTMLDialogElement).showModal(); | ||||||
|     if (user.length < 1) { |  | ||||||
|       setDialog("who are you?"); |  | ||||||
|     } else { |  | ||||||
|     setDialog("sending..."); |     setDialog("sending..."); | ||||||
|       let _user = user.map(({ name }) => name)[0]; |     let mvps = rankedPlayers.map(({ display_name }) => display_name); | ||||||
|       let mvps = rankedPlayers.map(({ name }) => name); |     const data = { user: user, mvps: mvps }; | ||||||
|       const data = { user: _user, mvps: mvps }; |     const response = await apiAuth("mvps", data); | ||||||
|       const response = await api("mvps", data); |  | ||||||
|     response.ok ? setDialog("success!") : setDialog("try sending again"); |     response.ok ? setDialog("success!") : setDialog("try sending again"); | ||||||
|   } |   } | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
| @@ -267,24 +189,25 @@ export function MVP({ user, players }: PlayerInfoProps) { | |||||||
| } | } | ||||||
|  |  | ||||||
| export default function Rankings() { | export default function Rankings() { | ||||||
|   const [user, setUser] = useState<Player[]>([]); |   const { user } = useSession(); | ||||||
|   const [players, setPlayers] = useState<Player[]>([]); |   const [players, setPlayers] = useState<User[]>([]); | ||||||
|   const [openTab, setOpenTab] = useState("Chemistry"); |   const [openTab, setOpenTab] = useState("Chemistry"); | ||||||
|  |  | ||||||
|   async function loadPlayers() { |   async function loadPlayers() { | ||||||
|     const response = await fetch(`${baseUrl}api/player/list`, { |     try { | ||||||
|       method: "GET", |       const data = await apiAuth("player/list", null, "GET"); | ||||||
|     }); |       setPlayers(data as User[]); | ||||||
|     const data = await response.json(); |     } catch (error) { | ||||||
|     setPlayers(data as Player[]); |       console.error(error); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   useMemo(() => { |   useEffect(() => { | ||||||
|     loadPlayers(); |     loadPlayers(); | ||||||
|   }, []); |   }, []); | ||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     user.length === 1 && openPage(openTab, "aliceblue"); |     openPage(openTab, "aliceblue"); | ||||||
|   }, [user]); |   }, [user]); | ||||||
|  |  | ||||||
|   function openPage(pageName: string, color: string) { |   function openPage(pageName: string, color: string) { | ||||||
| @@ -313,9 +236,6 @@ export default function Rankings() { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <> |  | ||||||
|       <SelectUser {...{ user, setUser, players, setPlayers }} /> |  | ||||||
|       {user.length === 1 && ( |  | ||||||
|     <> |     <> | ||||||
|       <div className="container navbar"> |       <div className="container navbar"> | ||||||
|         <button |         <button | ||||||
| @@ -334,17 +254,18 @@ export default function Rankings() { | |||||||
|         </button> |         </button> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
|           <span className="grey">assign as many or as few players as you want<br /> |       <span className="grey"> | ||||||
|             and don't forget to <b>submit</b> (💾) when you're done :)</span> |         assign as many or as few players as you want | ||||||
|  |         <br /> | ||||||
|  |         and don't forget to <b>submit</b> (💾) when you're done :) | ||||||
|  |       </span> | ||||||
|  |  | ||||||
|       <div id="Chemistry" className="tabcontent"> |       <div id="Chemistry" className="tabcontent"> | ||||||
|             <Chemistry {...{ user, players }} /> |         {user && <Chemistry {...{ user, players }} />} | ||||||
|       </div> |       </div> | ||||||
|       <div id="MVP" className="tabcontent"> |       <div id="MVP" className="tabcontent"> | ||||||
|             <MVP {...{ user, players }} /> |         {user && <MVP {...{ user, players }} />} | ||||||
|       </div> |       </div> | ||||||
|     </> |     </> | ||||||
|       )} |  | ||||||
|     </> |  | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,36 +1,87 @@ | |||||||
| import { createContext, ReactNode, useContext, useLayoutEffect, useState } from "react"; | import { | ||||||
| import { currentUser, User } from "./api"; |   createContext, | ||||||
|  |   ReactNode, | ||||||
|  |   useContext, | ||||||
|  |   useEffect, | ||||||
|  |   useState, | ||||||
|  | } from "react"; | ||||||
|  | import { currentUser, logout, User } from "./api"; | ||||||
| import { Login } from "./Login"; | import { Login } from "./Login"; | ||||||
|  | import Header from "./Header"; | ||||||
|  |  | ||||||
| export interface SessionProviderProps { | export interface SessionProviderProps { | ||||||
|   children: ReactNode; |   children: ReactNode; | ||||||
| } | } | ||||||
|  |  | ||||||
| const sessionContext = createContext<User | null>(null); | export interface Session { | ||||||
|  |   user: User | null; | ||||||
|  |   onLogout: () => void; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const sessionContext = createContext<Session>({ | ||||||
|  |   user: null, | ||||||
|  |   onLogout: () => {}, | ||||||
|  | }); | ||||||
|  |  | ||||||
| export function SessionProvider(props: SessionProviderProps) { | export function SessionProvider(props: SessionProviderProps) { | ||||||
|   const { children } = props; |   const { children } = props; | ||||||
|  |  | ||||||
|   const [user, setUser] = useState<User | null>(null); |   const [user, setUser] = useState<User | null>(null); | ||||||
|   const [err, setErr] = useState<unknown>(null); |   const [err, setErr] = useState<unknown>(null); | ||||||
|  |   const [loading, setLoading] = useState(false); | ||||||
|  |  | ||||||
|   function loadUser() { |   function loadUser() { | ||||||
|  |     setLoading(true); | ||||||
|     currentUser() |     currentUser() | ||||||
|       .then((user) => { setUser(user); setErr(null); }) |       .then((user) => { | ||||||
|       .catch((err) => { setUser(null); setErr(err); }); |         setUser(user); | ||||||
|  |         setErr(null); | ||||||
|  |       }) | ||||||
|  |       .catch((err) => { | ||||||
|  |         setUser(null); | ||||||
|  |         setErr(err); | ||||||
|  |       }) | ||||||
|  |       .finally(() => setLoading(false)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   useLayoutEffect(() => { loadUser(); }, [err]); |   useEffect(() => { | ||||||
|  |     loadUser(); | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|   function onLogin(user: User) { |   function onLogin(user: User) { | ||||||
|     setUser(user); |     setUser(user); | ||||||
|     setErr(null); |     setErr(null); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   async function onLogout() { | ||||||
|  |     try { | ||||||
|  |       logout(); | ||||||
|  |       setUser(null); | ||||||
|  |       setErr({ message: "Logged out successfully" }); | ||||||
|  |       console.log("logged out."); | ||||||
|  |     } catch (e) { | ||||||
|  |       console.error(e); | ||||||
|  |       setErr(e); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   console.log("sanity", user); | ||||||
|  |  | ||||||
|   let content: ReactNode; |   let content: ReactNode; | ||||||
|   if (!err && !user) content = <span className="loader" />; |   if (loading || (!err && !user)) | ||||||
|   else if (err) content = <Login onLogin={onLogin} />; |     content = ( | ||||||
|   else content = <sessionContext.Provider value={user}>{children}</sessionContext.Provider>; |       <> | ||||||
|  |         <Header /> | ||||||
|  |         <span className="loader" /> | ||||||
|  |       </> | ||||||
|  |     ); | ||||||
|  |   else if (err) { | ||||||
|  |     content = <Login onLogin={onLogin} />; | ||||||
|  |   } else | ||||||
|  |     content = ( | ||||||
|  |       <sessionContext.Provider value={{ user, onLogout }}> | ||||||
|  |         {children} | ||||||
|  |       </sessionContext.Provider> | ||||||
|  |     ); | ||||||
|  |  | ||||||
|   return content; |   return content; | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										126
									
								
								src/SetPassword.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										126
									
								
								src/SetPassword.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,126 @@ | |||||||
|  | import { InvalidTokenError, jwtDecode, JwtPayload } from "jwt-decode"; | ||||||
|  | import { useEffect, useState } from "react"; | ||||||
|  | import { baseUrl } from "./api"; | ||||||
|  | import { redirect, useNavigate } from "react-router"; | ||||||
|  |  | ||||||
|  | interface SetPassToken extends JwtPayload { | ||||||
|  |   name: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const SetPassword = () => { | ||||||
|  |   const [name, setName] = useState("after getting your token."); | ||||||
|  |   const [username, setUsername] = useState(""); | ||||||
|  |   const [password, setPassword] = useState(""); | ||||||
|  |   const [passwordr, setPasswordr] = useState(""); | ||||||
|  |   const [token, setToken] = useState(""); | ||||||
|  |   const [error, setError] = useState(""); | ||||||
|  |   const [loading, setLoading] = useState(false); | ||||||
|  |   const navigate = useNavigate(); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     const params = new URLSearchParams(window.location.search); | ||||||
|  |     const token = params.get("token"); | ||||||
|  |     if (token) { | ||||||
|  |       setToken(token); | ||||||
|  |       try { | ||||||
|  |         const payload = jwtDecode<SetPassToken>(token); | ||||||
|  |         if (payload.name) setName(payload.name); | ||||||
|  |         else if (payload.sub) setName(payload.sub); | ||||||
|  |         else setName("Mr. I-have-no Token"); | ||||||
|  |         payload.sub && setUsername(payload.sub); | ||||||
|  |       } catch (InvalidTokenError) { | ||||||
|  |         setName("Mr. I-have-no-valid Token"); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|  |   async function handleSubmit(e: React.FormEvent) { | ||||||
|  |     e.preventDefault(); | ||||||
|  |     if (password === passwordr) { | ||||||
|  |       setLoading(true); | ||||||
|  |       const req = new Request(`${baseUrl}api/set_password`, { | ||||||
|  |         method: "POST", | ||||||
|  |         headers: { | ||||||
|  |           "Content-Type": "application/json", | ||||||
|  |         }, | ||||||
|  |         body: JSON.stringify({ token: token, password: password }), | ||||||
|  |       }); | ||||||
|  |       let resp: Response; | ||||||
|  |       try { | ||||||
|  |         resp = await fetch(req); | ||||||
|  |       } catch (e) { | ||||||
|  |         throw new Error(`request failed: ${e}`); | ||||||
|  |       } | ||||||
|  |       setLoading(false); | ||||||
|  |  | ||||||
|  |       if (resp.ok) { | ||||||
|  |         console.log(resp); | ||||||
|  |         navigate("/", { | ||||||
|  |           replace: true, | ||||||
|  |           state: { username: username, password: password }, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (!resp.ok) { | ||||||
|  |         if (resp.status === 401) { | ||||||
|  |           resp.statusText | ||||||
|  |             ? setError(resp.statusText) | ||||||
|  |             : setError("unauthorized"); | ||||||
|  |           throw new Error("Unauthorized"); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } else setError("passwords are not the same"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <> | ||||||
|  |       <h2> | ||||||
|  |         set your password, | ||||||
|  |         <br /> | ||||||
|  |         {name} | ||||||
|  |       </h2> | ||||||
|  |       {username && ( | ||||||
|  |         <span> | ||||||
|  |           your username is: <i>{username}</i> | ||||||
|  |         </span> | ||||||
|  |       )} | ||||||
|  |       <form onSubmit={handleSubmit}> | ||||||
|  |         <div> | ||||||
|  |           <input | ||||||
|  |             type="password" | ||||||
|  |             id="password" | ||||||
|  |             name="password" | ||||||
|  |             placeholder="password" | ||||||
|  |             minLength={8} | ||||||
|  |             value={password} | ||||||
|  |             required | ||||||
|  |             onChange={(evt) => { | ||||||
|  |               setError(""); | ||||||
|  |               setPassword(evt.target.value); | ||||||
|  |             }} | ||||||
|  |           /> | ||||||
|  |         </div> | ||||||
|  |         <div> | ||||||
|  |           <input | ||||||
|  |             type="password" | ||||||
|  |             id="password-repeat" | ||||||
|  |             name="password-repeat" | ||||||
|  |             placeholder="repeat password" | ||||||
|  |             minLength={8} | ||||||
|  |             value={passwordr} | ||||||
|  |             required | ||||||
|  |             onChange={(evt) => { | ||||||
|  |               setError(""); | ||||||
|  |               setPasswordr(evt.target.value); | ||||||
|  |             }} | ||||||
|  |           /> | ||||||
|  |         </div> | ||||||
|  |         <div>{error && <span style={{ color: "red" }}>{error}</span>}</div> | ||||||
|  |         <button type="submit" value="login" style={{ fontSize: "small" }}> | ||||||
|  |           login | ||||||
|  |         </button> | ||||||
|  |         {loading && <span className="loader" />} | ||||||
|  |       </form> | ||||||
|  |     </> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										108
									
								
								src/api.ts
									
									
									
									
									
								
							
							
						
						
									
										108
									
								
								src/api.ts
									
									
									
									
									
								
							| @@ -1,35 +1,18 @@ | |||||||
| export const baseUrl = import.meta.env.VITE_BASE_URL as string; | export const baseUrl = import.meta.env.VITE_BASE_URL as string; | ||||||
| export const token = () => localStorage.getItem("access_token") as string; |  | ||||||
|  |  | ||||||
| export default async function api(path: string, data: any): Promise<any> { | export async function apiAuth( | ||||||
|   const request = new Request(`${baseUrl}${path}/`, { |   path: string, | ||||||
|     method: "POST", |   data: any, | ||||||
|  |   method: string = "GET" | ||||||
|  | ): Promise<any> { | ||||||
|  |   const req = new Request(`${baseUrl}api/${path}`, { | ||||||
|  |     method: method, | ||||||
|     headers: { |     headers: { | ||||||
|       "Content-Type": "application/json", |       "Content-Type": "application/json", | ||||||
|     }, |     }, | ||||||
|     body: JSON.stringify(data), |     credentials: "include", | ||||||
|  |     ...(data && { body: JSON.stringify(data) }), | ||||||
|   }); |   }); | ||||||
|   let response: Response; |  | ||||||
|   try { |  | ||||||
|     response = await fetch(request); |  | ||||||
|   } catch (e) { |  | ||||||
|     throw new Error(`request failed: ${e}`); |  | ||||||
|   } |  | ||||||
|   return response; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function apiAuth(path: string, data: any, method: string = "GET"): Promise<any> { |  | ||||||
|  |  | ||||||
|   const req = new Request(`${baseUrl}api/${path}`, |  | ||||||
|     { |  | ||||||
|       method: method, |  | ||||||
|       headers: { |  | ||||||
|         "Authorization": `Bearer ${token()} `, |  | ||||||
|         'Content-Type': 'application/json' |  | ||||||
|       }, |  | ||||||
|       ...(data && { body: JSON.stringify(data) }) |  | ||||||
|     } |  | ||||||
|   ); |  | ||||||
|   let resp: Response; |   let resp: Response; | ||||||
|   try { |   try { | ||||||
|     resp = await fetch(req); |     resp = await fetch(req); | ||||||
| @@ -39,25 +22,31 @@ export async function apiAuth(path: string, data: any, method: string = "GET"): | |||||||
|  |  | ||||||
|   if (!resp.ok) { |   if (!resp.ok) { | ||||||
|     if (resp.status === 401) { |     if (resp.status === 401) { | ||||||
|       logout() |       logout(); | ||||||
|       throw new Error('Unauthorized'); |       throw new Error("Unauthorized"); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   return resp.json() |   return resp.json(); | ||||||
| } | } | ||||||
|  |  | ||||||
| export type User = { | export type User = { | ||||||
|  |   id: number; | ||||||
|   username: string; |   username: string; | ||||||
|   fullName: string; |   display_name: string; | ||||||
| } |   full_name: string; | ||||||
|  |   email: string; | ||||||
|  |   player_id: number; | ||||||
|  |   number: string; | ||||||
|  |   scopes: string; | ||||||
|  | }; | ||||||
|  |  | ||||||
| export async function currentUser(): Promise<User> { | export async function currentUser(): Promise<User> { | ||||||
|   if (!token()) throw new Error("you have no access token") |   const req = new Request(`${baseUrl}api/player/me`, { | ||||||
|   const req = new Request(`${baseUrl}api/users/me/`, { |     method: "GET", | ||||||
|     method: "GET", headers: { |     headers: { | ||||||
|       "Authorization": `Bearer ${token()} `, |       "Content-Type": "application/json", | ||||||
|       'Content-Type': 'application/json' |     }, | ||||||
|     } |     credentials: "include", | ||||||
|   }); |   }); | ||||||
|   let resp: Response; |   let resp: Response; | ||||||
|   try { |   try { | ||||||
| @@ -68,8 +57,8 @@ export async function currentUser(): Promise<User> { | |||||||
|  |  | ||||||
|   if (!resp.ok) { |   if (!resp.ok) { | ||||||
|     if (resp.status === 401) { |     if (resp.status === 401) { | ||||||
|       logout() |       logout(); | ||||||
|       throw new Error('Unauthorized'); |       throw new Error("Unauthorized"); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   return resp.json() as Promise<User>; |   return resp.json() as Promise<User>; | ||||||
| @@ -79,18 +68,33 @@ export type LoginRequest = { | |||||||
|   username: string; |   username: string; | ||||||
|   password: string; |   password: string; | ||||||
| }; | }; | ||||||
| export type Token = { |  | ||||||
|   access_token: string; | export const login = async (req: LoginRequest): Promise<void> => { | ||||||
|   token_type: string; |   try { | ||||||
|  |     const response = await fetch(`${baseUrl}api/token`, { | ||||||
|  |       method: "POST", | ||||||
|  |       headers: { | ||||||
|  |         "Content-Type": "application/x-www-form-urlencoded", | ||||||
|  |       }, | ||||||
|  |       body: new URLSearchParams(req).toString(), | ||||||
|  |       credentials: "include", | ||||||
|  |     }); | ||||||
|  |     if (!response.ok) { | ||||||
|  |       throw new Error(`HTTP error! status: ${response.status}`); | ||||||
|  |     } | ||||||
|  |   } catch (e) { | ||||||
|  |     console.error(e); | ||||||
|  |     throw e; // rethrow the error so it can be caught by the caller | ||||||
|  |   } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const login = (req: LoginRequest) => { | export const logout = async () => { | ||||||
|   fetch(`${baseUrl}api/token`, { |   try { | ||||||
|     method: "POST", headers: { |     await fetch(`${baseUrl}api/logout`, { | ||||||
|       'Content-Type': 'application/x-www-form-urlencoded', |       method: "POST", | ||||||
|     }, body: new URLSearchParams(req).toString() |       credentials: "include", | ||||||
|   }).then(resp => resp.json() as Promise<Token>).then(token => token ? localStorage.setItem("access_token", token.access_token) : console.log("token not acquired")).catch((e) => console.log("catch error " + e + " in login")); |     }); | ||||||
|   return Promise<void> |   } catch (e) { | ||||||
| } |     console.error(e); | ||||||
|  |   } | ||||||
| export const logout = () => localStorage.removeItem("access_token"); | }; | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								src/types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								src/types.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | export interface Edge { | ||||||
|  |   from: string; | ||||||
|  |   to: string; | ||||||
|  |   color: string; | ||||||
|  |   relation: "likes" | "dislikes"; | ||||||
|  | } | ||||||
|  | export interface Node { | ||||||
|  |   id: string; | ||||||
|  | } | ||||||
|  | export default interface NetworkData { | ||||||
|  |   nodes: Node[]; | ||||||
|  |   edges: Edge[]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface PlayerRanking { | ||||||
|  |   name: string; | ||||||
|  |   rank: number; | ||||||
|  |   std: number; | ||||||
|  |   n: number; | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user