Compare commits
	
		
			27 Commits
		
	
	
		
			feat/secur
			...
			d61bea3c86
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| d61bea3c86 | |||
| 70a4ece5bc | |||
| 406ea9ffdd | |||
| 104ec70695 | |||
| 9d65c1d1df | |||
| de79970987 | |||
| a52dae5605 | |||
| a46427c6b8 | |||
| fd323db6d0 | |||
| c2d94c0400 | |||
| f94c3402c2 | |||
| 5c21cf1fc3 | |||
| 5cd793b278 | |||
| de8688133f | |||
| d6e5d0334c | |||
| 5fef47f692 | |||
| 978aafc204 | |||
| 47fd9bd859 | |||
| 13bb965b28 | |||
| 5405c3e12f | |||
| 1eab163e10 | |||
| 7c054d6ba3 | |||
| 4a46cd505d | |||
| 1fa91a7228 | |||
| 8e91724462 | |||
| 1a1b44743a | |||
| 827eceed2b | 
							
								
								
									
										93
									
								
								analysis.py
									
									
									
									
									
								
							
							
						
						
									
										93
									
								
								analysis.py
									
									
									
									
									
								
							| @@ -6,8 +6,9 @@ from fastapi.responses import JSONResponse | ||||
| from pydantic import BaseModel, Field | ||||
| from sqlmodel import Session, func, select | ||||
| from sqlmodel.sql.expression import SelectOfScalar | ||||
| from db import Chemistry, Player, engine | ||||
| from db import Chemistry, MVPRanking, Player, engine | ||||
| import networkx as nx | ||||
| import numpy as np | ||||
| import matplotlib | ||||
|  | ||||
| matplotlib.use("agg") | ||||
| @@ -18,16 +19,17 @@ analysis_router = APIRouter(prefix="/analysis") | ||||
|  | ||||
|  | ||||
| C = Chemistry | ||||
| R = MVPRanking | ||||
| P = Player | ||||
|  | ||||
|  | ||||
| def sociogram_json(): | ||||
|     nodes = [] | ||||
|     necessary_nodes = set() | ||||
|     links = [] | ||||
|     edges = [] | ||||
|     with Session(engine) as session: | ||||
|         for p in session.exec(select(P)).fetchall(): | ||||
|             nodes.append({"id": p.name, "appearance": 1}) | ||||
|             nodes.append({"id": p.name, "label": p.name}) | ||||
|         subquery = ( | ||||
|             select(C.user, func.max(C.time).label("latest")) | ||||
|             .where(C.time > datetime(2025, 2, 1, 10)) | ||||
| @@ -44,9 +46,62 @@ def sociogram_json(): | ||||
|                 # G.add_edge(c.user, p) | ||||
|                 # p_id = session.exec(select(P.id).where(P.name == p)).one() | ||||
|                 necessary_nodes.add(p) | ||||
|                 links.append({"source": c.user, "target": p}) | ||||
|                 edges.append({"from": c.user, "to": p, "relation": "likes"}) | ||||
|             for p in c.hate: | ||||
|                 edges.append({"from": c.user, "to": p, "relation": "dislikes"}) | ||||
|     # nodes = [n for n in nodes if n["name"] in necessary_nodes] | ||||
|     return JSONResponse({"nodes": nodes, "links": links}) | ||||
|     return JSONResponse({"nodes": nodes, "edges": edges}) | ||||
|  | ||||
|  | ||||
| def graph_json(): | ||||
|     nodes = [] | ||||
|     edges = [] | ||||
|     with Session(engine) as session: | ||||
|         for p in session.exec(select(P)).fetchall(): | ||||
|             nodes.append({"id": p.name, "label": p.name}) | ||||
|         subquery = ( | ||||
|             select(C.user, func.max(C.time).label("latest")) | ||||
|             .where(C.time > datetime(2025, 2, 1, 10)) | ||||
|             .group_by(C.user) | ||||
|             .subquery() | ||||
|         ) | ||||
|         statement2 = select(C).join( | ||||
|             subquery, (C.user == subquery.c.user) & (C.time == subquery.c.latest) | ||||
|         ) | ||||
|         for c in session.exec(statement2): | ||||
|             for i, p in enumerate(c.love): | ||||
|                 edges.append( | ||||
|                     { | ||||
|                         "id": f"{c.user}->{p}", | ||||
|                         "source": c.user, | ||||
|                         "target": p, | ||||
|                         "size": max(1.0 - 0.1 * i, 0.3), | ||||
|                         "data": { | ||||
|                             "relation": 2, | ||||
|                             "origSize": max(1.0 - 0.1 * i, 0.3), | ||||
|                             "origFill": "#bed4ff", | ||||
|                         }, | ||||
|                     } | ||||
|                 ) | ||||
|             for p in c.hate: | ||||
|                 edges.append( | ||||
|                     { | ||||
|                         "id": f"{c.user}-x>{p}", | ||||
|                         "source": c.user, | ||||
|                         "target": p, | ||||
|                         "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}) | ||||
|  | ||||
|  | ||||
| def sociogram_data(show: int | None = 2): | ||||
| @@ -143,8 +198,36 @@ async def render_sociogram(params: Params): | ||||
|     return {"image": encoded_image} | ||||
|  | ||||
|  | ||||
| def mvp(): | ||||
|     ranks = dict() | ||||
|     with Session(engine) as session: | ||||
|         subquery = ( | ||||
|             select(R.user, func.max(R.time).label("latest")) | ||||
|             .where(R.time > datetime(2025, 2, 8)) | ||||
|             .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 in enumerate(r.mvps): | ||||
|                 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("/graph_json", endpoint=graph_json, methods=["GET"]) | ||||
| analysis_router.add_api_route("/image", endpoint=render_sociogram, methods=["POST"]) | ||||
| analysis_router.add_api_route("/mvp", endpoint=mvp, methods=["GET"]) | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     with Session(engine) as session: | ||||
|   | ||||
							
								
								
									
										2
									
								
								db.py
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								db.py
									
									
									
									
									
								
							| @@ -12,7 +12,7 @@ from sqlmodel import ( | ||||
| with open("db.secrets", "r") as f: | ||||
|     db_secrets = f.readline().strip() | ||||
|  | ||||
| engine = create_engine(db_secrets) | ||||
| engine = create_engine(db_secrets, connect_args={"connect_timeout": 8}) | ||||
| del db_secrets | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										20
									
								
								main.py
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								main.py
									
									
									
									
									
								
							| @@ -18,10 +18,8 @@ from security import ( | ||||
| app = FastAPI(title="cutt") | ||||
| api_router = APIRouter(prefix="/api") | ||||
| origins = [ | ||||
|     "*", | ||||
|     "http://localhost", | ||||
|     "http://localhost:3000", | ||||
|     "http://localhost:8000", | ||||
|     "https://cutt.0124816.xyz", | ||||
|     "http://localhost:5173", | ||||
| ] | ||||
|  | ||||
| app.add_middleware( | ||||
| @@ -66,11 +64,21 @@ def list_teams(): | ||||
|  | ||||
| player_router = APIRouter(prefix="/player") | ||||
| 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"], | ||||
|     dependencies=[Depends(get_current_active_user)], | ||||
| ) | ||||
|  | ||||
| team_router = APIRouter(prefix="/team") | ||||
| team_router.add_api_route("/list", endpoint=list_teams, methods=["GET"]) | ||||
| team_router.add_api_route("/add", endpoint=add_team, methods=["POST"]) | ||||
| team_router.add_api_route( | ||||
|     "/add", | ||||
|     endpoint=add_team, | ||||
|     methods=["POST"], | ||||
|     dependencies=[Depends(get_current_active_user)], | ||||
| ) | ||||
|  | ||||
|  | ||||
| @app.post("/mvps/", status_code=status.HTTP_200_OK) | ||||
|   | ||||
							
								
								
									
										11
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								package.json
									
									
									
									
									
								
							| @@ -10,17 +10,16 @@ | ||||
|     "preview": "vite preview" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "d3": "^7.9.0", | ||||
|     "react": "^18.3.1", | ||||
|     "react-dom": "^18.3.1", | ||||
|     "react": "18.3.1", | ||||
|     "react-dom": "18.3.1", | ||||
|     "react-sortablejs": "^6.1.4", | ||||
|     "reagraph": "^4.21.2", | ||||
|     "sortablejs": "^1.15.6" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@eslint/js": "^9.17.0", | ||||
|     "@types/d3": "^7.4.3", | ||||
|     "@types/react": "^18.3.18", | ||||
|     "@types/react-dom": "^18.3.5", | ||||
|     "@types/react": "18.3.18", | ||||
|     "@types/react-dom": "18.3.5", | ||||
|     "@types/sortablejs": "^1.15.8", | ||||
|     "@vitejs/plugin-react": "^4.3.4", | ||||
|     "eslint": "^9.17.0", | ||||
|   | ||||
							
								
								
									
										12
									
								
								security.py
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								security.py
									
									
									
									
									
								
							| @@ -9,6 +9,7 @@ from db import engine, User | ||||
| from fastapi.security import OAuth2PasswordBearer, OAuth2PasswordRequestForm | ||||
| from pydantic_settings import BaseSettings, SettingsConfigDict | ||||
| from passlib.context import CryptContext | ||||
| from sqlalchemy.exc import OperationalError | ||||
|  | ||||
|  | ||||
| class Config(BaseSettings): | ||||
| @@ -47,10 +48,13 @@ def get_password_hash(password): | ||||
|  | ||||
| def get_user(username: str | None): | ||||
|     if username: | ||||
|         with Session(engine) as session: | ||||
|             return session.exec( | ||||
|                 select(User).where(User.username == username) | ||||
|             ).one_or_none() | ||||
|         try: | ||||
|             with Session(engine) as session: | ||||
|                 return session.exec( | ||||
|                     select(User).where(User.username == username) | ||||
|                 ).one_or_none() | ||||
|         except OperationalError: | ||||
|             return | ||||
|  | ||||
|  | ||||
| def authenticate_user(username: str, password: str): | ||||
|   | ||||
| @@ -42,7 +42,7 @@ interface DeferredProps { | ||||
| } | ||||
|  | ||||
|  | ||||
| let timeoutID: number | null = null; | ||||
| let timeoutID: NodeJS.Timeout | null = null; | ||||
| export default function Analysis() { | ||||
|   const [image, setImage] = useState(""); | ||||
|   const [params, setParams] = useState<Params>({ | ||||
|   | ||||
							
								
								
									
										131
									
								
								src/App.css
									
									
									
									
									
								
							
							
						
						
									
										131
									
								
								src/App.css
									
									
									
									
									
								
							| @@ -23,6 +23,115 @@ footer { | ||||
|   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 { | ||||
|   color: #444; | ||||
| @@ -168,13 +277,24 @@ button, | ||||
|   #control-panel { | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|   } | ||||
|  | ||||
|   .control { | ||||
|     font-size: 80%; | ||||
|     margin: 0px; | ||||
|   } | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| @media only screen and (max-width: 768px) { | ||||
|   #control-panel { | ||||
|     grid-template-columns: 1fr; | ||||
|   } | ||||
|  | ||||
|   .networkroute { | ||||
|     display: none; | ||||
|   } | ||||
|  | ||||
|   .submit_text { | ||||
|     display: none; | ||||
|   } | ||||
| @@ -241,6 +361,8 @@ button, | ||||
|   font-size: 150%; | ||||
| } | ||||
|  | ||||
| /*======LOGO=======*/ | ||||
|  | ||||
| .logo { | ||||
|   position: relative; | ||||
|   text-align: center; | ||||
| @@ -268,6 +390,15 @@ button, | ||||
|   } | ||||
| } | ||||
|  | ||||
| .networkroute { | ||||
|   z-index: 10; | ||||
|   position: absolute; | ||||
|   top: 24px; | ||||
|   left: 48px; | ||||
| } | ||||
|  | ||||
| /*======SPINNER=======*/ | ||||
|  | ||||
| .loader { | ||||
|   display: block; | ||||
|   position: relative; | ||||
|   | ||||
							
								
								
									
										22
									
								
								src/App.tsx
									
									
									
									
									
								
							
							
						
						
									
										22
									
								
								src/App.tsx
									
									
									
									
									
								
							| @@ -5,24 +5,34 @@ import Header from "./Header"; | ||||
| import Rankings from "./Rankings"; | ||||
| import { BrowserRouter, Routes, Route } from "react-router"; | ||||
| import { SessionProvider } from "./Session"; | ||||
| import { GraphComponent } from "./Network"; | ||||
| import MVPChart from "./MVPChart"; | ||||
|  | ||||
| function App() { | ||||
|   //const [data, setData] = useState({ nodes: [], links: [] } as SociogramData); | ||||
|   //async function loadData() { | ||||
|   //  await fetch(`${baseUrl}api/analysis/json`, { method: "GET" }).then(resp => resp.json() as unknown as SociogramData).then(json => { setData(json) }) | ||||
|   //} | ||||
|   //useEffect(() => { loadData() }, []) | ||||
|   // | ||||
|   return ( | ||||
|     <BrowserRouter> | ||||
|       <Header /> | ||||
|       <Routes> | ||||
|         <Route index element={<Rankings />} /> | ||||
|  | ||||
|         <Route path="/network" element={ | ||||
|           <SessionProvider> | ||||
|             <GraphComponent /> | ||||
|           </SessionProvider> | ||||
|         } /> | ||||
|  | ||||
|         <Route path="/analysis" element={ | ||||
|           <SessionProvider> | ||||
|             <Analysis /> | ||||
|           </SessionProvider> | ||||
|         } /> | ||||
|  | ||||
|         <Route path="/mvp" element={ | ||||
|           <SessionProvider> | ||||
|             <MVPChart /> | ||||
|           </SessionProvider> | ||||
|         } /> | ||||
|  | ||||
|       </Routes> | ||||
|       <Footer /> | ||||
|     </BrowserRouter> | ||||
|   | ||||
							
								
								
									
										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,21 +1,31 @@ | ||||
| import { Link } from "react-router"; | ||||
|  | ||||
| export default function Footer() { | ||||
|         return <footer> | ||||
|                 <div className="navbar"> | ||||
|                         <Link to="/" ><span>Form</span></Link> | ||||
|                         <span>|</span> | ||||
|                         <Link to="/analysis" ><span>Trainer Analysis</span></Link> | ||||
|                 </div> | ||||
|                 <p className="grey extra-margin"> | ||||
|                         something not working? | ||||
|                         <br /> | ||||
|                         message <a href="https://t.me/x0124816">me</a>. | ||||
|                         <br /> | ||||
|                         or fix it here:{" "} | ||||
|                         <a href="https://git.0124816.xyz/julius/cutt" key="gitea"> | ||||
|                                 <img src="gitea.svg" alt="gitea" height="16" /> | ||||
|                         </a> | ||||
|                 </p> | ||||
|         </footer> | ||||
|   return ( | ||||
|     <footer> | ||||
|       <div className="navbar"> | ||||
|         <a href="/"> | ||||
|           <span>Form</span> | ||||
|         </a> | ||||
|         <span>|</span> | ||||
|         <a href="/network"> | ||||
|           <span>Trainer Analysis</span> | ||||
|         </a> | ||||
|         <span>|</span> | ||||
|         <a href="/mvp"> | ||||
|           <span>MVP</span> | ||||
|         </a> | ||||
|       </div> | ||||
|       <p className="grey extra-margin"> | ||||
|         something not working? | ||||
|         <br /> | ||||
|         message <a href="https://t.me/x0124816">me</a>. | ||||
|         <br /> | ||||
|         or fix it here:{" "} | ||||
|         <a href="https://git.0124816.xyz/julius/cutt" key="gitea"> | ||||
|           <img src="gitea.svg" alt="gitea" height="16" /> | ||||
|         </a> | ||||
|       </p> | ||||
|     </footer> | ||||
|   ); | ||||
| } | ||||
|   | ||||
| @@ -1,8 +1,6 @@ | ||||
| import { baseUrl } from "./api"; | ||||
|  | ||||
| export default function Header() { | ||||
|   return <div className="logo"> | ||||
|     <a href={baseUrl}> | ||||
|   return <div className="logo" id="logo"> | ||||
|     <a href={"/"}> | ||||
|       <img alt="logo" height="66%" src="logo.svg" /> | ||||
|       <h3 className="centered">cutt</h3> | ||||
|     </a> | ||||
|   | ||||
							
								
								
									
										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; | ||||
							
								
								
									
										285
									
								
								src/Network.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										285
									
								
								src/Network.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,285 @@ | ||||
| import { useEffect, useRef, useState } from "react"; | ||||
| import { apiAuth } from "./api"; | ||||
| import { | ||||
|   GraphCanvas, | ||||
|   GraphCanvasRef, | ||||
|   GraphEdge, | ||||
|   GraphNode, | ||||
|   SelectionProps, | ||||
|   SelectionResult, | ||||
|   useSelection, | ||||
| } from "reagraph"; | ||||
| import { customTheme } from "./NetworkTheme"; | ||||
|  | ||||
| interface NetworkData { | ||||
|   nodes: GraphNode[]; | ||||
|   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 = () => { | ||||
|   const [data, setData] = useState({ nodes: [], edges: [] } as NetworkData); | ||||
|   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); | ||||
|  | ||||
|   const logo = document.getElementById("logo"); | ||||
|   if (logo) { | ||||
|     logo.className = "logo networkroute"; | ||||
|   } | ||||
|   const footer = document.getElementsByTagName("footer"); | ||||
|   if (footer) { | ||||
|     (footer.item(0) as HTMLElement).className = "fixed-footer"; | ||||
|   } | ||||
|  | ||||
|   async function loadData() { | ||||
|     setLoading(true); | ||||
|     await apiAuth("analysis/graph_json", null) | ||||
|       .then((json) => json as Promise<NetworkData>) | ||||
|       .then((json) => { | ||||
|         setData(json); | ||||
|       }); | ||||
|     setLoading(false); | ||||
|   } | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadData(); | ||||
|   }, []); | ||||
|  | ||||
|   const graphRef = useRef<GraphCanvasRef | null>(null); | ||||
|  | ||||
|   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, | ||||
|     nodes: data.nodes, | ||||
|     edges: data.edges.filter((edge) => edge.data.relation === likes), | ||||
|     ignore: data.edges.map((edge) => { | ||||
|       if (likes === 1 && edge.data.relation !== 2) return edge; | ||||
|     }), | ||||
|     pathSelectionType: "out", | ||||
|     pathHoverType: "in", | ||||
|     type: "multiModifier", | ||||
|   }); | ||||
|  | ||||
|   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} /> | ||||
|             <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} /> | ||||
|             <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} /> | ||||
|             <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 | ||||
|           draggable | ||||
|           cameraMode={threed ? "rotate" : "pan"} | ||||
|           layoutType={threed ? "forceDirected3d" : "forceDirected2d"} | ||||
|           layoutOverrides={{ | ||||
|             nodeStrength: -200, | ||||
|             linkDistance: 100, | ||||
|           }} | ||||
|           labelType="nodes" | ||||
|           sizingType="attribute" | ||||
|           sizingAttribute={popularity ? "inDegree" : undefined} | ||||
|           ref={graphRef} | ||||
|           theme={customTheme} | ||||
|           nodes={data.nodes} | ||||
|           edges={data.edges.filter( | ||||
|             (edge) => edge.data.relation === likes || likes === 1 | ||||
|           )} | ||||
|           selections={selections} | ||||
|           actives={actives} | ||||
|           onCanvasClick={onCanvasClick} | ||||
|           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' | ||||
|     } | ||||
|   } | ||||
| }; | ||||
							
								
								
									
										98
									
								
								src/RaceChart.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								src/RaceChart.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | ||||
| import { FC, useEffect, useState } from "react"; | ||||
| import { PlayerRanking } from "./types"; | ||||
|  | ||||
| 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 / 20); | ||||
|  | ||||
|   return ( | ||||
|     <svg width={width} height={height}> | ||||
|       {players.map((player, index) => ( | ||||
|         <rect | ||||
|           key={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> | ||||
|           <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; | ||||
							
								
								
									
										17
									
								
								src/api.ts
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								src/api.ts
									
									
									
									
									
								
							| @@ -20,13 +20,16 @@ export default async function api(path: string, data: any): Promise<any> { | ||||
|  | ||||
| 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' | ||||
|     }, | ||||
|     body: JSON.stringify(data), | ||||
|   }); | ||||
|   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; | ||||
|   try { | ||||
|     resp = await fetch(req); | ||||
|   | ||||
							
								
								
									
										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