Deploy mirror + built React game (Gemma 3 4B, PORT=7860)
Browse files- .gitattributes +3 -0
- 3d-game/dist/assets/almendra-latin-400-normal-1m0mD8kt.woff +0 -0
- 3d-game/dist/assets/almendra-latin-400-normal-CLadfHti.woff2 +0 -0
- 3d-game/dist/assets/almendra-latin-700-normal-CcSzIg09.woff2 +0 -0
- 3d-game/dist/assets/almendra-latin-700-normal-Kc8Sb4gi.woff +0 -0
- 3d-game/dist/assets/almendra-latin-ext-400-normal-CuH8qsFK.woff2 +0 -0
- 3d-game/dist/assets/almendra-latin-ext-400-normal-Cy7PBgqb.woff +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-700-normal-CnX1aK6s.woff2 +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-700-normal-ov6XPGpn.woff +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-900-normal-BBdTCSMn.woff +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-900-normal-MnIZQgjg.woff2 +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-ext-700-normal-BaOC-94C.woff +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-ext-700-normal-Dul5pKgq.woff2 +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-ext-900-normal-BhxIA4xV.woff2 +0 -0
- 3d-game/dist/assets/cinzel-decorative-latin-ext-900-normal-CTiVNQCf.woff +0 -0
- 3d-game/dist/assets/cinzel-latin-400-normal-C8jUSQqm.woff +0 -0
- 3d-game/dist/assets/cinzel-latin-400-normal-DnUIPmzd.woff2 +0 -0
- 3d-game/dist/assets/cinzel-latin-700-normal-C-gK7hA8.woff +0 -0
- 3d-game/dist/assets/cinzel-latin-700-normal-Dkw14w9r.woff2 +0 -0
- 3d-game/dist/assets/cinzel-latin-900-normal-BI3z7Tow.woff2 +0 -0
- 3d-game/dist/assets/cinzel-latin-900-normal-t_fSDEbn.woff +0 -0
- 3d-game/dist/assets/cinzel-latin-ext-400-normal-DJ0Lq8y-.woff +0 -0
- 3d-game/dist/assets/cinzel-latin-ext-400-normal-XQK_CSAr.woff2 +0 -0
- 3d-game/dist/assets/cinzel-latin-ext-700-normal-C24KFjuG.woff2 +0 -0
- 3d-game/dist/assets/cinzel-latin-ext-700-normal-CORa-yIv.woff +0 -0
- 3d-game/dist/assets/cinzel-latin-ext-900-normal-BlZZvP7K.woff +0 -0
- 3d-game/dist/assets/cinzel-latin-ext-900-normal-CWXxiu5r.woff2 +0 -0
- 3d-game/dist/assets/index-BhyGHDNH.js +0 -0
- 3d-game/dist/assets/index-COcwujCt.css +1 -0
- 3d-game/dist/assets/jim-nightshade-latin-400-normal-BWmRK4d7.woff2 +0 -0
- 3d-game/dist/assets/jim-nightshade-latin-400-normal-igVHAMgk.woff +0 -0
- 3d-game/dist/assets/jim-nightshade-latin-ext-400-normal-CBRHjeH7.woff2 +0 -0
- 3d-game/dist/assets/jim-nightshade-latin-ext-400-normal-MG59TXRF.woff +0 -0
- 3d-game/dist/favicon.svg +1 -0
- 3d-game/dist/icons.svg +24 -0
- 3d-game/dist/index.html +97 -0
- 3d-game/dist/models/characters/ATTRIBUTION.md +20 -0
- 3d-game/dist/models/characters/robot_expressive.glb +3 -0
- 3d-game/dist/models/characters/soldier.glb +3 -0
- 3d-game/dist/models/characters/xbot.glb +3 -0
- 3d_scene.html +0 -0
- Dockerfile +4 -0
- README.md +9 -78
- app.py +392 -696
- static/assets/index-DUDqKLMt.css +1 -0
- static/assets/index-Dp4pBtge.js +0 -0
- static/favicon.svg +1 -0
- static/icons.svg +24 -0
- static/index.html +30 -0
- three.min.js +0 -0
.gitattributes
CHANGED
|
@@ -33,3 +33,6 @@ saved_model/**/* filter=lfs diff=lfs merge=lfs -text
|
|
| 33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
| 34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
| 35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
|
|
|
|
|
|
|
|
|
|
|
| 33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
| 34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
| 35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
| 36 |
+
3d-game/dist/models/characters/robot_expressive.glb filter=lfs diff=lfs merge=lfs -text
|
| 37 |
+
3d-game/dist/models/characters/soldier.glb filter=lfs diff=lfs merge=lfs -text
|
| 38 |
+
3d-game/dist/models/characters/xbot.glb filter=lfs diff=lfs merge=lfs -text
|
3d-game/dist/assets/almendra-latin-400-normal-1m0mD8kt.woff
ADDED
|
Binary file (16 kB). View file
|
|
|
3d-game/dist/assets/almendra-latin-400-normal-CLadfHti.woff2
ADDED
|
Binary file (12.2 kB). View file
|
|
|
3d-game/dist/assets/almendra-latin-700-normal-CcSzIg09.woff2
ADDED
|
Binary file (11.5 kB). View file
|
|
|
3d-game/dist/assets/almendra-latin-700-normal-Kc8Sb4gi.woff
ADDED
|
Binary file (15.1 kB). View file
|
|
|
3d-game/dist/assets/almendra-latin-ext-400-normal-CuH8qsFK.woff2
ADDED
|
Binary file (6.73 kB). View file
|
|
|
3d-game/dist/assets/almendra-latin-ext-400-normal-Cy7PBgqb.woff
ADDED
|
Binary file (8.96 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-700-normal-CnX1aK6s.woff2
ADDED
|
Binary file (15.5 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-700-normal-ov6XPGpn.woff
ADDED
|
Binary file (18.3 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-900-normal-BBdTCSMn.woff
ADDED
|
Binary file (17.4 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-900-normal-MnIZQgjg.woff2
ADDED
|
Binary file (14.6 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-ext-700-normal-BaOC-94C.woff
ADDED
|
Binary file (10.5 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-ext-700-normal-Dul5pKgq.woff2
ADDED
|
Binary file (8.6 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-ext-900-normal-BhxIA4xV.woff2
ADDED
|
Binary file (8.06 kB). View file
|
|
|
3d-game/dist/assets/cinzel-decorative-latin-ext-900-normal-CTiVNQCf.woff
ADDED
|
Binary file (9.99 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-400-normal-C8jUSQqm.woff
ADDED
|
Binary file (16.6 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-400-normal-DnUIPmzd.woff2
ADDED
|
Binary file (14.1 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-700-normal-C-gK7hA8.woff
ADDED
|
Binary file (17.5 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-700-normal-Dkw14w9r.woff2
ADDED
|
Binary file (15.2 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-900-normal-BI3z7Tow.woff2
ADDED
|
Binary file (14.8 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-900-normal-t_fSDEbn.woff
ADDED
|
Binary file (17.1 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-ext-400-normal-DJ0Lq8y-.woff
ADDED
|
Binary file (9.29 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-ext-400-normal-XQK_CSAr.woff2
ADDED
|
Binary file (7.86 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-ext-700-normal-C24KFjuG.woff2
ADDED
|
Binary file (8.26 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-ext-700-normal-CORa-yIv.woff
ADDED
|
Binary file (9.81 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-ext-900-normal-BlZZvP7K.woff
ADDED
|
Binary file (9.54 kB). View file
|
|
|
3d-game/dist/assets/cinzel-latin-ext-900-normal-CWXxiu5r.woff2
ADDED
|
Binary file (8.06 kB). View file
|
|
|
3d-game/dist/assets/index-BhyGHDNH.js
ADDED
|
The diff for this file is too large to render.
See raw diff
|
|
|
3d-game/dist/assets/index-COcwujCt.css
ADDED
|
@@ -0,0 +1 @@
|
|
|
|
|
|
|
| 1 |
+
@font-face{font-family:Cinzel;font-style:normal;font-display:swap;font-weight:400;src:url(./cinzel-latin-ext-400-normal-XQK_CSAr.woff2)format("woff2"),url(./cinzel-latin-ext-400-normal-DJ0Lq8y-.woff)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Cinzel;font-style:normal;font-display:swap;font-weight:400;src:url(./cinzel-latin-400-normal-DnUIPmzd.woff2)format("woff2"),url(./cinzel-latin-400-normal-C8jUSQqm.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Cinzel;font-style:normal;font-display:swap;font-weight:700;src:url(./cinzel-latin-ext-700-normal-C24KFjuG.woff2)format("woff2"),url(./cinzel-latin-ext-700-normal-CORa-yIv.woff)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Cinzel;font-style:normal;font-display:swap;font-weight:700;src:url(./cinzel-latin-700-normal-Dkw14w9r.woff2)format("woff2"),url(./cinzel-latin-700-normal-C-gK7hA8.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Cinzel;font-style:normal;font-display:swap;font-weight:900;src:url(./cinzel-latin-ext-900-normal-CWXxiu5r.woff2)format("woff2"),url(./cinzel-latin-ext-900-normal-BlZZvP7K.woff)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Cinzel;font-style:normal;font-display:swap;font-weight:900;src:url(./cinzel-latin-900-normal-BI3z7Tow.woff2)format("woff2"),url(./cinzel-latin-900-normal-t_fSDEbn.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Cinzel Decorative;font-style:normal;font-display:swap;font-weight:700;src:url(./cinzel-decorative-latin-ext-700-normal-Dul5pKgq.woff2)format("woff2"),url(./cinzel-decorative-latin-ext-700-normal-BaOC-94C.woff)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Cinzel Decorative;font-style:normal;font-display:swap;font-weight:700;src:url(./cinzel-decorative-latin-700-normal-CnX1aK6s.woff2)format("woff2"),url(./cinzel-decorative-latin-700-normal-ov6XPGpn.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Cinzel Decorative;font-style:normal;font-display:swap;font-weight:900;src:url(./cinzel-decorative-latin-ext-900-normal-BhxIA4xV.woff2)format("woff2"),url(./cinzel-decorative-latin-ext-900-normal-CTiVNQCf.woff)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Cinzel Decorative;font-style:normal;font-display:swap;font-weight:900;src:url(./cinzel-decorative-latin-900-normal-MnIZQgjg.woff2)format("woff2"),url(./cinzel-decorative-latin-900-normal-BBdTCSMn.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Almendra;font-style:normal;font-display:swap;font-weight:400;src:url(./almendra-latin-ext-400-normal-CuH8qsFK.woff2)format("woff2"),url(./almendra-latin-ext-400-normal-Cy7PBgqb.woff)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Almendra;font-style:normal;font-display:swap;font-weight:400;src:url(./almendra-latin-400-normal-CLadfHti.woff2)format("woff2"),url(./almendra-latin-400-normal-1m0mD8kt.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Almendra;font-style:normal;font-display:swap;font-weight:700;src:url(data:font/woff2;base64,d09GMgABAAAAAAgsAA8AAAAAEvgAAAfYAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGhYbDBwYBmAAfBEICplAkyILOAABNgIkA2wEIAWDcAeCAQwHG50OUZQNUhXg52FsLH0qjCbNqra42xHUlceL0Gh855QNLYhk2Wb2vneEbE3j6MIiWzEaYZowWIKumiBBGFAIW2943Na/kSNN2hgGc2aAXsysAUZhBWDfNaBX20VzWUHcj06QCbcCtcDhr37/nz9mumi6HVgkD+TOAs9oUW9Zf7ui6gSbXkRSW7VQROj0SOh0SMtToRaov2zuAYEKFmNgLx6dCEpUihl6LSN0ZBgxAhgWT/yOe1mI89YAwIbendsgMVjNgHifNuvZhoYBAun95Pzf4DCnFvErNpHENRBiepjE6KEzidXTGewZe7xtIYdc3g4xU0w3ByKgWlpa3E894OYRNdE0pxJ0Hz39TNu/39X5+Ruv34dqEITVOpl6qOPHsEuM40DyBdmZtiDDKQ0qzegx5t9i3HBh7lqlXmIvqUjM9+WLeL68FEEyP4gvgYNgSTo3jptAYwlUggSeyvkY+gJaxKFud1q3e8KlQ7dibr1sbfoWBxCddmsoJ7CAmrPh0ntkMuS27uKSjnhLoF1mF5kjt/c5bFqzYn+aBHe6cyrFXWoze3RUHOYoIV1cMjrSrXfWW3X7VBClFzlsSxtp7mZggeSZaiGr2c5J5UMyTuEgOCriljrzhhdh+17JNz9H9z5thGAsPuZYWo4oVy841Ww7JXJqpTp82XoJJf5x/4UuiyM9wRmzlwfecsRW3R4udP+2VfWEsMiCI5wQgeWOLjsQsdYBRG491bDDsfTyWHEtaPNz9GEURIS7IQMcGV+Xu02+j0zXOrP2hgMHIbae0q31t0TQc1fyWtdSQr4fTSluRPrYS6HbQ3mWLDTUJVJ+IRuSrsAWcZDbrRHGQ1ZAJ0jYzLRQvmztBstWp1YCWV15irtJKtmQmZh7vVhdOn9gcWZkLefurctOidY5ZS3yrCxMgeNJkPcxBzW3UYUMZkdBrW8q+eFMC+EG8yi3oS0uGVaeYJ+/xcNeuif3uAGOcJdTGSDfknJJwhJ4NiWFDJArhM1LY9F6eh18QopdOoVWL3MQUwk3Uz6Ft8QHgH/HIiJrJjKKDIKP/BbhScIiO2uVDz62tqCa2STHL+ZN8r5Ur0zMNcysnd5VB8NsMLYsZGwRLvKrf+eyjzQ1F71IYt+FZsOnVaF/gqBuXslmQVNnIZYPH7ygrhfzDWahUVyvvnAwH8YKee9q9pmLOj5SLEoKzFhozm9rCSlLBA3Tp7Cnz/pJPG7y6/SvLRJp7QNB7o0krkJ5FXjg8bVLoBwLc3mkg/yWdsu9WkZ5pno9aPFz4UP+e6+ajNfO3jEdnjNvH75kD5o2Zr6lyrtSgh9cVhm+a0R/6tupw2Bgt6ej06uLVmO70om5uP1xFhXa9RZQUyXd13HTqAy6PqgyZ3Rusf1pWSe065nmnQL/DcVElRiakPrPz8tFS5lXJ/GMdYN8id+5jpn3lIZ7y7DvQnH4TCxs1qv8Fq/gWJD4YouyKv7FMu9b4yujhovV60360TodLrQr3rG+dYNWk5PwNiXtpi1cabxbkZCy0/tv/yoaLkW7pUh91mDR4qoxIJ6g53hNRhwrWnQ1j5pMsN8EH7atJZuayD2nS8i+ksCJRr68jKNe+rnQbMgRYedyHgvZ5aq6oysLAs3L+TrltcKn6icqCk8dLN0xt906sX1O6IMCZd8JH2rUXGNrcOWL7cwi1yy+oFKufhl8vv5g/fqEpiOI5VrYgYUXsI3nCjinVEJC+zf2wJYbVb1/oyehdLNoR3lnIVYFjxe5QOvnwvFxyRd0W/DQCmE2hzndvhwH+nSl9Jvb9q3U1hyfC6t4gftblhUJcptn25SVy4VlXGaLfU42UC1OgY3v5SnbUzIJ9YrQNLUa3jmp9InN0WwblqfzolRFabYaP3+OfqIizNAgmLXl4/NCG9IasGKUV9eyJ0VHJ/k0guMkT34VY/XWVDJnlJhJ/eJ9yy6S7vlh2Zz7BaQfLqi0ctZPH8cIKcYgtU5uiqhLTZtOnHV/ObWGVbCkQvoNPhdyyMELKOvbsC1ws1fmbzCP/iUA8HrvoRIAeI96csfzImUmh+kkgCgQnLzXn3CBwGxpA0D0ljFwxhO0BYCZpe5YrABIq+dXCzxv4m8obQkQQTM9n5jleRF0UIwk4AN9CFAzSx87EMrxfBU37eKGAn6sAGivgU6aF4CuAb6hnjeeUA701b4RgxNOsfa6ZWSpAjvomCkCz7yYXaThEgSOeOeSMMPmkmGkG7JmYhk2R0yG2gQACSPUwUwtLvawNkN67GDETI81DP16MLlAFyZ2uTwlph+xB3QwaneLbRADZ2qN7XSzeIgTTAgjIoEf4mnZyXwkJCfusmpoFdMro/GHp88bBdkTMpsZB3FX3GIjIrTB9KGJ4iVQacxpTBTtgqYzhsZYzdCFd+rQH39kZuIXGzDy4SnxNF0GGNIF2mY0AJMG4/bGEd0nVGHFwykROqZD+wR5ndgYIlTG0LWNQGMgQ99AWU3PMofIxj50s/uGiRmz5sxbsGjJshWrfPCp0LoNffTVT39FfJQ3tr5HXIDR3WoYYhWO9PS3d7B6tH3SCotoPb28ot8Px/87vGvwes06ft9vJFMZ2j4owBw57zx5/1dngjZuSbtpxE3J6D8UrLIuYnSkg2XU4qtq8atElBzqdKyqP9kbGVlDXTDcbhp5g5ym94KeBbeZZQ==)format("woff2"),url(data:font/woff;base64,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)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Almendra;font-style:normal;font-display:swap;font-weight:700;src:url(./almendra-latin-700-normal-CcSzIg09.woff2)format("woff2"),url(./almendra-latin-700-normal-Kc8Sb4gi.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Jim Nightshade;font-style:normal;font-display:swap;font-weight:400;src:url(./jim-nightshade-latin-ext-400-normal-CBRHjeH7.woff2)format("woff2"),url(./jim-nightshade-latin-ext-400-normal-MG59TXRF.woff)format("woff");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Jim Nightshade;font-style:normal;font-display:swap;font-weight:400;src:url(./jim-nightshade-latin-400-normal-BWmRK4d7.woff2)format("woff2"),url(./jim-nightshade-latin-400-normal-igVHAMgk.woff)format("woff");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-translate-x:0;--tw-translate-y:0;--tw-translate-z:0;--tw-scale-x:1;--tw-scale-y:1;--tw-scale-z:1;--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-y-reverse:0;--tw-border-style:solid;--tw-gradient-position:initial;--tw-gradient-from:#0000;--tw-gradient-via:#0000;--tw-gradient-to:#0000;--tw-gradient-stops:initial;--tw-gradient-via-stops:initial;--tw-gradient-from-position:0%;--tw-gradient-via-position:50%;--tw-gradient-to-position:100%;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-ordinal:initial;--tw-slashed-zero:initial;--tw-numeric-figure:initial;--tw-numeric-spacing:initial;--tw-numeric-fraction:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial;--tw-ease:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-amber-200:oklch(92.4% .12 95.746);--color-black:#000;--spacing:.25rem;--container-2xl:42rem;--container-6xl:72rem;--container-7xl:80rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-base:1rem;--text-base--line-height:calc(1.5 / 1);--text-lg:1.125rem;--text-lg--line-height:calc(1.75 / 1.125);--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--text-2xl:1.5rem;--text-2xl--line-height:calc(2 / 1.5);--text-3xl:1.875rem;--text-3xl--line-height:calc(2.25 / 1.875);--text-4xl:2.25rem;--text-4xl--line-height:calc(2.5 / 2.25);--text-5xl:3rem;--text-5xl--line-height:1;--text-6xl:3.75rem;--text-6xl--line-height:1;--text-7xl:4.5rem;--text-7xl--line-height:1;--text-8xl:6rem;--text-8xl--line-height:1;--font-weight-semibold:600;--font-weight-bold:700;--font-weight-extrabold:800;--font-weight-black:900;--tracking-normal:0em;--tracking-wide:.025em;--tracking-wider:.05em;--tracking-widest:.1em;--leading-tight:1.25;--leading-snug:1.375;--leading-relaxed:1.625;--radius-sm:.25rem;--drop-shadow-md:0 3px 3px #0000001f;--ease-out:cubic-bezier(0, 0, .2, 1);--ease-in-out:cubic-bezier(.4, 0, .2, 1);--blur-sm:8px;--blur-md:12px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono);--color-ink-900:#05040a;--color-ink-800:#0b0912;--color-ink-700:#181325;--color-ink-600:#251e32;--color-parchment:#f0e6d2;--color-parchment-dim:#b8a88a;--color-parchment-dark:#8a7b63;--color-aged-paper:#d1bfa0;--color-gold:#d4af37;--color-gold-dim:#8a6e20;--color-gold-bright:#f9d76c;--color-player:#2ec4b6;--color-player-deep:#0b4f6c;--color-enemy:#e71d36;--color-enemy-deep:#590d22;--color-mana:#7b68ee;--color-stamina:#f4a261;--color-stamina-exhausted:#c53030;--color-crit:gold;--font-fantasy:"Cinzel", "Cinzel Decorative", "Georgia", "Times New Roman", serif;--font-title:"Cinzel Decorative", "Cinzel", "Georgia", serif;--font-body:"Almendra", "Georgia", "Times New Roman", serif;--font-brush:"Jim Nightshade", "Brush Script MT", cursive;--shadow-gold:0 0 12px 2px #d4af3773, 0 0 4px 1px #d4af37b3;--animate-pulse-glow:pulse-glow 2.5s ease-in-out infinite;--animate-flame:flame .8s ease-in-out infinite alternate}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab, red, red)){::placeholder{color:color-mix(in oklab, currentcolor 50%, transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){appearance:button}::file-selector-button{appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.pointer-events-auto{pointer-events:auto}.pointer-events-none{pointer-events:none}.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.inset-0{inset:0}.inset-1{inset:var(--spacing)}.inset-x-0{inset-inline:0}.-top-3{top:calc(var(--spacing) * -3)}.-top-\[2px\]{top:-2px}.-top-px{top:-1px}.top-0{top:0}.top-1{top:var(--spacing)}.top-1\/2{top:50%}.top-2{top:calc(var(--spacing) * 2)}.top-3{top:calc(var(--spacing) * 3)}.top-5{top:calc(var(--spacing) * 5)}.top-6{top:calc(var(--spacing) * 6)}.-right-12{right:calc(var(--spacing) * -12)}.-right-\[2px\]{right:-2px}.-right-px{right:-1px}.right-1{right:var(--spacing)}.right-2{right:calc(var(--spacing) * 2)}.right-3{right:calc(var(--spacing) * 3)}.right-5{right:calc(var(--spacing) * 5)}.right-6{right:calc(var(--spacing) * 6)}.-bottom-3{bottom:calc(var(--spacing) * -3)}.-bottom-7{bottom:calc(var(--spacing) * -7)}.-bottom-\[2px\]{bottom:-2px}.-bottom-px{bottom:-1px}.bottom-0{bottom:0}.bottom-1{bottom:var(--spacing)}.bottom-2{bottom:calc(var(--spacing) * 2)}.bottom-3{bottom:calc(var(--spacing) * 3)}.bottom-6{bottom:calc(var(--spacing) * 6)}.bottom-20{bottom:calc(var(--spacing) * 20)}.-left-\[2px\]{left:-2px}.-left-px{left:-1px}.left-0{left:0}.left-1{left:var(--spacing)}.left-1\/2{left:50%}.left-2{left:calc(var(--spacing) * 2)}.left-3{left:calc(var(--spacing) * 3)}.left-5{left:calc(var(--spacing) * 5)}.left-6{left:calc(var(--spacing) * 6)}.left-\[1\.35rem\]{left:1.35rem}.left-\[2px\]{left:2px}.-z-10{z-index:calc(10 * -1)}.z-0{z-index:0}.z-10{z-index:10}.z-30{z-index:30}.z-\[1\]{z-index:1}.z-\[2\]{z-index:2}.z-\[3\]{z-index:3}.z-\[24\]{z-index:24}.z-\[25\]{z-index:25}.z-\[35\]{z-index:35}.z-\[90\]{z-index:90}.z-\[100\]{z-index:100}.z-\[110\]{z-index:110}.z-\[120\]{z-index:120}.z-\[150\]{z-index:150}.mx-auto{margin-inline:auto}.my-3{margin-block:calc(var(--spacing) * 3)}.my-6{margin-block:calc(var(--spacing) * 6)}.my-auto{margin-block:auto}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.mt-1{margin-top:var(--spacing)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-6{margin-top:calc(var(--spacing) * 6)}.mt-8{margin-top:calc(var(--spacing) * 8)}.mt-10{margin-top:calc(var(--spacing) * 10)}.mt-14{margin-top:calc(var(--spacing) * 14)}.mb-1{margin-bottom:var(--spacing)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.mb-3{margin-bottom:calc(var(--spacing) * 3)}.mb-4{margin-bottom:calc(var(--spacing) * 4)}.mb-5{margin-bottom:calc(var(--spacing) * 5)}.mb-6{margin-bottom:calc(var(--spacing) * 6)}.mb-8{margin-bottom:calc(var(--spacing) * 8)}.mb-10{margin-bottom:calc(var(--spacing) * 10)}.ml-1{margin-left:var(--spacing)}.ml-1\.5{margin-left:calc(var(--spacing) * 1.5)}.ml-2{margin-left:calc(var(--spacing) * 2)}.ml-6{margin-left:calc(var(--spacing) * 6)}.line-clamp-2{-webkit-line-clamp:2;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.table{display:table}.h-1{height:var(--spacing)}.h-1\.5{height:calc(var(--spacing) * 1.5)}.h-2{height:calc(var(--spacing) * 2)}.h-3{height:calc(var(--spacing) * 3)}.h-4{height:calc(var(--spacing) * 4)}.h-5{height:calc(var(--spacing) * 5)}.h-6{height:calc(var(--spacing) * 6)}.h-8{height:calc(var(--spacing) * 8)}.h-10{height:calc(var(--spacing) * 10)}.h-16{height:calc(var(--spacing) * 16)}.h-32{height:calc(var(--spacing) * 32)}.h-44{height:calc(var(--spacing) * 44)}.h-72{height:calc(var(--spacing) * 72)}.h-\[1px\]{height:1px}.h-\[2px\]{height:2px}.h-\[460px\]{height:460px}.h-\[520px\]{height:520px}.h-\[min\(85vh\,720px\)\]{height:min(85vh,720px)}.h-full{height:100%}.w-2{width:calc(var(--spacing) * 2)}.w-3{width:calc(var(--spacing) * 3)}.w-4{width:calc(var(--spacing) * 4)}.w-5{width:calc(var(--spacing) * 5)}.w-6{width:calc(var(--spacing) * 6)}.w-8{width:calc(var(--spacing) * 8)}.w-10{width:calc(var(--spacing) * 10)}.w-11{width:calc(var(--spacing) * 11)}.w-12{width:calc(var(--spacing) * 12)}.w-16{width:calc(var(--spacing) * 16)}.w-40{width:calc(var(--spacing) * 40)}.w-48{width:calc(var(--spacing) * 48)}.w-56{width:calc(var(--spacing) * 56)}.w-72{width:calc(var(--spacing) * 72)}.w-80{width:calc(var(--spacing) * 80)}.w-\[1px\]{width:1px}.w-\[3px\]{width:3px}.w-\[520px\]{width:520px}.w-\[min\(92vw\,1100px\)\]{width:min(92vw,1100px)}.w-full{width:100%}.max-w-2xl{max-width:var(--container-2xl)}.max-w-6xl{max-width:var(--container-6xl)}.max-w-7xl{max-width:var(--container-7xl)}.min-w-0{min-width:0}.min-w-\[1\.4rem\]{min-width:1.4rem}.min-w-\[1\.6rem\]{min-width:1.6rem}.min-w-\[2ch\]{min-width:2ch}.min-w-\[260px\]{min-width:260px}.flex-1{flex:1}.flex-\[1\.6\]{flex:1.6}.flex-shrink-0,.shrink-0{flex-shrink:0}.-translate-x-1\/2{--tw-translate-x:calc(calc(1 / 2 * 100%) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.-translate-x-full{--tw-translate-x:-100%;translate:var(--tw-translate-x) var(--tw-translate-y)}.-translate-y-1{--tw-translate-y:calc(var(--spacing) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.-translate-y-1\/2{--tw-translate-y:calc(calc(1 / 2 * 100%) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-y-0{--tw-translate-y:0;translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-y-1{--tw-translate-y:var(--spacing);translate:var(--tw-translate-x) var(--tw-translate-y)}.-scale-100{--tw-scale-x:calc(100% * -1);--tw-scale-y:calc(100% * -1);--tw-scale-z:calc(100% * -1);scale:var(--tw-scale-x) var(--tw-scale-y)}.-scale-x-100{--tw-scale-x:calc(100% * -1);scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-x-0{--tw-scale-x:0%;scale:var(--tw-scale-x) var(--tw-scale-y)}.-scale-y-100{--tw-scale-y:calc(100% * -1);scale:var(--tw-scale-x) var(--tw-scale-y)}.-rotate-90{rotate:-90deg}.rotate-90{rotate:90deg}.rotate-180{rotate:180deg}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.text-glow-gold-sheen{color:#0000;text-shadow:0 4px 24px #000000e6;filter:drop-shadow(0 0 10px #d4af3759);background:linear-gradient(100deg,#8a6e20 0%,#d4af37 22%,#f9d76c 48%,#d4af37 72%,#8a6e20 100%) 0 0/200% 100%;-webkit-background-clip:text;background-clip:text;animation:3s linear infinite gold-sheen}.animate-flame{animation:var(--animate-flame)}.animate-pulse-glow{animation:var(--animate-pulse-glow)}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.content-start{align-content:flex-start}.items-baseline{align-items:baseline}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.items-stretch{align-items:stretch}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-0\.5{gap:calc(var(--spacing) * .5)}.gap-1{gap:var(--spacing)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-5{gap:calc(var(--spacing) * 5)}.gap-6{gap:calc(var(--spacing) * 6)}.gap-7{gap:calc(var(--spacing) * 7)}.gap-8{gap:calc(var(--spacing) * 8)}:where(.space-y-1\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1.5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1.5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2.5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2.5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-6>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 6) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 6) * calc(1 - var(--tw-space-y-reverse)))}.self-start{align-self:flex-start}.overflow-hidden{overflow:hidden}.overflow-visible{overflow:visible}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-sm{border-radius:var(--radius-sm)}.rounded-l-sm{border-top-left-radius:var(--radius-sm);border-bottom-left-radius:var(--radius-sm)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-t-2{border-top-style:var(--tw-border-style);border-top-width:2px}.border-r{border-right-style:var(--tw-border-style);border-right-width:1px}.border-r-2{border-right-style:var(--tw-border-style);border-right-width:2px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-b-2{border-bottom-style:var(--tw-border-style);border-bottom-width:2px}.border-l-2{border-left-style:var(--tw-border-style);border-left-width:2px}.border-dashed{--tw-border-style:dashed;border-style:dashed}.border-\[\#6c8e6a\]\/60{border-color:oklab(61.0859% -.0527358 .0392579/.6)}.border-\[\#8a6e20\]{border-color:#8a6e20}.border-\[\#8a6e20\]\/60{border-color:oklab(55.1309% .00315212 .0994465/.6)}.border-\[\#8a7b63\]\/40{border-color:oklab(59.0067% .00723079 .038814/.4)}.border-\[\#8c1a1a\]{border-color:#8c1a1a}.border-\[\#d4af37\]{border-color:#d4af37}.border-\[rgba\(138\,110\,32\,0\.3\)\]{border-color:#8a6e204d}.border-\[rgba\(138\,110\,32\,0\.4\)\]{border-color:#8a6e2066}.border-\[rgba\(138\,110\,32\,0\.6\)\]{border-color:#8a6e2099}.border-\[rgba\(138\,110\,32\,0\.25\)\]{border-color:#8a6e2040}.border-\[rgba\(138\,110\,32\,0\.35\)\]{border-color:#8a6e2059}.border-\[rgba\(138\,110\,32\,0\.45\)\]{border-color:#8a6e2073}.border-\[rgba\(138\,110\,32\,0\.55\)\]{border-color:#8a6e208c}.border-\[rgba\(212\,175\,55\,0\.75\)\]{border-color:#d4af37bf}.border-crit\/60{border-color:#ffd70099}@supports (color:color-mix(in lab, red, red)){.border-crit\/60{border-color:color-mix(in oklab, var(--color-crit) 60%, transparent)}}.border-gold{border-color:var(--color-gold)}.border-gold\/10{border-color:#d4af371a}@supports (color:color-mix(in lab, red, red)){.border-gold\/10{border-color:color-mix(in oklab, var(--color-gold) 10%, transparent)}}.border-gold\/15{border-color:#d4af3726}@supports (color:color-mix(in lab, red, red)){.border-gold\/15{border-color:color-mix(in oklab, var(--color-gold) 15%, transparent)}}.border-gold\/20{border-color:#d4af3733}@supports (color:color-mix(in lab, red, red)){.border-gold\/20{border-color:color-mix(in oklab, var(--color-gold) 20%, transparent)}}.border-gold\/25{border-color:#d4af3740}@supports (color:color-mix(in lab, red, red)){.border-gold\/25{border-color:color-mix(in oklab, var(--color-gold) 25%, transparent)}}.border-gold\/30{border-color:#d4af374d}@supports (color:color-mix(in lab, red, red)){.border-gold\/30{border-color:color-mix(in oklab, var(--color-gold) 30%, transparent)}}.border-gold\/40{border-color:#d4af3766}@supports (color:color-mix(in lab, red, red)){.border-gold\/40{border-color:color-mix(in oklab, var(--color-gold) 40%, transparent)}}.border-gold\/45{border-color:#d4af3773}@supports (color:color-mix(in lab, red, red)){.border-gold\/45{border-color:color-mix(in oklab, var(--color-gold) 45%, transparent)}}.border-gold\/50{border-color:#d4af3780}@supports (color:color-mix(in lab, red, red)){.border-gold\/50{border-color:color-mix(in oklab, var(--color-gold) 50%, transparent)}}.border-gold\/55{border-color:#d4af378c}@supports (color:color-mix(in lab, red, red)){.border-gold\/55{border-color:color-mix(in oklab, var(--color-gold) 55%, transparent)}}.border-gold\/60{border-color:#d4af3799}@supports (color:color-mix(in lab, red, red)){.border-gold\/60{border-color:color-mix(in oklab, var(--color-gold) 60%, transparent)}}.border-gold\/70{border-color:#d4af37b3}@supports (color:color-mix(in lab, red, red)){.border-gold\/70{border-color:color-mix(in oklab, var(--color-gold) 70%, transparent)}}.border-gold\/80{border-color:#d4af37cc}@supports (color:color-mix(in lab, red, red)){.border-gold\/80{border-color:color-mix(in oklab, var(--color-gold) 80%, transparent)}}.border-parchment-dark\/35{border-color:#8a7b6359}@supports (color:color-mix(in lab, red, red)){.border-parchment-dark\/35{border-color:color-mix(in oklab, var(--color-parchment-dark) 35%, transparent)}}.border-parchment-dark\/40{border-color:#8a7b6366}@supports (color:color-mix(in lab, red, red)){.border-parchment-dark\/40{border-color:color-mix(in oklab, var(--color-parchment-dark) 40%, transparent)}}.bg-\[\#3a0a0a\]\/70{background-color:oklab(23.4013% .0680873 .0319487/.7)}.bg-\[\#6c8e6a\]{background-color:#6c8e6a}.bg-\[\#6c8e6a\]\/10{background-color:oklab(61.0859% -.0527358 .0392579/.1)}.bg-\[rgba\(5\,4\,10\,0\.55\)\]{background-color:#05040a8c}.bg-\[rgba\(138\,110\,32\,0\.1\)\]{background-color:#8a6e201a}.bg-\[rgba\(138\,110\,32\,0\.3\)\]{background-color:#8a6e204d}.bg-\[rgba\(209\,191\,160\,0\.5\)\]{background-color:#d1bfa080}.bg-\[rgba\(209\,191\,160\,0\.6\)\]{background-color:#d1bfa099}.bg-\[rgba\(209\,191\,160\,0\.7\)\]{background-color:#d1bfa0b3}.bg-\[rgba\(209\,191\,160\,0\.8\)\]{background-color:#d1bfa0cc}.bg-\[rgba\(212\,175\,55\,0\.18\)\]{background-color:#d4af372e}.bg-\[rgba\(240\,230\,210\,0\.6\)\]{background-color:#f0e6d299}.bg-\[rgba\(240\,230\,210\,0\.7\)\]{background-color:#f0e6d2b3}.bg-\[rgba\(240\,230\,210\,0\.55\)\]{background-color:#f0e6d28c}.bg-\[rgba\(240\,230\,210\,0\.65\)\]{background-color:#f0e6d2a6}.bg-\[rgba\(240\,230\,210\,0\.85\)\]{background-color:#f0e6d2d9}.bg-\[rgba\(240\,230\,210\,0\.95\)\]{background-color:#f0e6d2f2}.bg-crit{background-color:var(--color-crit)}.bg-crit\/10{background-color:#ffd7001a}@supports (color:color-mix(in lab, red, red)){.bg-crit\/10{background-color:color-mix(in oklab, var(--color-crit) 10%, transparent)}}.bg-gold-bright{background-color:var(--color-gold-bright)}.bg-gold\/20{background-color:#d4af3733}@supports (color:color-mix(in lab, red, red)){.bg-gold\/20{background-color:color-mix(in oklab, var(--color-gold) 20%, transparent)}}.bg-gold\/30{background-color:#d4af374d}@supports (color:color-mix(in lab, red, red)){.bg-gold\/30{background-color:color-mix(in oklab, var(--color-gold) 30%, transparent)}}.bg-gold\/40{background-color:#d4af3766}@supports (color:color-mix(in lab, red, red)){.bg-gold\/40{background-color:color-mix(in oklab, var(--color-gold) 40%, transparent)}}.bg-ink-600\/40{background-color:#251e3266}@supports (color:color-mix(in lab, red, red)){.bg-ink-600\/40{background-color:color-mix(in oklab, var(--color-ink-600) 40%, transparent)}}.bg-ink-800\/40{background-color:#0b091266}@supports (color:color-mix(in lab, red, red)){.bg-ink-800\/40{background-color:color-mix(in oklab, var(--color-ink-800) 40%, transparent)}}.bg-ink-800\/60{background-color:#0b091299}@supports (color:color-mix(in lab, red, red)){.bg-ink-800\/60{background-color:color-mix(in oklab, var(--color-ink-800) 60%, transparent)}}.bg-ink-800\/80{background-color:#0b0912cc}@supports (color:color-mix(in lab, red, red)){.bg-ink-800\/80{background-color:color-mix(in oklab, var(--color-ink-800) 80%, transparent)}}.bg-ink-900{background-color:var(--color-ink-900)}.bg-ink-900\/30{background-color:#05040a4d}@supports (color:color-mix(in lab, red, red)){.bg-ink-900\/30{background-color:color-mix(in oklab, var(--color-ink-900) 30%, transparent)}}.bg-ink-900\/60{background-color:#05040a99}@supports (color:color-mix(in lab, red, red)){.bg-ink-900\/60{background-color:color-mix(in oklab, var(--color-ink-900) 60%, transparent)}}.bg-ink-900\/70{background-color:#05040ab3}@supports (color:color-mix(in lab, red, red)){.bg-ink-900\/70{background-color:color-mix(in oklab, var(--color-ink-900) 70%, transparent)}}.bg-ink-900\/80{background-color:#05040acc}@supports (color:color-mix(in lab, red, red)){.bg-ink-900\/80{background-color:color-mix(in oklab, var(--color-ink-900) 80%, transparent)}}.bg-ink-900\/85{background-color:#05040ad9}@supports (color:color-mix(in lab, red, red)){.bg-ink-900\/85{background-color:color-mix(in oklab, var(--color-ink-900) 85%, transparent)}}.bg-ink-900\/95{background-color:#05040af2}@supports (color:color-mix(in lab, red, red)){.bg-ink-900\/95{background-color:color-mix(in oklab, var(--color-ink-900) 95%, transparent)}}.bg-parchment-dim\/70{background-color:#b8a88ab3}@supports (color:color-mix(in lab, red, red)){.bg-parchment-dim\/70{background-color:color-mix(in oklab, var(--color-parchment-dim) 70%, transparent)}}.bg-gradient-to-b{--tw-gradient-position:to bottom in oklab;background-image:linear-gradient(var(--tw-gradient-stops))}.bg-gradient-to-br{--tw-gradient-position:to bottom right in oklab;background-image:linear-gradient(var(--tw-gradient-stops))}.bg-gradient-to-r{--tw-gradient-position:to right in oklab;background-image:linear-gradient(var(--tw-gradient-stops))}.bg-gradient-to-t{--tw-gradient-position:to top in oklab;background-image:linear-gradient(var(--tw-gradient-stops))}.bg-\[linear-gradient\(90deg\,\#8a6e20\,\#d4af37\,\#f1c232\)\]{background-image:linear-gradient(90deg,#8a6e20,#d4af37,#f1c232)}.bg-\[linear-gradient\(90deg\,\#b8860b\,\#d4af37\)\]{background-image:linear-gradient(90deg,#b8860b,#d4af37)}.bg-\[linear-gradient\(135deg\,\#fff8e1\,\#d4af37\)\]{background-image:linear-gradient(135deg,#fff8e1,#d4af37)}.bg-\[linear-gradient\(135deg\,\#fff8e1\,\#f1c232\)\]{background-image:linear-gradient(135deg,#fff8e1,#f1c232)}.bg-\[linear-gradient\(135deg\,rgba\(240\,230\,210\,0\.97\)\,rgba\(209\,191\,160\,0\.94\)\)\]{background-image:linear-gradient(135deg,#f0e6d2f7,#d1bfa0f0)}.from-aged-paper\/90{--tw-gradient-from:#d1bfa0e6}@supports (color:color-mix(in lab, red, red)){.from-aged-paper\/90{--tw-gradient-from:color-mix(in oklab, var(--color-aged-paper) 90%, transparent)}}.from-aged-paper\/90{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-aged-paper\/95{--tw-gradient-from:#d1bfa0f2}@supports (color:color-mix(in lab, red, red)){.from-aged-paper\/95{--tw-gradient-from:color-mix(in oklab, var(--color-aged-paper) 95%, transparent)}}.from-aged-paper\/95{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-gold\/20{--tw-gradient-from:#d4af3733}@supports (color:color-mix(in lab, red, red)){.from-gold\/20{--tw-gradient-from:color-mix(in oklab, var(--color-gold) 20%, transparent)}}.from-gold\/20{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-gold\/40{--tw-gradient-from:#d4af3766}@supports (color:color-mix(in lab, red, red)){.from-gold\/40{--tw-gradient-from:color-mix(in oklab, var(--color-gold) 40%, transparent)}}.from-gold\/40{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-ink-700\/95{--tw-gradient-from:#181325f2}@supports (color:color-mix(in lab, red, red)){.from-ink-700\/95{--tw-gradient-from:color-mix(in oklab, var(--color-ink-700) 95%, transparent)}}.from-ink-700\/95{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-ink-800\/40{--tw-gradient-from:#0b091266}@supports (color:color-mix(in lab, red, red)){.from-ink-800\/40{--tw-gradient-from:color-mix(in oklab, var(--color-ink-800) 40%, transparent)}}.from-ink-800\/40{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-ink-800\/70{--tw-gradient-from:#0b0912b3}@supports (color:color-mix(in lab, red, red)){.from-ink-800\/70{--tw-gradient-from:color-mix(in oklab, var(--color-ink-800) 70%, transparent)}}.from-ink-800\/70{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-ink-800\/95{--tw-gradient-from:#0b0912f2}@supports (color:color-mix(in lab, red, red)){.from-ink-800\/95{--tw-gradient-from:color-mix(in oklab, var(--color-ink-800) 95%, transparent)}}.from-ink-800\/95{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-transparent{--tw-gradient-from:transparent;--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.via-\[rgba\(173\,255\,47\,0\.35\)\]{--tw-gradient-via:#adff2f59;--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-\[rgba\(231\,29\,54\,0\.18\)\]{--tw-gradient-via:#e71d362e;--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-gold{--tw-gradient-via:var(--color-gold);--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-gold\/15{--tw-gradient-via:#d4af3726}@supports (color:color-mix(in lab, red, red)){.via-gold\/15{--tw-gradient-via:color-mix(in oklab, var(--color-gold) 15%, transparent)}}.via-gold\/15{--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-gold\/30{--tw-gradient-via:#d4af374d}@supports (color:color-mix(in lab, red, red)){.via-gold\/30{--tw-gradient-via:color-mix(in oklab, var(--color-gold) 30%, transparent)}}.via-gold\/30{--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-gold\/60{--tw-gradient-via:#d4af3799}@supports (color:color-mix(in lab, red, red)){.via-gold\/60{--tw-gradient-via:color-mix(in oklab, var(--color-gold) 60%, transparent)}}.via-gold\/60{--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-gold\/70{--tw-gradient-via:#d4af37b3}@supports (color:color-mix(in lab, red, red)){.via-gold\/70{--tw-gradient-via:color-mix(in oklab, var(--color-gold) 70%, transparent)}}.via-gold\/70{--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-parchment-dark\/50{--tw-gradient-via:#8a7b6380}@supports (color:color-mix(in lab, red, red)){.via-parchment-dark\/50{--tw-gradient-via:color-mix(in oklab, var(--color-parchment-dark) 50%, transparent)}}.via-parchment-dark\/50{--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.via-parchment\/10{--tw-gradient-via:#f0e6d21a}@supports (color:color-mix(in lab, red, red)){.via-parchment\/10{--tw-gradient-via:color-mix(in oklab, var(--color-parchment) 10%, transparent)}}.via-parchment\/10{--tw-gradient-via-stops:var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-via) var(--tw-gradient-via-position), var(--tw-gradient-to) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-via-stops)}.to-gold\/5{--tw-gradient-to:#d4af370d}@supports (color:color-mix(in lab, red, red)){.to-gold\/5{--tw-gradient-to:color-mix(in oklab, var(--color-gold) 5%, transparent)}}.to-gold\/5{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-ink-800\/95{--tw-gradient-to:#0b0912f2}@supports (color:color-mix(in lab, red, red)){.to-ink-800\/95{--tw-gradient-to:color-mix(in oklab, var(--color-ink-800) 95%, transparent)}}.to-ink-800\/95{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-ink-900\/60{--tw-gradient-to:#05040a99}@supports (color:color-mix(in lab, red, red)){.to-ink-900\/60{--tw-gradient-to:color-mix(in oklab, var(--color-ink-900) 60%, transparent)}}.to-ink-900\/60{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-ink-900\/80{--tw-gradient-to:#05040acc}@supports (color:color-mix(in lab, red, red)){.to-ink-900\/80{--tw-gradient-to:color-mix(in oklab, var(--color-ink-900) 80%, transparent)}}.to-ink-900\/80{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-ink-900\/95{--tw-gradient-to:#05040af2}@supports (color:color-mix(in lab, red, red)){.to-ink-900\/95{--tw-gradient-to:color-mix(in oklab, var(--color-ink-900) 95%, transparent)}}.to-ink-900\/95{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-parchment-dark\/30{--tw-gradient-to:#8a7b634d}@supports (color:color-mix(in lab, red, red)){.to-parchment-dark\/30{--tw-gradient-to:color-mix(in oklab, var(--color-parchment-dark) 30%, transparent)}}.to-parchment-dark\/30{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-parchment\/90{--tw-gradient-to:#f0e6d2e6}@supports (color:color-mix(in lab, red, red)){.to-parchment\/90{--tw-gradient-to:color-mix(in oklab, var(--color-parchment) 90%, transparent)}}.to-parchment\/90{--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-transparent{--tw-gradient-to:transparent;--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.p-1{padding:var(--spacing)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.p-8{padding:calc(var(--spacing) * 8)}.p-10{padding:calc(var(--spacing) * 10)}.px-1{padding-inline:var(--spacing)}.px-1\.5{padding-inline:calc(var(--spacing) * 1.5)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.px-6{padding-inline:calc(var(--spacing) * 6)}.px-8{padding-inline:calc(var(--spacing) * 8)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:var(--spacing)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-4{padding-block:calc(var(--spacing) * 4)}.py-6{padding-block:calc(var(--spacing) * 6)}.py-8{padding-block:calc(var(--spacing) * 8)}.pt-2{padding-top:calc(var(--spacing) * 2)}.pt-3{padding-top:calc(var(--spacing) * 3)}.pt-5{padding-top:calc(var(--spacing) * 5)}.pt-10{padding-top:calc(var(--spacing) * 10)}.pr-0{padding-right:0}.pr-1{padding-right:var(--spacing)}.pb-1{padding-bottom:var(--spacing)}.pb-2{padding-bottom:calc(var(--spacing) * 2)}.pb-8{padding-bottom:calc(var(--spacing) * 8)}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.font-body{font-family:var(--font-body)}.font-brush{font-family:var(--font-brush)}.font-title{font-family:var(--font-title)}.text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.text-3xl{font-size:var(--text-3xl);line-height:var(--tw-leading,var(--text-3xl--line-height))}.text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}.text-5xl{font-size:var(--text-5xl);line-height:var(--tw-leading,var(--text-5xl--line-height))}.text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}.text-7xl{font-size:var(--text-7xl);line-height:var(--tw-leading,var(--text-7xl--line-height))}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[0\.7rem\]{font-size:.7rem}.text-\[0\.62rem\]{font-size:.62rem}.text-\[0\.65rem\]{font-size:.65rem}.text-\[0\.95rem\]{font-size:.95rem}.text-\[1\.5rem\]{font-size:1.5rem}.text-\[2\.1rem\]{font-size:2.1rem}.text-\[2\.5rem\]{font-size:2.5rem}.text-\[9px\]{font-size:9px}.text-\[10px\]{font-size:10px}.text-\[11px\]{font-size:11px}.text-\[12px\]{font-size:12px}.text-\[clamp\(1rem\,2\.5vw\,1\.5rem\)\]{font-size:clamp(1rem,2.5vw,1.5rem)}.text-\[clamp\(3rem\,12vw\,10rem\)\]{font-size:clamp(3rem,12vw,10rem)}.leading-none{--tw-leading:1;line-height:1}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.leading-snug{--tw-leading:var(--leading-snug);line-height:var(--leading-snug)}.leading-tight{--tw-leading:var(--leading-tight);line-height:var(--leading-tight)}.font-black{--tw-font-weight:var(--font-weight-black);font-weight:var(--font-weight-black)}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-extrabold{--tw-font-weight:var(--font-weight-extrabold);font-weight:var(--font-weight-extrabold)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-\[0\.2em\]{--tw-tracking:.2em;letter-spacing:.2em}.tracking-\[0\.3em\]{--tw-tracking:.3em;letter-spacing:.3em}.tracking-\[0\.4em\]{--tw-tracking:.4em;letter-spacing:.4em}.tracking-\[0\.5em\]{--tw-tracking:.5em;letter-spacing:.5em}.tracking-\[0\.12em\]{--tw-tracking:.12em;letter-spacing:.12em}.tracking-\[0\.15em\]{--tw-tracking:.15em;letter-spacing:.15em}.tracking-\[0\.18em\]{--tw-tracking:.18em;letter-spacing:.18em}.tracking-\[0\.25em\]{--tw-tracking:.25em;letter-spacing:.25em}.tracking-\[0\.28em\]{--tw-tracking:.28em;letter-spacing:.28em}.tracking-\[0\.35em\]{--tw-tracking:.35em;letter-spacing:.35em}.tracking-\[0\.45em\]{--tw-tracking:.45em;letter-spacing:.45em}.tracking-normal{--tw-tracking:var(--tracking-normal);letter-spacing:var(--tracking-normal)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.tracking-widest{--tw-tracking:var(--tracking-widest);letter-spacing:var(--tracking-widest)}.whitespace-nowrap{white-space:nowrap}.text-\[\#3a2e1a\]{color:#3a2e1a}.text-\[\#5c4d3c\]{color:#5c4d3c}.text-\[\#6c8e6a\]{color:#6c8e6a}.text-\[\#8a6e20\]{color:#8a6e20}.text-\[\#8a7b63\]{color:#8a7b63}.text-\[\#05040a\]{color:#05040a}.text-\[\#a89878\]{color:#a89878}.text-\[\#b8a88a\]{color:#b8a88a}.text-\[\#d4af37\]{color:#d4af37}.text-\[\#d9a3a3\]{color:#d9a3a3}.text-\[\#e71d36\]{color:#e71d36}.text-\[var\(--color-parchment\)\]{color:var(--color-parchment)}.text-\[var\(--color-parchment-dim\)\]{color:var(--color-parchment-dim)}.text-amber-200{color:var(--color-amber-200)}.text-crit{color:var(--color-crit)}.text-enemy{color:var(--color-enemy)}.text-gold{color:var(--color-gold)}.text-gold-bright{color:var(--color-gold-bright)}.text-gold-dim{color:var(--color-gold-dim)}.text-gold\/30{color:#d4af374d}@supports (color:color-mix(in lab, red, red)){.text-gold\/30{color:color-mix(in oklab, var(--color-gold) 30%, transparent)}}.text-gold\/35{color:#d4af3759}@supports (color:color-mix(in lab, red, red)){.text-gold\/35{color:color-mix(in oklab, var(--color-gold) 35%, transparent)}}.text-gold\/55{color:#d4af378c}@supports (color:color-mix(in lab, red, red)){.text-gold\/55{color:color-mix(in oklab, var(--color-gold) 55%, transparent)}}.text-gold\/60{color:#d4af3799}@supports (color:color-mix(in lab, red, red)){.text-gold\/60{color:color-mix(in oklab, var(--color-gold) 60%, transparent)}}.text-ink-700{color:var(--color-ink-700)}.text-ink-900{color:var(--color-ink-900)}.text-parchment{color:var(--color-parchment)}.text-parchment-dark{color:var(--color-parchment-dark)}.text-parchment-dark\/35{color:#8a7b6359}@supports (color:color-mix(in lab, red, red)){.text-parchment-dark\/35{color:color-mix(in oklab, var(--color-parchment-dark) 35%, transparent)}}.text-parchment-dark\/70{color:#8a7b63b3}@supports (color:color-mix(in lab, red, red)){.text-parchment-dark\/70{color:color-mix(in oklab, var(--color-parchment-dark) 70%, transparent)}}.text-parchment-dim{color:var(--color-parchment-dim)}.text-parchment\/80{color:#f0e6d2cc}@supports (color:color-mix(in lab, red, red)){.text-parchment\/80{color:color-mix(in oklab, var(--color-parchment) 80%, transparent)}}.text-parchment\/90{color:#f0e6d2e6}@supports (color:color-mix(in lab, red, red)){.text-parchment\/90{color:color-mix(in oklab, var(--color-parchment) 90%, transparent)}}.uppercase{text-transform:uppercase}.italic{font-style:italic}.tabular-nums{--tw-numeric-spacing:tabular-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.opacity-0{opacity:0}.opacity-25{opacity:.25}.opacity-30{opacity:.3}.opacity-40{opacity:.4}.opacity-50{opacity:.5}.opacity-60{opacity:.6}.opacity-70{opacity:.7}.opacity-100{opacity:1}.opacity-\[0\.08\]{opacity:.08}.shadow{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-2xl{--tw-shadow:0 25px 50px -12px var(--tw-shadow-color,#00000040);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_6px_rgba\(0\,0\,0\,0\.45\)\]{--tw-shadow:0 0 6px var(--tw-shadow-color,#00000073);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_8px_rgba\(212\,175\,55\,0\.5\)\]{--tw-shadow:0 0 8px var(--tw-shadow-color,#d4af3780);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_8px_rgba\(212\,175\,55\,0\.25\)\]{--tw-shadow:0 0 8px var(--tw-shadow-color,#d4af3740);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_8px_rgba\(212\,175\,55\,0\.55\)\]{--tw-shadow:0 0 8px var(--tw-shadow-color,#d4af378c);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_10px_rgba\(140\,26\,26\,0\.6\)\]{--tw-shadow:0 0 10px var(--tw-shadow-color,#8c1a1a99);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_12px_rgba\(212\,175\,55\,0\.55\)\]{--tw-shadow:0 0 12px var(--tw-shadow-color,#d4af378c);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_14px_rgba\(212\,175\,55\,0\.6\)\]{--tw-shadow:0 0 14px var(--tw-shadow-color,#d4af3799);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_0_24px_rgba\(0\,0\,0\,0\.6\)\]{--tw-shadow:0 0 24px var(--tw-shadow-color,#0009);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_12px_50px_rgba\(0\,0\,0\,0\.55\)\]{--tw-shadow:0 12px 50px var(--tw-shadow-color,#0000008c);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_24px_80px_rgba\(0\,0\,0\,0\.7\)\,inset_0_1px_0_rgba\(255\,255\,255\,0\.25\)\]{--tw-shadow:0 24px 80px var(--tw-shadow-color,#000000b3), inset 0 1px 0 var(--tw-shadow-color,#ffffff40);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[0_24px_80px_rgba\(0\,0\,0\,0\.75\)\]{--tw-shadow:0 24px 80px var(--tw-shadow-color,#000000bf);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-\[inset_0_1px_2px_rgba\(0\,0\,0\,0\.25\)\]{--tw-shadow:inset 0 1px 2px var(--tw-shadow-color,#00000040);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-black\/70{--tw-shadow-color:#000000b3}@supports (color:color-mix(in lab, red, red)){.shadow-black\/70{--tw-shadow-color:color-mix(in oklab, color-mix(in oklab, var(--color-black) 70%, transparent) var(--tw-shadow-alpha), transparent)}}.blur{--tw-blur:blur(8px);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.drop-shadow-\[0_0_18px_rgba\(212\,175\,55\,0\.7\)\]{--tw-drop-shadow-size:drop-shadow(0 0 18px var(--tw-drop-shadow-color,#d4af37b3));--tw-drop-shadow:var(--tw-drop-shadow-size);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.drop-shadow-\[0_0_18px_rgba\(231\,29\,54\,0\.7\)\]{--tw-drop-shadow-size:drop-shadow(0 0 18px var(--tw-drop-shadow-color,#e71d36b3));--tw-drop-shadow:var(--tw-drop-shadow-size);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.drop-shadow-md{--tw-drop-shadow-size:drop-shadow(0 3px 3px var(--tw-drop-shadow-color,#0000001f));--tw-drop-shadow:drop-shadow(var(--drop-shadow-md));filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.filter{filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.backdrop-blur-md{--tw-backdrop-blur:blur(var(--blur-md));-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-blur-sm{--tw-backdrop-blur:blur(var(--blur-sm));-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.transition{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to,opacity,box-shadow,transform,translate,scale,rotate,filter,-webkit-backdrop-filter,backdrop-filter,display,content-visibility,overlay,pointer-events;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-\[background\]{transition-property:background;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-\[left\]{transition-property:left;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-transform{transition-property:transform,translate,scale,rotate;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-75{--tw-duration:75ms;transition-duration:75ms}.duration-300{--tw-duration:.3s;transition-duration:.3s}.duration-500{--tw-duration:.5s;transition-duration:.5s}.duration-700{--tw-duration:.7s;transition-duration:.7s}.duration-1000{--tw-duration:1s;transition-duration:1s}.ease-in-out{--tw-ease:var(--ease-in-out);transition-timing-function:var(--ease-in-out)}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}.select-none{-webkit-user-select:none;user-select:none}.text-glow-enemy{text-shadow:0 0 12px #e71d36b3,0 0 24px #e71d3666}.text-glow-exhausted{text-shadow:0 0 10px #c53030d9,0 0 22px #c5303080,0 2px 6px #000000f2}.text-glow-gold{text-shadow:0 0 6px #d4af37d9,0 0 22px #d4af3780,0 0 48px #d4af3733,0 4px 24px #000000e6}.text-glow-player{text-shadow:0 0 12px #2ec4b6b3,0 0 24px #2ec4b666}.text-glow-stamina{text-shadow:0 0 10px #f4a261b3,0 0 22px #f4a26166,0 2px 6px #000000e6}@media (hover:hover){.group-hover\:translate-x-full:is(:where(.group):hover *){--tw-translate-x:100%;translate:var(--tw-translate-x) var(--tw-translate-y)}.group-hover\:scale-x-100:is(:where(.group):hover *){--tw-scale-x:100%;scale:var(--tw-scale-x) var(--tw-scale-y)}.group-hover\:border-\[\#8a7b63\]\/70:is(:where(.group):hover *){border-color:oklab(59.0067% .00723079 .038814/.7)}.group-hover\:border-gold:is(:where(.group):hover *){border-color:var(--color-gold)}.group-hover\:border-parchment-dim\/70:is(:where(.group):hover *){border-color:#b8a88ab3}@supports (color:color-mix(in lab, red, red)){.group-hover\:border-parchment-dim\/70:is(:where(.group):hover *){border-color:color-mix(in oklab, var(--color-parchment-dim) 70%, transparent)}}.group-hover\:from-ink-700\/60:is(:where(.group):hover *){--tw-gradient-from:#18132599}@supports (color:color-mix(in lab, red, red)){.group-hover\:from-ink-700\/60:is(:where(.group):hover *){--tw-gradient-from:color-mix(in oklab, var(--color-ink-700) 60%, transparent)}}.group-hover\:from-ink-700\/60:is(:where(.group):hover *){--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.group-hover\:from-ink-700\/80:is(:where(.group):hover *){--tw-gradient-from:#181325cc}@supports (color:color-mix(in lab, red, red)){.group-hover\:from-ink-700\/80:is(:where(.group):hover *){--tw-gradient-from:color-mix(in oklab, var(--color-ink-700) 80%, transparent)}}.group-hover\:from-ink-700\/80:is(:where(.group):hover *){--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.group-hover\:to-ink-800\/60:is(:where(.group):hover *){--tw-gradient-to:#0b091299}@supports (color:color-mix(in lab, red, red)){.group-hover\:to-ink-800\/60:is(:where(.group):hover *){--tw-gradient-to:color-mix(in oklab, var(--color-ink-800) 60%, transparent)}}.group-hover\:to-ink-800\/60:is(:where(.group):hover *){--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.group-hover\:to-ink-800\/80:is(:where(.group):hover *){--tw-gradient-to:#0b0912cc}@supports (color:color-mix(in lab, red, red)){.group-hover\:to-ink-800\/80:is(:where(.group):hover *){--tw-gradient-to:color-mix(in oklab, var(--color-ink-800) 80%, transparent)}}.group-hover\:to-ink-800\/80:is(:where(.group):hover *){--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.group-hover\:text-\[\#5c4d3c\]:is(:where(.group):hover *){color:#5c4d3c}.group-hover\:text-gold-bright:is(:where(.group):hover *){color:var(--color-gold-bright)}.group-hover\:opacity-100:is(:where(.group):hover *){opacity:1}.group-hover\:shadow-\[0_0_16px_2px_rgba\(184\,168\,138\,0\.18\)\]:is(:where(.group):hover *){--tw-shadow:0 0 16px 2px var(--tw-shadow-color,#b8a88a2e);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.group-hover\:shadow-\[0_0_24px_2px_rgba\(212\,175\,55\,0\.35\)\]:is(:where(.group):hover *){--tw-shadow:0 0 24px 2px var(--tw-shadow-color,#d4af3759);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.group-hover\:\[text-shadow\:0_0_8px_rgba\(184\,168\,138\,0\.45\)\]:is(:where(.group):hover *){text-shadow:0 0 8px #b8a88a73}.group-hover\:\[text-shadow\:0_0_10px_rgba\(212\,175\,55\,0\.7\)\]:is(:where(.group):hover *){text-shadow:0 0 10px #d4af37b3}.group-hover\:\[text-shadow\:0_0_14px_rgba\(212\,175\,55\,0\.9\)\,0_0_28px_rgba\(212\,175\,55\,0\.5\)\]:is(:where(.group):hover *){text-shadow:0 0 14px #d4af37e6,0 0 28px #d4af3780}.hover\:-translate-y-0\.5:hover{--tw-translate-y:calc(var(--spacing) * -.5);translate:var(--tw-translate-x) var(--tw-translate-y)}.hover\:-translate-y-1:hover{--tw-translate-y:calc(var(--spacing) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.hover\:border-\[rgba\(138\,110\,32\,0\.75\)\]:hover{border-color:#8a6e20bf}.hover\:border-\[rgba\(138\,110\,32\,0\.85\)\]:hover{border-color:#8a6e20d9}.hover\:border-gold:hover{border-color:var(--color-gold)}.hover\:border-gold\/50:hover{border-color:#d4af3780}@supports (color:color-mix(in lab, red, red)){.hover\:border-gold\/50:hover{border-color:color-mix(in oklab, var(--color-gold) 50%, transparent)}}.hover\:border-gold\/70:hover{border-color:#d4af37b3}@supports (color:color-mix(in lab, red, red)){.hover\:border-gold\/70:hover{border-color:color-mix(in oklab, var(--color-gold) 70%, transparent)}}.hover\:bg-\[rgba\(240\,230\,210\,0\.8\)\]:hover{background-color:#f0e6d2cc}.hover\:bg-\[rgba\(240\,230\,210\,0\.9\)\]:hover{background-color:#f0e6d2e6}.hover\:bg-ink-700\/70:hover{background-color:#181325b3}@supports (color:color-mix(in lab, red, red)){.hover\:bg-ink-700\/70:hover{background-color:color-mix(in oklab, var(--color-ink-700) 70%, transparent)}}.hover\:text-\[\#590d22\]:hover{color:#590d22}.hover\:text-\[\#05040a\]:hover{color:#05040a}.hover\:text-gold-bright:hover{color:var(--color-gold-bright)}.hover\:text-parchment:hover{color:var(--color-parchment)}.hover\:shadow-\[0_0_12px_rgba\(212\,175\,55\,0\.45\)\]:hover{--tw-shadow:0 0 12px var(--tw-shadow-color,#d4af3773);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.hover\:shadow-\[0_0_20px_2px_rgba\(212\,175\,55\,0\.25\)\]:hover{--tw-shadow:0 0 20px 2px var(--tw-shadow-color,#d4af3740);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.hover\:shadow-\[0_0_20px_2px_rgba\(212\,175\,55\,0\.35\)\]:hover{--tw-shadow:0 0 20px 2px var(--tw-shadow-color,#d4af3759);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}}.active\:scale-95:active{--tw-scale-x:95%;--tw-scale-y:95%;--tw-scale-z:95%;scale:var(--tw-scale-x) var(--tw-scale-y)}.active\:scale-\[0\.97\]:active{scale:.97}@media (width>=40rem){.sm\:h-52{height:calc(var(--spacing) * 52)}.sm\:h-56{height:calc(var(--spacing) * 56)}.sm\:w-56{width:calc(var(--spacing) * 56)}.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.sm\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.sm\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.sm\:grid-cols-\[1fr\,1fr\]{grid-template-columns:1fr,1fr}.sm\:flex-row{flex-direction:row}.sm\:items-end{align-items:flex-end}.sm\:gap-4{gap:calc(var(--spacing) * 4)}.sm\:gap-14{gap:calc(var(--spacing) * 14)}.sm\:px-10{padding-inline:calc(var(--spacing) * 10)}.sm\:pt-12{padding-top:calc(var(--spacing) * 12)}.sm\:pb-10{padding-bottom:calc(var(--spacing) * 10)}.sm\:text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.sm\:text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}.sm\:text-5xl{font-size:var(--text-5xl);line-height:var(--tw-leading,var(--text-5xl--line-height))}.sm\:text-7xl{font-size:var(--text-7xl);line-height:var(--tw-leading,var(--text-7xl--line-height))}.sm\:text-8xl{font-size:var(--text-8xl);line-height:var(--tw-leading,var(--text-8xl--line-height))}.sm\:text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.sm\:text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.sm\:text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}}@media (width>=48rem){.md\:grid{display:grid}.md\:h-\[420px\]{height:420px}.md\:h-auto{height:auto}.md\:w-1\/2{width:50%}.md\:w-\[460px\]{width:460px}.md\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.md\:flex-row{flex-direction:row}.md\:px-10{padding-inline:calc(var(--spacing) * 10)}.md\:text-5xl{font-size:var(--text-5xl);line-height:var(--tw-leading,var(--text-5xl--line-height))}.md\:text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}.md\:text-8xl{font-size:var(--text-8xl);line-height:var(--tw-leading,var(--text-8xl--line-height))}}@media (width>=64rem){.lg\:w-\[500px\]{width:500px}.lg\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}}}@keyframes pulse-glow{0%,to{filter:drop-shadow(0 0 4px #d4af3799)}50%{filter:drop-shadow(0 0 14px #d4af37f2)}}@keyframes sheen{0%{background-position:-200%}to{background-position:200%}}@keyframes flame{0%{opacity:.85;transform:scale(1)rotate(-1deg)}to{opacity:1;transform:scale(1.08)rotate(1deg)}}@keyframes ink-bleed{0%{clip-path:inset(0 100% 0 0);opacity:0}to{clip-path:inset(0);opacity:1}}@keyframes float-up{0%{opacity:0;transform:translate(-50%,-50%)scale(.4)rotate(-6deg)}15%{opacity:1;transform:translate(-50%,-90%)scale(1.2)rotate(4deg)}to{opacity:0;transform:translate(-50%,-170%)scale(.9)rotate(0)}}@keyframes shake{0%,to{transform:translate(0)}10%{transform:translate(-4px,2px)}20%{transform:translate(4px,-2px)}30%{transform:translate(-3px,-3px)}40%{transform:translate(3px,3px)}50%{transform:translate(-2px,1px)}60%{transform:translate(2px,-1px)}70%{transform:translate(-1px,-2px)}80%{transform:translate(1px,2px)}}@keyframes page-turn{0%{opacity:0;transform:rotateY(-90deg)}to{opacity:1;transform:rotateY(0)}}@keyframes gold-sheen{0%{background-position:0%}to{background-position:200%}}@keyframes drift-ember{0%{opacity:0;transform:translateY(100vh)scale(.6)}10%{opacity:1}85%{opacity:.85}to{transform:translate3d(var(--drift,30px), -12vh, 0) scale(1.2);opacity:0}}@keyframes gold-shower{0%{opacity:0;transform:translateY(-10vh)scale(.4)rotate(0)}10%{opacity:1}90%{opacity:.9}to{transform:translate3d(var(--drift,0), 110vh, 0) scale(1.1) rotate(360deg);opacity:0}}@keyframes rise-laurel{0%{opacity:0;transform:translateY(80px)scale(.85)}60%{opacity:1}to{opacity:1;transform:translateY(0)scale(1)}}@keyframes ambient-breath{0%,to{opacity:.1}50%{opacity:.22}}@keyframes spin-slow{0%{transform:rotate(0)}to{transform:rotate(360deg)}}@keyframes arena-confirm-fade{0%{background:#05040a00}60%{background:var(--arena-accent,#d4af37)}to{background:#05040a}}.gradient-border{position:relative}.gradient-border:before{content:"";border-radius:inherit;background:linear-gradient(135deg, var(--color-gold), var(--color-mana));-webkit-mask-composite:xor;pointer-events:none;-webkit-mask-composite:xor;-webkit-mask-source-type:auto,auto;-webkit-mask-composite:xor;-webkit-mask-source-type:auto,auto;padding:1.5px;position:absolute;inset:0;-webkit-mask-image:linear-gradient(#fff 0 0),linear-gradient(#fff 0 0);mask-image:linear-gradient(#fff 0 0),linear-gradient(#fff 0 0);-webkit-mask-position:0 0,0 0;mask-position:0 0,0 0;-webkit-mask-size:auto,auto;mask-size:auto,auto;-webkit-mask-repeat:repeat,repeat;mask-repeat:repeat,repeat;-webkit-mask-clip:content-box,border-box;mask-clip:content-box,border-box;-webkit-mask-origin:content-box,border-box;mask-origin:content-box,border-box;-webkit-mask-composite:xor;mask-composite:exclude;-webkit-mask-source-type:auto,auto;mask-mode:match-source,match-source}.ornate-corner{position:relative}.ornate-corner:before,.ornate-corner:after{content:"";pointer-events:none;border-style:solid;border-color:#d4af3799;width:16px;height:16px;position:absolute}.ornate-corner:before{border-width:2px 0 0 2px;top:-2px;left:-2px}.ornate-corner:after{border-width:0 2px 2px 0;bottom:-2px;right:-2px}*{box-sizing:border-box;margin:0;padding:0}html,body,#root{background:var(--color-ink-900);width:100%;height:100%;font-family:var(--font-body);color:var(--color-parchment);-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;overflow:hidden}.game-container{width:100vw;height:100vh;position:relative}::-webkit-scrollbar{width:8px}::-webkit-scrollbar-track{background:#05040a4d}::-webkit-scrollbar-thumb{background:#d4af3766;border-radius:4px}::-webkit-scrollbar-thumb:hover{background:#d4af3799}.gpu-idle *,.gpu-idle :before,.gpu-idle :after{animation-play-state:paused!important}.gpu-idle .ember-layer,.gpu-idle .ember,.gpu-idle .ember-ash,.gpu-idle .ember-defeat,.gpu-idle .ember-victory{display:none!important}.gpu-idle .text-glow-gold-sheen,.gpu-idle .ambient-breath,.gpu-idle .drift-ember,.gpu-idle .gold-shower,.gpu-idle .gold-sheen,.gpu-idle .rise-laurel,.gpu-idle .spin-slow,.gpu-idle .arena-confirm-fade{animation:none!important}.gpu-status{z-index:9999;font-family:var(--font-fantasy);letter-spacing:.18em;text-transform:uppercase;pointer-events:none;-webkit-backdrop-filter:blur(6px);backdrop-filter:blur(6px);color:var(--color-parchment);background:#05040a8c;border:1px solid #d4af3759;border-radius:999px;align-items:center;gap:6px;padding:4px 10px;font-size:11px;display:inline-flex;position:fixed;top:12px;right:12px;box-shadow:0 4px 12px #0006}.gpu-status .dot{border-radius:50%;width:8px;height:8px;box-shadow:0 0 8px}.gpu-status.is-idle{color:#6c8e6a;border-color:#6c8e6a80}.gpu-status.is-idle .dot{background:#6c8e6a}.gpu-status.is-active{color:var(--color-crit);border-color:#ffd7008c}.gpu-status.is-active .dot{background:var(--color-crit);animation:1.2s ease-in-out infinite gpu-pulse}@keyframes gpu-pulse{0%,to{opacity:1;transform:scale(1)}50%{opacity:.6;transform:scale(1.4)}}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-scale-x{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-y{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-z{syntax:"*";inherits:false;initial-value:1}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-gradient-position{syntax:"*";inherits:false}@property --tw-gradient-from{syntax:"<color>";inherits:false;initial-value:#0000}@property --tw-gradient-via{syntax:"<color>";inherits:false;initial-value:#0000}@property --tw-gradient-to{syntax:"<color>";inherits:false;initial-value:#0000}@property --tw-gradient-stops{syntax:"*";inherits:false}@property --tw-gradient-via-stops{syntax:"*";inherits:false}@property --tw-gradient-from-position{syntax:"<length-percentage>";inherits:false;initial-value:0%}@property --tw-gradient-via-position{syntax:"<length-percentage>";inherits:false;initial-value:50%}@property --tw-gradient-to-position{syntax:"<length-percentage>";inherits:false;initial-value:100%}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-ordinal{syntax:"*";inherits:false}@property --tw-slashed-zero{syntax:"*";inherits:false}@property --tw-numeric-figure{syntax:"*";inherits:false}@property --tw-numeric-spacing{syntax:"*";inherits:false}@property --tw-numeric-fraction{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@property --tw-ease{syntax:"*";inherits:false}.game-container{background:#05040a;width:100vw;height:100vh;position:relative;overflow:hidden}.game-container canvas{display:block;width:100%!important;height:100%!important}.shake-light{animation:.4s ease-out shake-light}.shake-heavy{animation:.55s ease-out shake-heavy}.shake-crit{animation:.6s cubic-bezier(.22,1,.36,1) shake-crit}@keyframes shake-light{0%,to{transform:translate(0)}10%{transform:translate(-3px,1px)}20%{transform:translate(3px,-1px)}30%{transform:translate(-2px,-2px)}40%{transform:translate(2px,2px)}50%{transform:translate(-1px,1px)}60%{transform:translate(1px,-1px)}70%{transform:translate(-2px,1px)}80%{transform:translate(2px,-1px)}}@keyframes shake-heavy{0%,to{transform:translate(0)}8%{transform:translate(-7px,3px)rotate(-.4deg)}18%{transform:translate(7px,-3px)rotate(.3deg)}28%{transform:translate(-6px,-5px)rotate(-.2deg)}38%{transform:translate(6px,5px)rotate(.3deg)}48%{transform:translate(-4px,3px)rotate(-.2deg)}58%{transform:translate(4px,-3px)rotate(.2deg)}68%{transform:translate(-3px,-4px)}78%{transform:translate(3px,4px)}88%{transform:translate(-2px,2px)}}@keyframes shake-crit{0%,to{transform:translate(0)scale(1)}10%{transform:translate(-10px,5px)scale(1.005)}20%{transform:translate(10px,-5px)scale(1.01)}30%{transform:translate(-8px,-6px)scale(1.005)}40%{transform:translate(8px,6px)scale(1.01)}50%{transform:translate(-5px,4px)scale(1.005)}60%{transform:translate(5px,-4px)}70%{transform:translate(-4px,-5px)}80%{transform:translate(4px,5px)}90%{transform:translate(-2px,2px)}}.crit-flash-base{animation:.22s ease-out forwards crit-flash-base}.crit-flash-radial{animation:.35s ease-out forwards crit-flash-radial}.crit-flash-vignette{background:radial-gradient(#0000 25%,#000000d9 95%);animation:.5s ease-in-out forwards crit-flash-vignette}.crit-ember-burst{animation:.65s ease-out forwards crit-ember-burst}@keyframes crit-flash-base{0%{opacity:.9}to{opacity:0}}@keyframes crit-flash-radial{0%{opacity:0;transform:scale(.4)}35%{opacity:.95}to{opacity:0;transform:scale(1.4)}}@keyframes crit-flash-vignette{0%{opacity:0}18%{opacity:.7}to{opacity:0}}@keyframes crit-ember-burst{0%{opacity:0;transform:scale(.4)rotate(0)}35%{opacity:1;transform:scale(1.15)rotate(20deg)}to{opacity:0;transform:scale(1.6)rotate(40deg)}}.lantern-pulse{animation:.6s ease-in-out infinite lantern-pulse}@keyframes lantern-pulse{0%,to{filter:drop-shadow(0 0 4px #d4af378c)drop-shadow(0 0 1px #ffffff4d)}50%{filter:drop-shadow(0 0 12px #e73c3cd9)drop-shadow(0 0 2px #ffc8508c)}}.lantern-pulse path.lantern-frame{animation:.6s ease-in-out infinite lantern-stroke}@keyframes lantern-stroke{0%,to{stroke:#d4af37}50%{stroke:#e74c3c}}.vignette-pulse{animation:1.4s ease-in-out infinite vignette-pulse}@keyframes vignette-pulse{0%,to{opacity:.35}50%{opacity:.85}}.stamina-exhausted-flicker{animation:.45s steps(2,end) infinite stamina-exhausted-flicker}@keyframes stamina-exhausted-flicker{0%,to{opacity:.85}50%{opacity:.4}}.hit-splash{font-family:var(--font-title);letter-spacing:.1em;text-transform:uppercase;pointer-events:none;z-index:30;text-shadow:0 0 30px,0 4px 24px #000c;font-size:clamp(28px,5vw,64px);font-weight:900;animation:.55s ease-out forwards hitFade;position:absolute;top:38%;left:50%;transform:translate(-50%,-50%)}.hit-splash .crit-star{margin-right:8px;font-size:.8em;animation:.55s ease-out forwards starSpin;display:inline-block}@keyframes hitFade{0%{opacity:1;transform:translate(-50%,-50%)scale(1.3)}to{opacity:0;transform:translate(-50%,-80%)scale(1)}}@keyframes starSpin{0%{transform:rotate(0)scale(0)}50%{transform:rotate(180deg)scale(1.3)}to{transform:rotate(360deg)scale(1)}}.hit-splash-ink{pointer-events:none;z-index:29;opacity:.85;mix-blend-mode:multiply;filter:drop-shadow(0 4px 12px #0009);animation:.55s ease-out forwards hitSplashInk;position:absolute;top:38%;left:50%;transform:translate(-50%,-50%)}@keyframes hitSplashInk{0%{opacity:0;transform:translate(-50%,-50%)scale(.4)rotate(-8deg)}20%{opacity:.9;transform:translate(-50%,-50%)scale(1.1)rotate(4deg)}to{opacity:0;transform:translate(-52%,-64%)scale(1.25)rotate(-2deg)}}.parchment-grain{pointer-events:none;opacity:.06;z-index:5;background-image:url("data:image/svg+xml,%3Csvg viewBox='0 0 200 200' xmlns='http://www.w3.org/2000/svg'%3E%3Cfilter id='noise'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.65' numOctaves='3' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23noise)'/%3E%3C/svg%3E");position:absolute;inset:0}.vignette{pointer-events:none;z-index:4;background:radial-gradient(circle at 50% 45%,#0000 0%,#05040a66 70%,#05040ad9 100%);position:absolute;inset:0}.ember-layer{pointer-events:none;z-index:3;position:absolute;inset:0;overflow:hidden}.ember{opacity:0;background:radial-gradient(circle,#f9d76c 0%,#d4af37 50%,#0000 100%);border-radius:50%;width:3px;height:3px;animation:linear infinite drift-ember;position:absolute;bottom:-10px;box-shadow:0 0 6px #f9d76cd9,0 0 14px #d4af3773}.ember-ash{background:radial-gradient(circle,#b8a88ab3 0%,#504b464d 70%,#0000 100%);box-shadow:0 0 4px #b8a88a66}.ember-defeat{background:radial-gradient(circle,#c14545 0%,#590d22 50%,#0000 100%);box-shadow:0 0 6px #c14545b3,0 0 14px #590d2280}.ember-victory{background:radial-gradient(circle,#fff4b8 0%,#d4af37 45%,#0000 100%);box-shadow:0 0 8px #fff4b8e6,0 0 18px #d4af378c}.filigree-corner{will-change:transform;transition:transform .4s cubic-bezier(.22,1,.36,1)}.card-tilt{transform-style:preserve-3d;transition:transform .5s cubic-bezier(.22,1,.36,1),box-shadow .5s;transform:perspective(1000px)rotateY(0)rotateX(0)}.card-tilt:hover{transform:perspective(1000px)rotateY(4deg)rotateX(-2deg)scale(1.02)}.torn-edge-top,.torn-edge-bottom{pointer-events:none;z-index:5;height:18px;position:absolute;left:0;right:0}.torn-edge-top{background-image:radial-gradient(circle at 6px 18px,#0000 5px,#d1bfa0f0 6px);background-repeat:repeat-x;background-size:14px 18px;top:0;-webkit-mask-image:linear-gradient(#000 0% 60%,#0000 100%);mask-image:linear-gradient(#000 0% 60%,#0000 100%)}.torn-edge-bottom{background-image:radial-gradient(circle at 6px 0,#0000 5px,#d1bfa0f0 6px);background-repeat:repeat-x;background-size:14px 18px;bottom:0;-webkit-mask-image:linear-gradient(#0000 0%,#000 40% 100%);mask-image:linear-gradient(#0000 0%,#000 40% 100%)}.ink-stain{filter:blur(2px);pointer-events:none;background:radial-gradient(circle,#281c0c2e 0%,#281c0c0f 45%,#0000 70%);border-radius:50%;position:absolute}.parchment-cracked{position:relative}.parchment-cracked:before,.parchment-cracked:after{content:"";pointer-events:none;z-index:4;mix-blend-mode:multiply;background:linear-gradient(105deg,#0000 30%,#140e068c 30.5%,#0000 31%),linear-gradient(75deg,#0000 18%,#140e0680 18.4%,#0000 18.8%),linear-gradient(135deg,#0000 55%,#140e0666 55.3%,#0000 55.7%),linear-gradient(45deg,#0000 70%,#140e0666 70.3%,#0000 70.7%);position:absolute;inset:0}.parchment-cracked:after{mix-blend-mode:multiply;background:linear-gradient(160deg,#0000 42%,#00000059 42.4%,#0000 42.8%),linear-gradient(20deg,#0000 88%,#0000004d 88.3%,#0000 88.7%)}.palette-desaturate{filter:saturate(.45)brightness(.85)}.glow-border-gold{isolation:isolate;position:relative}.glow-border-gold:after{content:"";border-radius:inherit;z-index:-1;filter:blur(6px);opacity:.85;background:linear-gradient(120deg,#8a6e20,#f9d76c,#d4af37,#8a6e20) 0 0/300% 100%;animation:3s linear infinite gold-sheen;position:absolute;inset:-3px}.atmos-card-temple{background:radial-gradient(circle at 50% 0,#e71d3659 0%,#0000 60%),linear-gradient(#2a201000 0%,#140c0499 100%),linear-gradient(135deg,#2a2010 0%,#5a3a10 45%,#d4af37 100%)}.atmos-card-bamboo{background:radial-gradient(circle at 30% 80%,#adff2f40 0%,#0000 55%),linear-gradient(#0d1f1700 0%,#050c0a8c 100%),linear-gradient(160deg,#0d1f17 0%,#1c3826 50%,#3a5f40 100%)}.atmos-card-coliseum{background:radial-gradient(circle at 70% 20%,#ff9d3e4d 0%,#0000 55%),linear-gradient(#2a1c4a00 0%,#0c081699 100%),linear-gradient(140deg,#2a1c4a 0%,#6a3520 55%,#b45a2b 100%)}.atmos-card-shrine{background:radial-gradient(circle at 50% 90%,#6ecfff4d 0%,#0000 60%),linear-gradient(#0a142400 0%,#04081099 100%),linear-gradient(150deg,#0a1424 0%,#1a3a52 50%,#3a6b8c 100%)}.arena-confirm-overlay{z-index:200;pointer-events:none;animation:.9s ease-in forwards arena-confirm-fade;position:fixed;inset:0}.float-up{animation:.45s ease-out forwards float-up-card}@keyframes float-up-card{0%{transform:translateY(8px)}to{transform:translateY(0)}}.player-low-hp-vignette{pointer-events:none;z-index:6;background:linear-gradient(90deg,#8c0c168c 0%,#8c0c162e 30%,#0000 55%);position:absolute;inset:0}.btn-base{font-family:var(--font-title);text-transform:uppercase;letter-spacing:.2em;cursor:pointer;-webkit-user-select:none;user-select:none;white-space:nowrap;border:1px solid #0000;border-radius:2px;outline:none;justify-content:center;align-items:center;gap:.5rem;transition:transform .22s,background .26s,color .26s,border-color .26s,box-shadow .26s,filter .26s;display:inline-flex;position:relative}.btn-base:focus-visible{box-shadow:0 0 0 2px #d4af3799}.btn-base:disabled{cursor:not-allowed;opacity:.55;filter:saturate(.6);box-shadow:none!important;transform:none!important}.btn-base:active:not(:disabled){transform:translateY(1px)scale(.97)}.btn-size-sm{min-width:0;padding:.55rem 1rem;font-size:.78rem}.btn-size-md{min-width:160px;padding:.7rem 1.4rem;font-size:.85rem}.btn-size-lg{min-width:220px;padding:.75rem 2rem;font-size:.9rem}.btn-primary{color:var(--color-parchment);background-image:linear-gradient(#d4af3752,#d4af371a);border-color:#d4af37cc;box-shadow:0 0 0 1px #d4af3740,0 0 14px #d4af3740}.btn-primary:hover:not(:disabled){color:var(--color-gold-bright);border-color:var(--color-gold);background-image:linear-gradient(#d4af3773,#d4af372e);box-shadow:0 0 0 1px #d4af3766,0 0 22px 2px #d4af3773}.btn-primary:hover:not(:disabled) .btn-sheen{transform:translate(100%)}.btn-secondary{color:var(--color-parchment-dim);background-image:linear-gradient(#181325d9,#0b0912d9);border-color:#8a7b6399;box-shadow:0 0 12px #0006}.btn-secondary:hover:not(:disabled){color:var(--color-parchment);border-color:var(--color-parchment-dim);box-shadow:0 0 16px 2px #b8a88a38}.btn-secondary:hover:not(:disabled) .btn-sheen{transform:translate(100%)}.btn-danger{color:var(--color-enemy-deep);font-family:var(--font-fantasy);background-color:#590d2224;background-image:linear-gradient(#590d222e,#590d2214);border-color:#590d228c;box-shadow:inset 0 0 0 1px #e71d361f}.btn-danger:hover:not(:disabled){color:#8a0e2a;background-image:linear-gradient(#590d2247,#590d2224);border-color:#e71d3699;box-shadow:0 0 14px #e71d3659,inset 0 0 0 1px #e71d3640}.btn-danger:hover:not(:disabled) .btn-sheen{transform:translate(100%)}.btn-danger:focus-visible{box-shadow:0 0 0 2px #e71d368c}.btn-ghost{color:var(--color-parchment-dim);box-shadow:none;background-color:#0000;border-color:#8a7b6366}.btn-ghost:hover:not(:disabled){color:var(--color-parchment);border-color:var(--color-parchment);background-color:#f0e6d20f;box-shadow:0 0 10px #b8a88a26}.btn-sheen{pointer-events:none;border-radius:inherit;background:linear-gradient(100deg,#0000 0%,#ffffff1f 50%,#0000 100%);transition:transform .7s;position:absolute;inset:0;transform:translate(-100%)}.btn-press{transition:transform .18s,box-shadow .22s}.btn-press:active:not(:disabled){transform:scale(.96)}
|
3d-game/dist/assets/jim-nightshade-latin-400-normal-BWmRK4d7.woff2
ADDED
|
Binary file (50.9 kB). View file
|
|
|
3d-game/dist/assets/jim-nightshade-latin-400-normal-igVHAMgk.woff
ADDED
|
Binary file (60.3 kB). View file
|
|
|
3d-game/dist/assets/jim-nightshade-latin-ext-400-normal-CBRHjeH7.woff2
ADDED
|
Binary file (24.2 kB). View file
|
|
|
3d-game/dist/assets/jim-nightshade-latin-ext-400-normal-MG59TXRF.woff
ADDED
|
Binary file (29.4 kB). View file
|
|
|
3d-game/dist/favicon.svg
ADDED
|
|
3d-game/dist/icons.svg
ADDED
|
|
3d-game/dist/index.html
ADDED
|
@@ -0,0 +1,97 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<!doctype html>
|
| 2 |
+
<html lang="en">
|
| 3 |
+
<head>
|
| 4 |
+
<meta charset="UTF-8" />
|
| 5 |
+
<link rel="icon" type="image/svg+xml" href="./favicon.svg" />
|
| 6 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
| 7 |
+
<meta name="theme-color" content="#05040a" />
|
| 8 |
+
<meta http-equiv="Cache-Control" content="no-cache, no-store, must-revalidate" />
|
| 9 |
+
<meta http-equiv="Pragma" content="no-cache" />
|
| 10 |
+
<title>Duel of Albion</title>
|
| 11 |
+
<style>
|
| 12 |
+
html, body {
|
| 13 |
+
margin: 0;
|
| 14 |
+
padding: 0;
|
| 15 |
+
width: 100%;
|
| 16 |
+
height: 100%;
|
| 17 |
+
background: #05040a;
|
| 18 |
+
color: #f0e6d2;
|
| 19 |
+
font-family: Cinzel, Georgia, serif;
|
| 20 |
+
overflow: hidden;
|
| 21 |
+
}
|
| 22 |
+
#root {
|
| 23 |
+
width: 100vw;
|
| 24 |
+
height: 100vh;
|
| 25 |
+
}
|
| 26 |
+
/* Pre-React splash so the user always sees something even if JS fails. */
|
| 27 |
+
#preload {
|
| 28 |
+
position: fixed;
|
| 29 |
+
inset: 0;
|
| 30 |
+
display: flex;
|
| 31 |
+
flex-direction: column;
|
| 32 |
+
align-items: center;
|
| 33 |
+
justify-content: center;
|
| 34 |
+
background: #05040a;
|
| 35 |
+
z-index: 1;
|
| 36 |
+
transition: opacity 0.4s ease;
|
| 37 |
+
text-align: center;
|
| 38 |
+
padding: 24px;
|
| 39 |
+
}
|
| 40 |
+
#preload .title {
|
| 41 |
+
font-size: clamp(32px, 6vw, 64px);
|
| 42 |
+
letter-spacing: 0.18em;
|
| 43 |
+
color: #d4af37;
|
| 44 |
+
text-shadow: 0 2px 18px rgba(0,0,0,0.9), 0 0 30px rgba(212,175,55,0.35);
|
| 45 |
+
margin: 0 0 12px;
|
| 46 |
+
}
|
| 47 |
+
#preload .sub {
|
| 48 |
+
font-size: clamp(14px, 1.6vw, 18px);
|
| 49 |
+
color: #b8a88a;
|
| 50 |
+
font-style: italic;
|
| 51 |
+
margin: 0;
|
| 52 |
+
}
|
| 53 |
+
#preload.hide { opacity: 0; pointer-events: none; }
|
| 54 |
+
.react-ready #preload { display: none; }
|
| 55 |
+
</style>
|
| 56 |
+
<script type="module" crossorigin src="./assets/index-BhyGHDNH.js"></script>
|
| 57 |
+
<link rel="stylesheet" crossorigin href="./assets/index-COcwujCt.css">
|
| 58 |
+
</head>
|
| 59 |
+
<body>
|
| 60 |
+
<div id="preload">
|
| 61 |
+
<h1 class="title">DUEL OF ALBION</h1>
|
| 62 |
+
<p class="sub">Loading the realm...</p>
|
| 63 |
+
</div>
|
| 64 |
+
<div id="root"></div>
|
| 65 |
+
<script>
|
| 66 |
+
// Hide the preload splash as soon as React mounts anything into #root.
|
| 67 |
+
// We watch for #root to have children and then fade it out.
|
| 68 |
+
(function () {
|
| 69 |
+
var root = document.getElementById('root');
|
| 70 |
+
var pre = document.getElementById('preload');
|
| 71 |
+
function hidePreload() {
|
| 72 |
+
if (pre) pre.classList.add('hide');
|
| 73 |
+
setTimeout(function () {
|
| 74 |
+
document.body.classList.add('react-ready');
|
| 75 |
+
}, 450);
|
| 76 |
+
}
|
| 77 |
+
if (!root) return;
|
| 78 |
+
var obs = new MutationObserver(function () {
|
| 79 |
+
if (root.children.length > 0) {
|
| 80 |
+
hidePreload();
|
| 81 |
+
obs.disconnect();
|
| 82 |
+
}
|
| 83 |
+
});
|
| 84 |
+
obs.observe(root, { childList: true });
|
| 85 |
+
// Safety: if React never mounts within 15s, show a hint.
|
| 86 |
+
setTimeout(function () {
|
| 87 |
+
if (root.children.length === 0) {
|
| 88 |
+
var hint = document.createElement('p');
|
| 89 |
+
hint.style.cssText = 'color:#e71d36;margin-top:24px;font-size:13px;letter-spacing:0.1em;';
|
| 90 |
+
hint.textContent = 'If this persists, open the browser console (F12) for details.';
|
| 91 |
+
pre.appendChild(hint);
|
| 92 |
+
}
|
| 93 |
+
}, 15000);
|
| 94 |
+
})();
|
| 95 |
+
</script>
|
| 96 |
+
</body>
|
| 97 |
+
</html>
|
3d-game/dist/models/characters/ATTRIBUTION.md
ADDED
|
@@ -0,0 +1,20 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
/**
|
| 2 |
+
* ATTRIBUTION
|
| 3 |
+
*
|
| 4 |
+
* Character models in this folder are from three.js's official example
|
| 5 |
+
* glTF asset library (https://github.com/mrdoob/three.js/tree/r184/examples/models/gltf):
|
| 6 |
+
*
|
| 7 |
+
* xbot.glb - "Xbot" humanoid rigged to the Mixamo skeleton
|
| 8 |
+
* (https://github.com/mrdoob/three.js/blob/r184/examples/models/gltf/Xbot.glb)
|
| 9 |
+
* soldier.glb - "Soldier" walking demo character
|
| 10 |
+
* robot_expressive.glb - "RobotExpressive" mech character
|
| 11 |
+
*
|
| 12 |
+
* License: MIT (three.js) -- free for personal and commercial use.
|
| 13 |
+
*
|
| 14 |
+
* The original Quaternius "Ultimate Animated Character" and "Ultimate
|
| 15 |
+
* Monsters" packs were the intended source for human Adventurer + Demon
|
| 16 |
+
* characters, but those packs are distributed only as .zip archives in
|
| 17 |
+
* FBX/OBJ/Blend format and do not ship glTF; the three.js official
|
| 18 |
+
* examples were used as a drop-in substitute. Replace these GLBs with
|
| 19 |
+
* a Quaternius import later if you want a less-generic look.
|
| 20 |
+
*/
|
3d-game/dist/models/characters/robot_expressive.glb
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:047f5e5fb3bb6d378bd1df16ca6137f2a596c99b3a1b5690b4020c05aaf6f319
|
| 3 |
+
size 463988
|
3d-game/dist/models/characters/soldier.glb
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:dfb230fc1f942f259dd00281a1186953ad602fc5d69067ce63e24b2aa439736b
|
| 3 |
+
size 2160468
|
3d-game/dist/models/characters/xbot.glb
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:002f8d269de68e5dce3d25195caf390d1aa359bbfaae3fcf4c8dc78ec36c3ba5
|
| 3 |
+
size 2930032
|
3d_scene.html
ADDED
|
The diff for this file is too large to render.
See raw diff
|
|
|
Dockerfile
CHANGED
|
@@ -2,10 +2,14 @@ FROM pytorch/pytorch:2.4.0-cuda12.1-cudnn9-runtime
|
|
| 2 |
|
| 3 |
WORKDIR /app
|
| 4 |
|
|
|
|
| 5 |
COPY requirements.txt .
|
| 6 |
RUN pip install --no-cache-dir -r requirements.txt --extra-index-url https://download.pytorch.org/whl/cu121
|
| 7 |
|
|
|
|
| 8 |
COPY app.py .
|
|
|
|
|
|
|
| 9 |
|
| 10 |
ENV PORT=7860
|
| 11 |
EXPOSE 7860
|
|
|
|
| 2 |
|
| 3 |
WORKDIR /app
|
| 4 |
|
| 5 |
+
# Install dependencies
|
| 6 |
COPY requirements.txt .
|
| 7 |
RUN pip install --no-cache-dir -r requirements.txt --extra-index-url https://download.pytorch.org/whl/cu121
|
| 8 |
|
| 9 |
+
# Copy app files
|
| 10 |
COPY app.py .
|
| 11 |
+
COPY 3d_scene.html .
|
| 12 |
+
COPY three.min.js .
|
| 13 |
|
| 14 |
ENV PORT=7860
|
| 15 |
EXPOSE 7860
|
README.md
CHANGED
|
@@ -1,85 +1,16 @@
|
|
| 1 |
---
|
| 2 |
-
title:
|
| 3 |
-
emoji:
|
| 4 |
-
colorFrom:
|
| 5 |
-
colorTo:
|
| 6 |
sdk: docker
|
| 7 |
-
hardware: a10g-small
|
| 8 |
-
app_file: app.py
|
| 9 |
pinned: false
|
| 10 |
---
|
| 11 |
|
| 12 |
-
#
|
| 13 |
|
| 14 |
-
A
|
| 15 |
-
|
| 16 |
|
| 17 |
-
##
|
| 18 |
-
|
| 19 |
-
Given the player's last 5 moves, the model recommends a counter-move from
|
| 20 |
-
the 9 legal options (jab, cross, low_kick, roundhouse, uppercut, parry,
|
| 21 |
-
backstep, clinch, throw), conditioned on both fighters' stats
|
| 22 |
-
(speed, power, range, weight, stance), stamina, distance, and round.
|
| 23 |
-
|
| 24 |
-
## API
|
| 25 |
-
|
| 26 |
-
`POST /predict` — counter-move recommendation.
|
| 27 |
-
|
| 28 |
-
```http
|
| 29 |
-
POST /predict
|
| 30 |
-
Content-Type: application/json
|
| 31 |
-
|
| 32 |
-
{
|
| 33 |
-
"sequence": "jab,cross,low_kick,jab,cross",
|
| 34 |
-
"player": {"name": "monk", "speed": 5, "power": 2, "range": 3, "weight": 0.8, "stance": "low", "stamina": 100, "hp": 100},
|
| 35 |
-
"npc": {"name": "brute","speed": 1, "power": 5, "range": 2, "weight": 1.4, "stance": "hunched", "stamina": 100, "hp": 100},
|
| 36 |
-
"round": 3,
|
| 37 |
-
"distance": "close",
|
| 38 |
-
"playerId": "ab12-...", // optional, enables online RL
|
| 39 |
-
"playerPrevMove": "jab" // optional, back-fills the previous log row
|
| 40 |
-
}
|
| 41 |
-
```
|
| 42 |
-
|
| 43 |
-
```json
|
| 44 |
-
{
|
| 45 |
-
"reasoning": "The player is alternating jab and cross before finishing low...",
|
| 46 |
-
"counterMove": "throw",
|
| 47 |
-
"sequence": "jab,cross,low_kick,jab,cross",
|
| 48 |
-
"adapterScope": "user" // "user" once you have a personalised adapter
|
| 49 |
-
}
|
| 50 |
-
```
|
| 51 |
-
|
| 52 |
-
`GET /health` — `{ready, has_token, online_rl_enabled, user_adapters_cached, buffered_users}`
|
| 53 |
-
`GET /me?playerId=...` — `{rounds_logged, next_retrain_in, cooldown_left_sec, adapter_scope, online_rl_enabled}`
|
| 54 |
-
`POST /forget` body `{"playerId": "..."}` — deletes the user's adapter + log (privacy / GDPR).
|
| 55 |
-
|
| 56 |
-
### Online RL
|
| 57 |
-
|
| 58 |
-
If the request includes a `playerId` and the Space was started with the
|
| 59 |
-
`MODAL_WEBHOOK_URL` and `MODAL_WEBHOOK_SECRET` env vars set, the Space
|
| 60 |
-
will:
|
| 61 |
-
|
| 62 |
-
1. Log `(state, model_move, player_next_move)` to the
|
| 63 |
-
`cyber-duel-tiny-logs` Hugging Face dataset (private).
|
| 64 |
-
2. After 25 fresh rows have been flushed, POST to the Modal webhook to
|
| 65 |
-
trigger a per-user DPO retrain.
|
| 66 |
-
3. The new LoRA delta is uploaded to `cyber-duel-tiny-users/<uid>/` and
|
| 67 |
-
loaded on the next `/predict` for that player.
|
| 68 |
-
|
| 69 |
-
The default base adapter (`Sathvik0101/cyber-duel-tiny-adapter`) is used
|
| 70 |
-
as the starting point for every per-user delta, and the global base
|
| 71 |
-
gets refreshed weekly by Modal's `retrain_global_base` job.
|
| 72 |
-
|
| 73 |
-
## Training
|
| 74 |
-
|
| 75 |
-
Trained on Modal with LoRA + DPO (verifiable rewards from the in-game
|
| 76 |
-
combat resolver). See `modal/app.py` in the [training repo](https://huggingface.co/Sathvik0101/cyber-duel-tiny-adapter).
|
| 77 |
-
|
| 78 |
-
## How to redeploy
|
| 79 |
-
|
| 80 |
-
1. Add `HF_TOKEN` as a Space Secret so the gated `gemma-3-270m-it` weights can be downloaded.
|
| 81 |
-
2. (Optional) Add `MODAL_WEBHOOK_URL` and `MODAL_WEBHOOK_SECRET` to enable
|
| 82 |
-
online per-user RL.
|
| 83 |
-
3. Update `ADAPTER_MODEL` env var to point to the latest adapter release.
|
| 84 |
-
4. The Space will hot-reload on push (you may need a manual restart to
|
| 85 |
-
pick up the new code if the env vars change).
|
|
|
|
| 1 |
---
|
| 2 |
+
title: Duel of Albion
|
| 3 |
+
emoji: ⚔️
|
| 4 |
+
colorFrom: purple
|
| 5 |
+
colorTo: blue
|
| 6 |
sdk: docker
|
|
|
|
|
|
|
| 7 |
pinned: false
|
| 8 |
---
|
| 9 |
|
| 10 |
+
# Duel of Albion
|
| 11 |
|
| 12 |
+
A 3D AI-powered cyberpunk fighting game built with FastAPI.
|
| 13 |
+
The AI opponent uses a fine-tuned Gemma 3 4B model to counter your moves in real time.
|
| 14 |
|
| 15 |
+
## Setup
|
| 16 |
+
Add your HF_TOKEN as a Secret in Space settings so the gated base model (google/gemma-3-4b-it) can be downloaded. The fine-tuned adapter is public at Sathvik0101/gemma-3-combat-npc-adapter.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
app.py
CHANGED
|
@@ -1,696 +1,392 @@
|
|
| 1 |
-
|
| 2 |
-
|
| 3 |
-
|
| 4 |
-
|
| 5 |
-
|
| 6 |
-
|
| 7 |
-
|
| 8 |
-
|
| 9 |
-
|
| 10 |
-
|
| 11 |
-
|
| 12 |
-
|
| 13 |
-
|
| 14 |
-
|
| 15 |
-
|
| 16 |
-
|
| 17 |
-
|
| 18 |
-
|
| 19 |
-
|
| 20 |
-
|
| 21 |
-
|
| 22 |
-
|
| 23 |
-
|
| 24 |
-
|
| 25 |
-
|
| 26 |
-
|
| 27 |
-
|
| 28 |
-
|
| 29 |
-
|
| 30 |
-
|
| 31 |
-
|
| 32 |
-
|
| 33 |
-
|
| 34 |
-
|
| 35 |
-
|
| 36 |
-
|
| 37 |
-
|
| 38 |
-
|
| 39 |
-
|
| 40 |
-
|
| 41 |
-
|
| 42 |
-
|
| 43 |
-
|
| 44 |
-
|
| 45 |
-
|
| 46 |
-
|
| 47 |
-
|
| 48 |
-
)
|
| 49 |
-
|
| 50 |
-
|
| 51 |
-
|
| 52 |
-
|
| 53 |
-
|
| 54 |
-
|
| 55 |
-
|
| 56 |
-
|
| 57 |
-
|
| 58 |
-
|
| 59 |
-
|
| 60 |
-
|
| 61 |
-
|
| 62 |
-
|
| 63 |
-
|
| 64 |
-
|
| 65 |
-
|
| 66 |
-
|
| 67 |
-
|
| 68 |
-
|
| 69 |
-
|
| 70 |
-
|
| 71 |
-
|
| 72 |
-
|
| 73 |
-
|
| 74 |
-
|
| 75 |
-
|
| 76 |
-
|
| 77 |
-
|
| 78 |
-
|
| 79 |
-
|
| 80 |
-
|
| 81 |
-
|
| 82 |
-
|
| 83 |
-
|
| 84 |
-
|
| 85 |
-
|
| 86 |
-
|
| 87 |
-
|
| 88 |
-
|
| 89 |
-
|
| 90 |
-
|
| 91 |
-
|
| 92 |
-
|
| 93 |
-
|
| 94 |
-
|
| 95 |
-
|
| 96 |
-
|
| 97 |
-
|
| 98 |
-
|
| 99 |
-
|
| 100 |
-
|
| 101 |
-
|
| 102 |
-
|
| 103 |
-
|
| 104 |
-
|
| 105 |
-
|
| 106 |
-
|
| 107 |
-
|
| 108 |
-
|
| 109 |
-
|
| 110 |
-
|
| 111 |
-
|
| 112 |
-
|
| 113 |
-
|
| 114 |
-
|
| 115 |
-
|
| 116 |
-
|
| 117 |
-
|
| 118 |
-
|
| 119 |
-
|
| 120 |
-
|
| 121 |
-
|
| 122 |
-
|
| 123 |
-
|
| 124 |
-
|
| 125 |
-
|
| 126 |
-
|
| 127 |
-
|
| 128 |
-
|
| 129 |
-
|
| 130 |
-
|
| 131 |
-
|
| 132 |
-
|
| 133 |
-
|
| 134 |
-
|
| 135 |
-
|
| 136 |
-
|
| 137 |
-
|
| 138 |
-
|
| 139 |
-
|
| 140 |
-
|
| 141 |
-
|
| 142 |
-
|
| 143 |
-
|
| 144 |
-
|
| 145 |
-
|
| 146 |
-
|
| 147 |
-
|
| 148 |
-
|
| 149 |
-
|
| 150 |
-
|
| 151 |
-
|
| 152 |
-
|
| 153 |
-
|
| 154 |
-
|
| 155 |
-
|
| 156 |
-
|
| 157 |
-
|
| 158 |
-
|
| 159 |
-
|
| 160 |
-
|
| 161 |
-
|
| 162 |
-
|
| 163 |
-
|
| 164 |
-
|
| 165 |
-
|
| 166 |
-
|
| 167 |
-
|
| 168 |
-
|
| 169 |
-
|
| 170 |
-
|
| 171 |
-
|
| 172 |
-
|
| 173 |
-
|
| 174 |
-
|
| 175 |
-
|
| 176 |
-
|
| 177 |
-
|
| 178 |
-
|
| 179 |
-
|
| 180 |
-
|
| 181 |
-
|
| 182 |
-
|
| 183 |
-
|
| 184 |
-
|
| 185 |
-
|
| 186 |
-
|
| 187 |
-
|
| 188 |
-
|
| 189 |
-
|
| 190 |
-
|
| 191 |
-
|
| 192 |
-
|
| 193 |
-
|
| 194 |
-
|
| 195 |
-
|
| 196 |
-
|
| 197 |
-
|
| 198 |
-
|
| 199 |
-
|
| 200 |
-
|
| 201 |
-
|
| 202 |
-
|
| 203 |
-
|
| 204 |
-
|
| 205 |
-
|
| 206 |
-
|
| 207 |
-
)
|
| 208 |
-
return
|
| 209 |
-
|
| 210 |
-
|
| 211 |
-
|
| 212 |
-
|
| 213 |
-
|
| 214 |
-
|
| 215 |
-
|
| 216 |
-
|
| 217 |
-
|
| 218 |
-
|
| 219 |
-
|
| 220 |
-
|
| 221 |
-
|
| 222 |
-
|
| 223 |
-
|
| 224 |
-
|
| 225 |
-
|
| 226 |
-
|
| 227 |
-
|
| 228 |
-
|
| 229 |
-
|
| 230 |
-
|
| 231 |
-
|
| 232 |
-
|
| 233 |
-
|
| 234 |
-
|
| 235 |
-
|
| 236 |
-
|
| 237 |
-
|
| 238 |
-
|
| 239 |
-
|
| 240 |
-
|
| 241 |
-
|
| 242 |
-
|
| 243 |
-
|
| 244 |
-
|
| 245 |
-
|
| 246 |
-
|
| 247 |
-
|
| 248 |
-
|
| 249 |
-
|
| 250 |
-
|
| 251 |
-
|
| 252 |
-
|
| 253 |
-
|
| 254 |
-
|
| 255 |
-
|
| 256 |
-
|
| 257 |
-
|
| 258 |
-
|
| 259 |
-
|
| 260 |
-
|
| 261 |
-
|
| 262 |
-
)
|
| 263 |
-
|
| 264 |
-
|
| 265 |
-
|
| 266 |
-
|
| 267 |
-
|
| 268 |
-
|
| 269 |
-
|
| 270 |
-
|
| 271 |
-
|
| 272 |
-
|
| 273 |
-
|
| 274 |
-
|
| 275 |
-
|
| 276 |
-
|
| 277 |
-
|
| 278 |
-
|
| 279 |
-
|
| 280 |
-
|
| 281 |
-
|
| 282 |
-
|
| 283 |
-
|
| 284 |
-
|
| 285 |
-
|
| 286 |
-
|
| 287 |
-
# ----
|
| 288 |
-
|
| 289 |
-
|
| 290 |
-
|
| 291 |
-
|
| 292 |
-
|
| 293 |
-
|
| 294 |
-
|
| 295 |
-
|
| 296 |
-
"
|
| 297 |
-
|
| 298 |
-
|
| 299 |
-
|
| 300 |
-
|
| 301 |
-
|
| 302 |
-
|
| 303 |
-
|
| 304 |
-
|
| 305 |
-
|
| 306 |
-
|
| 307 |
-
|
| 308 |
-
|
| 309 |
-
|
| 310 |
-
|
| 311 |
-
|
| 312 |
-
|
| 313 |
-
|
| 314 |
-
|
| 315 |
-
|
| 316 |
-
|
| 317 |
-
|
| 318 |
-
|
| 319 |
-
|
| 320 |
-
|
| 321 |
-
|
| 322 |
-
|
| 323 |
-
|
| 324 |
-
|
| 325 |
-
|
| 326 |
-
|
| 327 |
-
|
| 328 |
-
|
| 329 |
-
|
| 330 |
-
|
| 331 |
-
|
| 332 |
-
|
| 333 |
-
|
| 334 |
-
|
| 335 |
-
|
| 336 |
-
|
| 337 |
-
|
| 338 |
-
|
| 339 |
-
|
| 340 |
-
|
| 341 |
-
|
| 342 |
-
|
| 343 |
-
|
| 344 |
-
|
| 345 |
-
|
| 346 |
-
|
| 347 |
-
|
| 348 |
-
|
| 349 |
-
|
| 350 |
-
|
| 351 |
-
|
| 352 |
-
|
| 353 |
-
|
| 354 |
-
|
| 355 |
-
|
| 356 |
-
|
| 357 |
-
|
| 358 |
-
|
| 359 |
-
|
| 360 |
-
|
| 361 |
-
|
| 362 |
-
|
| 363 |
-
|
| 364 |
-
|
| 365 |
-
|
| 366 |
-
|
| 367 |
-
|
| 368 |
-
|
| 369 |
-
|
| 370 |
-
|
| 371 |
-
|
| 372 |
-
|
| 373 |
-
|
| 374 |
-
|
| 375 |
-
|
| 376 |
-
|
| 377 |
-
|
| 378 |
-
|
| 379 |
-
|
| 380 |
-
|
| 381 |
-
|
| 382 |
-
|
| 383 |
-
|
| 384 |
-
|
| 385 |
-
|
| 386 |
-
|
| 387 |
-
|
| 388 |
-
|
| 389 |
-
|
| 390 |
-
|
| 391 |
-
|
| 392 |
-
|
| 393 |
-
now = time.time()
|
| 394 |
-
with state_lock:
|
| 395 |
-
last_ts = LAST_RETRAIN_REQUEST.get(uid, 0)
|
| 396 |
-
flushed = FLUSHED_COUNT.get(uid, 0)
|
| 397 |
-
if now - last_ts < RETRAIN_COOLDOWN_SEC:
|
| 398 |
-
return
|
| 399 |
-
if flushed < RETRAIN_THRESHOLD:
|
| 400 |
-
return
|
| 401 |
-
with state_lock:
|
| 402 |
-
LAST_RETRAIN_REQUEST[uid] = now
|
| 403 |
-
log.info("Triggering retrain for %s (flushed=%d)", uid, flushed)
|
| 404 |
-
_post_webhook("/retrain", {"uid": uid})
|
| 405 |
-
|
| 406 |
-
|
| 407 |
-
# ---- FastAPI app ----------------------------------------------------------
|
| 408 |
-
app = FastAPI(title="cyber-duel-tiny")
|
| 409 |
-
app.add_middleware(
|
| 410 |
-
CORSMiddleware, allow_origins=["*"], allow_methods=["*"], allow_headers=["*"],
|
| 411 |
-
)
|
| 412 |
-
|
| 413 |
-
|
| 414 |
-
@app.on_event("startup")
|
| 415 |
-
def _startup():
|
| 416 |
-
load_global_model()
|
| 417 |
-
if ONLINE_RL_ENABLED:
|
| 418 |
-
log.info("Online RL enabled (webhook=%s, logs_repo=%s, users_repo=%s)",
|
| 419 |
-
MODAL_WEBHOOK_URL, LOGS_REPO, USERS_REPO)
|
| 420 |
-
else:
|
| 421 |
-
log.info("Online RL disabled (no MODAL_WEBHOOK_URL/SECRET set)")
|
| 422 |
-
|
| 423 |
-
|
| 424 |
-
@app.get("/health")
|
| 425 |
-
def health():
|
| 426 |
-
return {
|
| 427 |
-
"ready": HAS_MODEL,
|
| 428 |
-
"has_token": get_hf_token() is not None,
|
| 429 |
-
"online_rl_enabled": ONLINE_RL_ENABLED,
|
| 430 |
-
"user_adapters_cached": len(USER_ADAPTER_CACHE),
|
| 431 |
-
"buffered_users": len(LOG_BUFFER),
|
| 432 |
-
}
|
| 433 |
-
|
| 434 |
-
|
| 435 |
-
@app.get("/me")
|
| 436 |
-
def me(playerId: str = ""):
|
| 437 |
-
if not ONLINE_RL_ENABLED:
|
| 438 |
-
return {"online_rl_enabled": False}
|
| 439 |
-
if not playerId:
|
| 440 |
-
return {"error": "missing playerId"}
|
| 441 |
-
with state_lock:
|
| 442 |
-
flushed = FLUSHED_COUNT.get(playerId, 0)
|
| 443 |
-
last_ts = LAST_RETRAIN_REQUEST.get(playerId, 0)
|
| 444 |
-
adapter_scope = "user" if playerId in USER_ADAPTER_CACHE else "global"
|
| 445 |
-
next_at = max(0, RETRAIN_THRESHOLD - flushed)
|
| 446 |
-
cooldown_left = max(0, int(RETRAIN_COOLDOWN_SEC - (time.time() - last_ts)))
|
| 447 |
-
return {
|
| 448 |
-
"playerId": playerId,
|
| 449 |
-
"rounds_logged": flushed,
|
| 450 |
-
"next_retrain_in": next_at,
|
| 451 |
-
"cooldown_left_sec": cooldown_left if last_ts else 0,
|
| 452 |
-
"adapter_scope": adapter_scope,
|
| 453 |
-
"online_rl_enabled": True,
|
| 454 |
-
}
|
| 455 |
-
|
| 456 |
-
|
| 457 |
-
@app.post("/forget")
|
| 458 |
-
async def forget(request: Request):
|
| 459 |
-
if not ONLINE_RL_ENABLED:
|
| 460 |
-
return JSONResponse(content={"ok": False, "reason": "online_rl_disabled"},
|
| 461 |
-
status_code=400)
|
| 462 |
-
try:
|
| 463 |
-
data = await request.json()
|
| 464 |
-
except Exception:
|
| 465 |
-
data = {}
|
| 466 |
-
uid = (data or {}).get("playerId", "")
|
| 467 |
-
if not uid or len(uid) < 4 or len(uid) > 64:
|
| 468 |
-
return JSONResponse(content={"ok": False, "reason": "bad playerId"},
|
| 469 |
-
status_code=400)
|
| 470 |
-
evict_user_cache(uid)
|
| 471 |
-
with state_lock:
|
| 472 |
-
LOG_BUFFER.pop(uid, None)
|
| 473 |
-
LAST_ROW.pop(uid, None)
|
| 474 |
-
FLUSHED_COUNT.pop(uid, None)
|
| 475 |
-
LAST_RETRAIN_REQUEST.pop(uid, None)
|
| 476 |
-
LAST_FLUSH_AT.pop(uid, None)
|
| 477 |
-
_post_webhook("/forget", {"uid": uid})
|
| 478 |
-
return {"ok": True, "uid": uid}
|
| 479 |
-
|
| 480 |
-
|
| 481 |
-
@app.post("/predict")
|
| 482 |
-
async def predict(request: Request):
|
| 483 |
-
sequence = ""
|
| 484 |
-
try:
|
| 485 |
-
try:
|
| 486 |
-
data = await request.json()
|
| 487 |
-
except Exception:
|
| 488 |
-
data = {}
|
| 489 |
-
sequence = (data or {}).get("sequence", "")
|
| 490 |
-
state = {
|
| 491 |
-
"sequence": sequence,
|
| 492 |
-
"player": (data or {}).get("player"),
|
| 493 |
-
"npc": (data or {}).get("npc"),
|
| 494 |
-
"round": (data or {}).get("round", 1),
|
| 495 |
-
"distance": (data or {}).get("distance", "close"),
|
| 496 |
-
"playerId": (data or {}).get("playerId", "") or "",
|
| 497 |
-
}
|
| 498 |
-
player_prev_move = (data or {}).get("playerPrevMove", "") or ""
|
| 499 |
-
|
| 500 |
-
if not HAS_MODEL:
|
| 501 |
-
return JSONResponse(
|
| 502 |
-
content={"reasoning": "(model not loaded)", "counterMove": "jab", "sequence": sequence},
|
| 503 |
-
status_code=503,
|
| 504 |
-
)
|
| 505 |
-
|
| 506 |
-
model, scope = get_model_for(state["playerId"] or None)
|
| 507 |
-
text, counter_move = _generate_with_model(model, state)
|
| 508 |
-
reasoning = text.split("counter_move:")[0].strip() if "counter_move:" in text else text.strip()
|
| 509 |
-
|
| 510 |
-
# ---- Online RL bookkeeping (only if a playerId was sent) ----
|
| 511 |
-
if ONLINE_RL_ENABLED and state["playerId"]:
|
| 512 |
-
uid = state["playerId"]
|
| 513 |
-
with state_lock:
|
| 514 |
-
# Backfill the previous row's player_next_move
|
| 515 |
-
if uid in LAST_ROW and player_prev_move in LEGAL_MOVES:
|
| 516 |
-
LAST_ROW[uid]["player_next_move"] = player_prev_move
|
| 517 |
-
# Save THIS row for the next call to backfill
|
| 518 |
-
LAST_ROW[uid] = _state_to_log(state, counter_move)
|
| 519 |
-
# Add a placeholder row carrying the player's own next move (will be
|
| 520 |
-
# overwritten when the next /predict arrives) so we still log even
|
| 521 |
-
# if the user never comes back.
|
| 522 |
-
LOG_BUFFER[uid].append(LAST_ROW[uid])
|
| 523 |
-
buf_size = len(LOG_BUFFER[uid])
|
| 524 |
-
flushed = FLUSHED_COUNT[uid]
|
| 525 |
-
|
| 526 |
-
# Flush if buffer is large or stale
|
| 527 |
-
now = time.time()
|
| 528 |
-
with state_lock:
|
| 529 |
-
last_flush = LAST_FLUSH_AT.get(uid, 0)
|
| 530 |
-
if buf_size >= LOG_BUFFER_FLUSH_ROWS or (
|
| 531 |
-
buf_size > 0 and now - last_flush > LOG_BUFFER_FLUSH_SEC
|
| 532 |
-
):
|
| 533 |
-
with state_lock:
|
| 534 |
-
LAST_FLUSH_AT[uid] = now
|
| 535 |
-
# Flush in a background thread so /predict isn't blocked
|
| 536 |
-
threading.Thread(target=_flush_user_log,
|
| 537 |
-
args=(uid,), daemon=True).start()
|
| 538 |
-
_maybe_trigger_retrain(uid)
|
| 539 |
-
|
| 540 |
-
return JSONResponse(content={
|
| 541 |
-
"reasoning": reasoning,
|
| 542 |
-
"counterMove": counter_move,
|
| 543 |
-
"sequence": sequence,
|
| 544 |
-
"adapterScope": scope,
|
| 545 |
-
})
|
| 546 |
-
except Exception as e:
|
| 547 |
-
log.exception("predict failed: %s", e)
|
| 548 |
-
return JSONResponse(
|
| 549 |
-
content={"reasoning": f"(error: {type(e).__name__}: {e})",
|
| 550 |
-
"counterMove": "jab", "sequence": sequence},
|
| 551 |
-
status_code=500,
|
| 552 |
-
)
|
| 553 |
-
|
| 554 |
-
|
| 555 |
-
# ---- Gradio UI (mounted on top of FastAPI for the Build-with-Gradio hackathon)
|
| 556 |
-
def _gradio_predict(sequence: str, round_n: float, distance: str):
|
| 557 |
-
"""Gradio-friendly wrapper that reuses the exact same inference path as
|
| 558 |
-
/predict — no double HTTP hop, single model instance, single LRU cache."""
|
| 559 |
-
if not HAS_MODEL:
|
| 560 |
-
return (
|
| 561 |
-
"⚠️ **Model not loaded yet.** Hit *Counter* again in a few seconds — "
|
| 562 |
-
"the 270M Gemma + LoRA adapter is warming up on first call.",
|
| 563 |
-
"—",
|
| 564 |
-
"model-not-loaded",
|
| 565 |
-
)
|
| 566 |
-
sequence = (sequence or "").strip()
|
| 567 |
-
if not sequence:
|
| 568 |
-
return "_No sequence provided._", "—", "no-input"
|
| 569 |
-
state = {
|
| 570 |
-
"sequence": sequence,
|
| 571 |
-
"player": dict(DEFAULT_PLAYER),
|
| 572 |
-
"npc": dict(DEFAULT_NPC),
|
| 573 |
-
"round": int(round_n) if round_n else 1,
|
| 574 |
-
"distance": distance or "close",
|
| 575 |
-
"playerId": "",
|
| 576 |
-
}
|
| 577 |
-
try:
|
| 578 |
-
model, scope = get_model_for(None)
|
| 579 |
-
text, counter = _generate_with_model(model, state)
|
| 580 |
-
reasoning = (
|
| 581 |
-
text.split("counter_move:")[0].strip()
|
| 582 |
-
if "counter_move:" in text
|
| 583 |
-
else text.strip()
|
| 584 |
-
)
|
| 585 |
-
if not reasoning:
|
| 586 |
-
reasoning = "_(no reasoning emitted)_"
|
| 587 |
-
return reasoning, counter, scope
|
| 588 |
-
except Exception as e:
|
| 589 |
-
log.exception("gradio predict failed: %s", e)
|
| 590 |
-
return (
|
| 591 |
-
f"⚠️ Inference error: `{type(e).__name__}: {e}`",
|
| 592 |
-
"jab",
|
| 593 |
-
"error",
|
| 594 |
-
)
|
| 595 |
-
|
| 596 |
-
|
| 597 |
-
def _service_status():
|
| 598 |
-
return {
|
| 599 |
-
"ready": HAS_MODEL,
|
| 600 |
-
"base": BASE_MODEL,
|
| 601 |
-
"adapter": ADAPTER_MODEL,
|
| 602 |
-
"online_rl": ONLINE_RL_ENABLED,
|
| 603 |
-
"legal_moves": list(LEGAL_MOVES),
|
| 604 |
-
}
|
| 605 |
-
|
| 606 |
-
|
| 607 |
-
with gr.Blocks(
|
| 608 |
-
title="Cyber Duel Tiny — Combat Advisor",
|
| 609 |
-
theme=gr.themes.Soft(primary_hue="purple", secondary_hue="blue"),
|
| 610 |
-
css="""
|
| 611 |
-
.counter-badge {font-size:1.6em;font-weight:800;letter-spacing:.04em;
|
| 612 |
-
color:#fff;
|
| 613 |
-
background:linear-gradient(135deg,#7c3aed,#2563eb);
|
| 614 |
-
padding:14px 18px;border-radius:12px;text-align:center;
|
| 615 |
-
text-transform:uppercase;}
|
| 616 |
-
.reasoning-box{font-family:ui-monospace,SFMono-Regular,Menlo,monospace;
|
| 617 |
-
font-size:0.95em;}
|
| 618 |
-
.legal-chip{display:inline-block;margin:2px 4px;padding:4px 10px;
|
| 619 |
-
background:#1f2937;color:#c4b5fd;border-radius:999px;
|
| 620 |
-
font-size:0.85em;font-family:ui-monospace,monospace;}
|
| 621 |
-
""",
|
| 622 |
-
) as demo:
|
| 623 |
-
gr.Markdown(
|
| 624 |
-
f"""
|
| 625 |
-
# ⚔️ Cyber Duel Tiny — Combat Advisor
|
| 626 |
-
Fine-tuned **Gemma 3 270M + LoRA** (`{ADAPTER_MODEL}`) trained on procedural
|
| 627 |
-
rollouts from the in-game combat resolver. Given the player's last 5 moves,
|
| 628 |
-
the model recommends one of the 9 legal counter-moves.
|
| 629 |
-
"""
|
| 630 |
-
)
|
| 631 |
-
gr.Markdown(
|
| 632 |
-
"<sub>Legal moves: "
|
| 633 |
-
+ " ".join(f'<span class="legal-chip">{m}</span>' for m in LEGAL_MOVES)
|
| 634 |
-
+ "</sub>"
|
| 635 |
-
)
|
| 636 |
-
|
| 637 |
-
with gr.Row():
|
| 638 |
-
status_box = gr.JSON(label="Service status", scale=2)
|
| 639 |
-
|
| 640 |
-
with gr.Row():
|
| 641 |
-
with gr.Column(scale=1):
|
| 642 |
-
sequence_in = gr.Textbox(
|
| 643 |
-
label="Player's last 5 moves (comma-separated)",
|
| 644 |
-
value="jab,cross,low_kick,roundhouse,uppercut",
|
| 645 |
-
placeholder="e.g. jab,cross,jab,cross,jab",
|
| 646 |
-
)
|
| 647 |
-
with gr.Row():
|
| 648 |
-
round_in = gr.Slider(1, 5, value=1, step=1, label="Round")
|
| 649 |
-
distance_in = gr.Radio(
|
| 650 |
-
["close", "mid", "far"], value="close", label="Distance"
|
| 651 |
-
)
|
| 652 |
-
run_btn = gr.Button("Counter ⚡", variant="primary", size="lg")
|
| 653 |
-
gr.Examples(
|
| 654 |
-
examples=[
|
| 655 |
-
["jab,jab,jab,jab,jab", 1, "close"],
|
| 656 |
-
["uppercut,uppercut,uppercut,uppercut,uppercut", 3, "close"],
|
| 657 |
-
["low_kick,low_kick,roundhouse,roundhouse,uppercut", 2, "mid"],
|
| 658 |
-
["parry,parry,backstep,parry,parry", 4, "mid"],
|
| 659 |
-
["clinch,clinch,clinch,clinch,clinch", 5, "close"],
|
| 660 |
-
],
|
| 661 |
-
inputs=[sequence_in, round_in, distance_in],
|
| 662 |
-
label="Try these patterns",
|
| 663 |
-
)
|
| 664 |
-
with gr.Column(scale=1):
|
| 665 |
-
move_out = gr.Textbox(
|
| 666 |
-
label="Counter move",
|
| 667 |
-
value="—",
|
| 668 |
-
interactive=False,
|
| 669 |
-
elem_classes=["counter-badge"],
|
| 670 |
-
)
|
| 671 |
-
scope_out = gr.Textbox(
|
| 672 |
-
label="Adapter scope", value="—", interactive=False
|
| 673 |
-
)
|
| 674 |
-
reasoning_out = gr.Markdown(
|
| 675 |
-
value="_Press *Counter ⚡* to see the model's reasoning._",
|
| 676 |
-
label="Reasoning",
|
| 677 |
-
elem_classes=["reasoning-box"],
|
| 678 |
-
)
|
| 679 |
-
|
| 680 |
-
run_btn.click(
|
| 681 |
-
_gradio_predict,
|
| 682 |
-
inputs=[sequence_in, round_in, distance_in],
|
| 683 |
-
outputs=[reasoning_out, move_out, scope_out],
|
| 684 |
-
).then(_service_status, outputs=status_box)
|
| 685 |
-
demo.load(_service_status, outputs=status_box)
|
| 686 |
-
|
| 687 |
-
|
| 688 |
-
# Mount Gradio at / on top of the existing FastAPI app.
|
| 689 |
-
# All FastAPI routes (/predict, /health, /me, /forget) keep their original paths
|
| 690 |
-
# — the 3D-game client in the parent project doesn't have to change anything.
|
| 691 |
-
app = gr.mount_gradio_app(app, demo, path="/")
|
| 692 |
-
|
| 693 |
-
|
| 694 |
-
if __name__ == "__main__":
|
| 695 |
-
import uvicorn
|
| 696 |
-
uvicorn.run(app, host="0.0.0.0", port=int(os.environ.get("PORT", 7860)))
|
|
|
|
| 1 |
+
import os
|
| 2 |
+
import sys
|
| 3 |
+
import json
|
| 4 |
+
import logging
|
| 5 |
+
|
| 6 |
+
import gradio as gr
|
| 7 |
+
from fastapi import FastAPI, Request
|
| 8 |
+
from fastapi.responses import JSONResponse, HTMLResponse, FileResponse, RedirectResponse
|
| 9 |
+
from fastapi.staticfiles import StaticFiles
|
| 10 |
+
from fastapi.middleware.cors import CORSMiddleware
|
| 11 |
+
|
| 12 |
+
# ------------------------------------------------------------------
|
| 13 |
+
# ML stack (optional — UI works fine in mock mode without it)
|
| 14 |
+
# ------------------------------------------------------------------
|
| 15 |
+
HAS_ML = False
|
| 16 |
+
try:
|
| 17 |
+
import torch # noqa: F401
|
| 18 |
+
from transformers import AutoTokenizer, AutoModelForCausalLM, AutoConfig # noqa: F401
|
| 19 |
+
from peft import PeftModel # noqa: F401
|
| 20 |
+
HAS_ML = True
|
| 21 |
+
except Exception:
|
| 22 |
+
HAS_ML = False
|
| 23 |
+
|
| 24 |
+
BASE_MODEL = "google/gemma-3-4b-it"
|
| 25 |
+
ADAPTER_MODEL = "Sathvik0101/gemma-3-combat-npc-adapter"
|
| 26 |
+
|
| 27 |
+
# ------------------------------------------------------------------
|
| 28 |
+
# Configuration
|
| 29 |
+
# ------------------------------------------------------------------
|
| 30 |
+
HOST = os.environ.get("HOST", "0.0.0.0")
|
| 31 |
+
PORT = int(os.environ.get("PORT", "7860"))
|
| 32 |
+
SKIP_MODEL_LOAD = os.environ.get("SKIP_MODEL_LOAD", "0") == "1"
|
| 33 |
+
|
| 34 |
+
logging.basicConfig(
|
| 35 |
+
level=os.environ.get("LOG_LEVEL", "INFO"),
|
| 36 |
+
format="%(asctime)s | %(levelname)s | %(message)s",
|
| 37 |
+
)
|
| 38 |
+
log = logging.getLogger("duel-of-albion")
|
| 39 |
+
|
| 40 |
+
|
| 41 |
+
# ------------------------------------------------------------------
|
| 42 |
+
# Token / model state
|
| 43 |
+
# ------------------------------------------------------------------
|
| 44 |
+
def get_hf_token():
|
| 45 |
+
token = os.environ.get("HF_TOKEN") or os.environ.get("HUGGINGFACE_TOKEN")
|
| 46 |
+
if token:
|
| 47 |
+
return token
|
| 48 |
+
token_path = os.path.expanduser("~/.cache/huggingface/token")
|
| 49 |
+
if os.path.exists(token_path):
|
| 50 |
+
try:
|
| 51 |
+
with open(token_path, "r") as f:
|
| 52 |
+
return f.read().strip()
|
| 53 |
+
except Exception:
|
| 54 |
+
pass
|
| 55 |
+
return None
|
| 56 |
+
|
| 57 |
+
|
| 58 |
+
hf_token = get_hf_token()
|
| 59 |
+
HAS_MODEL = False
|
| 60 |
+
MODEL_ERROR = ""
|
| 61 |
+
model = None
|
| 62 |
+
tokenizer = None
|
| 63 |
+
|
| 64 |
+
# ------------------------------------------------------------------
|
| 65 |
+
# Model loading (skipped when SKIP_MODEL_LOAD=1, when transformers/peft
|
| 66 |
+
# is missing, or when there is no HF token — UI testing never needs it)
|
| 67 |
+
# ------------------------------------------------------------------
|
| 68 |
+
if not HAS_ML:
|
| 69 |
+
log.info("ML stack not installed — running in MOCK MODE (UI only).")
|
| 70 |
+
elif SKIP_MODEL_LOAD:
|
| 71 |
+
log.info("SKIP_MODEL_LOAD=1 — skipping model load (UI only).")
|
| 72 |
+
elif not hf_token:
|
| 73 |
+
log.info("No HF_TOKEN found — running in MOCK MODE. Set HF_TOKEN to enable the AI model.")
|
| 74 |
+
else:
|
| 75 |
+
log.info("Loading tokenizer and base model...")
|
| 76 |
+
try:
|
| 77 |
+
from huggingface_hub import snapshot_download
|
| 78 |
+
import torch
|
| 79 |
+
|
| 80 |
+
tokenizer = AutoTokenizer.from_pretrained(BASE_MODEL, token=hf_token)
|
| 81 |
+
|
| 82 |
+
device_arg = "auto" if torch.cuda.is_available() else None
|
| 83 |
+
dtype = torch.bfloat16 if torch.cuda.is_available() else torch.float32
|
| 84 |
+
|
| 85 |
+
config = AutoConfig.from_pretrained(BASE_MODEL, token=hf_token)
|
| 86 |
+
if hasattr(config, "vision_config") and config.vision_config is not None:
|
| 87 |
+
config.vision_config = None
|
| 88 |
+
log.info("Stripped vision_config to force text-only load path.")
|
| 89 |
+
|
| 90 |
+
base_model = AutoModelForCausalLM.from_pretrained(
|
| 91 |
+
BASE_MODEL,
|
| 92 |
+
config=config,
|
| 93 |
+
token=hf_token,
|
| 94 |
+
torch_dtype=dtype,
|
| 95 |
+
device_map=device_arg,
|
| 96 |
+
)
|
| 97 |
+
adapter_path = snapshot_download(repo_id=ADAPTER_MODEL, token=hf_token, force_download=True)
|
| 98 |
+
model = PeftModel.from_pretrained(base_model, adapter_path)
|
| 99 |
+
model.eval()
|
| 100 |
+
|
| 101 |
+
warmup_prompt = (
|
| 102 |
+
"<start_of_turn>user\nYou are an expert fighting game NPC AI. "
|
| 103 |
+
"The user has performed this sequence of 5 moves: jab,cross,low_kick,roundhouse,uppercut.\n"
|
| 104 |
+
"Decide on the best counter-move from: jab, cross, low_kick, roundhouse, uppercut, parry, backstep, clinch, throw.\n"
|
| 105 |
+
"Respond in this format:\n[reasoning]\ncounter_move: [move]"
|
| 106 |
+
"<end_of_turn>\n<start_of_turn>model\n"
|
| 107 |
+
)
|
| 108 |
+
warmup_inputs = tokenizer(warmup_prompt, return_tensors="pt").to(model.device)
|
| 109 |
+
with torch.no_grad():
|
| 110 |
+
_ = model.generate(
|
| 111 |
+
**warmup_inputs,
|
| 112 |
+
max_new_tokens=20,
|
| 113 |
+
do_sample=False,
|
| 114 |
+
pad_token_id=tokenizer.eos_token_id,
|
| 115 |
+
)
|
| 116 |
+
|
| 117 |
+
HAS_MODEL = True
|
| 118 |
+
log.info("Model loaded and warmed up.")
|
| 119 |
+
except Exception as e:
|
| 120 |
+
import traceback
|
| 121 |
+
MODEL_ERROR = f"{type(e).__name__}: {e}"
|
| 122 |
+
log.warning("Model load failed: %s", MODEL_ERROR)
|
| 123 |
+
log.debug(traceback.format_exc())
|
| 124 |
+
model = None
|
| 125 |
+
tokenizer = None
|
| 126 |
+
|
| 127 |
+
|
| 128 |
+
# ------------------------------------------------------------------
|
| 129 |
+
# Inference helper
|
| 130 |
+
# ------------------------------------------------------------------
|
| 131 |
+
MOCK_COUNTERS = ["jab", "cross", "low_kick", "roundhouse", "uppercut", "parry", "backstep", "clinch", "throw"]
|
| 132 |
+
|
| 133 |
+
def run_gemma(moves_sequence: str) -> str:
|
| 134 |
+
if not HAS_MODEL:
|
| 135 |
+
import time, random
|
| 136 |
+
time.sleep(0.25)
|
| 137 |
+
# Offline mode: return a clean scripted counter without leaking any
|
| 138 |
+
# raw model-loader diagnostics to the player UI.
|
| 139 |
+
reasoning = (
|
| 140 |
+
f"Mock Analysis: player performed {moves_sequence}. "
|
| 141 |
+
"AI opponent is offline — using scripted counters."
|
| 142 |
+
)
|
| 143 |
+
return json.dumps({
|
| 144 |
+
"reasoning": reasoning,
|
| 145 |
+
"counterMove": random.choice(MOCK_COUNTERS),
|
| 146 |
+
"sequence": moves_sequence,
|
| 147 |
+
})
|
| 148 |
+
|
| 149 |
+
prompt = (
|
| 150 |
+
f"<start_of_turn>user\n"
|
| 151 |
+
f"You are an expert fighting game NPC AI. "
|
| 152 |
+
f"The user has performed this sequence of 5 moves: {moves_sequence}.\n"
|
| 153 |
+
f"Observe the pattern and decide on the best counter-move from: "
|
| 154 |
+
f"jab, cross, low_kick, roundhouse, uppercut, parry, backstep, clinch, throw.\n"
|
| 155 |
+
f"Respond in this format:\n"
|
| 156 |
+
f"[Your reasoning about the player's pattern and tendencies]\n"
|
| 157 |
+
f"counter_move: [your chosen counter move]"
|
| 158 |
+
f"<end_of_turn>\n<start_of_turn>model\n"
|
| 159 |
+
)
|
| 160 |
+
import torch
|
| 161 |
+
inputs = tokenizer(prompt, return_tensors="pt").to(model.device)
|
| 162 |
+
with torch.no_grad():
|
| 163 |
+
outputs = model.generate(
|
| 164 |
+
**inputs,
|
| 165 |
+
max_new_tokens=80,
|
| 166 |
+
temperature=0.2,
|
| 167 |
+
do_sample=True,
|
| 168 |
+
pad_token_id=tokenizer.eos_token_id,
|
| 169 |
+
)
|
| 170 |
+
text = tokenizer.decode(outputs[0][inputs["input_ids"].shape[-1]:], skip_special_tokens=True)
|
| 171 |
+
|
| 172 |
+
reasoning = "Unable to process reasoning."
|
| 173 |
+
counter_move = "jab"
|
| 174 |
+
if "counter_move:" in text:
|
| 175 |
+
parts = text.split("counter_move:")
|
| 176 |
+
reasoning = parts[0].strip()
|
| 177 |
+
counter_move = parts[1].strip()
|
| 178 |
+
else:
|
| 179 |
+
reasoning = text.strip()
|
| 180 |
+
|
| 181 |
+
return json.dumps({"reasoning": reasoning, "counterMove": counter_move, "sequence": moves_sequence})
|
| 182 |
+
|
| 183 |
+
|
| 184 |
+
# ------------------------------------------------------------------
|
| 185 |
+
# FastAPI app setup
|
| 186 |
+
# ------------------------------------------------------------------
|
| 187 |
+
app = FastAPI(title="Duel of Albion - Gemma AI Fighter")
|
| 188 |
+
|
| 189 |
+
app.add_middleware(
|
| 190 |
+
CORSMiddleware,
|
| 191 |
+
allow_origins=["*"],
|
| 192 |
+
allow_credentials=True,
|
| 193 |
+
allow_methods=["*"],
|
| 194 |
+
allow_headers=["*"],
|
| 195 |
+
)
|
| 196 |
+
|
| 197 |
+
# ------------------------------------------------------------------
|
| 198 |
+
# Static files: the built React game
|
| 199 |
+
# ------------------------------------------------------------------
|
| 200 |
+
PROJECT_ROOT = os.path.dirname(os.path.abspath(__file__))
|
| 201 |
+
STATIC_DIR = os.path.join(PROJECT_ROOT, "3d-game", "dist")
|
| 202 |
+
STATIC_DIR_EXISTS = os.path.isdir(STATIC_DIR)
|
| 203 |
+
|
| 204 |
+
if STATIC_DIR_EXISTS:
|
| 205 |
+
# IMPORTANT: do NOT mount anything at /assets.
|
| 206 |
+
# Gradio serves its own JS/CSS bundles at /assets/* (e.g.
|
| 207 |
+
# /assets/index-DputZZxm.js). Mounting at /assets would shadow
|
| 208 |
+
# Gradio's handler and return 404 for those, leaving the Gradio
|
| 209 |
+
# shell blank. The React build's own assets are served via the
|
| 210 |
+
# /game/{path:path} catch-all below.
|
| 211 |
+
|
| 212 |
+
@app.get("/favicon.ico", include_in_schema=False)
|
| 213 |
+
async def favicon_ico():
|
| 214 |
+
return FileResponse(
|
| 215 |
+
os.path.join(STATIC_DIR, "favicon.svg"),
|
| 216 |
+
media_type="image/svg+xml",
|
| 217 |
+
)
|
| 218 |
+
|
| 219 |
+
@app.get("/favicon.svg", include_in_schema=False)
|
| 220 |
+
async def favicon_svg():
|
| 221 |
+
return FileResponse(os.path.join(STATIC_DIR, "favicon.svg"))
|
| 222 |
+
|
| 223 |
+
@app.get("/manifest.json", include_in_schema=False)
|
| 224 |
+
async def manifest():
|
| 225 |
+
return JSONResponse(content={
|
| 226 |
+
"name": "Duel of Albion",
|
| 227 |
+
"short_name": "Duel of Albion",
|
| 228 |
+
"start_url": "/game/",
|
| 229 |
+
"display": "fullscreen",
|
| 230 |
+
"background_color": "#05040a",
|
| 231 |
+
"theme_color": "#05040a",
|
| 232 |
+
"icons": [],
|
| 233 |
+
})
|
| 234 |
+
|
| 235 |
+
NO_CACHE = {
|
| 236 |
+
"Cache-Control": "no-cache, no-store, must-revalidate",
|
| 237 |
+
"Pragma": "no-cache",
|
| 238 |
+
"Expires": "0",
|
| 239 |
+
}
|
| 240 |
+
|
| 241 |
+
@app.get("/models/{path:path}", include_in_schema=False)
|
| 242 |
+
async def game_models(path: str):
|
| 243 |
+
static_root = os.path.normpath(STATIC_DIR)
|
| 244 |
+
candidate = os.path.normpath(os.path.join(STATIC_DIR, "models", path))
|
| 245 |
+
if not candidate.startswith(static_root):
|
| 246 |
+
return HTMLResponse("Forbidden", status_code=403)
|
| 247 |
+
if os.path.isfile(candidate):
|
| 248 |
+
return FileResponse(candidate)
|
| 249 |
+
return HTMLResponse("Not Found", status_code=404)
|
| 250 |
+
|
| 251 |
+
@app.get("/game", include_in_schema=False)
|
| 252 |
+
@app.get("/game/", include_in_schema=False)
|
| 253 |
+
async def game_index():
|
| 254 |
+
return FileResponse(
|
| 255 |
+
os.path.join(STATIC_DIR, "index.html"),
|
| 256 |
+
headers=NO_CACHE,
|
| 257 |
+
)
|
| 258 |
+
|
| 259 |
+
@app.get("/game/{path:path}", include_in_schema=False)
|
| 260 |
+
async def game_spa(path: str):
|
| 261 |
+
# Path-traversal guard
|
| 262 |
+
static_root = os.path.normpath(STATIC_DIR)
|
| 263 |
+
candidate = os.path.normpath(os.path.join(STATIC_DIR, path))
|
| 264 |
+
if not candidate.startswith(static_root):
|
| 265 |
+
return HTMLResponse("Forbidden", status_code=403)
|
| 266 |
+
if os.path.isfile(candidate):
|
| 267 |
+
return FileResponse(candidate)
|
| 268 |
+
# SPA fallback — unknown path -> index.html
|
| 269 |
+
return FileResponse(
|
| 270 |
+
os.path.join(STATIC_DIR, "index.html"),
|
| 271 |
+
headers=NO_CACHE,
|
| 272 |
+
)
|
| 273 |
+
else:
|
| 274 |
+
@app.get("/game", include_in_schema=False)
|
| 275 |
+
@app.get("/game/", include_in_schema=False)
|
| 276 |
+
async def game_not_built():
|
| 277 |
+
return HTMLResponse(
|
| 278 |
+
"<h1 style='font-family:sans-serif;color:#f0e6d2;background:#05040a;"
|
| 279 |
+
"padding:40px;'>React game not built</h1>"
|
| 280 |
+
"<p style='font-family:sans-serif;color:#b8a88a;background:#05040a;"
|
| 281 |
+
"padding:0 40px 40px;'>Run <code>npm run build</code> in "
|
| 282 |
+
"<code>3d-game/</code> to produce the dist directory.</p>",
|
| 283 |
+
status_code=404,
|
| 284 |
+
)
|
| 285 |
+
|
| 286 |
+
|
| 287 |
+
# ------------------------------------------------------------------
|
| 288 |
+
# API endpoints used by the React game
|
| 289 |
+
# ------------------------------------------------------------------
|
| 290 |
+
@app.get("/health")
|
| 291 |
+
async def health():
|
| 292 |
+
return JSONResponse(content={
|
| 293 |
+
"ready": HAS_MODEL,
|
| 294 |
+
"has_ml": HAS_ML,
|
| 295 |
+
"skip_model_load": SKIP_MODEL_LOAD,
|
| 296 |
+
"has_token": bool(hf_token),
|
| 297 |
+
})
|
| 298 |
+
|
| 299 |
+
|
| 300 |
+
@app.post("/predict")
|
| 301 |
+
async def predict(request: Request):
|
| 302 |
+
try:
|
| 303 |
+
data = await request.json()
|
| 304 |
+
except Exception:
|
| 305 |
+
data = {}
|
| 306 |
+
sequence = data.get("sequence", "")
|
| 307 |
+
result_str = run_gemma(sequence)
|
| 308 |
+
try:
|
| 309 |
+
result = json.loads(result_str)
|
| 310 |
+
except Exception:
|
| 311 |
+
result = {"reasoning": result_str, "counterMove": "jab", "sequence": sequence}
|
| 312 |
+
return JSONResponse(content=result)
|
| 313 |
+
|
| 314 |
+
|
| 315 |
+
# ------------------------------------------------------------------
|
| 316 |
+
# Gradio UI — full-screen game shell
|
| 317 |
+
# ------------------------------------------------------------------
|
| 318 |
+
# The game is always served through a Gradio container. The React app is
|
| 319 |
+
# embedded full-screen in an iframe so it keeps its own rendering / input
|
| 320 |
+
# logic, while Gradio provides the hosting layer (HF Spaces, local, etc.).
|
| 321 |
+
css = """
|
| 322 |
+
/* Kill every Gradio container from html down — nothing should add
|
| 323 |
+
padding, margin, gaps, borders, or constrained height. */
|
| 324 |
+
html,body,#root,.app,.gradio-container,
|
| 325 |
+
.gradio-container>.main,.gradio-container>.main>.wrap,
|
| 326 |
+
.gradio-container .column,.gradio-container .column>.form,
|
| 327 |
+
.gradio-container [class*="container"],
|
| 328 |
+
.gradio-container [class*="panel"],
|
| 329 |
+
.gradio-container [class*="gap"] {
|
| 330 |
+
background:#05040a!important;
|
| 331 |
+
padding:0!important;margin:0!important;
|
| 332 |
+
max-width:none!important;width:100%!important;height:100%!important;
|
| 333 |
+
min-height:100vh!important;
|
| 334 |
+
border:none!important;box-shadow:none!important;gap:0!important;
|
| 335 |
+
overflow:hidden!important;
|
| 336 |
+
}
|
| 337 |
+
/* Hide all Gradio chrome: splash, footer, loader, status bar */
|
| 338 |
+
#app_splash,.splash,.loading,.loader,
|
| 339 |
+
.progress,.progress-bar,.meta-loader,
|
| 340 |
+
footer,.footer,.gradio-footer,.built-with,
|
| 341 |
+
#component-status,.meta,[class*="built-with"],
|
| 342 |
+
[class*="splash"],[class*="loader"],
|
| 343 |
+
.svelte-1ipelgc {
|
| 344 |
+
display:none!important;visibility:hidden!important;opacity:0!important;
|
| 345 |
+
height:0!important;width:0!important;overflow:hidden!important;
|
| 346 |
+
}
|
| 347 |
+
/* The Column with class game-wrap fills the viewport */
|
| 348 |
+
.game-wrap,
|
| 349 |
+
.game-wrap>.form {
|
| 350 |
+
position:fixed!important;inset:0!important;
|
| 351 |
+
width:100vw!important;height:100vh!important;
|
| 352 |
+
padding:0!important;margin:0!important;
|
| 353 |
+
overflow:hidden!important;background:#05040a!important;
|
| 354 |
+
z-index:2147483647;
|
| 355 |
+
}
|
| 356 |
+
/* The iframe itself is also fixed so it ignores any intermediate wrappers
|
| 357 |
+
Gradio may insert around gr.HTML */
|
| 358 |
+
#game-iframe,.game-wrap iframe {
|
| 359 |
+
position:fixed!important;top:0!important;left:0!important;
|
| 360 |
+
width:100vw!important;height:100vh!important;
|
| 361 |
+
border:none!important;display:block!important;
|
| 362 |
+
background:#05040a!important;z-index:2147483647;
|
| 363 |
+
}
|
| 364 |
+
"""
|
| 365 |
+
with gr.Blocks(title="Duel of Albion", css=css, theme=gr.themes.Soft()) as demo:
|
| 366 |
+
if STATIC_DIR_EXISTS:
|
| 367 |
+
game_url = f"/game/?v={os.environ.get('BUILD_ID', int.from_bytes(os.urandom(2), 'big'))}"
|
| 368 |
+
else:
|
| 369 |
+
game_url = "/game"
|
| 370 |
+
with gr.Column(elem_classes="game-wrap"):
|
| 371 |
+
gr.HTML(
|
| 372 |
+
f'<iframe id="game-iframe" src="{game_url}" allowfullscreen '
|
| 373 |
+
'allow="autoplay; fullscreen; gamepad; xr-spatial-tracking" '
|
| 374 |
+
'sandbox="allow-scripts allow-same-origin allow-forms allow-popups" '
|
| 375 |
+
'style="background:#05040a;"></iframe>'
|
| 376 |
+
)
|
| 377 |
+
app = gr.mount_gradio_app(app, demo, path="/")
|
| 378 |
+
log.info("Mounted Gradio shell at /.")
|
| 379 |
+
|
| 380 |
+
|
| 381 |
+
if __name__ == "__main__":
|
| 382 |
+
log.info("=" * 60)
|
| 383 |
+
log.info("Duel of Albion — Gradio server")
|
| 384 |
+
log.info(" Project root : %s", PROJECT_ROOT)
|
| 385 |
+
log.info(" Static dir : %s (exists=%s)", STATIC_DIR, STATIC_DIR_EXISTS)
|
| 386 |
+
log.info(" Has ML stack: %s", HAS_ML)
|
| 387 |
+
log.info(" Has model : %s", HAS_MODEL)
|
| 388 |
+
log.info(" HF token : %s", "yes" if hf_token else "no")
|
| 389 |
+
log.info(" URL : http://%s:%s/", "localhost" if HOST == "0.0.0.0" else HOST, PORT)
|
| 390 |
+
log.info("=" * 60)
|
| 391 |
+
import uvicorn
|
| 392 |
+
uvicorn.run(app, host=HOST, port=PORT, log_level="info")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static/assets/index-DUDqKLMt.css
ADDED
|
@@ -0,0 +1 @@
|
|
|
|
|
|
|
| 1 |
+
body{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;background-color:#050510;width:100vw;height:100vh;margin:0;font-family:Segoe UI,system-ui,-apple-system,sans-serif;overflow:hidden}#root{width:100%;height:100%}*{box-sizing:border-box;margin:0;padding:0}html,body,#root{color:#fff;background:#050510;width:100%;height:100%;font-family:Segoe UI,system-ui,-apple-system,sans-serif;overflow:hidden}.game-container{width:100vw;height:100vh;position:relative}.game-container canvas{display:block;width:100%!important;height:100%!important}.hud{pointer-events:none;z-index:10;justify-content:space-between;align-items:flex-start;gap:16px;width:100%;padding:16px 24px;display:flex;position:absolute;top:0;left:0}.hp-section{width:35%}.hp-label{text-transform:uppercase;letter-spacing:2px;margin-bottom:4px;font-size:11px;font-weight:700}.hp-label.player{color:#00d4ff;text-shadow:0 0 10px #00d4ff66}.hp-label.npc{color:#f34;text-align:right;text-shadow:0 0 10px #f346}.hp-bar-bg{background:#1a1a2e;border:1px solid #333;border-radius:3px;width:100%;height:14px;overflow:hidden}.hp-bar{border-radius:2px;height:100%;transition:width .3s}.hp-bar.player{background:linear-gradient(90deg,#00d4ff,#08f);box-shadow:0 0 10px #00d4ff44}.hp-bar.npc{float:right;background:linear-gradient(90deg,#f34,#f64);box-shadow:0 0 10px #f344}.center-info{text-align:center;pointer-events:none;z-index:10;position:absolute;top:12px;left:50%;transform:translate(-50%)}.round-info{color:#555;letter-spacing:2px;text-transform:uppercase;font-size:10px}.score{letter-spacing:4px;margin-top:4px;font-size:22px;font-weight:900}.score .player{color:#00d4ff}.score .npc{color:#f34}.score .dash{color:#444}.hit-splash{pointer-events:none;z-index:20;text-shadow:0 0 20px #ffffff80;font-size:30px;font-weight:900;animation:.5s ease-out forwards hitFade;position:absolute;top:38%;left:50%;transform:translate(-50%,-50%)}@keyframes hitFade{0%{opacity:1;transform:translate(-50%,-50%)scale(1.3)}to{opacity:0;transform:translate(-50%,-70%)scale(1)}}.controls-hint{color:#444;pointer-events:none;z-index:10;font-size:10px;line-height:1.8;position:absolute;bottom:14px;left:16px}.controls-hint kbd{color:#666;background:#1a1a2e;border:1px solid #333;border-radius:3px;margin:0 1px;padding:2px 5px;font-family:inherit;font-size:9px;display:inline-block}.combo-info{color:#444;pointer-events:none;z-index:10;text-align:right;font-size:10px;line-height:1.6;position:absolute;bottom:14px;right:16px}.overlay{z-index:100;cursor:pointer;background:#050510eb;flex-direction:column;justify-content:center;align-items:center;width:100%;height:100%;display:flex;position:absolute;top:0;left:0}.overlay h1{letter-spacing:6px;text-transform:uppercase;background:linear-gradient(135deg,#00d4ff,#a4f,#f34);-webkit-text-fill-color:transparent;text-align:center;-webkit-background-clip:text;background-clip:text;margin-bottom:20px;font-size:42px;font-weight:900}.overlay .subtitle{color:#666;text-align:center;max-width:420px;margin-bottom:24px;font-size:14px;line-height:1.7}.overlay .start-hint{color:#555;font-size:12px;animation:2s ease-in-out infinite pulse}@keyframes pulse{0%,to{opacity:.5}50%{opacity:1}}.game-over{z-index:50;background:#050510f0;flex-direction:column;justify-content:center;align-items:center;width:100%;height:100%;display:flex;position:absolute;top:0;left:0}.game-over h1{letter-spacing:4px;margin-bottom:12px;font-size:48px;font-weight:900}.game-over .subtitle{color:#888;margin-bottom:32px;font-size:15px}.game-over button{text-transform:uppercase;letter-spacing:2px;cursor:pointer;color:#fff;pointer-events:all;background:linear-gradient(135deg,#00d4ff,#06f);border:none;border-radius:4px;padding:14px 44px;font-size:14px;font-weight:700;transition:filter .2s,transform .2s}.game-over button:hover{filter:brightness(1.2);transform:scale(1.05)}.round-overlay{z-index:40;pointer-events:none;background:#000000bf;flex-direction:column;justify-content:center;align-items:center;width:100%;height:100%;animation:.3s ease-out roundFadeIn;display:flex;position:absolute;top:0;left:0}@keyframes roundFadeIn{0%{opacity:0}to{opacity:1}}.round-overlay h2{letter-spacing:3px;font-size:40px;font-weight:900}.round-overlay p{color:#888;margin-top:8px;font-size:16px}
|
static/assets/index-Dp4pBtge.js
ADDED
|
The diff for this file is too large to render.
See raw diff
|
|
|
static/favicon.svg
ADDED
|
|
static/icons.svg
ADDED
|
|
static/index.html
ADDED
|
@@ -0,0 +1,30 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<!doctype html>
|
| 2 |
+
<html lang="en">
|
| 3 |
+
<head>
|
| 4 |
+
<meta charset="UTF-8" />
|
| 5 |
+
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
|
| 6 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
| 7 |
+
<title>Shadow Duel AI</title>
|
| 8 |
+
<script type="module" crossorigin src="/assets/index-Dp4pBtge.js"></script>
|
| 9 |
+
<link rel="stylesheet" crossorigin href="/assets/index-DUDqKLMt.css">
|
| 10 |
+
<script>
|
| 11 |
+
// AI Bridge: receives AI_RESPONSE from Gradio parent and forwards to game
|
| 12 |
+
window.addEventListener("message", function(e) {
|
| 13 |
+
if (e.data && e.data.type === "AI_RESPONSE") {
|
| 14 |
+
window.dispatchEvent(new CustomEvent("ai-response", { detail: e.data }));
|
| 15 |
+
}
|
| 16 |
+
});
|
| 17 |
+
window.sendAIRequest = function(sequence) {
|
| 18 |
+
try {
|
| 19 |
+
window.parent.postMessage({
|
| 20 |
+
type: 'AI_REQUEST',
|
| 21 |
+
sequence: sequence
|
| 22 |
+
}, '*');
|
| 23 |
+
} catch(e) {}
|
| 24 |
+
};
|
| 25 |
+
</script>
|
| 26 |
+
</head>
|
| 27 |
+
<body>
|
| 28 |
+
<div id="root"></div>
|
| 29 |
+
</body>
|
| 30 |
+
</html>
|
three.min.js
ADDED
|
The diff for this file is too large to render.
See raw diff
|
|
|