From fdc2d4a971d608bb230cd8aa1e602197c7775231 Mon Sep 17 00:00:00 2001 From: crupest Date: Wed, 30 Jun 2021 22:46:42 +0800 Subject: ... --- FrontEnd/src/views/center/TimelineBoard.tsx | 26 ++++++++++++++----------- FrontEnd/src/views/common/Spinner.css | 13 +++++++++++++ FrontEnd/src/views/common/Spinner.tsx | 30 ++++++++++++++++++++++++++--- FrontEnd/vite.config.js | 2 +- 4 files changed, 56 insertions(+), 15 deletions(-) create mode 100644 FrontEnd/src/views/common/Spinner.css diff --git a/FrontEnd/src/views/center/TimelineBoard.tsx b/FrontEnd/src/views/center/TimelineBoard.tsx index d7aa39ab..3961a7bc 100644 --- a/FrontEnd/src/views/center/TimelineBoard.tsx +++ b/FrontEnd/src/views/center/TimelineBoard.tsx @@ -208,7 +208,8 @@ const TimelineBoardItemContainer: React.FC = ({ interface TimelineBoardUIProps { title?: string; - timelines: HttpTimelineInfo[] | "offline" | "loading"; + state: "offline" | "loading" | "loaded"; + timelines: HttpTimelineInfo[]; onReload: () => void; className?: string; editHandler?: { @@ -218,7 +219,7 @@ interface TimelineBoardUIProps { } const TimelineBoardUI: React.FC = (props) => { - const { title, timelines, className, editHandler } = props; + const { title, state, timelines, className, editHandler } = props; const editable = editHandler != null; @@ -246,13 +247,13 @@ const TimelineBoardUI: React.FC = (props) => { ))} {(() => { - if (timelines === "loading") { + if (state === "loading") { return (
); - } else if (timelines === "offline") { + } else if (state === "offline") { return (
@@ -301,36 +302,39 @@ const TimelineBoard: React.FC = ({ load, editHandler, }) => { - const [timelines, setTimelines] = React.useState< - HttpTimelineInfo[] | "offline" | "loading" - >("loading"); + const [state, setState] = React.useState<"offline" | "loading" | "loaded">( + "loading" + ); + const [timelines, setTimelines] = React.useState([]); React.useEffect(() => { let subscribe = true; - if (timelines === "loading") { + if (state === "loading") { void load().then( (timelines) => { if (subscribe) { + setState("loaded"); setTimelines(timelines); } }, () => { - setTimelines("offline"); + setState("offline"); } ); } return () => { subscribe = false; }; - }, [load, timelines]); + }, [load, state]); return ( { - setTimelines("loading"); + setState("loaded"); }} editHandler={ typeof timelines === "object" && editHandler != null diff --git a/FrontEnd/src/views/common/Spinner.css b/FrontEnd/src/views/common/Spinner.css new file mode 100644 index 00000000..a1de68d2 --- /dev/null +++ b/FrontEnd/src/views/common/Spinner.css @@ -0,0 +1,13 @@ +@keyframes cru-spinner-animation { + from { + transform: scale(0,0); + } +} + +.cru-spinner { + display: inline-block; + animation: cru-spinner-animation 0.5s infinite alternate; + background-color: currentColor; + border-radius: 50%; + transform-origin: center; +} diff --git a/FrontEnd/src/views/common/Spinner.tsx b/FrontEnd/src/views/common/Spinner.tsx index 783c9be2..b591d8ab 100644 --- a/FrontEnd/src/views/common/Spinner.tsx +++ b/FrontEnd/src/views/common/Spinner.tsx @@ -1,13 +1,37 @@ -import { PaletteColorType } from "@/palette"; import React from "react"; +import classnames from "classnames"; + +import { PaletteColorType } from "@/palette"; + +import "./Spinner.css"; export interface SpinnerProps { - size?: "sm" | "md" | "lg" | number; + size?: "sm" | "md" | "lg" | number | string; color?: PaletteColorType; } export default function Spinner( props: SpinnerProps ): React.ReactElement | null { - return ; + const { size, color } = props; + const calculatedSize = + size === "sm" + ? "18px" + : size === "md" + ? "30px" + : size === "lg" + ? "42px" + : typeof size === "number" + ? size + : size == null + ? "20px" + : size; + const calculatedColor = color ?? "primary"; + + return ( + + ); } diff --git a/FrontEnd/vite.config.js b/FrontEnd/vite.config.js index 88529a33..64e0457f 100644 --- a/FrontEnd/vite.config.js +++ b/FrontEnd/vite.config.js @@ -22,7 +22,7 @@ export default defineConfig({ alias: [{ find: "@", replacement: "/src" }], }, server: { - port: 13000, + port: 10030, proxy: { "/api": { target: "http://localhost:5000", -- cgit v1.2.3