@capytale/meta-player 0.8.2 → 0.8.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.prettierrc.json +4 -4
- package/LICENSE +674 -674
- package/README.md +12 -12
- package/eslint.config.js +28 -28
- package/index.html +15 -15
- package/package.json +60 -60
- package/public/themes/lara-dark-blue/theme.css +7015 -7015
- package/public/themes/lara-light-blue/theme.css +7005 -7005
- package/src/App.tsx +139 -139
- package/src/AppRedux.css +39 -39
- package/src/MetaPlayer.tsx +170 -170
- package/src/activityJs.ts +7 -7
- package/src/app/createAppSlice.ts +6 -6
- package/src/app/hooks.ts +12 -12
- package/src/app/store.ts +52 -52
- package/src/app.module.scss +80 -80
- package/src/demo.tsx +87 -87
- package/src/external/prime.ts +5 -5
- package/src/features/activityData/ExitWarning.ts +28 -28
- package/src/features/activityData/IsAntiCheatExitDetectionDisabledSetter.tsx +19 -19
- package/src/features/activityData/IsDirtySetter.tsx +17 -17
- package/src/features/activityData/MetaPlayerOptionsSetter.tsx +41 -41
- package/src/features/activityData/activityDataSlice.ts +256 -256
- package/src/features/activityData/activityJsData.ts +82 -82
- package/src/features/activityData/hooks.ts +34 -34
- package/src/features/activityData/metaPlayerOptions.ts +23 -23
- package/src/features/activityData/uiState.ts +20 -20
- package/src/features/activityJS/ActivityJSProvider.tsx +339 -339
- package/src/features/activityJS/AfterResetAction.tsx +23 -23
- package/src/features/activityJS/AfterSaveAction.tsx +23 -23
- package/src/features/activityJS/BeforeResetAction.tsx +23 -23
- package/src/features/activityJS/BeforeSaveAction.tsx +23 -23
- package/src/features/activityJS/Saver.tsx +167 -167
- package/src/features/activityJS/hooks.ts +85 -85
- package/src/features/activityJS/internal-hooks.ts +96 -96
- package/src/features/activityJS/saverSlice.ts +96 -96
- package/src/features/activitySettings/ActivitySettingsSetter.tsx +21 -21
- package/src/features/activitySettings/hooks.ts +12 -12
- package/src/features/activitySettings/index.tsx +43 -43
- package/src/features/activitySettings/store.ts +108 -108
- package/src/features/activitySettings/style.module.scss +8 -8
- package/src/features/activitySettings/types.ts +140 -140
- package/src/features/activitySettings/ui.tsx +299 -299
- package/src/features/functionalities/AttachedFilesFunctionality.ts +23 -23
- package/src/features/functionalities/PreviewDialog.tsx +83 -83
- package/src/features/functionalities/functionalitiesSlice.ts +98 -98
- package/src/features/functionalities/hooks.ts +70 -70
- package/src/features/layout/hooks.ts +7 -7
- package/src/features/layout/layoutSlice.ts +90 -90
- package/src/features/navbar/activity-info/index.tsx +54 -54
- package/src/features/navbar/activity-info/style.module.scss +38 -38
- package/src/features/navbar/activity-menu/ActivityQuickActions.tsx +57 -57
- package/src/features/navbar/activity-menu/index.tsx +154 -153
- package/src/features/navbar/activity-menu/style.module.scss +7 -7
- package/src/features/navbar/capytale-menu/CloneDialog.tsx +75 -75
- package/src/features/navbar/capytale-menu/Countdown.tsx +312 -312
- package/src/features/navbar/capytale-menu/CountdownAndSaveButton.tsx +115 -115
- package/src/features/navbar/capytale-menu/index.tsx +260 -260
- package/src/features/navbar/capytale-menu/style.module.scss +8 -8
- package/src/features/navbar/index.tsx +39 -39
- package/src/features/navbar/navbarSlice.ts +79 -79
- package/src/features/navbar/review-navbar/GradingNav.tsx +128 -128
- package/src/features/navbar/review-navbar/index.tsx +18 -18
- package/src/features/navbar/review-navbar/style.module.scss +22 -22
- package/src/features/navbar/sidebars/ActivitySidebarActions.tsx +57 -57
- package/src/features/navbar/sidebars/AttachedFilesSidebarContent.module.scss +43 -43
- package/src/features/navbar/sidebars/AttachedFilesSidebarContent.tsx +181 -181
- package/src/features/navbar/sidebars/SettingsSidebarContent.tsx +192 -192
- package/src/features/navbar/sidebars/style.module.scss +15 -15
- package/src/features/navbar/student-utils.ts +11 -11
- package/src/features/navbar/style.module.scss +65 -65
- package/src/features/pedago/InstructionsEditor.tsx +102 -102
- package/src/features/pedago/PdfEditor.tsx +91 -91
- package/src/features/pedago/PedagoCommands.tsx +353 -353
- package/src/features/pedago/SharedNotesEditor.tsx +144 -144
- package/src/features/pedago/index.tsx +207 -204
- package/src/features/pedago/style.module.scss +233 -233
- package/src/features/theming/ThemeSwitcher.tsx +51 -51
- package/src/features/theming/hooks.ts +6 -6
- package/src/features/theming/themingSlice.ts +93 -93
- package/src/features/toast.tsx +38 -38
- package/src/hooks/index.ts +16 -16
- package/src/index.css +132 -132
- package/src/index.ts +90 -90
- package/src/logo.svg +1 -1
- package/src/my_json_data.js +4146 -4146
- package/src/settings.ts +6 -6
- package/src/types.ts +9 -9
- package/src/utils/ButtonDoubleIcon.tsx +35 -35
- package/src/utils/CardSelector.tsx +41 -41
- package/src/utils/ErrorBoundary.tsx +41 -41
- package/src/utils/PopupButton.tsx +134 -134
- package/src/utils/activity.ts +8 -8
- package/src/utils/breakpoints.ts +4 -4
- package/src/utils/capytale.ts +10 -10
- package/src/utils/clipboard.ts +11 -11
- package/src/utils/download.ts +7 -7
- package/src/utils/equality.ts +32 -32
- package/src/utils/server-clock.ts +42 -42
- package/src/utils/style.module.scss +45 -45
- package/src/utils/useFullscreen.ts +65 -65
- package/src/vite-env.d.ts +1 -1
- package/tsconfig.json +28 -28
- package/tsconfig.node.json +9 -9
- package/vite.config.ts +11 -11
package/src/App.tsx
CHANGED
|
@@ -1,139 +1,139 @@
|
|
|
1
|
-
import { Splitter, SplitterPanel } from "primereact/splitter";
|
|
2
|
-
import Navbar from "./features/navbar";
|
|
3
|
-
|
|
4
|
-
import styles from "./app.module.scss";
|
|
5
|
-
import Pedago from "./features/pedago";
|
|
6
|
-
import { useAppDispatch, useAppSelector } from "./app/hooks";
|
|
7
|
-
import { selectThemeIsDark } from "./features/theming/themingSlice";
|
|
8
|
-
|
|
9
|
-
import { Ripple } from "primereact/ripple";
|
|
10
|
-
import { classNames } from "primereact/utils";
|
|
11
|
-
import {
|
|
12
|
-
selectIsPedagoVisible,
|
|
13
|
-
selectOrientation,
|
|
14
|
-
selectShowSaveForStudents,
|
|
15
|
-
toggleIsPedagoVisible,
|
|
16
|
-
} from "./features/layout/layoutSlice";
|
|
17
|
-
import { FC, KeyboardEvent, PropsWithChildren, useCallback } from "react";
|
|
18
|
-
import { Tooltip } from "primereact/tooltip";
|
|
19
|
-
import {
|
|
20
|
-
selectHasGradingOrComments,
|
|
21
|
-
selectHasInstructions,
|
|
22
|
-
selectIsDirty,
|
|
23
|
-
selectMode,
|
|
24
|
-
selectShowSaveButton,
|
|
25
|
-
} from "./features/activityData/activityDataSlice";
|
|
26
|
-
import settings from "./settings";
|
|
27
|
-
import ReviewNavbar from "./features/navbar/review-navbar";
|
|
28
|
-
import { useSave } from "./features/activityData/hooks";
|
|
29
|
-
import PreviewDialog from "./features/functionalities/PreviewDialog";
|
|
30
|
-
|
|
31
|
-
type AppProps = PropsWithChildren<{}>;
|
|
32
|
-
|
|
33
|
-
const App: FC<AppProps> = (props) => {
|
|
34
|
-
const isDark = useAppSelector(selectThemeIsDark) as boolean;
|
|
35
|
-
const mode = useAppSelector(selectMode);
|
|
36
|
-
const isHorizontal = useAppSelector(selectOrientation) === "horizontal";
|
|
37
|
-
const isPedagoVisible = useAppSelector(selectIsPedagoVisible) as boolean;
|
|
38
|
-
const hasInstructions = useAppSelector(selectHasInstructions);
|
|
39
|
-
const hasGradingOrComments = useAppSelector(selectHasGradingOrComments);
|
|
40
|
-
const hasPedago =
|
|
41
|
-
hasInstructions || hasGradingOrComments || mode === "review";
|
|
42
|
-
const dispatch = useAppDispatch();
|
|
43
|
-
const showPedago = hasPedago && isPedagoVisible;
|
|
44
|
-
const isDirty = useAppSelector(selectIsDirty);
|
|
45
|
-
const save = useSave();
|
|
46
|
-
|
|
47
|
-
const showSaveButton = useAppSelector(selectShowSaveButton);
|
|
48
|
-
const showSaveForStudents = useAppSelector(selectShowSaveForStudents);
|
|
49
|
-
const hasSaveButton =
|
|
50
|
-
showSaveButton && !(mode === "assignment" && !showSaveForStudents);
|
|
51
|
-
|
|
52
|
-
const handleCtrlS = useCallback(
|
|
53
|
-
(e: KeyboardEvent<HTMLDivElement>) => {
|
|
54
|
-
if ((e.ctrlKey || e.metaKey) && e.key === "s") {
|
|
55
|
-
e.preventDefault();
|
|
56
|
-
if (hasSaveButton && isDirty) save(); // Checks if can save inside of save()
|
|
57
|
-
}
|
|
58
|
-
},
|
|
59
|
-
[hasSaveButton, isDirty, save],
|
|
60
|
-
);
|
|
61
|
-
|
|
62
|
-
const pedagoOpenLabel = hasPedago
|
|
63
|
-
? "Afficher les consignes"
|
|
64
|
-
: mode === "create"
|
|
65
|
-
? "Ce type d'activité n'accepte pas de consigne"
|
|
66
|
-
: "Pas de consignes ni de note";
|
|
67
|
-
|
|
68
|
-
return (
|
|
69
|
-
<div
|
|
70
|
-
className={classNames(
|
|
71
|
-
styles.app,
|
|
72
|
-
isDark ? "dark-theme" : "light-theme",
|
|
73
|
-
isHorizontal ? "layout-horizontal" : "layout-vertical",
|
|
74
|
-
)}
|
|
75
|
-
onKeyDown={handleCtrlS}
|
|
76
|
-
>
|
|
77
|
-
<div className={styles.navbarContainer}>
|
|
78
|
-
<Navbar />
|
|
79
|
-
{mode === "review" && <ReviewNavbar />}
|
|
80
|
-
</div>
|
|
81
|
-
<div className={styles.pedagoContainer} data-show-pedago={showPedago}>
|
|
82
|
-
<div
|
|
83
|
-
className={classNames(
|
|
84
|
-
styles.hiddenPedago,
|
|
85
|
-
hasPedago ? null : styles.noPedago,
|
|
86
|
-
)}
|
|
87
|
-
>
|
|
88
|
-
<div
|
|
89
|
-
className={classNames(
|
|
90
|
-
styles.hiddenPedagoButton,
|
|
91
|
-
hasPedago ? "p-ripple" : null,
|
|
92
|
-
)}
|
|
93
|
-
onClick={
|
|
94
|
-
hasPedago ? () => dispatch(toggleIsPedagoVisible()) : undefined
|
|
95
|
-
}
|
|
96
|
-
data-pr-tooltip={pedagoOpenLabel}
|
|
97
|
-
aria-label={pedagoOpenLabel}
|
|
98
|
-
role={hasPedago ? "button" : "note"}
|
|
99
|
-
>
|
|
100
|
-
<i
|
|
101
|
-
className={classNames(
|
|
102
|
-
"pi",
|
|
103
|
-
hasPedago
|
|
104
|
-
? isHorizontal
|
|
105
|
-
? "pi-angle-double-down"
|
|
106
|
-
: "pi-angle-double-right"
|
|
107
|
-
: "pi-minus-circle",
|
|
108
|
-
)}
|
|
109
|
-
/>
|
|
110
|
-
<Ripple />
|
|
111
|
-
</div>
|
|
112
|
-
<Tooltip
|
|
113
|
-
target={"." + styles.hiddenPedagoButton}
|
|
114
|
-
showDelay={settings.TOOLTIP_SHOW_DELAY}
|
|
115
|
-
position={isHorizontal ? "bottom" : "right"}
|
|
116
|
-
mouseTrack={!isHorizontal}
|
|
117
|
-
/>
|
|
118
|
-
</div>
|
|
119
|
-
<Splitter
|
|
120
|
-
className={styles.appPedagoSplitter}
|
|
121
|
-
layout={isHorizontal ? "vertical" : "horizontal"}
|
|
122
|
-
>
|
|
123
|
-
<SplitterPanel minSize={15} size={30} className={styles.pedagoPanel}>
|
|
124
|
-
<Pedago key="pedago" />
|
|
125
|
-
</SplitterPanel>
|
|
126
|
-
<SplitterPanel minSize={40} size={70} className={styles.contentPanel}>
|
|
127
|
-
<div className="meta-player-content-cover"></div>
|
|
128
|
-
<div id="meta-player-content" aria-label="Activité">
|
|
129
|
-
{props.children}
|
|
130
|
-
</div>
|
|
131
|
-
</SplitterPanel>
|
|
132
|
-
</Splitter>
|
|
133
|
-
</div>
|
|
134
|
-
<PreviewDialog />
|
|
135
|
-
</div>
|
|
136
|
-
);
|
|
137
|
-
};
|
|
138
|
-
|
|
139
|
-
export default App;
|
|
1
|
+
import { Splitter, SplitterPanel } from "primereact/splitter";
|
|
2
|
+
import Navbar from "./features/navbar";
|
|
3
|
+
|
|
4
|
+
import styles from "./app.module.scss";
|
|
5
|
+
import Pedago from "./features/pedago";
|
|
6
|
+
import { useAppDispatch, useAppSelector } from "./app/hooks";
|
|
7
|
+
import { selectThemeIsDark } from "./features/theming/themingSlice";
|
|
8
|
+
|
|
9
|
+
import { Ripple } from "primereact/ripple";
|
|
10
|
+
import { classNames } from "primereact/utils";
|
|
11
|
+
import {
|
|
12
|
+
selectIsPedagoVisible,
|
|
13
|
+
selectOrientation,
|
|
14
|
+
selectShowSaveForStudents,
|
|
15
|
+
toggleIsPedagoVisible,
|
|
16
|
+
} from "./features/layout/layoutSlice";
|
|
17
|
+
import { FC, KeyboardEvent, PropsWithChildren, useCallback } from "react";
|
|
18
|
+
import { Tooltip } from "primereact/tooltip";
|
|
19
|
+
import {
|
|
20
|
+
selectHasGradingOrComments,
|
|
21
|
+
selectHasInstructions,
|
|
22
|
+
selectIsDirty,
|
|
23
|
+
selectMode,
|
|
24
|
+
selectShowSaveButton,
|
|
25
|
+
} from "./features/activityData/activityDataSlice";
|
|
26
|
+
import settings from "./settings";
|
|
27
|
+
import ReviewNavbar from "./features/navbar/review-navbar";
|
|
28
|
+
import { useSave } from "./features/activityData/hooks";
|
|
29
|
+
import PreviewDialog from "./features/functionalities/PreviewDialog";
|
|
30
|
+
|
|
31
|
+
type AppProps = PropsWithChildren<{}>;
|
|
32
|
+
|
|
33
|
+
const App: FC<AppProps> = (props) => {
|
|
34
|
+
const isDark = useAppSelector(selectThemeIsDark) as boolean;
|
|
35
|
+
const mode = useAppSelector(selectMode);
|
|
36
|
+
const isHorizontal = useAppSelector(selectOrientation) === "horizontal";
|
|
37
|
+
const isPedagoVisible = useAppSelector(selectIsPedagoVisible) as boolean;
|
|
38
|
+
const hasInstructions = useAppSelector(selectHasInstructions);
|
|
39
|
+
const hasGradingOrComments = useAppSelector(selectHasGradingOrComments);
|
|
40
|
+
const hasPedago =
|
|
41
|
+
hasInstructions || hasGradingOrComments || mode === "review";
|
|
42
|
+
const dispatch = useAppDispatch();
|
|
43
|
+
const showPedago = hasPedago && isPedagoVisible;
|
|
44
|
+
const isDirty = useAppSelector(selectIsDirty);
|
|
45
|
+
const save = useSave();
|
|
46
|
+
|
|
47
|
+
const showSaveButton = useAppSelector(selectShowSaveButton);
|
|
48
|
+
const showSaveForStudents = useAppSelector(selectShowSaveForStudents);
|
|
49
|
+
const hasSaveButton =
|
|
50
|
+
showSaveButton && !(mode === "assignment" && !showSaveForStudents);
|
|
51
|
+
|
|
52
|
+
const handleCtrlS = useCallback(
|
|
53
|
+
(e: KeyboardEvent<HTMLDivElement>) => {
|
|
54
|
+
if ((e.ctrlKey || e.metaKey) && e.key === "s") {
|
|
55
|
+
e.preventDefault();
|
|
56
|
+
if (hasSaveButton && isDirty) save(); // Checks if can save inside of save()
|
|
57
|
+
}
|
|
58
|
+
},
|
|
59
|
+
[hasSaveButton, isDirty, save],
|
|
60
|
+
);
|
|
61
|
+
|
|
62
|
+
const pedagoOpenLabel = hasPedago
|
|
63
|
+
? "Afficher les consignes"
|
|
64
|
+
: mode === "create"
|
|
65
|
+
? "Ce type d'activité n'accepte pas de consigne"
|
|
66
|
+
: "Pas de consignes ni de note";
|
|
67
|
+
|
|
68
|
+
return (
|
|
69
|
+
<div
|
|
70
|
+
className={classNames(
|
|
71
|
+
styles.app,
|
|
72
|
+
isDark ? "dark-theme" : "light-theme",
|
|
73
|
+
isHorizontal ? "layout-horizontal" : "layout-vertical",
|
|
74
|
+
)}
|
|
75
|
+
onKeyDown={handleCtrlS}
|
|
76
|
+
>
|
|
77
|
+
<div className={styles.navbarContainer}>
|
|
78
|
+
<Navbar />
|
|
79
|
+
{mode === "review" && <ReviewNavbar />}
|
|
80
|
+
</div>
|
|
81
|
+
<div className={styles.pedagoContainer} data-show-pedago={showPedago}>
|
|
82
|
+
<div
|
|
83
|
+
className={classNames(
|
|
84
|
+
styles.hiddenPedago,
|
|
85
|
+
hasPedago ? null : styles.noPedago,
|
|
86
|
+
)}
|
|
87
|
+
>
|
|
88
|
+
<div
|
|
89
|
+
className={classNames(
|
|
90
|
+
styles.hiddenPedagoButton,
|
|
91
|
+
hasPedago ? "p-ripple" : null,
|
|
92
|
+
)}
|
|
93
|
+
onClick={
|
|
94
|
+
hasPedago ? () => dispatch(toggleIsPedagoVisible()) : undefined
|
|
95
|
+
}
|
|
96
|
+
data-pr-tooltip={pedagoOpenLabel}
|
|
97
|
+
aria-label={pedagoOpenLabel}
|
|
98
|
+
role={hasPedago ? "button" : "note"}
|
|
99
|
+
>
|
|
100
|
+
<i
|
|
101
|
+
className={classNames(
|
|
102
|
+
"pi",
|
|
103
|
+
hasPedago
|
|
104
|
+
? isHorizontal
|
|
105
|
+
? "pi-angle-double-down"
|
|
106
|
+
: "pi-angle-double-right"
|
|
107
|
+
: "pi-minus-circle",
|
|
108
|
+
)}
|
|
109
|
+
/>
|
|
110
|
+
<Ripple />
|
|
111
|
+
</div>
|
|
112
|
+
<Tooltip
|
|
113
|
+
target={"." + styles.hiddenPedagoButton}
|
|
114
|
+
showDelay={settings.TOOLTIP_SHOW_DELAY}
|
|
115
|
+
position={isHorizontal ? "bottom" : "right"}
|
|
116
|
+
mouseTrack={!isHorizontal}
|
|
117
|
+
/>
|
|
118
|
+
</div>
|
|
119
|
+
<Splitter
|
|
120
|
+
className={styles.appPedagoSplitter}
|
|
121
|
+
layout={isHorizontal ? "vertical" : "horizontal"}
|
|
122
|
+
>
|
|
123
|
+
<SplitterPanel minSize={15} size={30} className={styles.pedagoPanel}>
|
|
124
|
+
<Pedago key="pedago" />
|
|
125
|
+
</SplitterPanel>
|
|
126
|
+
<SplitterPanel minSize={40} size={70} className={styles.contentPanel}>
|
|
127
|
+
<div className="meta-player-content-cover"></div>
|
|
128
|
+
<div id="meta-player-content" aria-label="Activité">
|
|
129
|
+
{props.children}
|
|
130
|
+
</div>
|
|
131
|
+
</SplitterPanel>
|
|
132
|
+
</Splitter>
|
|
133
|
+
</div>
|
|
134
|
+
<PreviewDialog />
|
|
135
|
+
</div>
|
|
136
|
+
);
|
|
137
|
+
};
|
|
138
|
+
|
|
139
|
+
export default App;
|
package/src/AppRedux.css
CHANGED
|
@@ -1,39 +1,39 @@
|
|
|
1
|
-
.App {
|
|
2
|
-
text-align: center;
|
|
3
|
-
}
|
|
4
|
-
|
|
5
|
-
.App-logo {
|
|
6
|
-
height: 40vmin;
|
|
7
|
-
pointer-events: none;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
@media (prefers-reduced-motion: no-preference) {
|
|
11
|
-
.App-logo {
|
|
12
|
-
animation: App-logo-float infinite 3s ease-in-out;
|
|
13
|
-
}
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
.App-header {
|
|
17
|
-
min-height: 100vh;
|
|
18
|
-
display: flex;
|
|
19
|
-
flex-direction: column;
|
|
20
|
-
align-items: center;
|
|
21
|
-
justify-content: center;
|
|
22
|
-
font-size: calc(10px + 2vmin);
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
.App-link {
|
|
26
|
-
color: rgb(112, 76, 182);
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
@keyframes App-logo-float {
|
|
30
|
-
0% {
|
|
31
|
-
transform: translateY(0);
|
|
32
|
-
}
|
|
33
|
-
50% {
|
|
34
|
-
transform: translateY(10px);
|
|
35
|
-
}
|
|
36
|
-
100% {
|
|
37
|
-
transform: translateY(0px);
|
|
38
|
-
}
|
|
39
|
-
}
|
|
1
|
+
.App {
|
|
2
|
+
text-align: center;
|
|
3
|
+
}
|
|
4
|
+
|
|
5
|
+
.App-logo {
|
|
6
|
+
height: 40vmin;
|
|
7
|
+
pointer-events: none;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
@media (prefers-reduced-motion: no-preference) {
|
|
11
|
+
.App-logo {
|
|
12
|
+
animation: App-logo-float infinite 3s ease-in-out;
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
.App-header {
|
|
17
|
+
min-height: 100vh;
|
|
18
|
+
display: flex;
|
|
19
|
+
flex-direction: column;
|
|
20
|
+
align-items: center;
|
|
21
|
+
justify-content: center;
|
|
22
|
+
font-size: calc(10px + 2vmin);
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
.App-link {
|
|
26
|
+
color: rgb(112, 76, 182);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
@keyframes App-logo-float {
|
|
30
|
+
0% {
|
|
31
|
+
transform: translateY(0);
|
|
32
|
+
}
|
|
33
|
+
50% {
|
|
34
|
+
transform: translateY(10px);
|
|
35
|
+
}
|
|
36
|
+
100% {
|
|
37
|
+
transform: translateY(0px);
|
|
38
|
+
}
|
|
39
|
+
}
|