diff --git a/.env.development b/.env.development deleted file mode 100644 index 5f323191..00000000 --- a/.env.development +++ /dev/null @@ -1 +0,0 @@ -BROWSER=none \ No newline at end of file diff --git a/CNAME b/CNAME new file mode 100644 index 00000000..89923aed --- /dev/null +++ b/CNAME @@ -0,0 +1 @@ +www.phantasmal.world \ No newline at end of file diff --git a/LICENSE b/LICENSE deleted file mode 100644 index 55a16a91..00000000 --- a/LICENSE +++ /dev/null @@ -1,7 +0,0 @@ -Copyright (c) 2016 Daan Vanden Bosch - -Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/README.md b/README.md deleted file mode 100644 index c5560659..00000000 --- a/README.md +++ /dev/null @@ -1,23 +0,0 @@ -# Phantasmal World - -Phantasmal World is a suite of tools for Phantasy Star Online. This project is in a very early stage of development. - -## For Developers - -React is used for the frontend, MobX is used for state management and THREE.js for 3D graphics. - -### Getting Started - -1. Install Yarn ([https://yarnpkg.com/](https://yarnpkg.com/)) -2. cd to the project directory -3. Launch server on [http://localhost:3000/](http://localhost:3000/) with `yarn start` -4. src/index.tsx is the application's entry point - -### Tests - -Run tests with `yarn test`. - -### Production Build - -1. Change the homepage field in package.json if you don't want to deploy to / -3. Build with `yarn build` diff --git a/antd.customize.less b/antd.customize.less deleted file mode 100644 index 7eba8e1b..00000000 --- a/antd.customize.less +++ /dev/null @@ -1,52 +0,0 @@ -@primary-color: hsl(200, 100%, 50%); - -@white: #000; -@black: #fff; - -@primary-1: fade(@primary-color, 50%); -@primary-2: fade(@primary-color, 40%); - -@body-background: hsl(200, 10%, 20%); -@component-background: @body-background; -@text-color: hsl(200, 10%, 90%); -@text-color-secondary: hsl(200, 20%, 80%); -@text-color-dark: fade(white, 85%); -@text-color-secondary-dark: fade(white, 65%); - -@heading-color: fade(@black, 85%); - -@border-radius-base: 2px; -@border-radius-sm: 0px; - -@background-color-light: lighten(@component-background, 20%); // background of header and selected item -@background-color-base: fade(@primary-color, 20%); // Default grey background color - -@item-active-bg: fade(@primary-color, 20%); -@item-hover-bg: fade(@primary-color, 10%); - -@border-color-base: lighten(@component-background, 20%); // base border outline a component -@border-color-split: lighten(@component-background, 10%); // split border inside a component - -// Disabled states -@disabled-color: fade(#fff, 50%); - -// Animation -@animation-duration-slow: 0.1s; // Modal -@animation-duration-base: 0.066s; -@animation-duration-fast: 0.033s; // Tooltip - -// Input -@input-bg: darken(@component-background, 5%); - -// Buttons -@btn-default-bg: lighten(@component-background, 10%); - -// Modal -@modal-mask-bg: fade(black, 80%); - -// Table -@table-selected-row-bg: @item-active-bg; -@table-row-hover-bg: @item-hover-bg; - -// Menu -@menu-dark-bg: @component-background; diff --git a/asset-manifest.json b/asset-manifest.json new file mode 100644 index 00000000..f7003ff0 --- /dev/null +++ b/asset-manifest.json @@ -0,0 +1,17 @@ +{ + "files": { + "main.css": "/static/css/main.fb4f588b.chunk.css", + "main.js": "/static/js/main.08895bb3.chunk.js", + "main.js.map": "/static/js/main.08895bb3.chunk.js.map", + "runtime~main.js": "/static/js/runtime~main.a8a9905a.js", + "runtime~main.js.map": "/static/js/runtime~main.a8a9905a.js.map", + "static/css/2.5fcf4e3c.chunk.css": "/static/css/2.5fcf4e3c.chunk.css", + "static/js/2.e8014737.chunk.js": "/static/js/2.e8014737.chunk.js", + "static/js/2.e8014737.chunk.js.map": "/static/js/2.e8014737.chunk.js.map", + "index.html": "/index.html", + "precache-manifest.be7f938c356ac9dff1a6bfa51f6f976d.js": "/precache-manifest.be7f938c356ac9dff1a6bfa51f6f976d.js", + "service-worker.js": "/service-worker.js", + "static/css/2.5fcf4e3c.chunk.css.map": "/static/css/2.5fcf4e3c.chunk.css.map", + "static/css/main.fb4f588b.chunk.css.map": "/static/css/main.fb4f588b.chunk.css.map" + } +} \ No newline at end of file diff --git a/public/boxDrops.ephinea.json b/boxDrops.ephinea.json similarity index 100% rename from public/boxDrops.ephinea.json rename to boxDrops.ephinea.json diff --git a/craco.config.js b/craco.config.js deleted file mode 100644 index 12473c5c..00000000 --- a/craco.config.js +++ /dev/null @@ -1,5 +0,0 @@ -const CracoAntDesignPlugin = require("craco-antd"); - -module.exports = { - plugins: [{ plugin: CracoAntDesignPlugin }] -}; \ No newline at end of file diff --git a/public/enemyDrops.ephinea.json b/enemyDrops.ephinea.json similarity index 100% rename from public/enemyDrops.ephinea.json rename to enemyDrops.ephinea.json diff --git a/index.html b/index.html new file mode 100644 index 00000000..6a355a6f --- /dev/null +++ b/index.html @@ -0,0 +1 @@ +
\ No newline at end of file diff --git a/public/items.ephinea.json b/items.ephinea.json similarity index 100% rename from public/items.ephinea.json rename to items.ephinea.json diff --git a/public/maps/map_ancient01_00c.rel b/maps/map_ancient01_00c.rel similarity index 100% rename from public/maps/map_ancient01_00c.rel rename to maps/map_ancient01_00c.rel diff --git a/public/maps/map_ancient01_00n.rel b/maps/map_ancient01_00n.rel similarity index 100% rename from public/maps/map_ancient01_00n.rel rename to maps/map_ancient01_00n.rel diff --git a/public/maps/map_ancient01_01c.rel b/maps/map_ancient01_01c.rel similarity index 100% rename from public/maps/map_ancient01_01c.rel rename to maps/map_ancient01_01c.rel diff --git a/public/maps/map_ancient01_01n.rel b/maps/map_ancient01_01n.rel similarity index 100% rename from public/maps/map_ancient01_01n.rel rename to maps/map_ancient01_01n.rel diff --git a/public/maps/map_ancient01_02c.rel b/maps/map_ancient01_02c.rel similarity index 100% rename from public/maps/map_ancient01_02c.rel rename to maps/map_ancient01_02c.rel diff --git a/public/maps/map_ancient01_02n.rel b/maps/map_ancient01_02n.rel similarity index 100% rename from public/maps/map_ancient01_02n.rel rename to maps/map_ancient01_02n.rel diff --git a/public/maps/map_ancient01_03c.rel b/maps/map_ancient01_03c.rel similarity index 100% rename from public/maps/map_ancient01_03c.rel rename to maps/map_ancient01_03c.rel diff --git a/public/maps/map_ancient01_03n.rel b/maps/map_ancient01_03n.rel similarity index 100% rename from public/maps/map_ancient01_03n.rel rename to maps/map_ancient01_03n.rel diff --git a/public/maps/map_ancient01_04c.rel b/maps/map_ancient01_04c.rel similarity index 100% rename from public/maps/map_ancient01_04c.rel rename to maps/map_ancient01_04c.rel diff --git a/public/maps/map_ancient01_04n.rel b/maps/map_ancient01_04n.rel similarity index 100% rename from public/maps/map_ancient01_04n.rel rename to maps/map_ancient01_04n.rel diff --git a/public/maps/map_ancient02_00c.rel b/maps/map_ancient02_00c.rel similarity index 100% rename from public/maps/map_ancient02_00c.rel rename to maps/map_ancient02_00c.rel diff --git a/public/maps/map_ancient02_00n.rel b/maps/map_ancient02_00n.rel similarity index 100% rename from public/maps/map_ancient02_00n.rel rename to maps/map_ancient02_00n.rel diff --git a/public/maps/map_ancient02_01c.rel b/maps/map_ancient02_01c.rel similarity index 100% rename from public/maps/map_ancient02_01c.rel rename to maps/map_ancient02_01c.rel diff --git a/public/maps/map_ancient02_01n.rel b/maps/map_ancient02_01n.rel similarity index 100% rename from public/maps/map_ancient02_01n.rel rename to maps/map_ancient02_01n.rel diff --git a/public/maps/map_ancient02_02c.rel b/maps/map_ancient02_02c.rel similarity index 100% rename from public/maps/map_ancient02_02c.rel rename to maps/map_ancient02_02c.rel diff --git a/public/maps/map_ancient02_02n.rel b/maps/map_ancient02_02n.rel similarity index 100% rename from public/maps/map_ancient02_02n.rel rename to maps/map_ancient02_02n.rel diff --git a/public/maps/map_ancient02_03c.rel b/maps/map_ancient02_03c.rel similarity index 100% rename from public/maps/map_ancient02_03c.rel rename to maps/map_ancient02_03c.rel diff --git a/public/maps/map_ancient02_03n.rel b/maps/map_ancient02_03n.rel similarity index 100% rename from public/maps/map_ancient02_03n.rel rename to maps/map_ancient02_03n.rel diff --git a/public/maps/map_ancient02_04c.rel b/maps/map_ancient02_04c.rel similarity index 100% rename from public/maps/map_ancient02_04c.rel rename to maps/map_ancient02_04c.rel diff --git a/public/maps/map_ancient02_04n.rel b/maps/map_ancient02_04n.rel similarity index 100% rename from public/maps/map_ancient02_04n.rel rename to maps/map_ancient02_04n.rel diff --git a/public/maps/map_ancient03_00c.rel b/maps/map_ancient03_00c.rel similarity index 100% rename from public/maps/map_ancient03_00c.rel rename to maps/map_ancient03_00c.rel diff --git a/public/maps/map_ancient03_00n.rel b/maps/map_ancient03_00n.rel similarity index 100% rename from public/maps/map_ancient03_00n.rel rename to maps/map_ancient03_00n.rel diff --git a/public/maps/map_ancient03_01c.rel b/maps/map_ancient03_01c.rel similarity index 100% rename from public/maps/map_ancient03_01c.rel rename to maps/map_ancient03_01c.rel diff --git a/public/maps/map_ancient03_01n.rel b/maps/map_ancient03_01n.rel similarity index 100% rename from public/maps/map_ancient03_01n.rel rename to maps/map_ancient03_01n.rel diff --git a/public/maps/map_ancient03_02c.rel b/maps/map_ancient03_02c.rel similarity index 100% rename from public/maps/map_ancient03_02c.rel rename to maps/map_ancient03_02c.rel diff --git a/public/maps/map_ancient03_02n.rel b/maps/map_ancient03_02n.rel similarity index 100% rename from public/maps/map_ancient03_02n.rel rename to maps/map_ancient03_02n.rel diff --git a/public/maps/map_ancient03_03c.rel b/maps/map_ancient03_03c.rel similarity index 100% rename from public/maps/map_ancient03_03c.rel rename to maps/map_ancient03_03c.rel diff --git a/public/maps/map_ancient03_03n.rel b/maps/map_ancient03_03n.rel similarity index 100% rename from public/maps/map_ancient03_03n.rel rename to maps/map_ancient03_03n.rel diff --git a/public/maps/map_ancient03_04c.rel b/maps/map_ancient03_04c.rel similarity index 100% rename from public/maps/map_ancient03_04c.rel rename to maps/map_ancient03_04c.rel diff --git a/public/maps/map_ancient03_04n.rel b/maps/map_ancient03_04n.rel similarity index 100% rename from public/maps/map_ancient03_04n.rel rename to maps/map_ancient03_04n.rel diff --git a/public/maps/map_boss01c.rel b/maps/map_boss01c.rel similarity index 100% rename from public/maps/map_boss01c.rel rename to maps/map_boss01c.rel diff --git a/public/maps/map_boss01n.rel b/maps/map_boss01n.rel similarity index 100% rename from public/maps/map_boss01n.rel rename to maps/map_boss01n.rel diff --git a/public/maps/map_boss02c.rel b/maps/map_boss02c.rel similarity index 100% rename from public/maps/map_boss02c.rel rename to maps/map_boss02c.rel diff --git a/public/maps/map_boss02n.rel b/maps/map_boss02n.rel similarity index 100% rename from public/maps/map_boss02n.rel rename to maps/map_boss02n.rel diff --git a/public/maps/map_boss03c.rel b/maps/map_boss03c.rel similarity index 100% rename from public/maps/map_boss03c.rel rename to maps/map_boss03c.rel diff --git a/public/maps/map_boss03n.rel b/maps/map_boss03n.rel similarity index 100% rename from public/maps/map_boss03n.rel rename to maps/map_boss03n.rel diff --git a/public/maps/map_boss05c.rel b/maps/map_boss05c.rel similarity index 100% rename from public/maps/map_boss05c.rel rename to maps/map_boss05c.rel diff --git a/public/maps/map_boss05n.rel b/maps/map_boss05n.rel similarity index 100% rename from public/maps/map_boss05n.rel rename to maps/map_boss05n.rel diff --git a/public/maps/map_boss06c.rel b/maps/map_boss06c.rel similarity index 100% rename from public/maps/map_boss06c.rel rename to maps/map_boss06c.rel diff --git a/public/maps/map_boss06n.rel b/maps/map_boss06n.rel similarity index 100% rename from public/maps/map_boss06n.rel rename to maps/map_boss06n.rel diff --git a/public/maps/map_boss07c.rel b/maps/map_boss07c.rel similarity index 100% rename from public/maps/map_boss07c.rel rename to maps/map_boss07c.rel diff --git a/public/maps/map_boss07n.rel b/maps/map_boss07n.rel similarity index 100% rename from public/maps/map_boss07n.rel rename to maps/map_boss07n.rel diff --git a/public/maps/map_boss08c.rel b/maps/map_boss08c.rel similarity index 100% rename from public/maps/map_boss08c.rel rename to maps/map_boss08c.rel diff --git a/public/maps/map_boss08n.rel b/maps/map_boss08n.rel similarity index 100% rename from public/maps/map_boss08n.rel rename to maps/map_boss08n.rel diff --git a/public/maps/map_boss09_00c.rel b/maps/map_boss09_00c.rel similarity index 100% rename from public/maps/map_boss09_00c.rel rename to maps/map_boss09_00c.rel diff --git a/public/maps/map_boss09_00n.rel b/maps/map_boss09_00n.rel similarity index 100% rename from public/maps/map_boss09_00n.rel rename to maps/map_boss09_00n.rel diff --git a/public/maps/map_cave01_00c.rel b/maps/map_cave01_00c.rel similarity index 100% rename from public/maps/map_cave01_00c.rel rename to maps/map_cave01_00c.rel diff --git a/public/maps/map_cave01_00n.rel b/maps/map_cave01_00n.rel similarity index 100% rename from public/maps/map_cave01_00n.rel rename to maps/map_cave01_00n.rel diff --git a/public/maps/map_cave01_01c.rel b/maps/map_cave01_01c.rel similarity index 100% rename from public/maps/map_cave01_01c.rel rename to maps/map_cave01_01c.rel diff --git a/public/maps/map_cave01_01n.rel b/maps/map_cave01_01n.rel similarity index 100% rename from public/maps/map_cave01_01n.rel rename to maps/map_cave01_01n.rel diff --git a/public/maps/map_cave01_02c.rel b/maps/map_cave01_02c.rel similarity index 100% rename from public/maps/map_cave01_02c.rel rename to maps/map_cave01_02c.rel diff --git a/public/maps/map_cave01_02n.rel b/maps/map_cave01_02n.rel similarity index 100% rename from public/maps/map_cave01_02n.rel rename to maps/map_cave01_02n.rel diff --git a/public/maps/map_cave01_03c.rel b/maps/map_cave01_03c.rel similarity index 100% rename from public/maps/map_cave01_03c.rel rename to maps/map_cave01_03c.rel diff --git a/public/maps/map_cave01_03n.rel b/maps/map_cave01_03n.rel similarity index 100% rename from public/maps/map_cave01_03n.rel rename to maps/map_cave01_03n.rel diff --git a/public/maps/map_cave01_04c.rel b/maps/map_cave01_04c.rel similarity index 100% rename from public/maps/map_cave01_04c.rel rename to maps/map_cave01_04c.rel diff --git a/public/maps/map_cave01_04n.rel b/maps/map_cave01_04n.rel similarity index 100% rename from public/maps/map_cave01_04n.rel rename to maps/map_cave01_04n.rel diff --git a/public/maps/map_cave01_05c.rel b/maps/map_cave01_05c.rel similarity index 100% rename from public/maps/map_cave01_05c.rel rename to maps/map_cave01_05c.rel diff --git a/public/maps/map_cave01_05n.rel b/maps/map_cave01_05n.rel similarity index 100% rename from public/maps/map_cave01_05n.rel rename to maps/map_cave01_05n.rel diff --git a/public/maps/map_cave02_00c.rel b/maps/map_cave02_00c.rel similarity index 100% rename from public/maps/map_cave02_00c.rel rename to maps/map_cave02_00c.rel diff --git a/public/maps/map_cave02_00n.rel b/maps/map_cave02_00n.rel similarity index 100% rename from public/maps/map_cave02_00n.rel rename to maps/map_cave02_00n.rel diff --git a/public/maps/map_cave02_01c.rel b/maps/map_cave02_01c.rel similarity index 100% rename from public/maps/map_cave02_01c.rel rename to maps/map_cave02_01c.rel diff --git a/public/maps/map_cave02_01n.rel b/maps/map_cave02_01n.rel similarity index 100% rename from public/maps/map_cave02_01n.rel rename to maps/map_cave02_01n.rel diff --git a/public/maps/map_cave02_02c.rel b/maps/map_cave02_02c.rel similarity index 100% rename from public/maps/map_cave02_02c.rel rename to maps/map_cave02_02c.rel diff --git a/public/maps/map_cave02_02n.rel b/maps/map_cave02_02n.rel similarity index 100% rename from public/maps/map_cave02_02n.rel rename to maps/map_cave02_02n.rel diff --git a/public/maps/map_cave02_03c.rel b/maps/map_cave02_03c.rel similarity index 100% rename from public/maps/map_cave02_03c.rel rename to maps/map_cave02_03c.rel diff --git a/public/maps/map_cave02_03n.rel b/maps/map_cave02_03n.rel similarity index 100% rename from public/maps/map_cave02_03n.rel rename to maps/map_cave02_03n.rel diff --git a/public/maps/map_cave02_04c.rel b/maps/map_cave02_04c.rel similarity index 100% rename from public/maps/map_cave02_04c.rel rename to maps/map_cave02_04c.rel diff --git a/public/maps/map_cave02_04n.rel b/maps/map_cave02_04n.rel similarity index 100% rename from public/maps/map_cave02_04n.rel rename to maps/map_cave02_04n.rel diff --git a/public/maps/map_cave03_00c.rel b/maps/map_cave03_00c.rel similarity index 100% rename from public/maps/map_cave03_00c.rel rename to maps/map_cave03_00c.rel diff --git a/public/maps/map_cave03_00n.rel b/maps/map_cave03_00n.rel similarity index 100% rename from public/maps/map_cave03_00n.rel rename to maps/map_cave03_00n.rel diff --git a/public/maps/map_cave03_01c.rel b/maps/map_cave03_01c.rel similarity index 100% rename from public/maps/map_cave03_01c.rel rename to maps/map_cave03_01c.rel diff --git a/public/maps/map_cave03_01n.rel b/maps/map_cave03_01n.rel similarity index 100% rename from public/maps/map_cave03_01n.rel rename to maps/map_cave03_01n.rel diff --git a/public/maps/map_cave03_02c.rel b/maps/map_cave03_02c.rel similarity index 100% rename from public/maps/map_cave03_02c.rel rename to maps/map_cave03_02c.rel diff --git a/public/maps/map_cave03_02n.rel b/maps/map_cave03_02n.rel similarity index 100% rename from public/maps/map_cave03_02n.rel rename to maps/map_cave03_02n.rel diff --git a/public/maps/map_cave03_03c.rel b/maps/map_cave03_03c.rel similarity index 100% rename from public/maps/map_cave03_03c.rel rename to maps/map_cave03_03c.rel diff --git a/public/maps/map_cave03_03n.rel b/maps/map_cave03_03n.rel similarity index 100% rename from public/maps/map_cave03_03n.rel rename to maps/map_cave03_03n.rel diff --git a/public/maps/map_cave03_04c.rel b/maps/map_cave03_04c.rel similarity index 100% rename from public/maps/map_cave03_04c.rel rename to maps/map_cave03_04c.rel diff --git a/public/maps/map_cave03_04n.rel b/maps/map_cave03_04n.rel similarity index 100% rename from public/maps/map_cave03_04n.rel rename to maps/map_cave03_04n.rel diff --git a/public/maps/map_cave03_05c.rel b/maps/map_cave03_05c.rel similarity index 100% rename from public/maps/map_cave03_05c.rel rename to maps/map_cave03_05c.rel diff --git a/public/maps/map_cave03_05n.rel b/maps/map_cave03_05n.rel similarity index 100% rename from public/maps/map_cave03_05n.rel rename to maps/map_cave03_05n.rel diff --git a/public/maps/map_city00_00c.rel b/maps/map_city00_00c.rel similarity index 100% rename from public/maps/map_city00_00c.rel rename to maps/map_city00_00c.rel diff --git a/public/maps/map_city00_00n.rel b/maps/map_city00_00n.rel similarity index 100% rename from public/maps/map_city00_00n.rel rename to maps/map_city00_00n.rel diff --git a/public/maps/map_city02_00c.rel b/maps/map_city02_00c.rel similarity index 100% rename from public/maps/map_city02_00c.rel rename to maps/map_city02_00c.rel diff --git a/public/maps/map_city02_00n.rel b/maps/map_city02_00n.rel similarity index 100% rename from public/maps/map_city02_00n.rel rename to maps/map_city02_00n.rel diff --git a/public/maps/map_crater01_00c.rel b/maps/map_crater01_00c.rel similarity index 100% rename from public/maps/map_crater01_00c.rel rename to maps/map_crater01_00c.rel diff --git a/public/maps/map_crater01_00n.rel b/maps/map_crater01_00n.rel similarity index 100% rename from public/maps/map_crater01_00n.rel rename to maps/map_crater01_00n.rel diff --git a/public/maps/map_darkfalz00c.rel b/maps/map_darkfalz00c.rel similarity index 100% rename from public/maps/map_darkfalz00c.rel rename to maps/map_darkfalz00c.rel diff --git a/public/maps/map_darkfalz00n.rel b/maps/map_darkfalz00n.rel similarity index 100% rename from public/maps/map_darkfalz00n.rel rename to maps/map_darkfalz00n.rel diff --git a/public/maps/map_desert01_00c.rel b/maps/map_desert01_00c.rel similarity index 100% rename from public/maps/map_desert01_00c.rel rename to maps/map_desert01_00c.rel diff --git a/public/maps/map_desert01_00n.rel b/maps/map_desert01_00n.rel similarity index 100% rename from public/maps/map_desert01_00n.rel rename to maps/map_desert01_00n.rel diff --git a/public/maps/map_desert01_01c.rel b/maps/map_desert01_01c.rel similarity index 100% rename from public/maps/map_desert01_01c.rel rename to maps/map_desert01_01c.rel diff --git a/public/maps/map_desert01_01n.rel b/maps/map_desert01_01n.rel similarity index 100% rename from public/maps/map_desert01_01n.rel rename to maps/map_desert01_01n.rel diff --git a/public/maps/map_desert01_02c.rel b/maps/map_desert01_02c.rel similarity index 100% rename from public/maps/map_desert01_02c.rel rename to maps/map_desert01_02c.rel diff --git a/public/maps/map_desert01_02n.rel b/maps/map_desert01_02n.rel similarity index 100% rename from public/maps/map_desert01_02n.rel rename to maps/map_desert01_02n.rel diff --git a/public/maps/map_desert02_00c.rel b/maps/map_desert02_00c.rel similarity index 100% rename from public/maps/map_desert02_00c.rel rename to maps/map_desert02_00c.rel diff --git a/public/maps/map_desert02_00n.rel b/maps/map_desert02_00n.rel similarity index 100% rename from public/maps/map_desert02_00n.rel rename to maps/map_desert02_00n.rel diff --git a/public/maps/map_desert02_01c.rel b/maps/map_desert02_01c.rel similarity index 100% rename from public/maps/map_desert02_01c.rel rename to maps/map_desert02_01c.rel diff --git a/public/maps/map_desert02_01n.rel b/maps/map_desert02_01n.rel similarity index 100% rename from public/maps/map_desert02_01n.rel rename to maps/map_desert02_01n.rel diff --git a/public/maps/map_desert02_02c.rel b/maps/map_desert02_02c.rel similarity index 100% rename from public/maps/map_desert02_02c.rel rename to maps/map_desert02_02c.rel diff --git a/public/maps/map_desert02_02n.rel b/maps/map_desert02_02n.rel similarity index 100% rename from public/maps/map_desert02_02n.rel rename to maps/map_desert02_02n.rel diff --git a/public/maps/map_desert03_00c.rel b/maps/map_desert03_00c.rel similarity index 100% rename from public/maps/map_desert03_00c.rel rename to maps/map_desert03_00c.rel diff --git a/public/maps/map_desert03_00n.rel b/maps/map_desert03_00n.rel similarity index 100% rename from public/maps/map_desert03_00n.rel rename to maps/map_desert03_00n.rel diff --git a/public/maps/map_desert03_01c.rel b/maps/map_desert03_01c.rel similarity index 100% rename from public/maps/map_desert03_01c.rel rename to maps/map_desert03_01c.rel diff --git a/public/maps/map_desert03_01n.rel b/maps/map_desert03_01n.rel similarity index 100% rename from public/maps/map_desert03_01n.rel rename to maps/map_desert03_01n.rel diff --git a/public/maps/map_desert03_02c.rel b/maps/map_desert03_02c.rel similarity index 100% rename from public/maps/map_desert03_02c.rel rename to maps/map_desert03_02c.rel diff --git a/public/maps/map_desert03_02n.rel b/maps/map_desert03_02n.rel similarity index 100% rename from public/maps/map_desert03_02n.rel rename to maps/map_desert03_02n.rel diff --git a/public/maps/map_forest01c.rel b/maps/map_forest01c.rel similarity index 100% rename from public/maps/map_forest01c.rel rename to maps/map_forest01c.rel diff --git a/public/maps/map_forest01n.rel b/maps/map_forest01n.rel similarity index 100% rename from public/maps/map_forest01n.rel rename to maps/map_forest01n.rel diff --git a/public/maps/map_forest02c.rel b/maps/map_forest02c.rel similarity index 100% rename from public/maps/map_forest02c.rel rename to maps/map_forest02c.rel diff --git a/public/maps/map_forest02n.rel b/maps/map_forest02n.rel similarity index 100% rename from public/maps/map_forest02n.rel rename to maps/map_forest02n.rel diff --git a/public/maps/map_jungle01_00c.rel b/maps/map_jungle01_00c.rel similarity index 100% rename from public/maps/map_jungle01_00c.rel rename to maps/map_jungle01_00c.rel diff --git a/public/maps/map_jungle01_00n.rel b/maps/map_jungle01_00n.rel similarity index 100% rename from public/maps/map_jungle01_00n.rel rename to maps/map_jungle01_00n.rel diff --git a/public/maps/map_jungle02_00c.rel b/maps/map_jungle02_00c.rel similarity index 100% rename from public/maps/map_jungle02_00c.rel rename to maps/map_jungle02_00c.rel diff --git a/public/maps/map_jungle02_00n.rel b/maps/map_jungle02_00n.rel similarity index 100% rename from public/maps/map_jungle02_00n.rel rename to maps/map_jungle02_00n.rel diff --git a/public/maps/map_jungle03_00c.rel b/maps/map_jungle03_00c.rel similarity index 100% rename from public/maps/map_jungle03_00c.rel rename to maps/map_jungle03_00c.rel diff --git a/public/maps/map_jungle03_00n.rel b/maps/map_jungle03_00n.rel similarity index 100% rename from public/maps/map_jungle03_00n.rel rename to maps/map_jungle03_00n.rel diff --git a/public/maps/map_jungle04_00c.rel b/maps/map_jungle04_00c.rel similarity index 100% rename from public/maps/map_jungle04_00c.rel rename to maps/map_jungle04_00c.rel diff --git a/public/maps/map_jungle04_00n.rel b/maps/map_jungle04_00n.rel similarity index 100% rename from public/maps/map_jungle04_00n.rel rename to maps/map_jungle04_00n.rel diff --git a/public/maps/map_jungle04_01c.rel b/maps/map_jungle04_01c.rel similarity index 100% rename from public/maps/map_jungle04_01c.rel rename to maps/map_jungle04_01c.rel diff --git a/public/maps/map_jungle04_01n.rel b/maps/map_jungle04_01n.rel similarity index 100% rename from public/maps/map_jungle04_01n.rel rename to maps/map_jungle04_01n.rel diff --git a/public/maps/map_jungle04_02c.rel b/maps/map_jungle04_02c.rel similarity index 100% rename from public/maps/map_jungle04_02c.rel rename to maps/map_jungle04_02c.rel diff --git a/public/maps/map_jungle04_02n.rel b/maps/map_jungle04_02n.rel similarity index 100% rename from public/maps/map_jungle04_02n.rel rename to maps/map_jungle04_02n.rel diff --git a/public/maps/map_jungle05_00c.rel b/maps/map_jungle05_00c.rel similarity index 100% rename from public/maps/map_jungle05_00c.rel rename to maps/map_jungle05_00c.rel diff --git a/public/maps/map_jungle05_00n.rel b/maps/map_jungle05_00n.rel similarity index 100% rename from public/maps/map_jungle05_00n.rel rename to maps/map_jungle05_00n.rel diff --git a/public/maps/map_jungle06_00c.rel b/maps/map_jungle06_00c.rel similarity index 100% rename from public/maps/map_jungle06_00c.rel rename to maps/map_jungle06_00c.rel diff --git a/public/maps/map_jungle06_00n.rel b/maps/map_jungle06_00n.rel similarity index 100% rename from public/maps/map_jungle06_00n.rel rename to maps/map_jungle06_00n.rel diff --git a/public/maps/map_jungle07_00c.rel b/maps/map_jungle07_00c.rel similarity index 100% rename from public/maps/map_jungle07_00c.rel rename to maps/map_jungle07_00c.rel diff --git a/public/maps/map_jungle07_00n.rel b/maps/map_jungle07_00n.rel similarity index 100% rename from public/maps/map_jungle07_00n.rel rename to maps/map_jungle07_00n.rel diff --git a/public/maps/map_jungle07_01c.rel b/maps/map_jungle07_01c.rel similarity index 100% rename from public/maps/map_jungle07_01c.rel rename to maps/map_jungle07_01c.rel diff --git a/public/maps/map_jungle07_01n.rel b/maps/map_jungle07_01n.rel similarity index 100% rename from public/maps/map_jungle07_01n.rel rename to maps/map_jungle07_01n.rel diff --git a/public/maps/map_jungle07_02c.rel b/maps/map_jungle07_02c.rel similarity index 100% rename from public/maps/map_jungle07_02c.rel rename to maps/map_jungle07_02c.rel diff --git a/public/maps/map_jungle07_02n.rel b/maps/map_jungle07_02n.rel similarity index 100% rename from public/maps/map_jungle07_02n.rel rename to maps/map_jungle07_02n.rel diff --git a/public/maps/map_jungle07_03c.rel b/maps/map_jungle07_03c.rel similarity index 100% rename from public/maps/map_jungle07_03c.rel rename to maps/map_jungle07_03c.rel diff --git a/public/maps/map_jungle07_03n.rel b/maps/map_jungle07_03n.rel similarity index 100% rename from public/maps/map_jungle07_03n.rel rename to maps/map_jungle07_03n.rel diff --git a/public/maps/map_jungle07_04c.rel b/maps/map_jungle07_04c.rel similarity index 100% rename from public/maps/map_jungle07_04c.rel rename to maps/map_jungle07_04c.rel diff --git a/public/maps/map_jungle07_04n.rel b/maps/map_jungle07_04n.rel similarity index 100% rename from public/maps/map_jungle07_04n.rel rename to maps/map_jungle07_04n.rel diff --git a/public/maps/map_labo00_00c.rel b/maps/map_labo00_00c.rel similarity index 100% rename from public/maps/map_labo00_00c.rel rename to maps/map_labo00_00c.rel diff --git a/public/maps/map_labo00_00n.rel b/maps/map_labo00_00n.rel similarity index 100% rename from public/maps/map_labo00_00n.rel rename to maps/map_labo00_00n.rel diff --git a/public/maps/map_lobby_00c.rel b/maps/map_lobby_00c.rel similarity index 100% rename from public/maps/map_lobby_00c.rel rename to maps/map_lobby_00c.rel diff --git a/public/maps/map_lobby_00n.rel b/maps/map_lobby_00n.rel similarity index 100% rename from public/maps/map_lobby_00n.rel rename to maps/map_lobby_00n.rel diff --git a/public/maps/map_lobby_01c.rel b/maps/map_lobby_01c.rel similarity index 100% rename from public/maps/map_lobby_01c.rel rename to maps/map_lobby_01c.rel diff --git a/public/maps/map_lobby_01n.rel b/maps/map_lobby_01n.rel similarity index 100% rename from public/maps/map_lobby_01n.rel rename to maps/map_lobby_01n.rel diff --git a/public/maps/map_lobby_02c.rel b/maps/map_lobby_02c.rel similarity index 100% rename from public/maps/map_lobby_02c.rel rename to maps/map_lobby_02c.rel diff --git a/public/maps/map_lobby_02n.rel b/maps/map_lobby_02n.rel similarity index 100% rename from public/maps/map_lobby_02n.rel rename to maps/map_lobby_02n.rel diff --git a/public/maps/map_lobby_03c.rel b/maps/map_lobby_03c.rel similarity index 100% rename from public/maps/map_lobby_03c.rel rename to maps/map_lobby_03c.rel diff --git a/public/maps/map_lobby_03n.rel b/maps/map_lobby_03n.rel similarity index 100% rename from public/maps/map_lobby_03n.rel rename to maps/map_lobby_03n.rel diff --git a/public/maps/map_lobby_04c.rel b/maps/map_lobby_04c.rel similarity index 100% rename from public/maps/map_lobby_04c.rel rename to maps/map_lobby_04c.rel diff --git a/public/maps/map_lobby_04n.rel b/maps/map_lobby_04n.rel similarity index 100% rename from public/maps/map_lobby_04n.rel rename to maps/map_lobby_04n.rel diff --git a/public/maps/map_lobby_05c.rel b/maps/map_lobby_05c.rel similarity index 100% rename from public/maps/map_lobby_05c.rel rename to maps/map_lobby_05c.rel diff --git a/public/maps/map_lobby_05n.rel b/maps/map_lobby_05n.rel similarity index 100% rename from public/maps/map_lobby_05n.rel rename to maps/map_lobby_05n.rel diff --git a/public/maps/map_lobby_06c.rel b/maps/map_lobby_06c.rel similarity index 100% rename from public/maps/map_lobby_06c.rel rename to maps/map_lobby_06c.rel diff --git a/public/maps/map_lobby_06n.rel b/maps/map_lobby_06n.rel similarity index 100% rename from public/maps/map_lobby_06n.rel rename to maps/map_lobby_06n.rel diff --git a/public/maps/map_lobby_07c.rel b/maps/map_lobby_07c.rel similarity index 100% rename from public/maps/map_lobby_07c.rel rename to maps/map_lobby_07c.rel diff --git a/public/maps/map_lobby_07n.rel b/maps/map_lobby_07n.rel similarity index 100% rename from public/maps/map_lobby_07n.rel rename to maps/map_lobby_07n.rel diff --git a/public/maps/map_lobby_08c.rel b/maps/map_lobby_08c.rel similarity index 100% rename from public/maps/map_lobby_08c.rel rename to maps/map_lobby_08c.rel diff --git a/public/maps/map_lobby_08n.rel b/maps/map_lobby_08n.rel similarity index 100% rename from public/maps/map_lobby_08n.rel rename to maps/map_lobby_08n.rel diff --git a/public/maps/map_lobby_09c.rel b/maps/map_lobby_09c.rel similarity index 100% rename from public/maps/map_lobby_09c.rel rename to maps/map_lobby_09c.rel diff --git a/public/maps/map_lobby_09n.rel b/maps/map_lobby_09n.rel similarity index 100% rename from public/maps/map_lobby_09n.rel rename to maps/map_lobby_09n.rel diff --git a/public/maps/map_lobby_10c.rel b/maps/map_lobby_10c.rel similarity index 100% rename from public/maps/map_lobby_10c.rel rename to maps/map_lobby_10c.rel diff --git a/public/maps/map_lobby_10n.rel b/maps/map_lobby_10n.rel similarity index 100% rename from public/maps/map_lobby_10n.rel rename to maps/map_lobby_10n.rel diff --git a/public/maps/map_lobby_green_be00c.rel b/maps/map_lobby_green_be00c.rel similarity index 100% rename from public/maps/map_lobby_green_be00c.rel rename to maps/map_lobby_green_be00c.rel diff --git a/public/maps/map_lobby_green_be00n.rel b/maps/map_lobby_green_be00n.rel similarity index 100% rename from public/maps/map_lobby_green_be00n.rel rename to maps/map_lobby_green_be00n.rel diff --git a/public/maps/map_lobby_red_be00c.rel b/maps/map_lobby_red_be00c.rel similarity index 100% rename from public/maps/map_lobby_red_be00c.rel rename to maps/map_lobby_red_be00c.rel diff --git a/public/maps/map_lobby_red_be00n.rel b/maps/map_lobby_red_be00n.rel similarity index 100% rename from public/maps/map_lobby_red_be00n.rel rename to maps/map_lobby_red_be00n.rel diff --git a/public/maps/map_lobby_yellow_be00c.rel b/maps/map_lobby_yellow_be00c.rel similarity index 100% rename from public/maps/map_lobby_yellow_be00c.rel rename to maps/map_lobby_yellow_be00c.rel diff --git a/public/maps/map_lobby_yellow_be00n.rel b/maps/map_lobby_yellow_be00n.rel similarity index 100% rename from public/maps/map_lobby_yellow_be00n.rel rename to maps/map_lobby_yellow_be00n.rel diff --git a/public/maps/map_machine01_00c.rel b/maps/map_machine01_00c.rel similarity index 100% rename from public/maps/map_machine01_00c.rel rename to maps/map_machine01_00c.rel diff --git a/public/maps/map_machine01_00n.rel b/maps/map_machine01_00n.rel similarity index 100% rename from public/maps/map_machine01_00n.rel rename to maps/map_machine01_00n.rel diff --git a/public/maps/map_machine01_01c.rel b/maps/map_machine01_01c.rel similarity index 100% rename from public/maps/map_machine01_01c.rel rename to maps/map_machine01_01c.rel diff --git a/public/maps/map_machine01_01n.rel b/maps/map_machine01_01n.rel similarity index 100% rename from public/maps/map_machine01_01n.rel rename to maps/map_machine01_01n.rel diff --git a/public/maps/map_machine01_02c.rel b/maps/map_machine01_02c.rel similarity index 100% rename from public/maps/map_machine01_02c.rel rename to maps/map_machine01_02c.rel diff --git a/public/maps/map_machine01_02n.rel b/maps/map_machine01_02n.rel similarity index 100% rename from public/maps/map_machine01_02n.rel rename to maps/map_machine01_02n.rel diff --git a/public/maps/map_machine01_03c.rel b/maps/map_machine01_03c.rel similarity index 100% rename from public/maps/map_machine01_03c.rel rename to maps/map_machine01_03c.rel diff --git a/public/maps/map_machine01_03n.rel b/maps/map_machine01_03n.rel similarity index 100% rename from public/maps/map_machine01_03n.rel rename to maps/map_machine01_03n.rel diff --git a/public/maps/map_machine01_04c.rel b/maps/map_machine01_04c.rel similarity index 100% rename from public/maps/map_machine01_04c.rel rename to maps/map_machine01_04c.rel diff --git a/public/maps/map_machine01_04n.rel b/maps/map_machine01_04n.rel similarity index 100% rename from public/maps/map_machine01_04n.rel rename to maps/map_machine01_04n.rel diff --git a/public/maps/map_machine01_05c.rel b/maps/map_machine01_05c.rel similarity index 100% rename from public/maps/map_machine01_05c.rel rename to maps/map_machine01_05c.rel diff --git a/public/maps/map_machine01_05n.rel b/maps/map_machine01_05n.rel similarity index 100% rename from public/maps/map_machine01_05n.rel rename to maps/map_machine01_05n.rel diff --git a/public/maps/map_machine02_00c.rel b/maps/map_machine02_00c.rel similarity index 100% rename from public/maps/map_machine02_00c.rel rename to maps/map_machine02_00c.rel diff --git a/public/maps/map_machine02_00n.rel b/maps/map_machine02_00n.rel similarity index 100% rename from public/maps/map_machine02_00n.rel rename to maps/map_machine02_00n.rel diff --git a/public/maps/map_machine02_01c.rel b/maps/map_machine02_01c.rel similarity index 100% rename from public/maps/map_machine02_01c.rel rename to maps/map_machine02_01c.rel diff --git a/public/maps/map_machine02_01n.rel b/maps/map_machine02_01n.rel similarity index 100% rename from public/maps/map_machine02_01n.rel rename to maps/map_machine02_01n.rel diff --git a/public/maps/map_machine02_02c.rel b/maps/map_machine02_02c.rel similarity index 100% rename from public/maps/map_machine02_02c.rel rename to maps/map_machine02_02c.rel diff --git a/public/maps/map_machine02_02n.rel b/maps/map_machine02_02n.rel similarity index 100% rename from public/maps/map_machine02_02n.rel rename to maps/map_machine02_02n.rel diff --git a/public/maps/map_machine02_03c.rel b/maps/map_machine02_03c.rel similarity index 100% rename from public/maps/map_machine02_03c.rel rename to maps/map_machine02_03c.rel diff --git a/public/maps/map_machine02_03n.rel b/maps/map_machine02_03n.rel similarity index 100% rename from public/maps/map_machine02_03n.rel rename to maps/map_machine02_03n.rel diff --git a/public/maps/map_machine02_04c.rel b/maps/map_machine02_04c.rel similarity index 100% rename from public/maps/map_machine02_04c.rel rename to maps/map_machine02_04c.rel diff --git a/public/maps/map_machine02_04n.rel b/maps/map_machine02_04n.rel similarity index 100% rename from public/maps/map_machine02_04n.rel rename to maps/map_machine02_04n.rel diff --git a/public/maps/map_machine02_05c.rel b/maps/map_machine02_05c.rel similarity index 100% rename from public/maps/map_machine02_05c.rel rename to maps/map_machine02_05c.rel diff --git a/public/maps/map_machine02_05n.rel b/maps/map_machine02_05n.rel similarity index 100% rename from public/maps/map_machine02_05n.rel rename to maps/map_machine02_05n.rel diff --git a/public/maps/map_ruins01_00c.rel b/maps/map_ruins01_00c.rel similarity index 100% rename from public/maps/map_ruins01_00c.rel rename to maps/map_ruins01_00c.rel diff --git a/public/maps/map_ruins01_00n.rel b/maps/map_ruins01_00n.rel similarity index 100% rename from public/maps/map_ruins01_00n.rel rename to maps/map_ruins01_00n.rel diff --git a/public/maps/map_ruins01_01c.rel b/maps/map_ruins01_01c.rel similarity index 100% rename from public/maps/map_ruins01_01c.rel rename to maps/map_ruins01_01c.rel diff --git a/public/maps/map_ruins01_01n.rel b/maps/map_ruins01_01n.rel similarity index 100% rename from public/maps/map_ruins01_01n.rel rename to maps/map_ruins01_01n.rel diff --git a/public/maps/map_ruins01_02c.rel b/maps/map_ruins01_02c.rel similarity index 100% rename from public/maps/map_ruins01_02c.rel rename to maps/map_ruins01_02c.rel diff --git a/public/maps/map_ruins01_02n.rel b/maps/map_ruins01_02n.rel similarity index 100% rename from public/maps/map_ruins01_02n.rel rename to maps/map_ruins01_02n.rel diff --git a/public/maps/map_ruins02_00c.rel b/maps/map_ruins02_00c.rel similarity index 100% rename from public/maps/map_ruins02_00c.rel rename to maps/map_ruins02_00c.rel diff --git a/public/maps/map_ruins02_00n.rel b/maps/map_ruins02_00n.rel similarity index 100% rename from public/maps/map_ruins02_00n.rel rename to maps/map_ruins02_00n.rel diff --git a/public/maps/map_ruins02_01c.rel b/maps/map_ruins02_01c.rel similarity index 100% rename from public/maps/map_ruins02_01c.rel rename to maps/map_ruins02_01c.rel diff --git a/public/maps/map_ruins02_01n.rel b/maps/map_ruins02_01n.rel similarity index 100% rename from public/maps/map_ruins02_01n.rel rename to maps/map_ruins02_01n.rel diff --git a/public/maps/map_ruins02_02c.rel b/maps/map_ruins02_02c.rel similarity index 100% rename from public/maps/map_ruins02_02c.rel rename to maps/map_ruins02_02c.rel diff --git a/public/maps/map_ruins02_02n.rel b/maps/map_ruins02_02n.rel similarity index 100% rename from public/maps/map_ruins02_02n.rel rename to maps/map_ruins02_02n.rel diff --git a/public/maps/map_seabed01_00c.rel b/maps/map_seabed01_00c.rel similarity index 100% rename from public/maps/map_seabed01_00c.rel rename to maps/map_seabed01_00c.rel diff --git a/public/maps/map_seabed01_00n.rel b/maps/map_seabed01_00n.rel similarity index 100% rename from public/maps/map_seabed01_00n.rel rename to maps/map_seabed01_00n.rel diff --git a/public/maps/map_seabed01_01c.rel b/maps/map_seabed01_01c.rel similarity index 100% rename from public/maps/map_seabed01_01c.rel rename to maps/map_seabed01_01c.rel diff --git a/public/maps/map_seabed01_01n.rel b/maps/map_seabed01_01n.rel similarity index 100% rename from public/maps/map_seabed01_01n.rel rename to maps/map_seabed01_01n.rel diff --git a/public/maps/map_seabed01_02c.rel b/maps/map_seabed01_02c.rel similarity index 100% rename from public/maps/map_seabed01_02c.rel rename to maps/map_seabed01_02c.rel diff --git a/public/maps/map_seabed01_02n.rel b/maps/map_seabed01_02n.rel similarity index 100% rename from public/maps/map_seabed01_02n.rel rename to maps/map_seabed01_02n.rel diff --git a/public/maps/map_seabed02_00c.rel b/maps/map_seabed02_00c.rel similarity index 100% rename from public/maps/map_seabed02_00c.rel rename to maps/map_seabed02_00c.rel diff --git a/public/maps/map_seabed02_00n.rel b/maps/map_seabed02_00n.rel similarity index 100% rename from public/maps/map_seabed02_00n.rel rename to maps/map_seabed02_00n.rel diff --git a/public/maps/map_seabed02_01c.rel b/maps/map_seabed02_01c.rel similarity index 100% rename from public/maps/map_seabed02_01c.rel rename to maps/map_seabed02_01c.rel diff --git a/public/maps/map_seabed02_01n.rel b/maps/map_seabed02_01n.rel similarity index 100% rename from public/maps/map_seabed02_01n.rel rename to maps/map_seabed02_01n.rel diff --git a/public/maps/map_seabed02_02c.rel b/maps/map_seabed02_02c.rel similarity index 100% rename from public/maps/map_seabed02_02c.rel rename to maps/map_seabed02_02c.rel diff --git a/public/maps/map_seabed02_02n.rel b/maps/map_seabed02_02n.rel similarity index 100% rename from public/maps/map_seabed02_02n.rel rename to maps/map_seabed02_02n.rel diff --git a/public/maps/map_soccer11c.rel b/maps/map_soccer11c.rel similarity index 100% rename from public/maps/map_soccer11c.rel rename to maps/map_soccer11c.rel diff --git a/public/maps/map_soccer11n.rel b/maps/map_soccer11n.rel similarity index 100% rename from public/maps/map_soccer11n.rel rename to maps/map_soccer11n.rel diff --git a/public/maps/map_soccer12c.rel b/maps/map_soccer12c.rel similarity index 100% rename from public/maps/map_soccer12c.rel rename to maps/map_soccer12c.rel diff --git a/public/maps/map_soccer12n.rel b/maps/map_soccer12n.rel similarity index 100% rename from public/maps/map_soccer12n.rel rename to maps/map_soccer12n.rel diff --git a/public/maps/map_space01_00c.rel b/maps/map_space01_00c.rel similarity index 100% rename from public/maps/map_space01_00c.rel rename to maps/map_space01_00c.rel diff --git a/public/maps/map_space01_00n.rel b/maps/map_space01_00n.rel similarity index 100% rename from public/maps/map_space01_00n.rel rename to maps/map_space01_00n.rel diff --git a/public/maps/map_space01_01c.rel b/maps/map_space01_01c.rel similarity index 100% rename from public/maps/map_space01_01c.rel rename to maps/map_space01_01c.rel diff --git a/public/maps/map_space01_01n.rel b/maps/map_space01_01n.rel similarity index 100% rename from public/maps/map_space01_01n.rel rename to maps/map_space01_01n.rel diff --git a/public/maps/map_space01_02c.rel b/maps/map_space01_02c.rel similarity index 100% rename from public/maps/map_space01_02c.rel rename to maps/map_space01_02c.rel diff --git a/public/maps/map_space01_02n.rel b/maps/map_space01_02n.rel similarity index 100% rename from public/maps/map_space01_02n.rel rename to maps/map_space01_02n.rel diff --git a/public/maps/map_space02_00c.rel b/maps/map_space02_00c.rel similarity index 100% rename from public/maps/map_space02_00c.rel rename to maps/map_space02_00c.rel diff --git a/public/maps/map_space02_00n.rel b/maps/map_space02_00n.rel similarity index 100% rename from public/maps/map_space02_00n.rel rename to maps/map_space02_00n.rel diff --git a/public/maps/map_space02_01c.rel b/maps/map_space02_01c.rel similarity index 100% rename from public/maps/map_space02_01c.rel rename to maps/map_space02_01c.rel diff --git a/public/maps/map_space02_01n.rel b/maps/map_space02_01n.rel similarity index 100% rename from public/maps/map_space02_01n.rel rename to maps/map_space02_01n.rel diff --git a/public/maps/map_space02_02c.rel b/maps/map_space02_02c.rel similarity index 100% rename from public/maps/map_space02_02c.rel rename to maps/map_space02_02c.rel diff --git a/public/maps/map_space02_02n.rel b/maps/map_space02_02n.rel similarity index 100% rename from public/maps/map_space02_02n.rel rename to maps/map_space02_02n.rel diff --git a/public/maps/map_test01_00c.rel b/maps/map_test01_00c.rel similarity index 100% rename from public/maps/map_test01_00c.rel rename to maps/map_test01_00c.rel diff --git a/public/maps/map_test01_00n.rel b/maps/map_test01_00n.rel similarity index 100% rename from public/maps/map_test01_00n.rel rename to maps/map_test01_00n.rel diff --git a/public/maps/map_vs01_00c.rel b/maps/map_vs01_00c.rel similarity index 100% rename from public/maps/map_vs01_00c.rel rename to maps/map_vs01_00c.rel diff --git a/public/maps/map_vs01_00n.rel b/maps/map_vs01_00n.rel similarity index 100% rename from public/maps/map_vs01_00n.rel rename to maps/map_vs01_00n.rel diff --git a/public/maps/map_vs01_01c.rel b/maps/map_vs01_01c.rel similarity index 100% rename from public/maps/map_vs01_01c.rel rename to maps/map_vs01_01c.rel diff --git a/public/maps/map_vs01_01n.rel b/maps/map_vs01_01n.rel similarity index 100% rename from public/maps/map_vs01_01n.rel rename to maps/map_vs01_01n.rel diff --git a/public/maps/map_vs01_02c.rel b/maps/map_vs01_02c.rel similarity index 100% rename from public/maps/map_vs01_02c.rel rename to maps/map_vs01_02c.rel diff --git a/public/maps/map_vs01_02n.rel b/maps/map_vs01_02n.rel similarity index 100% rename from public/maps/map_vs01_02n.rel rename to maps/map_vs01_02n.rel diff --git a/public/maps/map_vs02_00c.rel b/maps/map_vs02_00c.rel similarity index 100% rename from public/maps/map_vs02_00c.rel rename to maps/map_vs02_00c.rel diff --git a/public/maps/map_vs02_00n.rel b/maps/map_vs02_00n.rel similarity index 100% rename from public/maps/map_vs02_00n.rel rename to maps/map_vs02_00n.rel diff --git a/public/maps/map_vs02_01c.rel b/maps/map_vs02_01c.rel similarity index 100% rename from public/maps/map_vs02_01c.rel rename to maps/map_vs02_01c.rel diff --git a/public/maps/map_vs02_01n.rel b/maps/map_vs02_01n.rel similarity index 100% rename from public/maps/map_vs02_01n.rel rename to maps/map_vs02_01n.rel diff --git a/public/maps/map_vs02_02c.rel b/maps/map_vs02_02c.rel similarity index 100% rename from public/maps/map_vs02_02c.rel rename to maps/map_vs02_02c.rel diff --git a/public/maps/map_vs02_02n.rel b/maps/map_vs02_02n.rel similarity index 100% rename from public/maps/map_vs02_02n.rel rename to maps/map_vs02_02n.rel diff --git a/public/maps/map_wilds01_00c.rel b/maps/map_wilds01_00c.rel similarity index 100% rename from public/maps/map_wilds01_00c.rel rename to maps/map_wilds01_00c.rel diff --git a/public/maps/map_wilds01_00n.rel b/maps/map_wilds01_00n.rel similarity index 100% rename from public/maps/map_wilds01_00n.rel rename to maps/map_wilds01_00n.rel diff --git a/public/maps/map_wilds01_01c.rel b/maps/map_wilds01_01c.rel similarity index 100% rename from public/maps/map_wilds01_01c.rel rename to maps/map_wilds01_01c.rel diff --git a/public/maps/map_wilds01_01n.rel b/maps/map_wilds01_01n.rel similarity index 100% rename from public/maps/map_wilds01_01n.rel rename to maps/map_wilds01_01n.rel diff --git a/public/maps/map_wilds01_02c.rel b/maps/map_wilds01_02c.rel similarity index 100% rename from public/maps/map_wilds01_02c.rel rename to maps/map_wilds01_02c.rel diff --git a/public/maps/map_wilds01_02n.rel b/maps/map_wilds01_02n.rel similarity index 100% rename from public/maps/map_wilds01_02n.rel rename to maps/map_wilds01_02n.rel diff --git a/public/maps/map_wilds01_03c.rel b/maps/map_wilds01_03c.rel similarity index 100% rename from public/maps/map_wilds01_03c.rel rename to maps/map_wilds01_03c.rel diff --git a/public/maps/map_wilds01_03n.rel b/maps/map_wilds01_03n.rel similarity index 100% rename from public/maps/map_wilds01_03n.rel rename to maps/map_wilds01_03n.rel diff --git a/public/npcs/AlRappy.nj b/npcs/AlRappy.nj similarity index 100% rename from public/npcs/AlRappy.nj rename to npcs/AlRappy.nj diff --git a/public/npcs/AlRappy.xvm b/npcs/AlRappy.xvm similarity index 100% rename from public/npcs/AlRappy.xvm rename to npcs/AlRappy.xvm diff --git a/public/npcs/Astark.nj b/npcs/Astark.nj similarity index 100% rename from public/npcs/Astark.nj rename to npcs/Astark.nj diff --git a/public/npcs/Astark.xvm b/npcs/Astark.xvm similarity index 100% rename from public/npcs/Astark.xvm rename to npcs/Astark.xvm diff --git a/public/npcs/BaBoota.nj b/npcs/BaBoota.nj similarity index 100% rename from public/npcs/BaBoota.nj rename to npcs/BaBoota.nj diff --git a/public/npcs/BaBoota.xvm b/npcs/BaBoota.xvm similarity index 100% rename from public/npcs/BaBoota.xvm rename to npcs/BaBoota.xvm diff --git a/public/npcs/BarbaRay.nj b/npcs/BarbaRay.nj similarity index 100% rename from public/npcs/BarbaRay.nj rename to npcs/BarbaRay.nj diff --git a/public/npcs/BarbaRay.xvm b/npcs/BarbaRay.xvm similarity index 100% rename from public/npcs/BarbaRay.xvm rename to npcs/BarbaRay.xvm diff --git a/public/npcs/BarbarousWolf.nj b/npcs/BarbarousWolf.nj similarity index 100% rename from public/npcs/BarbarousWolf.nj rename to npcs/BarbarousWolf.nj diff --git a/public/npcs/BarbarousWolf.xvm b/npcs/BarbarousWolf.xvm similarity index 100% rename from public/npcs/BarbarousWolf.xvm rename to npcs/BarbarousWolf.xvm diff --git a/public/npcs/BlueSoldier.nj b/npcs/BlueSoldier.nj similarity index 100% rename from public/npcs/BlueSoldier.nj rename to npcs/BlueSoldier.nj diff --git a/public/npcs/BlueSoldier.xvm b/npcs/BlueSoldier.xvm similarity index 100% rename from public/npcs/BlueSoldier.xvm rename to npcs/BlueSoldier.xvm diff --git a/public/npcs/Booma.nj b/npcs/Booma.nj similarity index 100% rename from public/npcs/Booma.nj rename to npcs/Booma.nj diff --git a/public/npcs/Booma.xvm b/npcs/Booma.xvm similarity index 100% rename from public/npcs/Booma.xvm rename to npcs/Booma.xvm diff --git a/public/npcs/Boota.nj b/npcs/Boota.nj similarity index 100% rename from public/npcs/Boota.nj rename to npcs/Boota.nj diff --git a/public/npcs/Boota.xvm b/npcs/Boota.xvm similarity index 100% rename from public/npcs/Boota.xvm rename to npcs/Boota.xvm diff --git a/public/npcs/Bulclaw.nj b/npcs/Bulclaw.nj similarity index 100% rename from public/npcs/Bulclaw.nj rename to npcs/Bulclaw.nj diff --git a/public/npcs/Bulclaw.xvm b/npcs/Bulclaw.xvm similarity index 100% rename from public/npcs/Bulclaw.xvm rename to npcs/Bulclaw.xvm diff --git a/public/npcs/Bulk.nj b/npcs/Bulk.nj similarity index 100% rename from public/npcs/Bulk.nj rename to npcs/Bulk.nj diff --git a/public/npcs/Bulk.xvm b/npcs/Bulk.xvm similarity index 100% rename from public/npcs/Bulk.xvm rename to npcs/Bulk.xvm diff --git a/public/npcs/Canadine.nj b/npcs/Canadine.nj similarity index 100% rename from public/npcs/Canadine.nj rename to npcs/Canadine.nj diff --git a/public/npcs/Canadine.xvm b/npcs/Canadine.xvm similarity index 100% rename from public/npcs/Canadine.xvm rename to npcs/Canadine.xvm diff --git a/public/npcs/Canane.nj b/npcs/Canane.nj similarity index 100% rename from public/npcs/Canane.nj rename to npcs/Canane.nj diff --git a/public/npcs/Canane.xvm b/npcs/Canane.xvm similarity index 100% rename from public/npcs/Canane.xvm rename to npcs/Canane.xvm diff --git a/public/npcs/ChaosBringer.nj b/npcs/ChaosBringer.nj similarity index 100% rename from public/npcs/ChaosBringer.nj rename to npcs/ChaosBringer.nj diff --git a/public/npcs/ChaosBringer.xvm b/npcs/ChaosBringer.xvm similarity index 100% rename from public/npcs/ChaosBringer.xvm rename to npcs/ChaosBringer.xvm diff --git a/public/npcs/ChaosSorcerer.nj b/npcs/ChaosSorcerer.nj similarity index 100% rename from public/npcs/ChaosSorcerer.nj rename to npcs/ChaosSorcerer.nj diff --git a/public/npcs/ChaosSorcerer.xvm b/npcs/ChaosSorcerer.xvm similarity index 100% rename from public/npcs/ChaosSorcerer.xvm rename to npcs/ChaosSorcerer.xvm diff --git a/public/npcs/Claw.nj b/npcs/Claw.nj similarity index 100% rename from public/npcs/Claw.nj rename to npcs/Claw.nj diff --git a/public/npcs/Claw.xvm b/npcs/Claw.xvm similarity index 100% rename from public/npcs/Claw.xvm rename to npcs/Claw.xvm diff --git a/public/npcs/DarkBelra.nj b/npcs/DarkBelra.nj similarity index 100% rename from public/npcs/DarkBelra.nj rename to npcs/DarkBelra.nj diff --git a/public/npcs/DarkBelra.xvm b/npcs/DarkBelra.xvm similarity index 100% rename from public/npcs/DarkBelra.xvm rename to npcs/DarkBelra.xvm diff --git a/public/npcs/DarkFalz.nj b/npcs/DarkFalz.nj similarity index 100% rename from public/npcs/DarkFalz.nj rename to npcs/DarkFalz.nj diff --git a/public/npcs/DarkFalz.xvm b/npcs/DarkFalz.xvm similarity index 100% rename from public/npcs/DarkFalz.xvm rename to npcs/DarkFalz.xvm diff --git a/public/npcs/DarkGunner.nj b/npcs/DarkGunner.nj similarity index 100% rename from public/npcs/DarkGunner.nj rename to npcs/DarkGunner.nj diff --git a/public/npcs/DarkGunner.xvm b/npcs/DarkGunner.xvm similarity index 100% rename from public/npcs/DarkGunner.xvm rename to npcs/DarkGunner.xvm diff --git a/public/npcs/DeRolLe.nj b/npcs/DeRolLe.nj similarity index 100% rename from public/npcs/DeRolLe.nj rename to npcs/DeRolLe.nj diff --git a/public/npcs/DeRolLe.xvm b/npcs/DeRolLe.xvm similarity index 100% rename from public/npcs/DeRolLe.xvm rename to npcs/DeRolLe.xvm diff --git a/public/npcs/DelLily.nj b/npcs/DelLily.nj similarity index 100% rename from public/npcs/DelLily.nj rename to npcs/DelLily.nj diff --git a/public/npcs/DelLily.xvm b/npcs/DelLily.xvm similarity index 100% rename from public/npcs/DelLily.xvm rename to npcs/DelLily.xvm diff --git a/public/npcs/DelRappy.nj b/npcs/DelRappy.nj similarity index 100% rename from public/npcs/DelRappy.nj rename to npcs/DelRappy.nj diff --git a/public/npcs/DelRappy.xvm b/npcs/DelRappy.xvm similarity index 100% rename from public/npcs/DelRappy.xvm rename to npcs/DelRappy.xvm diff --git a/public/npcs/Delbiter.nj b/npcs/Delbiter.nj similarity index 100% rename from public/npcs/Delbiter.nj rename to npcs/Delbiter.nj diff --git a/public/npcs/Delbiter.xvm b/npcs/Delbiter.xvm similarity index 100% rename from public/npcs/Delbiter.xvm rename to npcs/Delbiter.xvm diff --git a/public/npcs/Deldepth.nj b/npcs/Deldepth.nj similarity index 100% rename from public/npcs/Deldepth.nj rename to npcs/Deldepth.nj diff --git a/public/npcs/Deldepth.xvm b/npcs/Deldepth.xvm similarity index 100% rename from public/npcs/Deldepth.xvm rename to npcs/Deldepth.xvm diff --git a/public/npcs/Delsaber.nj b/npcs/Delsaber.nj similarity index 100% rename from public/npcs/Delsaber.nj rename to npcs/Delsaber.nj diff --git a/public/npcs/Delsaber.xvm b/npcs/Delsaber.xvm similarity index 100% rename from public/npcs/Delsaber.xvm rename to npcs/Delsaber.xvm diff --git a/public/npcs/Dimenian.nj b/npcs/Dimenian.nj similarity index 100% rename from public/npcs/Dimenian.nj rename to npcs/Dimenian.nj diff --git a/public/npcs/Dimenian.xvm b/npcs/Dimenian.xvm similarity index 100% rename from public/npcs/Dimenian.xvm rename to npcs/Dimenian.xvm diff --git a/public/npcs/Dolmdarl.nj b/npcs/Dolmdarl.nj similarity index 100% rename from public/npcs/Dolmdarl.nj rename to npcs/Dolmdarl.nj diff --git a/public/npcs/Dolmdarl.xvm b/npcs/Dolmdarl.xvm similarity index 100% rename from public/npcs/Dolmdarl.xvm rename to npcs/Dolmdarl.xvm diff --git a/public/npcs/Dolmolm.nj b/npcs/Dolmolm.nj similarity index 100% rename from public/npcs/Dolmolm.nj rename to npcs/Dolmolm.nj diff --git a/public/npcs/Dolmolm.xvm b/npcs/Dolmolm.xvm similarity index 100% rename from public/npcs/Dolmolm.xvm rename to npcs/Dolmolm.xvm diff --git a/public/npcs/Dorphon.nj b/npcs/Dorphon.nj similarity index 100% rename from public/npcs/Dorphon.nj rename to npcs/Dorphon.nj diff --git a/public/npcs/Dorphon.xvm b/npcs/Dorphon.xvm similarity index 100% rename from public/npcs/Dorphon.xvm rename to npcs/Dorphon.xvm diff --git a/public/npcs/DorphonEclair.nj b/npcs/DorphonEclair.nj similarity index 100% rename from public/npcs/DorphonEclair.nj rename to npcs/DorphonEclair.nj diff --git a/public/npcs/DorphonEclair.xvm b/npcs/DorphonEclair.xvm similarity index 100% rename from public/npcs/DorphonEclair.xvm rename to npcs/DorphonEclair.xvm diff --git a/public/npcs/Dragon.nj b/npcs/Dragon.nj similarity index 100% rename from public/npcs/Dragon.nj rename to npcs/Dragon.nj diff --git a/public/npcs/Dragon.xvm b/npcs/Dragon.xvm similarity index 100% rename from public/npcs/Dragon.xvm rename to npcs/Dragon.xvm diff --git a/public/npcs/Dubchic.nj b/npcs/Dubchic.nj similarity index 100% rename from public/npcs/Dubchic.nj rename to npcs/Dubchic.nj diff --git a/public/npcs/Dubchic.xvm b/npcs/Dubchic.xvm similarity index 100% rename from public/npcs/Dubchic.xvm rename to npcs/Dubchic.xvm diff --git a/public/npcs/Dubswitch.xj b/npcs/Dubswitch.xj similarity index 100% rename from public/npcs/Dubswitch.xj rename to npcs/Dubswitch.xj diff --git a/public/npcs/Dubswitch.xvm b/npcs/Dubswitch.xvm similarity index 100% rename from public/npcs/Dubswitch.xvm rename to npcs/Dubswitch.xvm diff --git a/public/npcs/Epsilon.nj b/npcs/Epsilon.nj similarity index 100% rename from public/npcs/Epsilon.nj rename to npcs/Epsilon.nj diff --git a/public/npcs/Epsilon.xvm b/npcs/Epsilon.xvm similarity index 100% rename from public/npcs/Epsilon.xvm rename to npcs/Epsilon.xvm diff --git a/public/npcs/EvilShark.nj b/npcs/EvilShark.nj similarity index 100% rename from public/npcs/EvilShark.nj rename to npcs/EvilShark.nj diff --git a/public/npcs/EvilShark.xvm b/npcs/EvilShark.xvm similarity index 100% rename from public/npcs/EvilShark.xvm rename to npcs/EvilShark.xvm diff --git a/public/npcs/FemaleFat.nj b/npcs/FemaleFat.nj similarity index 100% rename from public/npcs/FemaleFat.nj rename to npcs/FemaleFat.nj diff --git a/public/npcs/FemaleFat.xvm b/npcs/FemaleFat.xvm similarity index 100% rename from public/npcs/FemaleFat.xvm rename to npcs/FemaleFat.xvm diff --git a/public/npcs/FemaleMacho.nj b/npcs/FemaleMacho.nj similarity index 100% rename from public/npcs/FemaleMacho.nj rename to npcs/FemaleMacho.nj diff --git a/public/npcs/FemaleMacho.xvm b/npcs/FemaleMacho.xvm similarity index 100% rename from public/npcs/FemaleMacho.xvm rename to npcs/FemaleMacho.xvm diff --git a/public/npcs/FemaleTall.nj b/npcs/FemaleTall.nj similarity index 100% rename from public/npcs/FemaleTall.nj rename to npcs/FemaleTall.nj diff --git a/public/npcs/FemaleTall.xvm b/npcs/FemaleTall.xvm similarity index 100% rename from public/npcs/FemaleTall.xvm rename to npcs/FemaleTall.xvm diff --git a/public/npcs/GalGryphon.nj b/npcs/GalGryphon.nj similarity index 100% rename from public/npcs/GalGryphon.nj rename to npcs/GalGryphon.nj diff --git a/public/npcs/GalGryphon.xvm b/npcs/GalGryphon.xvm similarity index 100% rename from public/npcs/GalGryphon.xvm rename to npcs/GalGryphon.xvm diff --git a/public/npcs/Garanz.nj b/npcs/Garanz.nj similarity index 100% rename from public/npcs/Garanz.nj rename to npcs/Garanz.nj diff --git a/public/npcs/Garanz.xvm b/npcs/Garanz.xvm similarity index 100% rename from public/npcs/Garanz.xvm rename to npcs/Garanz.xvm diff --git a/public/npcs/Gee.nj b/npcs/Gee.nj similarity index 100% rename from public/npcs/Gee.nj rename to npcs/Gee.nj diff --git a/public/npcs/Gee.xvm b/npcs/Gee.xvm similarity index 100% rename from public/npcs/Gee.xvm rename to npcs/Gee.xvm diff --git a/public/npcs/GiGue.nj b/npcs/GiGue.nj similarity index 100% rename from public/npcs/GiGue.nj rename to npcs/GiGue.nj diff --git a/public/npcs/GiGue.xvm b/npcs/GiGue.xvm similarity index 100% rename from public/npcs/GiGue.xvm rename to npcs/GiGue.xvm diff --git a/public/npcs/Gibbles.nj b/npcs/Gibbles.nj similarity index 100% rename from public/npcs/Gibbles.nj rename to npcs/Gibbles.nj diff --git a/public/npcs/Gibbles.xvm b/npcs/Gibbles.xvm similarity index 100% rename from public/npcs/Gibbles.xvm rename to npcs/Gibbles.xvm diff --git a/public/npcs/Gigobooma.nj b/npcs/Gigobooma.nj similarity index 100% rename from public/npcs/Gigobooma.nj rename to npcs/Gigobooma.nj diff --git a/public/npcs/Gigobooma.xvm b/npcs/Gigobooma.xvm similarity index 100% rename from public/npcs/Gigobooma.xvm rename to npcs/Gigobooma.xvm diff --git a/public/npcs/Gilchic.nj b/npcs/Gilchic.nj similarity index 100% rename from public/npcs/Gilchic.nj rename to npcs/Gilchic.nj diff --git a/public/npcs/Gilchic.xvm b/npcs/Gilchic.xvm similarity index 100% rename from public/npcs/Gilchic.xvm rename to npcs/Gilchic.xvm diff --git a/public/npcs/Girtablulu.nj b/npcs/Girtablulu.nj similarity index 100% rename from public/npcs/Girtablulu.nj rename to npcs/Girtablulu.nj diff --git a/public/npcs/Girtablulu.xvm b/npcs/Girtablulu.xvm similarity index 100% rename from public/npcs/Girtablulu.xvm rename to npcs/Girtablulu.xvm diff --git a/public/npcs/Gobooma.nj b/npcs/Gobooma.nj similarity index 100% rename from public/npcs/Gobooma.nj rename to npcs/Gobooma.nj diff --git a/public/npcs/Gobooma.xvm b/npcs/Gobooma.xvm similarity index 100% rename from public/npcs/Gobooma.xvm rename to npcs/Gobooma.xvm diff --git a/public/npcs/GolDragon.nj b/npcs/GolDragon.nj similarity index 100% rename from public/npcs/GolDragon.nj rename to npcs/GolDragon.nj diff --git a/public/npcs/GolDragon.xvm b/npcs/GolDragon.xvm similarity index 100% rename from public/npcs/GolDragon.xvm rename to npcs/GolDragon.xvm diff --git a/public/npcs/Goran.nj b/npcs/Goran.nj similarity index 100% rename from public/npcs/Goran.nj rename to npcs/Goran.nj diff --git a/public/npcs/Goran.xvm b/npcs/Goran.xvm similarity index 100% rename from public/npcs/Goran.xvm rename to npcs/Goran.xvm diff --git a/public/npcs/GoranDetonator.nj b/npcs/GoranDetonator.nj similarity index 100% rename from public/npcs/GoranDetonator.nj rename to npcs/GoranDetonator.nj diff --git a/public/npcs/GoranDetonator.xvm b/npcs/GoranDetonator.xvm similarity index 100% rename from public/npcs/GoranDetonator.xvm rename to npcs/GoranDetonator.xvm diff --git a/public/npcs/GrassAssassin.nj b/npcs/GrassAssassin.nj similarity index 100% rename from public/npcs/GrassAssassin.nj rename to npcs/GrassAssassin.nj diff --git a/public/npcs/GrassAssassin.xvm b/npcs/GrassAssassin.xvm similarity index 100% rename from public/npcs/GrassAssassin.xvm rename to npcs/GrassAssassin.xvm diff --git a/public/npcs/GuilShark.nj b/npcs/GuilShark.nj similarity index 100% rename from public/npcs/GuilShark.nj rename to npcs/GuilShark.nj diff --git a/public/npcs/GuilShark.xvm b/npcs/GuilShark.xvm similarity index 100% rename from public/npcs/GuilShark.xvm rename to npcs/GuilShark.xvm diff --git a/public/npcs/GuildLady.nj b/npcs/GuildLady.nj similarity index 100% rename from public/npcs/GuildLady.nj rename to npcs/GuildLady.nj diff --git a/public/npcs/GuildLady.xvm b/npcs/GuildLady.xvm similarity index 100% rename from public/npcs/GuildLady.xvm rename to npcs/GuildLady.xvm diff --git a/public/npcs/Hildebear.nj b/npcs/Hildebear.nj similarity index 100% rename from public/npcs/Hildebear.nj rename to npcs/Hildebear.nj diff --git a/public/npcs/Hildebear.xvm b/npcs/Hildebear.xvm similarity index 100% rename from public/npcs/Hildebear.xvm rename to npcs/Hildebear.xvm diff --git a/public/npcs/Hildeblue.nj b/npcs/Hildeblue.nj similarity index 100% rename from public/npcs/Hildeblue.nj rename to npcs/Hildeblue.nj diff --git a/public/npcs/Hildeblue.xvm b/npcs/Hildeblue.xvm similarity index 100% rename from public/npcs/Hildeblue.xvm rename to npcs/Hildeblue.xvm diff --git a/public/npcs/IllGill.nj b/npcs/IllGill.nj similarity index 100% rename from public/npcs/IllGill.nj rename to npcs/IllGill.nj diff --git a/public/npcs/IllGill.xvm b/npcs/IllGill.xvm similarity index 100% rename from public/npcs/IllGill.xvm rename to npcs/IllGill.xvm diff --git a/public/npcs/Irene.nj b/npcs/Irene.nj similarity index 100% rename from public/npcs/Irene.nj rename to npcs/Irene.nj diff --git a/public/npcs/Irene.xvm b/npcs/Irene.xvm similarity index 100% rename from public/npcs/Irene.xvm rename to npcs/Irene.xvm diff --git a/public/npcs/ItemShop.nj b/npcs/ItemShop.nj similarity index 100% rename from public/npcs/ItemShop.nj rename to npcs/ItemShop.nj diff --git a/public/npcs/ItemShop.xvm b/npcs/ItemShop.xvm similarity index 100% rename from public/npcs/ItemShop.xvm rename to npcs/ItemShop.xvm diff --git a/public/npcs/Kondrieu.nj b/npcs/Kondrieu.nj similarity index 100% rename from public/npcs/Kondrieu.nj rename to npcs/Kondrieu.nj diff --git a/public/npcs/Kondrieu.xvm b/npcs/Kondrieu.xvm similarity index 100% rename from public/npcs/Kondrieu.xvm rename to npcs/Kondrieu.xvm diff --git a/public/npcs/LaDimenian.nj b/npcs/LaDimenian.nj similarity index 100% rename from public/npcs/LaDimenian.nj rename to npcs/LaDimenian.nj diff --git a/public/npcs/LaDimenian.xvm b/npcs/LaDimenian.xvm similarity index 100% rename from public/npcs/LaDimenian.xvm rename to npcs/LaDimenian.xvm diff --git a/public/npcs/LoveRappy.nj b/npcs/LoveRappy.nj similarity index 100% rename from public/npcs/LoveRappy.nj rename to npcs/LoveRappy.nj diff --git a/public/npcs/LoveRappy.xvm b/npcs/LoveRappy.xvm similarity index 100% rename from public/npcs/LoveRappy.xvm rename to npcs/LoveRappy.xvm diff --git a/public/npcs/MaleDwarf.nj b/npcs/MaleDwarf.nj similarity index 100% rename from public/npcs/MaleDwarf.nj rename to npcs/MaleDwarf.nj diff --git a/public/npcs/MaleDwarf.xvm b/npcs/MaleDwarf.xvm similarity index 100% rename from public/npcs/MaleDwarf.xvm rename to npcs/MaleDwarf.xvm diff --git a/public/npcs/MaleFat.nj b/npcs/MaleFat.nj similarity index 100% rename from public/npcs/MaleFat.nj rename to npcs/MaleFat.nj diff --git a/public/npcs/MaleFat.xvm b/npcs/MaleFat.xvm similarity index 100% rename from public/npcs/MaleFat.xvm rename to npcs/MaleFat.xvm diff --git a/public/npcs/MaleMacho.nj b/npcs/MaleMacho.nj similarity index 100% rename from public/npcs/MaleMacho.nj rename to npcs/MaleMacho.nj diff --git a/public/npcs/MaleMacho.xvm b/npcs/MaleMacho.xvm similarity index 100% rename from public/npcs/MaleMacho.xvm rename to npcs/MaleMacho.xvm diff --git a/public/npcs/MaleOld.nj b/npcs/MaleOld.nj similarity index 100% rename from public/npcs/MaleOld.nj rename to npcs/MaleOld.nj diff --git a/public/npcs/MaleOld.xvm b/npcs/MaleOld.xvm similarity index 100% rename from public/npcs/MaleOld.xvm rename to npcs/MaleOld.xvm diff --git a/public/npcs/Mericarol.nj b/npcs/Mericarol.nj similarity index 100% rename from public/npcs/Mericarol.nj rename to npcs/Mericarol.nj diff --git a/public/npcs/Mericarol.xvm b/npcs/Mericarol.xvm similarity index 100% rename from public/npcs/Mericarol.xvm rename to npcs/Mericarol.xvm diff --git a/public/npcs/Mericus.nj b/npcs/Mericus.nj similarity index 100% rename from public/npcs/Mericus.nj rename to npcs/Mericus.nj diff --git a/public/npcs/Mericus.xvm b/npcs/Mericus.xvm similarity index 100% rename from public/npcs/Mericus.xvm rename to npcs/Mericus.xvm diff --git a/public/npcs/Merikle.nj b/npcs/Merikle.nj similarity index 100% rename from public/npcs/Merikle.nj rename to npcs/Merikle.nj diff --git a/public/npcs/Merikle.xvm b/npcs/Merikle.xvm similarity index 100% rename from public/npcs/Merikle.xvm rename to npcs/Merikle.xvm diff --git a/public/npcs/Merillia.nj b/npcs/Merillia.nj similarity index 100% rename from public/npcs/Merillia.nj rename to npcs/Merillia.nj diff --git a/public/npcs/Merillia.xvm b/npcs/Merillia.xvm similarity index 100% rename from public/npcs/Merillia.xvm rename to npcs/Merillia.xvm diff --git a/public/npcs/Meriltas.nj b/npcs/Meriltas.nj similarity index 100% rename from public/npcs/Meriltas.nj rename to npcs/Meriltas.nj diff --git a/public/npcs/Meriltas.xvm b/npcs/Meriltas.xvm similarity index 100% rename from public/npcs/Meriltas.xvm rename to npcs/Meriltas.xvm diff --git a/public/npcs/MerissaA.nj b/npcs/MerissaA.nj similarity index 100% rename from public/npcs/MerissaA.nj rename to npcs/MerissaA.nj diff --git a/public/npcs/MerissaA.xvm b/npcs/MerissaA.xvm similarity index 100% rename from public/npcs/MerissaA.xvm rename to npcs/MerissaA.xvm diff --git a/public/npcs/MerissaAA.nj b/npcs/MerissaAA.nj similarity index 100% rename from public/npcs/MerissaAA.nj rename to npcs/MerissaAA.nj diff --git a/public/npcs/MerissaAA.xvm b/npcs/MerissaAA.xvm similarity index 100% rename from public/npcs/MerissaAA.xvm rename to npcs/MerissaAA.xvm diff --git a/public/npcs/Monest.nj b/npcs/Monest.nj similarity index 100% rename from public/npcs/Monest.nj rename to npcs/Monest.nj diff --git a/public/npcs/Monest.xvm b/npcs/Monest.xvm similarity index 100% rename from public/npcs/Monest.xvm rename to npcs/Monest.xvm diff --git a/public/npcs/Morfos.nj b/npcs/Morfos.nj similarity index 100% rename from public/npcs/Morfos.nj rename to npcs/Morfos.nj diff --git a/public/npcs/Morfos.xvm b/npcs/Morfos.xvm similarity index 100% rename from public/npcs/Morfos.xvm rename to npcs/Morfos.xvm diff --git a/public/npcs/Mothmant.nj b/npcs/Mothmant.nj similarity index 100% rename from public/npcs/Mothmant.nj rename to npcs/Mothmant.nj diff --git a/public/npcs/Mothmant.xvm b/npcs/Mothmant.xvm similarity index 100% rename from public/npcs/Mothmant.xvm rename to npcs/Mothmant.xvm diff --git a/public/npcs/NanoDragon.nj b/npcs/NanoDragon.nj similarity index 100% rename from public/npcs/NanoDragon.nj rename to npcs/NanoDragon.nj diff --git a/public/npcs/NanoDragon.xvm b/npcs/NanoDragon.xvm similarity index 100% rename from public/npcs/NanoDragon.xvm rename to npcs/NanoDragon.xvm diff --git a/public/npcs/NarLily.nj b/npcs/NarLily.nj similarity index 100% rename from public/npcs/NarLily.nj rename to npcs/NarLily.nj diff --git a/public/npcs/NarLily.xvm b/npcs/NarLily.xvm similarity index 100% rename from public/npcs/NarLily.xvm rename to npcs/NarLily.xvm diff --git a/public/npcs/Nurse.nj b/npcs/Nurse.nj similarity index 100% rename from public/npcs/Nurse.nj rename to npcs/Nurse.nj diff --git a/public/npcs/Nurse.xvm b/npcs/Nurse.xvm similarity index 100% rename from public/npcs/Nurse.xvm rename to npcs/Nurse.xvm diff --git a/public/npcs/Nurse2.nj b/npcs/Nurse2.nj similarity index 100% rename from public/npcs/Nurse2.nj rename to npcs/Nurse2.nj diff --git a/public/npcs/Nurse2.xvm b/npcs/Nurse2.xvm similarity index 100% rename from public/npcs/Nurse2.xvm rename to npcs/Nurse2.xvm diff --git a/public/npcs/OlgaFlow.nj b/npcs/OlgaFlow.nj similarity index 100% rename from public/npcs/OlgaFlow.nj rename to npcs/OlgaFlow.nj diff --git a/public/npcs/OlgaFlow.xvm b/npcs/OlgaFlow.xvm similarity index 100% rename from public/npcs/OlgaFlow.xvm rename to npcs/OlgaFlow.xvm diff --git a/public/npcs/PalShark.nj b/npcs/PalShark.nj similarity index 100% rename from public/npcs/PalShark.nj rename to npcs/PalShark.nj diff --git a/public/npcs/PalShark.xvm b/npcs/PalShark.xvm similarity index 100% rename from public/npcs/PalShark.xvm rename to npcs/PalShark.xvm diff --git a/public/npcs/PanArms.nj b/npcs/PanArms.nj similarity index 100% rename from public/npcs/PanArms.nj rename to npcs/PanArms.nj diff --git a/public/npcs/PanArms.xvm b/npcs/PanArms.xvm similarity index 100% rename from public/npcs/PanArms.xvm rename to npcs/PanArms.xvm diff --git a/public/npcs/Pazuzu.nj b/npcs/Pazuzu.nj similarity index 100% rename from public/npcs/Pazuzu.nj rename to npcs/Pazuzu.nj diff --git a/public/npcs/Pazuzu.xvm b/npcs/Pazuzu.xvm similarity index 100% rename from public/npcs/Pazuzu.xvm rename to npcs/Pazuzu.xvm diff --git a/public/npcs/PofuillySlime.nj b/npcs/PofuillySlime.nj similarity index 100% rename from public/npcs/PofuillySlime.nj rename to npcs/PofuillySlime.nj diff --git a/public/npcs/PofuillySlime.xvm b/npcs/PofuillySlime.xvm similarity index 100% rename from public/npcs/PofuillySlime.xvm rename to npcs/PofuillySlime.xvm diff --git a/public/npcs/PoisonLily.nj b/npcs/PoisonLily.nj similarity index 100% rename from public/npcs/PoisonLily.nj rename to npcs/PoisonLily.nj diff --git a/public/npcs/PoisonLily.xvm b/npcs/PoisonLily.xvm similarity index 100% rename from public/npcs/PoisonLily.xvm rename to npcs/PoisonLily.xvm diff --git a/public/npcs/PouillySlime.nj b/npcs/PouillySlime.nj similarity index 100% rename from public/npcs/PouillySlime.nj rename to npcs/PouillySlime.nj diff --git a/public/npcs/PouillySlime.xvm b/npcs/PouillySlime.xvm similarity index 100% rename from public/npcs/PouillySlime.xvm rename to npcs/PouillySlime.xvm diff --git a/public/npcs/Principal.nj b/npcs/Principal.nj similarity index 100% rename from public/npcs/Principal.nj rename to npcs/Principal.nj diff --git a/public/npcs/Principal.xvm b/npcs/Principal.xvm similarity index 100% rename from public/npcs/Principal.xvm rename to npcs/Principal.xvm diff --git a/public/npcs/PyroGoran.nj b/npcs/PyroGoran.nj similarity index 100% rename from public/npcs/PyroGoran.nj rename to npcs/PyroGoran.nj diff --git a/public/npcs/PyroGoran.xvm b/npcs/PyroGoran.xvm similarity index 100% rename from public/npcs/PyroGoran.xvm rename to npcs/PyroGoran.xvm diff --git a/public/npcs/RagRappy.nj b/npcs/RagRappy.nj similarity index 100% rename from public/npcs/RagRappy.nj rename to npcs/RagRappy.nj diff --git a/public/npcs/RagRappy.xvm b/npcs/RagRappy.xvm similarity index 100% rename from public/npcs/RagRappy.xvm rename to npcs/RagRappy.xvm diff --git a/public/npcs/Recobox.nj b/npcs/Recobox.nj similarity index 100% rename from public/npcs/Recobox.nj rename to npcs/Recobox.nj diff --git a/public/npcs/Recobox.xvm b/npcs/Recobox.xvm similarity index 100% rename from public/npcs/Recobox.xvm rename to npcs/Recobox.xvm diff --git a/public/npcs/RedSoldier.nj b/npcs/RedSoldier.nj similarity index 100% rename from public/npcs/RedSoldier.nj rename to npcs/RedSoldier.nj diff --git a/public/npcs/RedSoldier.xvm b/npcs/RedSoldier.xvm similarity index 100% rename from public/npcs/RedSoldier.xvm rename to npcs/RedSoldier.xvm diff --git a/public/npcs/SaintMillion.nj b/npcs/SaintMillion.nj similarity index 100% rename from public/npcs/SaintMillion.nj rename to npcs/SaintMillion.nj diff --git a/public/npcs/SaintMillion.xvm b/npcs/SaintMillion.xvm similarity index 100% rename from public/npcs/SaintMillion.xvm rename to npcs/SaintMillion.xvm diff --git a/public/npcs/SandRappy.nj b/npcs/SandRappy.nj similarity index 100% rename from public/npcs/SandRappy.nj rename to npcs/SandRappy.nj diff --git a/public/npcs/SandRappy.xvm b/npcs/SandRappy.xvm similarity index 100% rename from public/npcs/SandRappy.xvm rename to npcs/SandRappy.xvm diff --git a/public/npcs/SatelliteLizard.nj b/npcs/SatelliteLizard.nj similarity index 100% rename from public/npcs/SatelliteLizard.nj rename to npcs/SatelliteLizard.nj diff --git a/public/npcs/SatelliteLizard.xvm b/npcs/SatelliteLizard.xvm similarity index 100% rename from public/npcs/SatelliteLizard.xvm rename to npcs/SatelliteLizard.xvm diff --git a/public/npcs/SavageWolf.nj b/npcs/SavageWolf.nj similarity index 100% rename from public/npcs/SavageWolf.nj rename to npcs/SavageWolf.nj diff --git a/public/npcs/SavageWolf.xvm b/npcs/SavageWolf.xvm similarity index 100% rename from public/npcs/SavageWolf.xvm rename to npcs/SavageWolf.xvm diff --git a/public/npcs/Scientist.nj b/npcs/Scientist.nj similarity index 100% rename from public/npcs/Scientist.nj rename to npcs/Scientist.nj diff --git a/public/npcs/Scientist.xvm b/npcs/Scientist.xvm similarity index 100% rename from public/npcs/Scientist.xvm rename to npcs/Scientist.xvm diff --git a/public/npcs/Shambertin.nj b/npcs/Shambertin.nj similarity index 100% rename from public/npcs/Shambertin.nj rename to npcs/Shambertin.nj diff --git a/public/npcs/Shambertin.xvm b/npcs/Shambertin.xvm similarity index 100% rename from public/npcs/Shambertin.xvm rename to npcs/Shambertin.xvm diff --git a/public/npcs/SinowBeat.nj b/npcs/SinowBeat.nj similarity index 100% rename from public/npcs/SinowBeat.nj rename to npcs/SinowBeat.nj diff --git a/public/npcs/SinowBeat.xvm b/npcs/SinowBeat.xvm similarity index 100% rename from public/npcs/SinowBeat.xvm rename to npcs/SinowBeat.xvm diff --git a/public/npcs/SinowBerill.nj b/npcs/SinowBerill.nj similarity index 100% rename from public/npcs/SinowBerill.nj rename to npcs/SinowBerill.nj diff --git a/public/npcs/SinowBerill.xvm b/npcs/SinowBerill.xvm similarity index 100% rename from public/npcs/SinowBerill.xvm rename to npcs/SinowBerill.xvm diff --git a/public/npcs/SinowGold.nj b/npcs/SinowGold.nj similarity index 100% rename from public/npcs/SinowGold.nj rename to npcs/SinowGold.nj diff --git a/public/npcs/SinowGold.xvm b/npcs/SinowGold.xvm similarity index 100% rename from public/npcs/SinowGold.xvm rename to npcs/SinowGold.xvm diff --git a/public/npcs/SinowSpigell.nj b/npcs/SinowSpigell.nj similarity index 100% rename from public/npcs/SinowSpigell.nj rename to npcs/SinowSpigell.nj diff --git a/public/npcs/SinowSpigell.xvm b/npcs/SinowSpigell.xvm similarity index 100% rename from public/npcs/SinowSpigell.xvm rename to npcs/SinowSpigell.xvm diff --git a/public/npcs/SinowZele.nj b/npcs/SinowZele.nj similarity index 100% rename from public/npcs/SinowZele.nj rename to npcs/SinowZele.nj diff --git a/public/npcs/SinowZele.xvm b/npcs/SinowZele.xvm similarity index 100% rename from public/npcs/SinowZele.xvm rename to npcs/SinowZele.xvm diff --git a/public/npcs/SinowZoa.nj b/npcs/SinowZoa.nj similarity index 100% rename from public/npcs/SinowZoa.nj rename to npcs/SinowZoa.nj diff --git a/public/npcs/SinowZoa.xvm b/npcs/SinowZoa.xvm similarity index 100% rename from public/npcs/SinowZoa.xvm rename to npcs/SinowZoa.xvm diff --git a/public/npcs/SoDimenian.nj b/npcs/SoDimenian.nj similarity index 100% rename from public/npcs/SoDimenian.nj rename to npcs/SoDimenian.nj diff --git a/public/npcs/SoDimenian.xvm b/npcs/SoDimenian.xvm similarity index 100% rename from public/npcs/SoDimenian.xvm rename to npcs/SoDimenian.xvm diff --git a/public/npcs/Tekker.nj b/npcs/Tekker.nj similarity index 100% rename from public/npcs/Tekker.nj rename to npcs/Tekker.nj diff --git a/public/npcs/Tekker.xvm b/npcs/Tekker.xvm similarity index 100% rename from public/npcs/Tekker.xvm rename to npcs/Tekker.xvm diff --git a/public/npcs/UlGibbon.nj b/npcs/UlGibbon.nj similarity index 100% rename from public/npcs/UlGibbon.nj rename to npcs/UlGibbon.nj diff --git a/public/npcs/UlGibbon.xvm b/npcs/UlGibbon.xvm similarity index 100% rename from public/npcs/UlGibbon.xvm rename to npcs/UlGibbon.xvm diff --git a/public/npcs/VolOpt.nj b/npcs/VolOpt.nj similarity index 100% rename from public/npcs/VolOpt.nj rename to npcs/VolOpt.nj diff --git a/public/npcs/VolOpt.xvm b/npcs/VolOpt.xvm similarity index 100% rename from public/npcs/VolOpt.xvm rename to npcs/VolOpt.xvm diff --git a/public/npcs/Yowie.nj b/npcs/Yowie.nj similarity index 100% rename from public/npcs/Yowie.nj rename to npcs/Yowie.nj diff --git a/public/npcs/Yowie.xvm b/npcs/Yowie.xvm similarity index 100% rename from public/npcs/Yowie.xvm rename to npcs/Yowie.xvm diff --git a/public/npcs/ZeBoota.nj b/npcs/ZeBoota.nj similarity index 100% rename from public/npcs/ZeBoota.nj rename to npcs/ZeBoota.nj diff --git a/public/npcs/ZeBoota.xvm b/npcs/ZeBoota.xvm similarity index 100% rename from public/npcs/ZeBoota.xvm rename to npcs/ZeBoota.xvm diff --git a/public/npcs/ZolGibbon.nj b/npcs/ZolGibbon.nj similarity index 100% rename from public/npcs/ZolGibbon.nj rename to npcs/ZolGibbon.nj diff --git a/public/npcs/ZolGibbon.xvm b/npcs/ZolGibbon.xvm similarity index 100% rename from public/npcs/ZolGibbon.xvm rename to npcs/ZolGibbon.xvm diff --git a/public/npcs/Zu.nj b/npcs/Zu.nj similarity index 100% rename from public/npcs/Zu.nj rename to npcs/Zu.nj diff --git a/public/npcs/Zu.xvm b/npcs/Zu.xvm similarity index 100% rename from public/npcs/Zu.xvm rename to npcs/Zu.xvm diff --git a/public/objects/10-2.xj b/objects/10-2.xj similarity index 100% rename from public/objects/10-2.xj rename to objects/10-2.xj diff --git a/public/objects/10-2.xvm b/objects/10-2.xvm similarity index 100% rename from public/objects/10-2.xvm rename to objects/10-2.xvm diff --git a/public/objects/10.xj b/objects/10.xj similarity index 100% rename from public/objects/10.xj rename to objects/10.xj diff --git a/public/objects/10.xvm b/objects/10.xvm similarity index 100% rename from public/objects/10.xvm rename to objects/10.xvm diff --git a/public/objects/11-2.xj b/objects/11-2.xj similarity index 100% rename from public/objects/11-2.xj rename to objects/11-2.xj diff --git a/public/objects/11-2.xvm b/objects/11-2.xvm similarity index 100% rename from public/objects/11-2.xvm rename to objects/11-2.xvm diff --git a/public/objects/11.xj b/objects/11.xj similarity index 100% rename from public/objects/11.xj rename to objects/11.xj diff --git a/public/objects/11.xvm b/objects/11.xvm similarity index 100% rename from public/objects/11.xvm rename to objects/11.xvm diff --git a/public/objects/12-2.xj b/objects/12-2.xj similarity index 100% rename from public/objects/12-2.xj rename to objects/12-2.xj diff --git a/public/objects/12-2.xvm b/objects/12-2.xvm similarity index 100% rename from public/objects/12-2.xvm rename to objects/12-2.xvm diff --git a/public/objects/12.xj b/objects/12.xj similarity index 100% rename from public/objects/12.xj rename to objects/12.xj diff --git a/public/objects/12.xvm b/objects/12.xvm similarity index 100% rename from public/objects/12.xvm rename to objects/12.xvm diff --git a/public/objects/128-2.xj b/objects/128-2.xj similarity index 100% rename from public/objects/128-2.xj rename to objects/128-2.xj diff --git a/public/objects/128-3.xj b/objects/128-3.xj similarity index 100% rename from public/objects/128-3.xj rename to objects/128-3.xj diff --git a/public/objects/128-4.xj b/objects/128-4.xj similarity index 100% rename from public/objects/128-4.xj rename to objects/128-4.xj diff --git a/public/objects/128-5.xj b/objects/128-5.xj similarity index 100% rename from public/objects/128-5.xj rename to objects/128-5.xj diff --git a/public/objects/128.xj b/objects/128.xj similarity index 100% rename from public/objects/128.xj rename to objects/128.xj diff --git a/public/objects/128.xvm b/objects/128.xvm similarity index 100% rename from public/objects/128.xvm rename to objects/128.xvm diff --git a/public/objects/129-2.xj b/objects/129-2.xj similarity index 100% rename from public/objects/129-2.xj rename to objects/129-2.xj diff --git a/public/objects/129-3.xj b/objects/129-3.xj similarity index 100% rename from public/objects/129-3.xj rename to objects/129-3.xj diff --git a/public/objects/129.xj b/objects/129.xj similarity index 100% rename from public/objects/129.xj rename to objects/129.xj diff --git a/public/objects/129.xvm b/objects/129.xvm similarity index 100% rename from public/objects/129.xvm rename to objects/129.xvm diff --git a/public/objects/13-2.xj b/objects/13-2.xj similarity index 100% rename from public/objects/13-2.xj rename to objects/13-2.xj diff --git a/public/objects/13-2.xvm b/objects/13-2.xvm similarity index 100% rename from public/objects/13-2.xvm rename to objects/13-2.xvm diff --git a/public/objects/13.xj b/objects/13.xj similarity index 100% rename from public/objects/13.xj rename to objects/13.xj diff --git a/public/objects/13.xvm b/objects/13.xvm similarity index 100% rename from public/objects/13.xvm rename to objects/13.xvm diff --git a/public/objects/130-2.xj b/objects/130-2.xj similarity index 100% rename from public/objects/130-2.xj rename to objects/130-2.xj diff --git a/public/objects/130.xj b/objects/130.xj similarity index 100% rename from public/objects/130.xj rename to objects/130.xj diff --git a/public/objects/130.xvm b/objects/130.xvm similarity index 100% rename from public/objects/130.xvm rename to objects/130.xvm diff --git a/public/objects/131-2.xj b/objects/131-2.xj similarity index 100% rename from public/objects/131-2.xj rename to objects/131-2.xj diff --git a/public/objects/131.xj b/objects/131.xj similarity index 100% rename from public/objects/131.xj rename to objects/131.xj diff --git a/public/objects/131.xvm b/objects/131.xvm similarity index 100% rename from public/objects/131.xvm rename to objects/131.xvm diff --git a/public/objects/132-2.xj b/objects/132-2.xj similarity index 100% rename from public/objects/132-2.xj rename to objects/132-2.xj diff --git a/public/objects/132-3.xj b/objects/132-3.xj similarity index 100% rename from public/objects/132-3.xj rename to objects/132-3.xj diff --git a/public/objects/132.xj b/objects/132.xj similarity index 100% rename from public/objects/132.xj rename to objects/132.xj diff --git a/public/objects/132.xvm b/objects/132.xvm similarity index 100% rename from public/objects/132.xvm rename to objects/132.xvm diff --git a/public/objects/133.xj b/objects/133.xj similarity index 100% rename from public/objects/133.xj rename to objects/133.xj diff --git a/public/objects/133.xvm b/objects/133.xvm similarity index 100% rename from public/objects/133.xvm rename to objects/133.xvm diff --git a/public/objects/134.xj b/objects/134.xj similarity index 100% rename from public/objects/134.xj rename to objects/134.xj diff --git a/public/objects/134.xvm b/objects/134.xvm similarity index 100% rename from public/objects/134.xvm rename to objects/134.xvm diff --git a/public/objects/135-0.xj b/objects/135-0.xj similarity index 100% rename from public/objects/135-0.xj rename to objects/135-0.xj diff --git a/public/objects/135-0.xvm b/objects/135-0.xvm similarity index 100% rename from public/objects/135-0.xvm rename to objects/135-0.xvm diff --git a/public/objects/135-1.xj b/objects/135-1.xj similarity index 100% rename from public/objects/135-1.xj rename to objects/135-1.xj diff --git a/public/objects/135-1.xvm b/objects/135-1.xvm similarity index 100% rename from public/objects/135-1.xvm rename to objects/135-1.xvm diff --git a/public/objects/135.xj b/objects/135.xj similarity index 100% rename from public/objects/135.xj rename to objects/135.xj diff --git a/public/objects/135.xvm b/objects/135.xvm similarity index 100% rename from public/objects/135.xvm rename to objects/135.xvm diff --git a/public/objects/136-2.xj b/objects/136-2.xj similarity index 100% rename from public/objects/136-2.xj rename to objects/136-2.xj diff --git a/public/objects/136-3.xj b/objects/136-3.xj similarity index 100% rename from public/objects/136-3.xj rename to objects/136-3.xj diff --git a/public/objects/136.xj b/objects/136.xj similarity index 100% rename from public/objects/136.xj rename to objects/136.xj diff --git a/public/objects/136.xvm b/objects/136.xvm similarity index 100% rename from public/objects/136.xvm rename to objects/136.xvm diff --git a/public/objects/137.xj b/objects/137.xj similarity index 100% rename from public/objects/137.xj rename to objects/137.xj diff --git a/public/objects/137.xvm b/objects/137.xvm similarity index 100% rename from public/objects/137.xvm rename to objects/137.xvm diff --git a/public/objects/139-0.xj b/objects/139-0.xj similarity index 100% rename from public/objects/139-0.xj rename to objects/139-0.xj diff --git a/public/objects/139-0.xvm b/objects/139-0.xvm similarity index 100% rename from public/objects/139-0.xvm rename to objects/139-0.xvm diff --git a/public/objects/139-1.xj b/objects/139-1.xj similarity index 100% rename from public/objects/139-1.xj rename to objects/139-1.xj diff --git a/public/objects/139-1.xvm b/objects/139-1.xvm similarity index 100% rename from public/objects/139-1.xvm rename to objects/139-1.xvm diff --git a/public/objects/139.xj b/objects/139.xj similarity index 100% rename from public/objects/139.xj rename to objects/139.xj diff --git a/public/objects/139.xvm b/objects/139.xvm similarity index 100% rename from public/objects/139.xvm rename to objects/139.xvm diff --git a/public/objects/140-2.xj b/objects/140-2.xj similarity index 100% rename from public/objects/140-2.xj rename to objects/140-2.xj diff --git a/public/objects/140.xj b/objects/140.xj similarity index 100% rename from public/objects/140.xj rename to objects/140.xj diff --git a/public/objects/140.xvm b/objects/140.xvm similarity index 100% rename from public/objects/140.xvm rename to objects/140.xvm diff --git a/public/objects/141.xj b/objects/141.xj similarity index 100% rename from public/objects/141.xj rename to objects/141.xj diff --git a/public/objects/141.xvm b/objects/141.xvm similarity index 100% rename from public/objects/141.xvm rename to objects/141.xvm diff --git a/public/objects/142-2.xj b/objects/142-2.xj similarity index 100% rename from public/objects/142-2.xj rename to objects/142-2.xj diff --git a/public/objects/142-2.xvm b/objects/142-2.xvm similarity index 100% rename from public/objects/142-2.xvm rename to objects/142-2.xvm diff --git a/public/objects/142.xj b/objects/142.xj similarity index 100% rename from public/objects/142.xj rename to objects/142.xj diff --git a/public/objects/142.xvm b/objects/142.xvm similarity index 100% rename from public/objects/142.xvm rename to objects/142.xvm diff --git a/public/objects/143.xj b/objects/143.xj similarity index 100% rename from public/objects/143.xj rename to objects/143.xj diff --git a/public/objects/143.xvm b/objects/143.xvm similarity index 100% rename from public/objects/143.xvm rename to objects/143.xvm diff --git a/public/objects/144-2.xj b/objects/144-2.xj similarity index 100% rename from public/objects/144-2.xj rename to objects/144-2.xj diff --git a/public/objects/144-a.xj b/objects/144-a.xj similarity index 100% rename from public/objects/144-a.xj rename to objects/144-a.xj diff --git a/public/objects/144-a.xvm b/objects/144-a.xvm similarity index 100% rename from public/objects/144-a.xvm rename to objects/144-a.xvm diff --git a/public/objects/144.xj b/objects/144.xj similarity index 100% rename from public/objects/144.xj rename to objects/144.xj diff --git a/public/objects/144.xvm b/objects/144.xvm similarity index 100% rename from public/objects/144.xvm rename to objects/144.xvm diff --git a/public/objects/145-2.xj b/objects/145-2.xj similarity index 100% rename from public/objects/145-2.xj rename to objects/145-2.xj diff --git a/public/objects/145-3.xj b/objects/145-3.xj similarity index 100% rename from public/objects/145-3.xj rename to objects/145-3.xj diff --git a/public/objects/145.xj b/objects/145.xj similarity index 100% rename from public/objects/145.xj rename to objects/145.xj diff --git a/public/objects/145.xvm b/objects/145.xvm similarity index 100% rename from public/objects/145.xvm rename to objects/145.xvm diff --git a/public/objects/146-2.xj b/objects/146-2.xj similarity index 100% rename from public/objects/146-2.xj rename to objects/146-2.xj diff --git a/public/objects/146-3.xj b/objects/146-3.xj similarity index 100% rename from public/objects/146-3.xj rename to objects/146-3.xj diff --git a/public/objects/146-4.xj b/objects/146-4.xj similarity index 100% rename from public/objects/146-4.xj rename to objects/146-4.xj diff --git a/public/objects/146.xj b/objects/146.xj similarity index 100% rename from public/objects/146.xj rename to objects/146.xj diff --git a/public/objects/146.xvm b/objects/146.xvm similarity index 100% rename from public/objects/146.xvm rename to objects/146.xvm diff --git a/public/objects/147-2.xj b/objects/147-2.xj similarity index 100% rename from public/objects/147-2.xj rename to objects/147-2.xj diff --git a/public/objects/147-3.xj b/objects/147-3.xj similarity index 100% rename from public/objects/147-3.xj rename to objects/147-3.xj diff --git a/public/objects/147-4.xj b/objects/147-4.xj similarity index 100% rename from public/objects/147-4.xj rename to objects/147-4.xj diff --git a/public/objects/147.xj b/objects/147.xj similarity index 100% rename from public/objects/147.xj rename to objects/147.xj diff --git a/public/objects/147.xvm b/objects/147.xvm similarity index 100% rename from public/objects/147.xvm rename to objects/147.xvm diff --git a/public/objects/149.xj b/objects/149.xj similarity index 100% rename from public/objects/149.xj rename to objects/149.xj diff --git a/public/objects/149.xvm b/objects/149.xvm similarity index 100% rename from public/objects/149.xvm rename to objects/149.xvm diff --git a/public/objects/15.xj b/objects/15.xj similarity index 100% rename from public/objects/15.xj rename to objects/15.xj diff --git a/public/objects/15.xvm b/objects/15.xvm similarity index 100% rename from public/objects/15.xvm rename to objects/15.xvm diff --git a/public/objects/150-2.xj b/objects/150-2.xj similarity index 100% rename from public/objects/150-2.xj rename to objects/150-2.xj diff --git a/public/objects/150.xj b/objects/150.xj similarity index 100% rename from public/objects/150.xj rename to objects/150.xj diff --git a/public/objects/150.xvm b/objects/150.xvm similarity index 100% rename from public/objects/150.xvm rename to objects/150.xvm diff --git a/public/objects/151-2.xj b/objects/151-2.xj similarity index 100% rename from public/objects/151-2.xj rename to objects/151-2.xj diff --git a/public/objects/151.xj b/objects/151.xj similarity index 100% rename from public/objects/151.xj rename to objects/151.xj diff --git a/public/objects/151.xvm b/objects/151.xvm similarity index 100% rename from public/objects/151.xvm rename to objects/151.xvm diff --git a/public/objects/19.xj b/objects/19.xj similarity index 100% rename from public/objects/19.xj rename to objects/19.xj diff --git a/public/objects/19.xvm b/objects/19.xvm similarity index 100% rename from public/objects/19.xvm rename to objects/19.xvm diff --git a/public/objects/192.xj b/objects/192.xj similarity index 100% rename from public/objects/192.xj rename to objects/192.xj diff --git a/public/objects/192.xvm b/objects/192.xvm similarity index 100% rename from public/objects/192.xvm rename to objects/192.xvm diff --git a/public/objects/193.xj b/objects/193.xj similarity index 100% rename from public/objects/193.xj rename to objects/193.xj diff --git a/public/objects/193.xvm b/objects/193.xvm similarity index 100% rename from public/objects/193.xvm rename to objects/193.xvm diff --git a/public/objects/194.xj b/objects/194.xj similarity index 100% rename from public/objects/194.xj rename to objects/194.xj diff --git a/public/objects/194.xvm b/objects/194.xvm similarity index 100% rename from public/objects/194.xvm rename to objects/194.xvm diff --git a/public/objects/195-2.xj b/objects/195-2.xj similarity index 100% rename from public/objects/195-2.xj rename to objects/195-2.xj diff --git a/public/objects/195-3.xj b/objects/195-3.xj similarity index 100% rename from public/objects/195-3.xj rename to objects/195-3.xj diff --git a/public/objects/195.xj b/objects/195.xj similarity index 100% rename from public/objects/195.xj rename to objects/195.xj diff --git a/public/objects/195.xvm b/objects/195.xvm similarity index 100% rename from public/objects/195.xvm rename to objects/195.xvm diff --git a/public/objects/196.xj b/objects/196.xj similarity index 100% rename from public/objects/196.xj rename to objects/196.xj diff --git a/public/objects/196.xvm b/objects/196.xvm similarity index 100% rename from public/objects/196.xvm rename to objects/196.xvm diff --git a/public/objects/197.xj b/objects/197.xj similarity index 100% rename from public/objects/197.xj rename to objects/197.xj diff --git a/public/objects/197.xvm b/objects/197.xvm similarity index 100% rename from public/objects/197.xvm rename to objects/197.xvm diff --git a/public/objects/198.xj b/objects/198.xj similarity index 100% rename from public/objects/198.xj rename to objects/198.xj diff --git a/public/objects/198.xvm b/objects/198.xvm similarity index 100% rename from public/objects/198.xvm rename to objects/198.xvm diff --git a/public/objects/199.xj b/objects/199.xj similarity index 100% rename from public/objects/199.xj rename to objects/199.xj diff --git a/public/objects/199.xvm b/objects/199.xvm similarity index 100% rename from public/objects/199.xvm rename to objects/199.xvm diff --git a/public/objects/2-2.xj b/objects/2-2.xj similarity index 100% rename from public/objects/2-2.xj rename to objects/2-2.xj diff --git a/public/objects/2.xj b/objects/2.xj similarity index 100% rename from public/objects/2.xj rename to objects/2.xj diff --git a/public/objects/2.xvm b/objects/2.xvm similarity index 100% rename from public/objects/2.xvm rename to objects/2.xvm diff --git a/public/objects/200.xj b/objects/200.xj similarity index 100% rename from public/objects/200.xj rename to objects/200.xj diff --git a/public/objects/200.xvm b/objects/200.xvm similarity index 100% rename from public/objects/200.xvm rename to objects/200.xvm diff --git a/public/objects/201.xj b/objects/201.xj similarity index 100% rename from public/objects/201.xj rename to objects/201.xj diff --git a/public/objects/201.xvm b/objects/201.xvm similarity index 100% rename from public/objects/201.xvm rename to objects/201.xvm diff --git a/public/objects/204.nj b/objects/204.nj similarity index 100% rename from public/objects/204.nj rename to objects/204.nj diff --git a/public/objects/204.xvm b/objects/204.xvm similarity index 100% rename from public/objects/204.xvm rename to objects/204.xvm diff --git a/public/objects/205.xj b/objects/205.xj similarity index 100% rename from public/objects/205.xj rename to objects/205.xj diff --git a/public/objects/205.xvm b/objects/205.xvm similarity index 100% rename from public/objects/205.xvm rename to objects/205.xvm diff --git a/public/objects/206.xj b/objects/206.xj similarity index 100% rename from public/objects/206.xj rename to objects/206.xj diff --git a/public/objects/206.xvm b/objects/206.xvm similarity index 100% rename from public/objects/206.xvm rename to objects/206.xvm diff --git a/public/objects/207-2.xj b/objects/207-2.xj similarity index 100% rename from public/objects/207-2.xj rename to objects/207-2.xj diff --git a/public/objects/207.xj b/objects/207.xj similarity index 100% rename from public/objects/207.xj rename to objects/207.xj diff --git a/public/objects/207.xvm b/objects/207.xvm similarity index 100% rename from public/objects/207.xvm rename to objects/207.xvm diff --git a/public/objects/208.xj b/objects/208.xj similarity index 100% rename from public/objects/208.xj rename to objects/208.xj diff --git a/public/objects/208.xvm b/objects/208.xvm similarity index 100% rename from public/objects/208.xvm rename to objects/208.xvm diff --git a/public/objects/209.xj b/objects/209.xj similarity index 100% rename from public/objects/209.xj rename to objects/209.xj diff --git a/public/objects/209.xvm b/objects/209.xvm similarity index 100% rename from public/objects/209.xvm rename to objects/209.xvm diff --git a/public/objects/210.xj b/objects/210.xj similarity index 100% rename from public/objects/210.xj rename to objects/210.xj diff --git a/public/objects/210.xvm b/objects/210.xvm similarity index 100% rename from public/objects/210.xvm rename to objects/210.xvm diff --git a/public/objects/211.xj b/objects/211.xj similarity index 100% rename from public/objects/211.xj rename to objects/211.xj diff --git a/public/objects/211.xvm b/objects/211.xvm similarity index 100% rename from public/objects/211.xvm rename to objects/211.xvm diff --git a/public/objects/212.xj b/objects/212.xj similarity index 100% rename from public/objects/212.xj rename to objects/212.xj diff --git a/public/objects/212.xvm b/objects/212.xvm similarity index 100% rename from public/objects/212.xvm rename to objects/212.xvm diff --git a/public/objects/213.xj b/objects/213.xj similarity index 100% rename from public/objects/213.xj rename to objects/213.xj diff --git a/public/objects/213.xvm b/objects/213.xvm similarity index 100% rename from public/objects/213.xvm rename to objects/213.xvm diff --git a/public/objects/214.xj b/objects/214.xj similarity index 100% rename from public/objects/214.xj rename to objects/214.xj diff --git a/public/objects/214.xvm b/objects/214.xvm similarity index 100% rename from public/objects/214.xvm rename to objects/214.xvm diff --git a/public/objects/215.xj b/objects/215.xj similarity index 100% rename from public/objects/215.xj rename to objects/215.xj diff --git a/public/objects/215.xvm b/objects/215.xvm similarity index 100% rename from public/objects/215.xvm rename to objects/215.xvm diff --git a/public/objects/216.xj b/objects/216.xj similarity index 100% rename from public/objects/216.xj rename to objects/216.xj diff --git a/public/objects/216.xvm b/objects/216.xvm similarity index 100% rename from public/objects/216.xvm rename to objects/216.xvm diff --git a/public/objects/217.xj b/objects/217.xj similarity index 100% rename from public/objects/217.xj rename to objects/217.xj diff --git a/public/objects/217.xvm b/objects/217.xvm similarity index 100% rename from public/objects/217.xvm rename to objects/217.xvm diff --git a/public/objects/218.xj b/objects/218.xj similarity index 100% rename from public/objects/218.xj rename to objects/218.xj diff --git a/public/objects/218.xvm b/objects/218.xvm similarity index 100% rename from public/objects/218.xvm rename to objects/218.xvm diff --git a/public/objects/219.xj b/objects/219.xj similarity index 100% rename from public/objects/219.xj rename to objects/219.xj diff --git a/public/objects/219.xvm b/objects/219.xvm similarity index 100% rename from public/objects/219.xvm rename to objects/219.xvm diff --git a/public/objects/220.xj b/objects/220.xj similarity index 100% rename from public/objects/220.xj rename to objects/220.xj diff --git a/public/objects/220.xvm b/objects/220.xvm similarity index 100% rename from public/objects/220.xvm rename to objects/220.xvm diff --git a/public/objects/222.xj b/objects/222.xj similarity index 100% rename from public/objects/222.xj rename to objects/222.xj diff --git a/public/objects/222.xvm b/objects/222.xvm similarity index 100% rename from public/objects/222.xvm rename to objects/222.xvm diff --git a/public/objects/223.xj b/objects/223.xj similarity index 100% rename from public/objects/223.xj rename to objects/223.xj diff --git a/public/objects/223.xvm b/objects/223.xvm similarity index 100% rename from public/objects/223.xvm rename to objects/223.xvm diff --git a/public/objects/224.xj b/objects/224.xj similarity index 100% rename from public/objects/224.xj rename to objects/224.xj diff --git a/public/objects/224.xvm b/objects/224.xvm similarity index 100% rename from public/objects/224.xvm rename to objects/224.xvm diff --git a/public/objects/225.xj b/objects/225.xj similarity index 100% rename from public/objects/225.xj rename to objects/225.xj diff --git a/public/objects/225.xvm b/objects/225.xvm similarity index 100% rename from public/objects/225.xvm rename to objects/225.xvm diff --git a/public/objects/25-2.xj b/objects/25-2.xj similarity index 100% rename from public/objects/25-2.xj rename to objects/25-2.xj diff --git a/public/objects/25.xj b/objects/25.xj similarity index 100% rename from public/objects/25.xj rename to objects/25.xj diff --git a/public/objects/25.xvm b/objects/25.xvm similarity index 100% rename from public/objects/25.xvm rename to objects/25.xvm diff --git a/public/objects/256.xj b/objects/256.xj similarity index 100% rename from public/objects/256.xj rename to objects/256.xj diff --git a/public/objects/256.xvm b/objects/256.xvm similarity index 100% rename from public/objects/256.xvm rename to objects/256.xvm diff --git a/public/objects/257.xj b/objects/257.xj similarity index 100% rename from public/objects/257.xj rename to objects/257.xj diff --git a/public/objects/257.xvm b/objects/257.xvm similarity index 100% rename from public/objects/257.xvm rename to objects/257.xvm diff --git a/public/objects/258.xj b/objects/258.xj similarity index 100% rename from public/objects/258.xj rename to objects/258.xj diff --git a/public/objects/258.xvm b/objects/258.xvm similarity index 100% rename from public/objects/258.xvm rename to objects/258.xvm diff --git a/public/objects/259.xj b/objects/259.xj similarity index 100% rename from public/objects/259.xj rename to objects/259.xj diff --git a/public/objects/259.xvm b/objects/259.xvm similarity index 100% rename from public/objects/259.xvm rename to objects/259.xvm diff --git a/public/objects/260.xj b/objects/260.xj similarity index 100% rename from public/objects/260.xj rename to objects/260.xj diff --git a/public/objects/260.xvm b/objects/260.xvm similarity index 100% rename from public/objects/260.xvm rename to objects/260.xvm diff --git a/public/objects/261.xj b/objects/261.xj similarity index 100% rename from public/objects/261.xj rename to objects/261.xj diff --git a/public/objects/261.xvm b/objects/261.xvm similarity index 100% rename from public/objects/261.xvm rename to objects/261.xvm diff --git a/public/objects/262.xj b/objects/262.xj similarity index 100% rename from public/objects/262.xj rename to objects/262.xj diff --git a/public/objects/262.xvm b/objects/262.xvm similarity index 100% rename from public/objects/262.xvm rename to objects/262.xvm diff --git a/public/objects/264-a.xj b/objects/264-a.xj similarity index 100% rename from public/objects/264-a.xj rename to objects/264-a.xj diff --git a/public/objects/264-a.xvm b/objects/264-a.xvm similarity index 100% rename from public/objects/264-a.xvm rename to objects/264-a.xvm diff --git a/public/objects/264-b.xj b/objects/264-b.xj similarity index 100% rename from public/objects/264-b.xj rename to objects/264-b.xj diff --git a/public/objects/264-b.xvm b/objects/264-b.xvm similarity index 100% rename from public/objects/264-b.xvm rename to objects/264-b.xvm diff --git a/public/objects/264.xj b/objects/264.xj similarity index 100% rename from public/objects/264.xj rename to objects/264.xj diff --git a/public/objects/264.xvm b/objects/264.xvm similarity index 100% rename from public/objects/264.xvm rename to objects/264.xvm diff --git a/public/objects/265-a.xj b/objects/265-a.xj similarity index 100% rename from public/objects/265-a.xj rename to objects/265-a.xj diff --git a/public/objects/265-a.xvm b/objects/265-a.xvm similarity index 100% rename from public/objects/265-a.xvm rename to objects/265-a.xvm diff --git a/public/objects/265-b.xj b/objects/265-b.xj similarity index 100% rename from public/objects/265-b.xj rename to objects/265-b.xj diff --git a/public/objects/265-b.xvm b/objects/265-b.xvm similarity index 100% rename from public/objects/265-b.xvm rename to objects/265-b.xvm diff --git a/public/objects/265.xj b/objects/265.xj similarity index 100% rename from public/objects/265.xj rename to objects/265.xj diff --git a/public/objects/265.xvm b/objects/265.xvm similarity index 100% rename from public/objects/265.xvm rename to objects/265.xvm diff --git a/public/objects/266-a.xj b/objects/266-a.xj similarity index 100% rename from public/objects/266-a.xj rename to objects/266-a.xj diff --git a/public/objects/266-a.xvm b/objects/266-a.xvm similarity index 100% rename from public/objects/266-a.xvm rename to objects/266-a.xvm diff --git a/public/objects/266-b.xj b/objects/266-b.xj similarity index 100% rename from public/objects/266-b.xj rename to objects/266-b.xj diff --git a/public/objects/266-b.xvm b/objects/266-b.xvm similarity index 100% rename from public/objects/266-b.xvm rename to objects/266-b.xvm diff --git a/public/objects/266.xj b/objects/266.xj similarity index 100% rename from public/objects/266.xj rename to objects/266.xj diff --git a/public/objects/266.xvm b/objects/266.xvm similarity index 100% rename from public/objects/266.xvm rename to objects/266.xvm diff --git a/public/objects/267.xj b/objects/267.xj similarity index 100% rename from public/objects/267.xj rename to objects/267.xj diff --git a/public/objects/267.xvm b/objects/267.xvm similarity index 100% rename from public/objects/267.xvm rename to objects/267.xvm diff --git a/public/objects/268.xj b/objects/268.xj similarity index 100% rename from public/objects/268.xj rename to objects/268.xj diff --git a/public/objects/268.xvm b/objects/268.xvm similarity index 100% rename from public/objects/268.xvm rename to objects/268.xvm diff --git a/public/objects/27-2.xj b/objects/27-2.xj similarity index 100% rename from public/objects/27-2.xj rename to objects/27-2.xj diff --git a/public/objects/27.xj b/objects/27.xj similarity index 100% rename from public/objects/27.xj rename to objects/27.xj diff --git a/public/objects/27.xvm b/objects/27.xvm similarity index 100% rename from public/objects/27.xvm rename to objects/27.xvm diff --git a/public/objects/28-2.xj b/objects/28-2.xj similarity index 100% rename from public/objects/28-2.xj rename to objects/28-2.xj diff --git a/public/objects/28.xj b/objects/28.xj similarity index 100% rename from public/objects/28.xj rename to objects/28.xj diff --git a/public/objects/28.xvm b/objects/28.xvm similarity index 100% rename from public/objects/28.xvm rename to objects/28.xvm diff --git a/public/objects/3-2.xj b/objects/3-2.xj similarity index 100% rename from public/objects/3-2.xj rename to objects/3-2.xj diff --git a/public/objects/3.xj b/objects/3.xj similarity index 100% rename from public/objects/3.xj rename to objects/3.xj diff --git a/public/objects/3.xvm b/objects/3.xvm similarity index 100% rename from public/objects/3.xvm rename to objects/3.xvm diff --git a/public/objects/304.nj b/objects/304.nj similarity index 100% rename from public/objects/304.nj rename to objects/304.nj diff --git a/public/objects/304.xvm b/objects/304.xvm similarity index 100% rename from public/objects/304.xvm rename to objects/304.xvm diff --git a/public/objects/320-2.xj b/objects/320-2.xj similarity index 100% rename from public/objects/320-2.xj rename to objects/320-2.xj diff --git a/public/objects/320-3.xj b/objects/320-3.xj similarity index 100% rename from public/objects/320-3.xj rename to objects/320-3.xj diff --git a/public/objects/320-4.xj b/objects/320-4.xj similarity index 100% rename from public/objects/320-4.xj rename to objects/320-4.xj diff --git a/public/objects/320-a.xj b/objects/320-a.xj similarity index 100% rename from public/objects/320-a.xj rename to objects/320-a.xj diff --git a/public/objects/320-a.xvm b/objects/320-a.xvm similarity index 100% rename from public/objects/320-a.xvm rename to objects/320-a.xvm diff --git a/public/objects/320-b.xj b/objects/320-b.xj similarity index 100% rename from public/objects/320-b.xj rename to objects/320-b.xj diff --git a/public/objects/320-b.xvm b/objects/320-b.xvm similarity index 100% rename from public/objects/320-b.xvm rename to objects/320-b.xvm diff --git a/public/objects/320-c.xj b/objects/320-c.xj similarity index 100% rename from public/objects/320-c.xj rename to objects/320-c.xj diff --git a/public/objects/320-c.xvm b/objects/320-c.xvm similarity index 100% rename from public/objects/320-c.xvm rename to objects/320-c.xvm diff --git a/public/objects/320.xj b/objects/320.xj similarity index 100% rename from public/objects/320.xj rename to objects/320.xj diff --git a/public/objects/320.xvm b/objects/320.xvm similarity index 100% rename from public/objects/320.xvm rename to objects/320.xvm diff --git a/public/objects/321-2.xj b/objects/321-2.xj similarity index 100% rename from public/objects/321-2.xj rename to objects/321-2.xj diff --git a/public/objects/321-a.xj b/objects/321-a.xj similarity index 100% rename from public/objects/321-a.xj rename to objects/321-a.xj diff --git a/public/objects/321-a.xvm b/objects/321-a.xvm similarity index 100% rename from public/objects/321-a.xvm rename to objects/321-a.xvm diff --git a/public/objects/321.xj b/objects/321.xj similarity index 100% rename from public/objects/321.xj rename to objects/321.xj diff --git a/public/objects/321.xvm b/objects/321.xvm similarity index 100% rename from public/objects/321.xvm rename to objects/321.xvm diff --git a/public/objects/322-2.xj b/objects/322-2.xj similarity index 100% rename from public/objects/322-2.xj rename to objects/322-2.xj diff --git a/public/objects/322-a.xj b/objects/322-a.xj similarity index 100% rename from public/objects/322-a.xj rename to objects/322-a.xj diff --git a/public/objects/322-a.xvm b/objects/322-a.xvm similarity index 100% rename from public/objects/322-a.xvm rename to objects/322-a.xvm diff --git a/public/objects/322.xj b/objects/322.xj similarity index 100% rename from public/objects/322.xj rename to objects/322.xj diff --git a/public/objects/322.xvm b/objects/322.xvm similarity index 100% rename from public/objects/322.xvm rename to objects/322.xvm diff --git a/public/objects/323.xj b/objects/323.xj similarity index 100% rename from public/objects/323.xj rename to objects/323.xj diff --git a/public/objects/323.xvm b/objects/323.xvm similarity index 100% rename from public/objects/323.xvm rename to objects/323.xvm diff --git a/public/objects/324.xj b/objects/324.xj similarity index 100% rename from public/objects/324.xj rename to objects/324.xj diff --git a/public/objects/324.xvm b/objects/324.xvm similarity index 100% rename from public/objects/324.xvm rename to objects/324.xvm diff --git a/public/objects/325.xj b/objects/325.xj similarity index 100% rename from public/objects/325.xj rename to objects/325.xj diff --git a/public/objects/325.xvm b/objects/325.xvm similarity index 100% rename from public/objects/325.xvm rename to objects/325.xvm diff --git a/public/objects/326.xj b/objects/326.xj similarity index 100% rename from public/objects/326.xj rename to objects/326.xj diff --git a/public/objects/326.xvm b/objects/326.xvm similarity index 100% rename from public/objects/326.xvm rename to objects/326.xvm diff --git a/public/objects/327.xj b/objects/327.xj similarity index 100% rename from public/objects/327.xj rename to objects/327.xj diff --git a/public/objects/327.xvm b/objects/327.xvm similarity index 100% rename from public/objects/327.xvm rename to objects/327.xvm diff --git a/public/objects/328.xj b/objects/328.xj similarity index 100% rename from public/objects/328.xj rename to objects/328.xj diff --git a/public/objects/328.xvm b/objects/328.xvm similarity index 100% rename from public/objects/328.xvm rename to objects/328.xvm diff --git a/public/objects/329.xj b/objects/329.xj similarity index 100% rename from public/objects/329.xj rename to objects/329.xj diff --git a/public/objects/329.xvm b/objects/329.xvm similarity index 100% rename from public/objects/329.xvm rename to objects/329.xvm diff --git a/public/objects/330.xj b/objects/330.xj similarity index 100% rename from public/objects/330.xj rename to objects/330.xj diff --git a/public/objects/330.xvm b/objects/330.xvm similarity index 100% rename from public/objects/330.xvm rename to objects/330.xvm diff --git a/public/objects/331.xj b/objects/331.xj similarity index 100% rename from public/objects/331.xj rename to objects/331.xj diff --git a/public/objects/331.xvm b/objects/331.xvm similarity index 100% rename from public/objects/331.xvm rename to objects/331.xvm diff --git a/public/objects/332.xj b/objects/332.xj similarity index 100% rename from public/objects/332.xj rename to objects/332.xj diff --git a/public/objects/332.xvm b/objects/332.xvm similarity index 100% rename from public/objects/332.xvm rename to objects/332.xvm diff --git a/public/objects/333.xj b/objects/333.xj similarity index 100% rename from public/objects/333.xj rename to objects/333.xj diff --git a/public/objects/333.xvm b/objects/333.xvm similarity index 100% rename from public/objects/333.xvm rename to objects/333.xvm diff --git a/public/objects/334.xj b/objects/334.xj similarity index 100% rename from public/objects/334.xj rename to objects/334.xj diff --git a/public/objects/334.xvm b/objects/334.xvm similarity index 100% rename from public/objects/334.xvm rename to objects/334.xvm diff --git a/public/objects/335.xj b/objects/335.xj similarity index 100% rename from public/objects/335.xj rename to objects/335.xj diff --git a/public/objects/335.xvm b/objects/335.xvm similarity index 100% rename from public/objects/335.xvm rename to objects/335.xvm diff --git a/public/objects/336.xj b/objects/336.xj similarity index 100% rename from public/objects/336.xj rename to objects/336.xj diff --git a/public/objects/336.xvm b/objects/336.xvm similarity index 100% rename from public/objects/336.xvm rename to objects/336.xvm diff --git a/public/objects/337.xj b/objects/337.xj similarity index 100% rename from public/objects/337.xj rename to objects/337.xj diff --git a/public/objects/337.xvm b/objects/337.xvm similarity index 100% rename from public/objects/337.xvm rename to objects/337.xvm diff --git a/public/objects/338.xj b/objects/338.xj similarity index 100% rename from public/objects/338.xj rename to objects/338.xj diff --git a/public/objects/338.xvm b/objects/338.xvm similarity index 100% rename from public/objects/338.xvm rename to objects/338.xvm diff --git a/public/objects/339-2.xj b/objects/339-2.xj similarity index 100% rename from public/objects/339-2.xj rename to objects/339-2.xj diff --git a/public/objects/339-3.xj b/objects/339-3.xj similarity index 100% rename from public/objects/339-3.xj rename to objects/339-3.xj diff --git a/public/objects/339-4.xj b/objects/339-4.xj similarity index 100% rename from public/objects/339-4.xj rename to objects/339-4.xj diff --git a/public/objects/339.xj b/objects/339.xj similarity index 100% rename from public/objects/339.xj rename to objects/339.xj diff --git a/public/objects/339.xvm b/objects/339.xvm similarity index 100% rename from public/objects/339.xvm rename to objects/339.xvm diff --git a/public/objects/341-2.xj b/objects/341-2.xj similarity index 100% rename from public/objects/341-2.xj rename to objects/341-2.xj diff --git a/public/objects/341-3.xj b/objects/341-3.xj similarity index 100% rename from public/objects/341-3.xj rename to objects/341-3.xj diff --git a/public/objects/341.xj b/objects/341.xj similarity index 100% rename from public/objects/341.xj rename to objects/341.xj diff --git a/public/objects/341.xvm b/objects/341.xvm similarity index 100% rename from public/objects/341.xvm rename to objects/341.xvm diff --git a/public/objects/342.xj b/objects/342.xj similarity index 100% rename from public/objects/342.xj rename to objects/342.xj diff --git a/public/objects/342.xvm b/objects/342.xvm similarity index 100% rename from public/objects/342.xvm rename to objects/342.xvm diff --git a/public/objects/345.xj b/objects/345.xj similarity index 100% rename from public/objects/345.xj rename to objects/345.xj diff --git a/public/objects/345.xvm b/objects/345.xvm similarity index 100% rename from public/objects/345.xvm rename to objects/345.xvm diff --git a/public/objects/346.xj b/objects/346.xj similarity index 100% rename from public/objects/346.xj rename to objects/346.xj diff --git a/public/objects/346.xvm b/objects/346.xvm similarity index 100% rename from public/objects/346.xvm rename to objects/346.xvm diff --git a/public/objects/347.xj b/objects/347.xj similarity index 100% rename from public/objects/347.xj rename to objects/347.xj diff --git a/public/objects/347.xvm b/objects/347.xvm similarity index 100% rename from public/objects/347.xvm rename to objects/347.xvm diff --git a/public/objects/348.xj b/objects/348.xj similarity index 100% rename from public/objects/348.xj rename to objects/348.xj diff --git a/public/objects/348.xvm b/objects/348.xvm similarity index 100% rename from public/objects/348.xvm rename to objects/348.xvm diff --git a/public/objects/349.xj b/objects/349.xj similarity index 100% rename from public/objects/349.xj rename to objects/349.xj diff --git a/public/objects/349.xvm b/objects/349.xvm similarity index 100% rename from public/objects/349.xvm rename to objects/349.xvm diff --git a/public/objects/350.xj b/objects/350.xj similarity index 100% rename from public/objects/350.xj rename to objects/350.xj diff --git a/public/objects/350.xvm b/objects/350.xvm similarity index 100% rename from public/objects/350.xvm rename to objects/350.xvm diff --git a/public/objects/351.xj b/objects/351.xj similarity index 100% rename from public/objects/351.xj rename to objects/351.xj diff --git a/public/objects/351.xvm b/objects/351.xvm similarity index 100% rename from public/objects/351.xvm rename to objects/351.xvm diff --git a/public/objects/353-2.xj b/objects/353-2.xj similarity index 100% rename from public/objects/353-2.xj rename to objects/353-2.xj diff --git a/public/objects/353-3.xj b/objects/353-3.xj similarity index 100% rename from public/objects/353-3.xj rename to objects/353-3.xj diff --git a/public/objects/353.xj b/objects/353.xj similarity index 100% rename from public/objects/353.xj rename to objects/353.xj diff --git a/public/objects/353.xvm b/objects/353.xvm similarity index 100% rename from public/objects/353.xvm rename to objects/353.xvm diff --git a/public/objects/354-2.xj b/objects/354-2.xj similarity index 100% rename from public/objects/354-2.xj rename to objects/354-2.xj diff --git a/public/objects/354-3.xj b/objects/354-3.xj similarity index 100% rename from public/objects/354-3.xj rename to objects/354-3.xj diff --git a/public/objects/354.xj b/objects/354.xj similarity index 100% rename from public/objects/354.xj rename to objects/354.xj diff --git a/public/objects/354.xvm b/objects/354.xvm similarity index 100% rename from public/objects/354.xvm rename to objects/354.xvm diff --git a/public/objects/358.xj b/objects/358.xj similarity index 100% rename from public/objects/358.xj rename to objects/358.xj diff --git a/public/objects/358.xvm b/objects/358.xvm similarity index 100% rename from public/objects/358.xvm rename to objects/358.xvm diff --git a/public/objects/359.xj b/objects/359.xj similarity index 100% rename from public/objects/359.xj rename to objects/359.xj diff --git a/public/objects/359.xvm b/objects/359.xvm similarity index 100% rename from public/objects/359.xvm rename to objects/359.xvm diff --git a/public/objects/368.xj b/objects/368.xj similarity index 100% rename from public/objects/368.xj rename to objects/368.xj diff --git a/public/objects/368.xvm b/objects/368.xvm similarity index 100% rename from public/objects/368.xvm rename to objects/368.xvm diff --git a/public/objects/369.xj b/objects/369.xj similarity index 100% rename from public/objects/369.xj rename to objects/369.xj diff --git a/public/objects/369.xvm b/objects/369.xvm similarity index 100% rename from public/objects/369.xvm rename to objects/369.xvm diff --git a/public/objects/370-0.xj b/objects/370-0.xj similarity index 100% rename from public/objects/370-0.xj rename to objects/370-0.xj diff --git a/public/objects/370-0.xvm b/objects/370-0.xvm similarity index 100% rename from public/objects/370-0.xvm rename to objects/370-0.xvm diff --git a/public/objects/370-1.xj b/objects/370-1.xj similarity index 100% rename from public/objects/370-1.xj rename to objects/370-1.xj diff --git a/public/objects/370-1.xvm b/objects/370-1.xvm similarity index 100% rename from public/objects/370-1.xvm rename to objects/370-1.xvm diff --git a/public/objects/370-2.xj b/objects/370-2.xj similarity index 100% rename from public/objects/370-2.xj rename to objects/370-2.xj diff --git a/public/objects/370-2.xvm b/objects/370-2.xvm similarity index 100% rename from public/objects/370-2.xvm rename to objects/370-2.xvm diff --git a/public/objects/370-3.xj b/objects/370-3.xj similarity index 100% rename from public/objects/370-3.xj rename to objects/370-3.xj diff --git a/public/objects/370-3.xvm b/objects/370-3.xvm similarity index 100% rename from public/objects/370-3.xvm rename to objects/370-3.xvm diff --git a/public/objects/385.xj b/objects/385.xj similarity index 100% rename from public/objects/385.xj rename to objects/385.xj diff --git a/public/objects/385.xvm b/objects/385.xvm similarity index 100% rename from public/objects/385.xvm rename to objects/385.xvm diff --git a/public/objects/386.xj b/objects/386.xj similarity index 100% rename from public/objects/386.xj rename to objects/386.xj diff --git a/public/objects/386.xvm b/objects/386.xvm similarity index 100% rename from public/objects/386.xvm rename to objects/386.xvm diff --git a/public/objects/390.xj b/objects/390.xj similarity index 100% rename from public/objects/390.xj rename to objects/390.xj diff --git a/public/objects/390.xvm b/objects/390.xvm similarity index 100% rename from public/objects/390.xvm rename to objects/390.xvm diff --git a/public/objects/391.xj b/objects/391.xj similarity index 100% rename from public/objects/391.xj rename to objects/391.xj diff --git a/public/objects/391.xvm b/objects/391.xvm similarity index 100% rename from public/objects/391.xvm rename to objects/391.xvm diff --git a/public/objects/395.xj b/objects/395.xj similarity index 100% rename from public/objects/395.xj rename to objects/395.xj diff --git a/public/objects/395.xvm b/objects/395.xvm similarity index 100% rename from public/objects/395.xvm rename to objects/395.xvm diff --git a/public/objects/396.xj b/objects/396.xj similarity index 100% rename from public/objects/396.xj rename to objects/396.xj diff --git a/public/objects/396.xvm b/objects/396.xvm similarity index 100% rename from public/objects/396.xvm rename to objects/396.xvm diff --git a/public/objects/401.xj b/objects/401.xj similarity index 100% rename from public/objects/401.xj rename to objects/401.xj diff --git a/public/objects/401.xvm b/objects/401.xvm similarity index 100% rename from public/objects/401.xvm rename to objects/401.xvm diff --git a/public/objects/402.xj b/objects/402.xj similarity index 100% rename from public/objects/402.xj rename to objects/402.xj diff --git a/public/objects/402.xvm b/objects/402.xvm similarity index 100% rename from public/objects/402.xvm rename to objects/402.xvm diff --git a/public/objects/403.xj b/objects/403.xj similarity index 100% rename from public/objects/403.xj rename to objects/403.xj diff --git a/public/objects/403.xvm b/objects/403.xvm similarity index 100% rename from public/objects/403.xvm rename to objects/403.xvm diff --git a/public/objects/416.xj b/objects/416.xj similarity index 100% rename from public/objects/416.xj rename to objects/416.xj diff --git a/public/objects/416.xvm b/objects/416.xvm similarity index 100% rename from public/objects/416.xvm rename to objects/416.xvm diff --git a/public/objects/417.xj b/objects/417.xj similarity index 100% rename from public/objects/417.xj rename to objects/417.xj diff --git a/public/objects/417.xvm b/objects/417.xvm similarity index 100% rename from public/objects/417.xvm rename to objects/417.xvm diff --git a/public/objects/418.xj b/objects/418.xj similarity index 100% rename from public/objects/418.xj rename to objects/418.xj diff --git a/public/objects/418.xvm b/objects/418.xvm similarity index 100% rename from public/objects/418.xvm rename to objects/418.xvm diff --git a/public/objects/419.xj b/objects/419.xj similarity index 100% rename from public/objects/419.xj rename to objects/419.xj diff --git a/public/objects/419.xvm b/objects/419.xvm similarity index 100% rename from public/objects/419.xvm rename to objects/419.xvm diff --git a/public/objects/420.xj b/objects/420.xj similarity index 100% rename from public/objects/420.xj rename to objects/420.xj diff --git a/public/objects/420.xvm b/objects/420.xvm similarity index 100% rename from public/objects/420.xvm rename to objects/420.xvm diff --git a/public/objects/421.xj b/objects/421.xj similarity index 100% rename from public/objects/421.xj rename to objects/421.xj diff --git a/public/objects/421.xvm b/objects/421.xvm similarity index 100% rename from public/objects/421.xvm rename to objects/421.xvm diff --git a/public/objects/422.xj b/objects/422.xj similarity index 100% rename from public/objects/422.xj rename to objects/422.xj diff --git a/public/objects/422.xvm b/objects/422.xvm similarity index 100% rename from public/objects/422.xvm rename to objects/422.xvm diff --git a/public/objects/423.xj b/objects/423.xj similarity index 100% rename from public/objects/423.xj rename to objects/423.xj diff --git a/public/objects/423.xvm b/objects/423.xvm similarity index 100% rename from public/objects/423.xvm rename to objects/423.xvm diff --git a/public/objects/425.xj b/objects/425.xj similarity index 100% rename from public/objects/425.xj rename to objects/425.xj diff --git a/public/objects/425.xvm b/objects/425.xvm similarity index 100% rename from public/objects/425.xvm rename to objects/425.xvm diff --git a/public/objects/426.xj b/objects/426.xj similarity index 100% rename from public/objects/426.xj rename to objects/426.xj diff --git a/public/objects/426.xvm b/objects/426.xvm similarity index 100% rename from public/objects/426.xvm rename to objects/426.xvm diff --git a/public/objects/427.xj b/objects/427.xj similarity index 100% rename from public/objects/427.xj rename to objects/427.xj diff --git a/public/objects/427.xvm b/objects/427.xvm similarity index 100% rename from public/objects/427.xvm rename to objects/427.xvm diff --git a/public/objects/448.xj b/objects/448.xj similarity index 100% rename from public/objects/448.xj rename to objects/448.xj diff --git a/public/objects/448.xvm b/objects/448.xvm similarity index 100% rename from public/objects/448.xvm rename to objects/448.xvm diff --git a/public/objects/512-2.xj b/objects/512-2.xj similarity index 100% rename from public/objects/512-2.xj rename to objects/512-2.xj diff --git a/public/objects/512-3.xj b/objects/512-3.xj similarity index 100% rename from public/objects/512-3.xj rename to objects/512-3.xj diff --git a/public/objects/512-4.xj b/objects/512-4.xj similarity index 100% rename from public/objects/512-4.xj rename to objects/512-4.xj diff --git a/public/objects/512.xj b/objects/512.xj similarity index 100% rename from public/objects/512.xj rename to objects/512.xj diff --git a/public/objects/512.xvm b/objects/512.xvm similarity index 100% rename from public/objects/512.xvm rename to objects/512.xvm diff --git a/public/objects/513-2.xj b/objects/513-2.xj similarity index 100% rename from public/objects/513-2.xj rename to objects/513-2.xj diff --git a/public/objects/513.xj b/objects/513.xj similarity index 100% rename from public/objects/513.xj rename to objects/513.xj diff --git a/public/objects/513.xvm b/objects/513.xvm similarity index 100% rename from public/objects/513.xvm rename to objects/513.xvm diff --git a/public/objects/514-2.xj b/objects/514-2.xj similarity index 100% rename from public/objects/514-2.xj rename to objects/514-2.xj diff --git a/public/objects/514.xj b/objects/514.xj similarity index 100% rename from public/objects/514.xj rename to objects/514.xj diff --git a/public/objects/514.xvm b/objects/514.xvm similarity index 100% rename from public/objects/514.xvm rename to objects/514.xvm diff --git a/public/objects/515-2.xj b/objects/515-2.xj similarity index 100% rename from public/objects/515-2.xj rename to objects/515-2.xj diff --git a/public/objects/515-3.xj b/objects/515-3.xj similarity index 100% rename from public/objects/515-3.xj rename to objects/515-3.xj diff --git a/public/objects/515-4.xj b/objects/515-4.xj similarity index 100% rename from public/objects/515-4.xj rename to objects/515-4.xj diff --git a/public/objects/515.xj b/objects/515.xj similarity index 100% rename from public/objects/515.xj rename to objects/515.xj diff --git a/public/objects/515.xvm b/objects/515.xvm similarity index 100% rename from public/objects/515.xvm rename to objects/515.xvm diff --git a/public/objects/516-2.xj b/objects/516-2.xj similarity index 100% rename from public/objects/516-2.xj rename to objects/516-2.xj diff --git a/public/objects/516-3.xj b/objects/516-3.xj similarity index 100% rename from public/objects/516-3.xj rename to objects/516-3.xj diff --git a/public/objects/516-4.xj b/objects/516-4.xj similarity index 100% rename from public/objects/516-4.xj rename to objects/516-4.xj diff --git a/public/objects/516.xj b/objects/516.xj similarity index 100% rename from public/objects/516.xj rename to objects/516.xj diff --git a/public/objects/516.xvm b/objects/516.xvm similarity index 100% rename from public/objects/516.xvm rename to objects/516.xvm diff --git a/public/objects/517-2.xj b/objects/517-2.xj similarity index 100% rename from public/objects/517-2.xj rename to objects/517-2.xj diff --git a/public/objects/517.xj b/objects/517.xj similarity index 100% rename from public/objects/517.xj rename to objects/517.xj diff --git a/public/objects/517.xvm b/objects/517.xvm similarity index 100% rename from public/objects/517.xvm rename to objects/517.xvm diff --git a/public/objects/518.xj b/objects/518.xj similarity index 100% rename from public/objects/518.xj rename to objects/518.xj diff --git a/public/objects/518.xvm b/objects/518.xvm similarity index 100% rename from public/objects/518.xvm rename to objects/518.xvm diff --git a/public/objects/519.xj b/objects/519.xj similarity index 100% rename from public/objects/519.xj rename to objects/519.xj diff --git a/public/objects/519.xvm b/objects/519.xvm similarity index 100% rename from public/objects/519.xvm rename to objects/519.xvm diff --git a/public/objects/520.xj b/objects/520.xj similarity index 100% rename from public/objects/520.xj rename to objects/520.xj diff --git a/public/objects/520.xvm b/objects/520.xvm similarity index 100% rename from public/objects/520.xvm rename to objects/520.xvm diff --git a/public/objects/521.xj b/objects/521.xj similarity index 100% rename from public/objects/521.xj rename to objects/521.xj diff --git a/public/objects/521.xvm b/objects/521.xvm similarity index 100% rename from public/objects/521.xvm rename to objects/521.xvm diff --git a/public/objects/527-0.xj b/objects/527-0.xj similarity index 100% rename from public/objects/527-0.xj rename to objects/527-0.xj diff --git a/public/objects/527-0.xvm b/objects/527-0.xvm similarity index 100% rename from public/objects/527-0.xvm rename to objects/527-0.xvm diff --git a/public/objects/527-1.xj b/objects/527-1.xj similarity index 100% rename from public/objects/527-1.xj rename to objects/527-1.xj diff --git a/public/objects/527-1.xvm b/objects/527-1.xvm similarity index 100% rename from public/objects/527-1.xvm rename to objects/527-1.xvm diff --git a/public/objects/527.xj b/objects/527.xj similarity index 100% rename from public/objects/527.xj rename to objects/527.xj diff --git a/public/objects/527.xvm b/objects/527.xvm similarity index 100% rename from public/objects/527.xvm rename to objects/527.xvm diff --git a/public/objects/528-0.xj b/objects/528-0.xj similarity index 100% rename from public/objects/528-0.xj rename to objects/528-0.xj diff --git a/public/objects/528-0.xvm b/objects/528-0.xvm similarity index 100% rename from public/objects/528-0.xvm rename to objects/528-0.xvm diff --git a/public/objects/528-1.xj b/objects/528-1.xj similarity index 100% rename from public/objects/528-1.xj rename to objects/528-1.xj diff --git a/public/objects/528-1.xvm b/objects/528-1.xvm similarity index 100% rename from public/objects/528-1.xvm rename to objects/528-1.xvm diff --git a/public/objects/528-2.xj b/objects/528-2.xj similarity index 100% rename from public/objects/528-2.xj rename to objects/528-2.xj diff --git a/public/objects/528.xj b/objects/528.xj similarity index 100% rename from public/objects/528.xj rename to objects/528.xj diff --git a/public/objects/528.xvm b/objects/528.xvm similarity index 100% rename from public/objects/528.xvm rename to objects/528.xvm diff --git a/public/objects/544.xj b/objects/544.xj similarity index 100% rename from public/objects/544.xj rename to objects/544.xj diff --git a/public/objects/544.xvm b/objects/544.xvm similarity index 100% rename from public/objects/544.xvm rename to objects/544.xvm diff --git a/public/objects/545.xj b/objects/545.xj similarity index 100% rename from public/objects/545.xj rename to objects/545.xj diff --git a/public/objects/545.xvm b/objects/545.xvm similarity index 100% rename from public/objects/545.xvm rename to objects/545.xvm diff --git a/public/objects/546.xj b/objects/546.xj similarity index 100% rename from public/objects/546.xj rename to objects/546.xj diff --git a/public/objects/546.xvm b/objects/546.xvm similarity index 100% rename from public/objects/546.xvm rename to objects/546.xvm diff --git a/public/objects/547-0.xj b/objects/547-0.xj similarity index 100% rename from public/objects/547-0.xj rename to objects/547-0.xj diff --git a/public/objects/547-0.xvm b/objects/547-0.xvm similarity index 100% rename from public/objects/547-0.xvm rename to objects/547-0.xvm diff --git a/public/objects/547-1.xj b/objects/547-1.xj similarity index 100% rename from public/objects/547-1.xj rename to objects/547-1.xj diff --git a/public/objects/547-1.xvm b/objects/547-1.xvm similarity index 100% rename from public/objects/547-1.xvm rename to objects/547-1.xvm diff --git a/public/objects/547.xj b/objects/547.xj similarity index 100% rename from public/objects/547.xj rename to objects/547.xj diff --git a/public/objects/547.xvm b/objects/547.xvm similarity index 100% rename from public/objects/547.xvm rename to objects/547.xvm diff --git a/public/objects/548.xj b/objects/548.xj similarity index 100% rename from public/objects/548.xj rename to objects/548.xj diff --git a/public/objects/548.xvm b/objects/548.xvm similarity index 100% rename from public/objects/548.xvm rename to objects/548.xvm diff --git a/public/objects/549.xj b/objects/549.xj similarity index 100% rename from public/objects/549.xj rename to objects/549.xj diff --git a/public/objects/549.xvm b/objects/549.xvm similarity index 100% rename from public/objects/549.xvm rename to objects/549.xvm diff --git a/public/objects/550.xj b/objects/550.xj similarity index 100% rename from public/objects/550.xj rename to objects/550.xj diff --git a/public/objects/550.xvm b/objects/550.xvm similarity index 100% rename from public/objects/550.xvm rename to objects/550.xvm diff --git a/public/objects/551.nj b/objects/551.nj similarity index 100% rename from public/objects/551.nj rename to objects/551.nj diff --git a/public/objects/551.xvm b/objects/551.xvm similarity index 100% rename from public/objects/551.xvm rename to objects/551.xvm diff --git a/public/objects/552-2.xj b/objects/552-2.xj similarity index 100% rename from public/objects/552-2.xj rename to objects/552-2.xj diff --git a/public/objects/552-3.xj b/objects/552-3.xj similarity index 100% rename from public/objects/552-3.xj rename to objects/552-3.xj diff --git a/public/objects/552.xj b/objects/552.xj similarity index 100% rename from public/objects/552.xj rename to objects/552.xj diff --git a/public/objects/552.xvm b/objects/552.xvm similarity index 100% rename from public/objects/552.xvm rename to objects/552.xvm diff --git a/public/objects/553.xj b/objects/553.xj similarity index 100% rename from public/objects/553.xj rename to objects/553.xj diff --git a/public/objects/553.xvm b/objects/553.xvm similarity index 100% rename from public/objects/553.xvm rename to objects/553.xvm diff --git a/public/objects/67-2.xj b/objects/67-2.xj similarity index 100% rename from public/objects/67-2.xj rename to objects/67-2.xj diff --git a/public/objects/67.xj b/objects/67.xj similarity index 100% rename from public/objects/67.xj rename to objects/67.xj diff --git a/public/objects/67.xvm b/objects/67.xvm similarity index 100% rename from public/objects/67.xvm rename to objects/67.xvm diff --git a/public/objects/68.xj b/objects/68.xj similarity index 100% rename from public/objects/68.xj rename to objects/68.xj diff --git a/public/objects/68.xvm b/objects/68.xvm similarity index 100% rename from public/objects/68.xvm rename to objects/68.xvm diff --git a/public/objects/688.xj b/objects/688.xj similarity index 100% rename from public/objects/688.xj rename to objects/688.xj diff --git a/public/objects/688.xvm b/objects/688.xvm similarity index 100% rename from public/objects/688.xvm rename to objects/688.xvm diff --git a/public/objects/689.xj b/objects/689.xj similarity index 100% rename from public/objects/689.xj rename to objects/689.xj diff --git a/public/objects/689.xvm b/objects/689.xvm similarity index 100% rename from public/objects/689.xvm rename to objects/689.xvm diff --git a/public/objects/69-0.xj b/objects/69-0.xj similarity index 100% rename from public/objects/69-0.xj rename to objects/69-0.xj diff --git a/public/objects/69-0.xvm b/objects/69-0.xvm similarity index 100% rename from public/objects/69-0.xvm rename to objects/69-0.xvm diff --git a/public/objects/69-1.xj b/objects/69-1.xj similarity index 100% rename from public/objects/69-1.xj rename to objects/69-1.xj diff --git a/public/objects/69-1.xvm b/objects/69-1.xvm similarity index 100% rename from public/objects/69-1.xvm rename to objects/69-1.xvm diff --git a/public/objects/69-2.xj b/objects/69-2.xj similarity index 100% rename from public/objects/69-2.xj rename to objects/69-2.xj diff --git a/public/objects/69-a.xj b/objects/69-a.xj similarity index 100% rename from public/objects/69-a.xj rename to objects/69-a.xj diff --git a/public/objects/69.xj b/objects/69.xj similarity index 100% rename from public/objects/69.xj rename to objects/69.xj diff --git a/public/objects/69.xvm b/objects/69.xvm similarity index 100% rename from public/objects/69.xvm rename to objects/69.xvm diff --git a/public/objects/690.xj b/objects/690.xj similarity index 100% rename from public/objects/690.xj rename to objects/690.xj diff --git a/public/objects/690.xvm b/objects/690.xvm similarity index 100% rename from public/objects/690.xvm rename to objects/690.xvm diff --git a/public/objects/691.xj b/objects/691.xj similarity index 100% rename from public/objects/691.xj rename to objects/691.xj diff --git a/public/objects/691.xvm b/objects/691.xvm similarity index 100% rename from public/objects/691.xvm rename to objects/691.xvm diff --git a/public/objects/692.xj b/objects/692.xj similarity index 100% rename from public/objects/692.xj rename to objects/692.xj diff --git a/public/objects/692.xvm b/objects/692.xvm similarity index 100% rename from public/objects/692.xvm rename to objects/692.xvm diff --git a/public/objects/693.xj b/objects/693.xj similarity index 100% rename from public/objects/693.xj rename to objects/693.xj diff --git a/public/objects/693.xvm b/objects/693.xvm similarity index 100% rename from public/objects/693.xvm rename to objects/693.xvm diff --git a/public/objects/694.xj b/objects/694.xj similarity index 100% rename from public/objects/694.xj rename to objects/694.xj diff --git a/public/objects/694.xvm b/objects/694.xvm similarity index 100% rename from public/objects/694.xvm rename to objects/694.xvm diff --git a/public/objects/696.xj b/objects/696.xj similarity index 100% rename from public/objects/696.xj rename to objects/696.xj diff --git a/public/objects/696.xvm b/objects/696.xvm similarity index 100% rename from public/objects/696.xvm rename to objects/696.xvm diff --git a/public/objects/699.xj b/objects/699.xj similarity index 100% rename from public/objects/699.xj rename to objects/699.xj diff --git a/public/objects/699.xvm b/objects/699.xvm similarity index 100% rename from public/objects/699.xvm rename to objects/699.xvm diff --git a/public/objects/70.xj b/objects/70.xj similarity index 100% rename from public/objects/70.xj rename to objects/70.xj diff --git a/public/objects/70.xvm b/objects/70.xvm similarity index 100% rename from public/objects/70.xvm rename to objects/70.xvm diff --git a/public/objects/701-2.xj b/objects/701-2.xj similarity index 100% rename from public/objects/701-2.xj rename to objects/701-2.xj diff --git a/public/objects/701.xj b/objects/701.xj similarity index 100% rename from public/objects/701.xj rename to objects/701.xj diff --git a/public/objects/701.xvm b/objects/701.xvm similarity index 100% rename from public/objects/701.xvm rename to objects/701.xvm diff --git a/public/objects/71.xj b/objects/71.xj similarity index 100% rename from public/objects/71.xj rename to objects/71.xj diff --git a/public/objects/71.xvm b/objects/71.xvm similarity index 100% rename from public/objects/71.xvm rename to objects/71.xvm diff --git a/public/objects/72-2.xj b/objects/72-2.xj similarity index 100% rename from public/objects/72-2.xj rename to objects/72-2.xj diff --git a/public/objects/72.xj b/objects/72.xj similarity index 100% rename from public/objects/72.xj rename to objects/72.xj diff --git a/public/objects/72.xvm b/objects/72.xvm similarity index 100% rename from public/objects/72.xvm rename to objects/72.xvm diff --git a/public/objects/73.xj b/objects/73.xj similarity index 100% rename from public/objects/73.xj rename to objects/73.xj diff --git a/public/objects/73.xvm b/objects/73.xvm similarity index 100% rename from public/objects/73.xvm rename to objects/73.xvm diff --git a/public/objects/74.xj b/objects/74.xj similarity index 100% rename from public/objects/74.xj rename to objects/74.xj diff --git a/public/objects/74.xvm b/objects/74.xvm similarity index 100% rename from public/objects/74.xvm rename to objects/74.xvm diff --git a/public/objects/75-2.nj b/objects/75-2.nj similarity index 100% rename from public/objects/75-2.nj rename to objects/75-2.nj diff --git a/public/objects/75.nj b/objects/75.nj similarity index 100% rename from public/objects/75.nj rename to objects/75.nj diff --git a/public/objects/75.xvm b/objects/75.xvm similarity index 100% rename from public/objects/75.xvm rename to objects/75.xvm diff --git a/public/objects/76-2.xj b/objects/76-2.xj similarity index 100% rename from public/objects/76-2.xj rename to objects/76-2.xj diff --git a/public/objects/76-3.xj b/objects/76-3.xj similarity index 100% rename from public/objects/76-3.xj rename to objects/76-3.xj diff --git a/public/objects/76.xj b/objects/76.xj similarity index 100% rename from public/objects/76.xj rename to objects/76.xj diff --git a/public/objects/76.xvm b/objects/76.xvm similarity index 100% rename from public/objects/76.xvm rename to objects/76.xvm diff --git a/public/objects/769-0.nj b/objects/769-0.nj similarity index 100% rename from public/objects/769-0.nj rename to objects/769-0.nj diff --git a/public/objects/769-0.xvm b/objects/769-0.xvm similarity index 100% rename from public/objects/769-0.xvm rename to objects/769-0.xvm diff --git a/public/objects/769-1.nj b/objects/769-1.nj similarity index 100% rename from public/objects/769-1.nj rename to objects/769-1.nj diff --git a/public/objects/769-1.xvm b/objects/769-1.xvm similarity index 100% rename from public/objects/769-1.xvm rename to objects/769-1.xvm diff --git a/public/objects/769-2.nj b/objects/769-2.nj similarity index 100% rename from public/objects/769-2.nj rename to objects/769-2.nj diff --git a/public/objects/769-2.xvm b/objects/769-2.xvm similarity index 100% rename from public/objects/769-2.xvm rename to objects/769-2.xvm diff --git a/public/objects/769.nj b/objects/769.nj similarity index 100% rename from public/objects/769.nj rename to objects/769.nj diff --git a/public/objects/769.xvm b/objects/769.xvm similarity index 100% rename from public/objects/769.xvm rename to objects/769.xvm diff --git a/public/objects/77-2.nj b/objects/77-2.nj similarity index 100% rename from public/objects/77-2.nj rename to objects/77-2.nj diff --git a/public/objects/77-3.nj b/objects/77-3.nj similarity index 100% rename from public/objects/77-3.nj rename to objects/77-3.nj diff --git a/public/objects/77-4.nj b/objects/77-4.nj similarity index 100% rename from public/objects/77-4.nj rename to objects/77-4.nj diff --git a/public/objects/77.nj b/objects/77.nj similarity index 100% rename from public/objects/77.nj rename to objects/77.nj diff --git a/public/objects/77.xvm b/objects/77.xvm similarity index 100% rename from public/objects/77.xvm rename to objects/77.xvm diff --git a/public/objects/770-0.nj b/objects/770-0.nj similarity index 100% rename from public/objects/770-0.nj rename to objects/770-0.nj diff --git a/public/objects/770-0.xvm b/objects/770-0.xvm similarity index 100% rename from public/objects/770-0.xvm rename to objects/770-0.xvm diff --git a/public/objects/770-1.nj b/objects/770-1.nj similarity index 100% rename from public/objects/770-1.nj rename to objects/770-1.nj diff --git a/public/objects/770-1.xvm b/objects/770-1.xvm similarity index 100% rename from public/objects/770-1.xvm rename to objects/770-1.xvm diff --git a/public/objects/770-2.nj b/objects/770-2.nj similarity index 100% rename from public/objects/770-2.nj rename to objects/770-2.nj diff --git a/public/objects/770-2.xvm b/objects/770-2.xvm similarity index 100% rename from public/objects/770-2.xvm rename to objects/770-2.xvm diff --git a/public/objects/78.nj b/objects/78.nj similarity index 100% rename from public/objects/78.nj rename to objects/78.nj diff --git a/public/objects/78.xvm b/objects/78.xvm similarity index 100% rename from public/objects/78.xvm rename to objects/78.xvm diff --git a/public/objects/79.xj b/objects/79.xj similarity index 100% rename from public/objects/79.xj rename to objects/79.xj diff --git a/public/objects/79.xvm b/objects/79.xvm similarity index 100% rename from public/objects/79.xvm rename to objects/79.xvm diff --git a/public/objects/80-2.nj b/objects/80-2.nj similarity index 100% rename from public/objects/80-2.nj rename to objects/80-2.nj diff --git a/public/objects/80.nj b/objects/80.nj similarity index 100% rename from public/objects/80.nj rename to objects/80.nj diff --git a/public/objects/80.xvm b/objects/80.xvm similarity index 100% rename from public/objects/80.xvm rename to objects/80.xvm diff --git a/public/objects/81-0.nj b/objects/81-0.nj similarity index 100% rename from public/objects/81-0.nj rename to objects/81-0.nj diff --git a/public/objects/81-0.xvm b/objects/81-0.xvm similarity index 100% rename from public/objects/81-0.xvm rename to objects/81-0.xvm diff --git a/public/objects/81-1.nj b/objects/81-1.nj similarity index 100% rename from public/objects/81-1.nj rename to objects/81-1.nj diff --git a/public/objects/81-1.xvm b/objects/81-1.xvm similarity index 100% rename from public/objects/81-1.xvm rename to objects/81-1.xvm diff --git a/public/objects/81-2.nj b/objects/81-2.nj similarity index 100% rename from public/objects/81-2.nj rename to objects/81-2.nj diff --git a/public/objects/81-2.xvm b/objects/81-2.xvm similarity index 100% rename from public/objects/81-2.xvm rename to objects/81-2.xvm diff --git a/public/objects/81-3.nj b/objects/81-3.nj similarity index 100% rename from public/objects/81-3.nj rename to objects/81-3.nj diff --git a/public/objects/81-3.xvm b/objects/81-3.xvm similarity index 100% rename from public/objects/81-3.xvm rename to objects/81-3.xvm diff --git a/public/objects/81.nj b/objects/81.nj similarity index 100% rename from public/objects/81.nj rename to objects/81.nj diff --git a/public/objects/81.xvm b/objects/81.xvm similarity index 100% rename from public/objects/81.xvm rename to objects/81.xvm diff --git a/public/objects/82-2.nj b/objects/82-2.nj similarity index 100% rename from public/objects/82-2.nj rename to objects/82-2.nj diff --git a/public/objects/82.nj b/objects/82.nj similarity index 100% rename from public/objects/82.nj rename to objects/82.nj diff --git a/public/objects/82.xvm b/objects/82.xvm similarity index 100% rename from public/objects/82.xvm rename to objects/82.xvm diff --git a/public/objects/84.xj b/objects/84.xj similarity index 100% rename from public/objects/84.xj rename to objects/84.xj diff --git a/public/objects/84.xvm b/objects/84.xvm similarity index 100% rename from public/objects/84.xvm rename to objects/84.xvm diff --git a/public/objects/86.xj b/objects/86.xj similarity index 100% rename from public/objects/86.xj rename to objects/86.xj diff --git a/public/objects/86.xvm b/objects/86.xvm similarity index 100% rename from public/objects/86.xvm rename to objects/86.xvm diff --git a/public/objects/896.nj b/objects/896.nj similarity index 100% rename from public/objects/896.nj rename to objects/896.nj diff --git a/public/objects/896.xvm b/objects/896.xvm similarity index 100% rename from public/objects/896.xvm rename to objects/896.xvm diff --git a/public/objects/902-0.nj b/objects/902-0.nj similarity index 100% rename from public/objects/902-0.nj rename to objects/902-0.nj diff --git a/public/objects/902-0.xvm b/objects/902-0.xvm similarity index 100% rename from public/objects/902-0.xvm rename to objects/902-0.xvm diff --git a/public/objects/902-1.nj b/objects/902-1.nj similarity index 100% rename from public/objects/902-1.nj rename to objects/902-1.nj diff --git a/public/objects/902-1.xvm b/objects/902-1.xvm similarity index 100% rename from public/objects/902-1.xvm rename to objects/902-1.xvm diff --git a/public/objects/902-2.nj b/objects/902-2.nj similarity index 100% rename from public/objects/902-2.nj rename to objects/902-2.nj diff --git a/public/objects/902-2.xvm b/objects/902-2.xvm similarity index 100% rename from public/objects/902-2.xvm rename to objects/902-2.xvm diff --git a/public/objects/902.nj b/objects/902.nj similarity index 100% rename from public/objects/902.nj rename to objects/902.nj diff --git a/public/objects/902.xvm b/objects/902.xvm similarity index 100% rename from public/objects/902.xvm rename to objects/902.xvm diff --git a/public/objects/907.nj b/objects/907.nj similarity index 100% rename from public/objects/907.nj rename to objects/907.nj diff --git a/public/objects/907.xvm b/objects/907.xvm similarity index 100% rename from public/objects/907.xvm rename to objects/907.xvm diff --git a/public/objects/909.nj b/objects/909.nj similarity index 100% rename from public/objects/909.nj rename to objects/909.nj diff --git a/public/objects/909.xvm b/objects/909.xvm similarity index 100% rename from public/objects/909.xvm rename to objects/909.xvm diff --git a/public/objects/911-0.nj b/objects/911-0.nj similarity index 100% rename from public/objects/911-0.nj rename to objects/911-0.nj diff --git a/public/objects/911-0.xvm b/objects/911-0.xvm similarity index 100% rename from public/objects/911-0.xvm rename to objects/911-0.xvm diff --git a/public/objects/911-1.nj b/objects/911-1.nj similarity index 100% rename from public/objects/911-1.nj rename to objects/911-1.nj diff --git a/public/objects/911-1.xvm b/objects/911-1.xvm similarity index 100% rename from public/objects/911-1.xvm rename to objects/911-1.xvm diff --git a/public/objects/911.nj b/objects/911.nj similarity index 100% rename from public/objects/911.nj rename to objects/911.nj diff --git a/public/objects/911.xvm b/objects/911.xvm similarity index 100% rename from public/objects/911.xvm rename to objects/911.xvm diff --git a/package.json b/package.json deleted file mode 100644 index 8ca63562..00000000 --- a/package.json +++ /dev/null @@ -1,56 +0,0 @@ -{ - "name": "phantasmal-world", - "version": "0.1.0", - "private": true, - "dependencies": { - "@craco/craco": "^5.2.1", - "@types/jest": "24.0.13", - "@types/lodash": "^4.14.132", - "@types/react": "16.8.18", - "@types/react-dom": "16.8.4", - "@types/react-virtualized": "^9.21.2", - "@types/text-encoding": "^0.0.35", - "antd": "^3.19.1", - "craco-antd": "^1.11.0", - "javascript-lp-solver": "^0.4.5", - "lodash": "^4.17.11", - "mobx": "^5.9.4", - "mobx-react": "^5.4.4", - "react": "^16.8.6", - "react-dom": "^16.8.6", - "react-scripts": "3.0.1", - "react-virtualized": "^9.21.1", - "text-encoding": "^0.7.0", - "three": "^0.104.0", - "three-orbit-controls": "^82.1.0", - "typescript": "3.4.5" - }, - "scripts": { - "start": "craco start", - "build": "craco build", - "test": "craco test", - "updateDropsEphinea": "ts-node --project=tsconfig-scripts.json static/updateDropsEphinea.ts" - }, - "eslintConfig": { - "extends": "react-app" - }, - "browserslist": { - "production": [ - ">0.2%", - "not dead", - "not op_mini all" - ], - "development": [ - "last 1 chrome version", - "last 1 firefox version", - "last 1 safari version" - ] - }, - "devDependencies": { - "@types/cheerio": "^0.22.0", - "@types/node": "^12.0.3", - "cheerio": "^0.22.0", - "isomorphic-fetch": "^2.2.1", - "ts-node": "^8.2.0" - } -} diff --git a/precache-manifest.be7f938c356ac9dff1a6bfa51f6f976d.js b/precache-manifest.be7f938c356ac9dff1a6bfa51f6f976d.js new file mode 100644 index 00000000..53ef7632 --- /dev/null +++ b/precache-manifest.be7f938c356ac9dff1a6bfa51f6f976d.js @@ -0,0 +1,26 @@ +self.__precacheManifest = (self.__precacheManifest || []).concat([ + { + "revision": "86d7d60af24d2379fe9695c5dfd9fd0b", + "url": "/index.html" + }, + { + "revision": "c5f8c7d1327154ee9df9", + "url": "/static/css/2.5fcf4e3c.chunk.css" + }, + { + "revision": "348ecb98a00d389d9700", + "url": "/static/css/main.fb4f588b.chunk.css" + }, + { + "revision": "c5f8c7d1327154ee9df9", + "url": "/static/js/2.e8014737.chunk.js" + }, + { + "revision": "348ecb98a00d389d9700", + "url": "/static/js/main.08895bb3.chunk.js" + }, + { + "revision": "42ac5946195a7306e2a5", + "url": "/static/js/runtime~main.a8a9905a.js" + } +]); \ No newline at end of file diff --git a/public/index.html b/public/index.html deleted file mode 100644 index 727393c7..00000000 --- a/public/index.html +++ /dev/null @@ -1,12 +0,0 @@ - - - - - - - - -
- - - \ No newline at end of file diff --git a/public/quests.ephinea.tsv b/quests.ephinea.tsv similarity index 100% rename from public/quests.ephinea.tsv rename to quests.ephinea.tsv diff --git a/service-worker.js b/service-worker.js new file mode 100644 index 00000000..6d2a8571 --- /dev/null +++ b/service-worker.js @@ -0,0 +1,39 @@ +/** + * Welcome to your Workbox-powered service worker! + * + * You'll need to register this file in your web app and you should + * disable HTTP caching for this file too. + * See https://goo.gl/nhQhGp + * + * The rest of the code is auto-generated. Please don't update this file + * directly; instead, make changes to your Workbox build configuration + * and re-run your build process. + * See https://goo.gl/2aRDsh + */ + +importScripts("https://storage.googleapis.com/workbox-cdn/releases/4.3.1/workbox-sw.js"); + +importScripts( + "/precache-manifest.be7f938c356ac9dff1a6bfa51f6f976d.js" +); + +self.addEventListener('message', (event) => { + if (event.data && event.data.type === 'SKIP_WAITING') { + self.skipWaiting(); + } +}); + +workbox.core.clientsClaim(); + +/** + * The workboxSW.precacheAndRoute() method efficiently caches and responds to + * requests for URLs in the manifest. + * See https://goo.gl/S9QRab + */ +self.__precacheManifest = [].concat(self.__precacheManifest || []); +workbox.precaching.precacheAndRoute(self.__precacheManifest, {}); + +workbox.routing.registerNavigationRoute(workbox.precaching.getCacheKeyForURL("/index.html"), { + + blacklist: [/^\/_/,/\/[^\/]+\.[^\/]+$/], +}); diff --git a/src/Loadable.ts b/src/Loadable.ts deleted file mode 100644 index e15b75a3..00000000 --- a/src/Loadable.ts +++ /dev/null @@ -1,133 +0,0 @@ -import { observable, computed } from "mobx"; -import { defer } from "lodash"; - -export enum LoadableState { - /** - * No attempt has been made to load data. - */ - Uninitialized, - - /** - * The first data load is underway. - */ - Initializing, - - /** - * Data was loaded at least once. The most recent load was successful. - */ - Nominal, - - /** - * Data was loaded at least once. The most recent load failed. - */ - Error, - - /** - * Data was loaded at least once. Another data load is underway. - */ - Reloading, -} - -/** - * Represents a value that can be loaded asynchronously. - * [state]{@link Loadable#state} represents the current state of this Loadable's value. - */ -export class Loadable { - @observable private _value: T; - @observable private _promise: Promise = new Promise(resolve => resolve(this._value)); - @observable private _state = LoadableState.Uninitialized; - private _load?: () => Promise; - @observable private _error?: Error; - - constructor(initialValue: T, load?: () => Promise) { - this._value = initialValue; - this._load = load; - } - - /** - * When this Loadable is uninitialized, a load will be triggered. - * Will return the initial value until a load has succeeded. - */ - @computed get value(): T { - // Load value on first use and return initial placeholder value. - if (this._state === LoadableState.Uninitialized) { - // Defer loading value to avoid side effects in computed value. - defer(() => this.loadValue()); - } - - return this._value; - } - - /** - * This property returns valid data as soon as possible. - * If the Loadable is uninitialized a data load will be triggered, otherwise the current value will be returned. - */ - get promise(): Promise { - // Load value on first use. - if (this._state === LoadableState.Uninitialized) { - return this.loadValue(); - } else { - return this._promise; - } - } - - @computed get state(): LoadableState { - return this._state; - } - - /** - * @returns true if the initial data load has happened. It may or may not have succeeded. - * Check [error]{@link Loadable#error} to know whether an error occurred. - */ - @computed get isInitialized(): boolean { - return this._state !== LoadableState.Uninitialized; - } - - /** - * @returns true if a data load is underway. This may be the initializing load or a later load. - */ - @computed get isLoading(): boolean { - switch (this._state) { - case LoadableState.Initializing: - case LoadableState.Reloading: - return true; - default: - return false; - } - } - - /** - * @returns an {@link Error} if an error occurred during the most recent data load. - */ - @computed get error(): Error | undefined { - return this._error; - } - - /** - * Load the data. Initializes the Loadable if it is uninitialized. - */ - load(): Promise { - return this.loadValue(); - } - - private async loadValue(): Promise { - if (this.isLoading) return this._promise; - - this._state = LoadableState.Initializing; - - try { - if (this._load) { - this._promise = this._load(); - this._value = await this._promise; - } - - this._state = LoadableState.Nominal; - this._error = undefined; - return this._value; - } catch (e) { - this._state = LoadableState.Error; - this._error = e; - throw e; - } - } -} diff --git a/src/bin-data/ArrayBufferCursor.test.ts b/src/bin-data/ArrayBufferCursor.test.ts deleted file mode 100644 index b9ca15d3..00000000 --- a/src/bin-data/ArrayBufferCursor.test.ts +++ /dev/null @@ -1,231 +0,0 @@ -import { ArrayBufferCursor } from './ArrayBufferCursor'; - -test('simple properties and invariants', () => { - const cursor = new ArrayBufferCursor(10, true); - - expect(cursor.size).toBe(cursor.position + cursor.bytesLeft); - expect(cursor.size).toBeLessThanOrEqual(cursor.capacity); - expect(cursor.size).toBe(0); - expect(cursor.capacity).toBe(10); - expect(cursor.position).toBe(0); - expect(cursor.bytesLeft).toBe(0); - expect(cursor.littleEndian).toBe(true); - - cursor.writeU8(99).writeU8(99).writeU8(99).writeU8(99); - cursor.seek(-1); - - expect(cursor.size).toBe(cursor.position + cursor.bytesLeft); - expect(cursor.size).toBeLessThanOrEqual(cursor.capacity); - expect(cursor.size).toBe(4); - expect(cursor.capacity).toBe(10); - expect(cursor.position).toBe(3); - expect(cursor.bytesLeft).toBe(1); - expect(cursor.littleEndian).toBe(true); -}); - -test('correct byte order handling', () => { - const buffer = new Uint8Array([1, 2, 3, 4]).buffer; - - expect(new ArrayBufferCursor(buffer, false).u32()).toBe(0x01020304); - expect(new ArrayBufferCursor(buffer, true).u32()).toBe(0x04030201); -}); - -test('reallocation of internal buffer when necessary', () => { - const cursor = new ArrayBufferCursor(3, true); - cursor.writeU8(99).writeU8(99).writeU8(99).writeU8(99); - - expect(cursor.size).toBe(4); - expect(cursor.capacity).toBeGreaterThanOrEqual(4); - expect(cursor.buffer.byteLength).toBeGreaterThanOrEqual(4); -}); - -function testIntegerRead(methodName: string) { - test(methodName, () => { - const bytes = parseInt(methodName.replace(/^[iu](\d+)$/, '$1'), 10) / 8; - let testNumber1 = 0; - let testNumber2 = 0; - // The "false" arrays are for big endian tests and the "true" arrays for little endian tests. - const testArrays: { [index: string]: number[] } = { false: [], true: [] }; - - for (let i = 1; i <= bytes; ++i) { - // Generates numbers of the form 0x010203... - testNumber1 <<= 8; - testNumber1 |= i; - testArrays['false'].push(i); - testArrays['true'].unshift(i); - } - - for (let i = bytes + 1; i <= 2 * bytes; ++i) { - testNumber2 <<= 8; - testNumber2 |= i; - testArrays['false'].push(i); - testArrays['true'].splice(bytes, 0, i); - } - - for (const littleEndian of [false, true]) { - const cursor = new ArrayBufferCursor( - new Uint8Array(testArrays[String(littleEndian)]).buffer, littleEndian); - - expect((cursor as any)[methodName]()).toBe(testNumber1); - expect(cursor.position).toBe(bytes); - - expect((cursor as any)[methodName]()).toBe(testNumber2); - expect(cursor.position).toBe(2 * bytes); - } - }); -} - -testIntegerRead('u8'); -testIntegerRead('u16'); -testIntegerRead('u32'); -testIntegerRead('i32'); - -test('u8Array', () => { - const cursor = new ArrayBufferCursor(new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8]).buffer, true); - - expect(cursor.u8Array(3)).toEqual([1, 2, 3]); - expect(cursor.seekStart(2).u8Array(4)).toEqual([3, 4, 5, 6]); - expect(cursor.seekStart(5).u8Array(3)).toEqual([6, 7, 8]); -}); - -function testStringRead(methodName: string, charSize: number) { - test(methodName, () => { - const charArray = [7, 65, 66, 0, 255, 13]; - - for (const littleEndian of [false, true]) { - const charArrayCopy = []; - - for (const char of charArray) { - if (littleEndian) charArrayCopy.push(char); - - for (let i = 0; i < charSize - 1; ++i) { - charArrayCopy.push(0); - } - - if (!littleEndian) charArrayCopy.push(char); - } - - const cursor = new ArrayBufferCursor( - new Uint8Array(charArrayCopy).buffer, littleEndian); - - cursor.seekStart(charSize); - expect((cursor as any)[methodName](4 * charSize, true, true)).toBe('AB'); - expect(cursor.position).toBe(5 * charSize); - cursor.seekStart(charSize); - expect((cursor as any)[methodName](2 * charSize, true, true)).toBe('AB'); - expect(cursor.position).toBe(3 * charSize); - - cursor.seekStart(charSize); - expect((cursor as any)[methodName](4 * charSize, true, false)).toBe('AB'); - expect(cursor.position).toBe(4 * charSize); - cursor.seekStart(charSize); - expect((cursor as any)[methodName](2 * charSize, true, false)).toBe('AB'); - expect(cursor.position).toBe(3 * charSize); - - cursor.seekStart(charSize); - expect((cursor as any)[methodName](4 * charSize, false, true)).toBe('AB\0ÿ'); - expect(cursor.position).toBe(5 * charSize); - - cursor.seekStart(charSize); - expect((cursor as any)[methodName](4 * charSize, false, false)).toBe('AB\0ÿ'); - expect(cursor.position).toBe(5 * charSize); - } - }); -} - -testStringRead('stringAscii', 1); -testStringRead('stringUtf16', 2); - -function testIntegerWrite(methodName: string) { - test(methodName, () => { - const bytes = parseInt(methodName.replace(/^write[IU](\d+)$/, '$1'), 10) / 8; - let testNumber1 = 0; - let testNumber2 = 0; - // The "false" arrays are for big endian tests and the "true" arrays for little endian tests. - const testArrays1: { [index: string]: number[] } = { false: [], true: [] }; - const testArrays2: { [index: string]: number[] } = { false: [], true: [] }; - - for (let i = 1; i <= bytes; ++i) { - // Generates numbers of the form 0x010203... - testNumber1 <<= 8; - testNumber1 |= i; - testNumber2 <<= 8; - testNumber2 |= i + bytes; - testArrays1['false'].push(i); - testArrays1['true'].unshift(i); - testArrays2['false'].push(i + bytes); - testArrays2['true'].unshift(i + bytes); - } - - for (const littleEndian of [false, true]) { - const cursor = new ArrayBufferCursor(0, littleEndian); - (cursor as any)[methodName](testNumber1); - - expect(cursor.position).toBe(bytes); - expect(cursor.seekStart(0).u8Array(bytes)) - .toEqual(testArrays1[String(littleEndian)]); - expect(cursor.position).toBe(bytes); - - (cursor as any)[methodName](testNumber2); - - expect(cursor.position).toBe(2 * bytes); - expect(cursor.seekStart(0).u8Array(2 * bytes)) - .toEqual(testArrays1[String(littleEndian)].concat(testArrays2[String(littleEndian)])); - } - }); -} - -testIntegerWrite('writeU8'); -testIntegerWrite('writeU16'); -testIntegerWrite('writeU32'); - -test('writeF32', () => { - for (const littleEndian of [false, true]) { - const cursor = new ArrayBufferCursor(0, littleEndian); - cursor.writeF32(1337.9001); - - expect(cursor.position).toBe(4); - expect(cursor.seek(-4).f32()).toBeCloseTo(1337.9001, 4); - expect(cursor.position).toBe(4); - - cursor.writeF32(103.502); - - expect(cursor.position).toBe(8); - expect(cursor.seek(-4).f32()).toBeCloseTo(103.502, 3); - } -}); - -test('writeU8Array', () => { - for (const littleEndian of [false, true]) { - const bytes = 10; - const cursor = new ArrayBufferCursor(2 * bytes, littleEndian); - const uint8Array = new Uint8Array(cursor.buffer); - const testArray1 = []; - const testArray2 = []; - - for (let i = 1; i <= bytes; ++i) { - testArray1.push(i); - testArray2.push(i + bytes); - } - - cursor.writeU8Array(testArray1); - - expect(cursor.position).toBe(bytes); - - for (let i = 0; i < bytes; ++i) { - expect(uint8Array[i]).toBe(testArray1[i]); - } - - cursor.writeU8Array(testArray2); - - expect(cursor.position).toBe(2 * bytes); - - for (let i = 0; i < bytes; ++i) { - expect(uint8Array[i]).toBe(testArray1[i]); - } - - for (let i = 0; i < bytes; ++i) { - expect(uint8Array[i + bytes]).toBe(testArray2[i]); - } - } -}); diff --git a/src/bin-data/ArrayBufferCursor.ts b/src/bin-data/ArrayBufferCursor.ts deleted file mode 100644 index 93be5f64..00000000 --- a/src/bin-data/ArrayBufferCursor.ts +++ /dev/null @@ -1,425 +0,0 @@ -// TODO: remove dependency on text-encoding because it is no longer maintained. -import { TextDecoder, TextEncoder } from 'text-encoding'; - -const ASCII_DECODER = new TextDecoder('ascii'); -const UTF_16BE_DECODER = new TextDecoder('utf-16be'); -const UTF_16LE_DECODER = new TextDecoder('utf-16le'); - -const ASCII_ENCODER = new TextEncoder('ascii'); -const UTF_16BE_ENCODER = new TextEncoder('utf-16be'); -const UTF_16LE_ENCODER = new TextEncoder('utf-16le'); - -/** - * A cursor for reading and writing binary data. - * Uses an ArrayBuffer internally. This buffer is reallocated if and only if a write beyond the current capacity happens. - */ -export class ArrayBufferCursor { - private _size: number = 0; - - /** - * The cursor's size. This value will always be non-negative and equal to or smaller than the cursor's capacity. - */ - get size(): number { - return this._size; - } - - set size(size: number) { - if (size < 0) { - throw new Error('Size should be non-negative.') - } - - this.ensureCapacity(size); - this._size = size; - } - - /** - * The position from where bytes will be read or written. - */ - position: number; - - /** - * Byte order mode. - */ - littleEndian: boolean; - - /** - * The amount of bytes left to read from the current position onward. - */ - get bytesLeft(): number { - return this.size - this.position; - } - - /** - * The size of the underlying buffer. This value will always be equal to or greater than the cursor's size. - */ - get capacity(): number { - return this.buffer.byteLength; - } - - buffer: ArrayBuffer; - - private dv: DataView; - private uint8Array: Uint8Array; - private utf16Decoder: TextDecoder; - private utf16Encoder: TextEncoder; - - /** - * @param bufferOrCapacity - If an ArrayBuffer is given, writes to the cursor will be reflected in this array buffer and vice versa until a cursor write that requires allocating a new internal buffer happens - * @param littleEndian - Decides in which byte order multi-byte integers and floats will be interpreted - */ - constructor(bufferOrCapacity: ArrayBuffer | number, littleEndian?: boolean) { - if (typeof bufferOrCapacity === 'number') { - this.buffer = new ArrayBuffer(bufferOrCapacity); - this.size = 0; - } else if (bufferOrCapacity instanceof ArrayBuffer) { - this.buffer = bufferOrCapacity; - this.size = this.buffer.byteLength; - } else { - throw new Error('buffer_or_capacity should be an ArrayBuffer or a number.'); - } - - this.littleEndian = !!littleEndian; - this.position = 0; - this.dv = new DataView(this.buffer); - this.uint8Array = new Uint8Array(this.buffer, 0, this.size); - this.utf16Decoder = littleEndian ? UTF_16LE_DECODER : UTF_16BE_DECODER; - this.utf16Encoder = littleEndian ? UTF_16LE_ENCODER : UTF_16BE_ENCODER; - } - - /** - * Seek forward or backward by a number of bytes. - * - * @param offset - if positive, seeks forward by offset bytes, otherwise seeks backward by -offset bytes. - */ - seek(offset: number) { - return this.seekStart(this.position + offset); - } - - /** - * Seek forward from the start of the cursor by a number of bytes. - * - * @param offset - greater or equal to 0 and smaller than size - */ - seekStart(offset: number) { - if (offset < 0 || offset > this.size) { - throw new Error(`Offset ${offset} is out of bounds.`); - } - - this.position = offset; - return this; - } - - /** - * Seek backward from the end of the cursor by a number of bytes. - * - * @param offset - greater or equal to 0 and smaller than size - */ - seekEnd(offset: number) { - if (offset < 0 || offset > this.size) { - throw new Error(`Offset ${offset} is out of bounds.`); - } - - this.position = this.size - offset; - return this; - } - - /** - * Reads an unsigned 8-bit integer and increments position by 1. - */ - u8() { - return this.dv.getUint8(this.position++); - } - - /** - * Reads an unsigned 16-bit integer and increments position by 2. - */ - u16() { - const r = this.dv.getUint16(this.position, this.littleEndian); - this.position += 2; - return r; - } - - /** - * Reads an unsigned 32-bit integer and increments position by 4. - */ - u32() { - const r = this.dv.getUint32(this.position, this.littleEndian); - this.position += 4; - return r; - } - - /** - * Reads a signed 16-bit integer and increments position by 2. - */ - i16() { - const r = this.dv.getInt16(this.position, this.littleEndian); - this.position += 2; - return r; - } - - /** - * Reads a signed 32-bit integer and increments position by 4. - */ - i32() { - const r = this.dv.getInt32(this.position, this.littleEndian); - this.position += 4; - return r; - } - - /** - * Reads a 32-bit floating point number and increments position by 4. - */ - f32() { - const r = this.dv.getFloat32(this.position, this.littleEndian); - this.position += 4; - return r; - } - - /** - * Reads n unsigned 8-bit integers and increments position by n. - */ - u8Array(n: number): number[] { - const array = []; - for (let i = 0; i < n; ++i) array.push(this.dv.getUint8(this.position++)); - return array; - } - - /** - * Reads n unsigned 16-bit integers and increments position by 2n. - */ - u16Array(n: number): number[] { - const array = []; - - for (let i = 0; i < n; ++i) { - array.push(this.dv.getUint16(this.position, this.littleEndian)); - this.position += 2; - } - - return array; - } - - /** - * Consumes a variable number of bytes. - * - * @param size - the amount bytes to consume. - * @returns a new cursor containing size bytes. - */ - take(size: number): ArrayBufferCursor { - if (size < 0 || size > this.size - this.position) { - throw new Error(`Size ${size} out of bounds.`); - } - - this.position += size; - return new ArrayBufferCursor( - this.buffer.slice(this.position - size, this.position), this.littleEndian); - } - - /** - * Consumes up to maxByteLength bytes. - */ - stringAscii(maxByteLength: number, nullTerminated: boolean, dropRemaining: boolean) { - const string_length = nullTerminated - ? this.indexOfU8(0, maxByteLength) - this.position - : maxByteLength; - - const r = ASCII_DECODER.decode( - new DataView(this.buffer, this.position, string_length)); - this.position += dropRemaining - ? maxByteLength - : Math.min(string_length + 1, maxByteLength); - return r; - } - - /** - * Consumes up to maxByteLength bytes. - */ - stringUtf16(maxByteLength: number, nullTerminated: boolean, dropRemaining: boolean) { - const stringLength = nullTerminated - ? this.indexOfU16(0, maxByteLength) - this.position - : Math.floor(maxByteLength / 2) * 2; - - const r = this.utf16Decoder.decode( - new DataView(this.buffer, this.position, stringLength)); - this.position += dropRemaining - ? maxByteLength - : Math.min(stringLength + 2, maxByteLength); - return r; - } - - /** - * Writes an unsigned 8-bit integer and increments position by 1. If necessary, grows the cursor and reallocates the underlying buffer. - */ - writeU8(value: number) { - this.ensureCapacity(this.position + 1); - - this.dv.setUint8(this.position++, value); - - if (this.position > this.size) { - this.size = this.position; - } - - return this; - } - - /** - * Writes an unsigned 16-bit integer and increments position by 2. If necessary, grows the cursor and reallocates the underlying buffer. - */ - writeU16(value: number) { - this.ensureCapacity(this.position + 2); - - this.dv.setUint16(this.position, value, this.littleEndian); - this.position += 2; - - if (this.position > this.size) { - this.size = this.position; - } - - return this; - } - - /** - * Writes an unsigned 32-bit integer and increments position by 4. If necessary, grows the cursor and reallocates the underlying buffer. - */ - writeU32(value: number) { - this.ensureCapacity(this.position + 4); - - this.dv.setUint32(this.position, value, this.littleEndian); - this.position += 4; - - if (this.position > this.size) { - this.size = this.position; - } - - return this; - } - - /** - * Writes a signed 32-bit integer and increments position by 4. If necessary, grows the cursor and reallocates the underlying buffer. - */ - writeI32(value: number) { - this.ensureCapacity(this.position + 4); - - this.dv.setInt32(this.position, value, this.littleEndian); - this.position += 4; - - if (this.position > this.size) { - this.size = this.position; - } - - return this; - } - - /** - * Writes a 32-bit floating point number and increments position by 4. If necessary, grows the cursor and reallocates the underlying buffer. - */ - writeF32(value: number) { - this.ensureCapacity(this.position + 4); - - this.dv.setFloat32(this.position, value, this.littleEndian); - this.position += 4; - - if (this.position > this.size) { - this.size = this.position; - } - - return this; - } - - /** - * Writes an array of unsigned 8-bit integers and increments position by the array's length. If necessary, grows the cursor and reallocates the underlying buffer. - */ - writeU8Array(array: number[]) { - this.ensureCapacity(this.position + array.length); - - new Uint8Array(this.buffer, this.position).set(new Uint8Array(array)); - this.position += array.length; - - if (this.position > this.size) { - this.size = this.position; - } - - return this; - } - - /** - * Writes the contents of other and increments position by the size of other. If necessary, grows the cursor and reallocates the underlying buffer. - */ - writeCursor(other: ArrayBufferCursor) { - this.ensureCapacity(this.position + other.size); - - new Uint8Array(this.buffer, this.position).set(new Uint8Array(other.buffer)); - this.position += other.size; - - if (this.position > this.size) { - this.size = this.position; - } - - return this; - } - - writeStringAscii(str: string, byteLength: number) { - let i = 0; - - for (const byte of ASCII_ENCODER.encode(str)) { - if (i < byteLength) { - this.writeU8(byte); - ++i; - } - } - - while (i < byteLength) { - this.writeU8(0); - ++i; - } - } - - /** - * @returns a Uint8Array that remains a write-through view of the underlying array buffer until the buffer is reallocated. - */ - uint8ArrayView(): Uint8Array { - return this.uint8Array; - } - - private indexOfU8(value: number, maxByteLength: number) { - const maxPos = Math.min(this.position + maxByteLength, this.size); - - for (let i = this.position; i < maxPos; ++i) { - if (this.dv.getUint8(i) === value) { - return i; - } - } - - return this.position + maxByteLength; - } - - private indexOfU16(value: number, maxByteLength: number) { - const maxPos = Math.min(this.position + maxByteLength, this.size); - - for (let i = this.position; i < maxPos; i += 2) { - if (this.dv.getUint16(i, this.littleEndian) === value) { - return i; - } - } - - return this.position + maxByteLength; - } - - /** - * Increases buffer size if necessary. - */ - private ensureCapacity(minNewSize: number) { - if (minNewSize > this.capacity) { - let newSize = this.capacity || minNewSize; - - do { - newSize *= 2; - } while (newSize < minNewSize); - - const newBuffer = new ArrayBuffer(newSize); - new Uint8Array(newBuffer).set(new Uint8Array(this.buffer, 0, this.size)); - this.buffer = newBuffer; - this.dv = new DataView(this.buffer); - this.uint8Array = new Uint8Array(this.buffer, 0, minNewSize); - } - } -} diff --git a/src/bin-data/compression/prs/compress.ts b/src/bin-data/compression/prs/compress.ts deleted file mode 100644 index 6e79a34e..00000000 --- a/src/bin-data/compression/prs/compress.ts +++ /dev/null @@ -1,298 +0,0 @@ -/** - * This code is based on the Sylverant PRS compression code written by Lawrence Sebald. - */ - -import { ArrayBufferCursor } from '../../ArrayBufferCursor'; - -export function compress(src: ArrayBufferCursor): ArrayBufferCursor { - const ctx = new Context(src); - const hashTable = new HashTable(); - - if (ctx.src.size <= 3) { - // Make a literal copy of the input. - while (ctx.src.bytesLeft) { - ctx.setBit(1); - ctx.copyLiteral(); - } - } else { - // Add the first two "strings" to the hash table. - hashTable.put(hashTable.hash(ctx.src), 0); - ctx.src.seek(1); - hashTable.put(hashTable.hash(ctx.src), 1); - ctx.src.seek(-1); - - // Copy the first two bytes as literals. - ctx.setBit(1); - ctx.copyLiteral(); - ctx.setBit(1); - ctx.copyLiteral(); - - while (ctx.src.bytesLeft > 1) { - let [offset, mlen] = ctx.findLongestMatch(hashTable, false); - - if (mlen > 0) { - ctx.src.seek(1); - const [offset2, mlen2] = ctx.findLongestMatch(hashTable, true); - ctx.src.seek(-1); - - // Did the "lazy match" produce something more compressed? - if (mlen2 > mlen) { - let copyLiteral = true; - // Check if it is a good idea to switch from a short match to a long one. - if (mlen >= 2 && mlen <= 5 && offset2 < offset) { - if (offset >= -256 && offset2 < -256) { - if (mlen2 - mlen < 3) { - copyLiteral = false; - } - } - } - - if (copyLiteral) { - ctx.setBit(1); - ctx.copyLiteral(); - continue; - } - } - - // What kind of match did we find? - if (mlen >= 2 && mlen <= 5 && offset >= -256) { - // Short match. - ctx.setBit(0); - ctx.setBit(0); - ctx.setBit((mlen - 2) & 0x02); - ctx.setBit((mlen - 2) & 0x01); - ctx.writeLiteral(offset & 0xFF); - ctx.addIntermediates(hashTable, mlen); - continue; - } else if (mlen >= 3 && mlen <= 9) { - // Long match, short length. - ctx.setBit(0); - ctx.setBit(1); - ctx.writeLiteral(((offset & 0x1F) << 3) | ((mlen - 2) & 0x07)); - ctx.writeLiteral(offset >> 5); - ctx.addIntermediates(hashTable, mlen); - continue; - } else if (mlen > 9) { - // Long match, long length. - if (mlen > 256) { - mlen = 256; - } - - ctx.setBit(0); - ctx.setBit(1); - ctx.writeLiteral((offset & 0x1F) << 3); - ctx.writeLiteral(offset >> 5); - ctx.writeLiteral(mlen - 1); - ctx.addIntermediates(hashTable, mlen); - continue; - } - } - - // If we get here, we didn't find a suitable match, so just we just make a literal copy. - ctx.setBit(1); - ctx.copyLiteral(); - } - - // If there's a left over byte at the end, make a literal copy. - if (ctx.src.bytesLeft) { - ctx.setBit(1); - ctx.copyLiteral(); - } - } - - ctx.writeEof(); - - return ctx.dst.seekStart(0); -} - -const MAX_WINDOW = 0x2000; -const WINDOW_MASK = MAX_WINDOW - 1; -const HASH_SIZE = 1 << 8; - -class Context { - src: ArrayBufferCursor; - dst: ArrayBufferCursor; - flags: number; - flagBitsLeft: number; - flagOffset: number; - - constructor(cursor: ArrayBufferCursor) { - this.src = cursor; - this.dst = new ArrayBufferCursor(cursor.size, cursor.littleEndian); - this.flags = 0; - this.flagBitsLeft = 0; - this.flagOffset = 0; - } - - setBit(bit: number): void { - if (!this.flagBitsLeft--) { - // Write out the flags to their position in the file, and store the next flags byte position. - const pos = this.dst.position; - this.dst - .seekStart(this.flagOffset) - .writeU8(this.flags) - .seekStart(pos) - .writeU8(0); // Placeholder for the next flags byte. - this.flagOffset = pos; - this.flagBitsLeft = 7; - } - - this.flags >>>= 1; - - if (bit) { - this.flags |= 0x80; - } - } - - copyLiteral(): void { - this.dst.writeU8(this.src.u8()); - } - - writeLiteral(value: number): void { - this.dst.writeU8(value); - } - - writeFinalFlags(): void { - this.flags >>>= this.flagBitsLeft; - const pos = this.dst.position; - this.dst - .seekStart(this.flagOffset) - .writeU8(this.flags) - .seekStart(pos); - } - - writeEof(): void { - this.setBit(0); - this.setBit(1); - - this.writeFinalFlags(); - - this.writeLiteral(0); - this.writeLiteral(0); - } - - matchLength(s2: number): number { - const array = this.src.uint8ArrayView(); - let len = 0; - let s1 = this.src.position; - - while (s1 < array.byteLength && array[s1] === array[s2]) { - ++len; - ++s1; - ++s2; - } - - return len; - } - - findLongestMatch(hashTable: HashTable, lazy: boolean): [number, number] { - if (!this.src.bytesLeft) { - return [0, 0]; - } - - // Figure out where we're looking. - const hash = hashTable.hash(this.src); - - // If there is nothing in the table at that point, bail out now. - let entry = hashTable.get(hash); - - if (entry === null) { - if (!lazy) { - hashTable.put(hash, this.src.position); - } - - return [0, 0]; - } - - // If we'd go outside the window, truncate the hash chain now. - if (this.src.position - entry > MAX_WINDOW) { - hashTable.hashToOffset[hash] = null; - - if (!lazy) { - hashTable.put(hash, this.src.position); - } - - return [0, 0]; - } - - // Ok, we have something in the hash table that matches the hash value. - // Follow the chain to see if we have an actual string match, and find the longest match. - let longestLength = 0; - let longestMatch = 0; - - while (entry != null) { - const mlen = this.matchLength(entry); - - if (mlen > longestLength || mlen >= 256) { - longestLength = mlen; - longestMatch = entry; - } - - // Follow the chain, making sure not to exceed a difference of MAX_WINDOW. - let entry2 = hashTable.prev(entry); - - if (entry2 !== null) { - // If we'd go outside the window, truncate the hash chain now. - if (this.src.position - entry2 > MAX_WINDOW) { - hashTable.setPrev(entry, null); - entry2 = null; - } - } - - entry = entry2; - } - - // Add our current string to the hash. - if (!lazy) { - hashTable.put(hash, this.src.position); - } - - // Did we find a match? - const offset = longestLength > 0 ? longestMatch - this.src.position : 0; - return [offset, longestLength]; - } - - addIntermediates(hashTable: HashTable, len: number): void { - this.src.seek(1); - - for (let i = 1; i < len; ++i) { - const hash = hashTable.hash(this.src); - hashTable.put(hash, this.src.position); - this.src.seek(1); - } - } -} - -class HashTable { - hashToOffset: Array = new Array(HASH_SIZE).fill(null); - maskedOffsetToPrev: Array = new Array(MAX_WINDOW).fill(null); - - hash(cursor: ArrayBufferCursor): number { - let hash = cursor.u8(); - - if (cursor.bytesLeft) { - hash ^= cursor.u8(); - cursor.seek(-1); - } - - cursor.seek(-1); - return hash; - } - - get(hash: number): number | null { - return this.hashToOffset[hash]; - } - - put(hash: number, offset: number): void { - this.setPrev(offset, this.hashToOffset[hash]); - this.hashToOffset[hash] = offset; - } - - prev(offset: number): number | null { - return this.maskedOffsetToPrev[offset & WINDOW_MASK]; - } - - setPrev(offset: number, prevOffset: number | null): void { - this.maskedOffsetToPrev[offset & WINDOW_MASK] = prevOffset; - } -} diff --git a/src/bin-data/compression/prs/decompress.ts b/src/bin-data/compression/prs/decompress.ts deleted file mode 100644 index 8e18543a..00000000 --- a/src/bin-data/compression/prs/decompress.ts +++ /dev/null @@ -1,117 +0,0 @@ -/** - * This code is based on the Sylverant PRS decompression code written by Lawrence Sebald. - */ - -/*eslint no-use-before-define: "off"*/ -import { ArrayBufferCursor } from '../../ArrayBufferCursor'; - -export function decompress(cursor: ArrayBufferCursor) { - const ctx = new Context(cursor); - - while (true) { - if (ctx.readFlagBit() === 1) { - // Single byte copy. - ctx.copyU8(); - } else { - // Multi byte copy. - let length; - let offset; - - if (ctx.readFlagBit() === 0) { - // Short copy. - length = ctx.readFlagBit() << 1; - length |= ctx.readFlagBit(); - length += 2; - - offset = ctx.readU8() - 256; - } else { - // Long copy or end of file. - offset = ctx.readU16(); - - // Two zero bytes implies that this is the end of the file. - if (offset === 0) { - break; - } - - // Do we need to read a length byte, or is it encoded in what we already have? - length = offset & 0b111; - offset >>>= 3; - - if (length === 0) { - length = ctx.readU8(); - length += 1; - } else { - length += 2; - } - - offset -= 8192; - } - - ctx.offsetCopy(offset, length); - } - } - - return ctx.dst.seekStart(0); -} - -class Context { - src: ArrayBufferCursor; - dst: ArrayBufferCursor; - flags: number; - flagBitsLeft: number; - - constructor(cursor: ArrayBufferCursor) { - this.src = cursor; - this.dst = new ArrayBufferCursor(4 * cursor.size, cursor.littleEndian); - this.flags = 0; - this.flagBitsLeft = 0; - } - - readFlagBit() { - // Fetch a new flag byte when the previous byte has been processed. - if (this.flagBitsLeft === 0) { - this.flags = this.readU8(); - this.flagBitsLeft = 8; - } - - let bit = this.flags & 1; - this.flags >>>= 1; - this.flagBitsLeft -= 1; - return bit; - } - - copyU8() { - this.dst.writeU8(this.readU8()); - } - - readU8() { - return this.src.u8(); - } - - readU16() { - return this.src.u16(); - } - - offsetCopy(offset: number, length: number) { - if (offset < -8192 || offset > 0) { - console.error(`offset was ${offset}, should be between -8192 and 0.`); - } - - if (length < 1 || length > 256) { - console.error(`length was ${length}, should be between 1 and 256.`); - } - - // The length can be larger than -offset, in that case we copy -offset bytes size/-offset times. - const bufSize = Math.min(-offset, length); - - this.dst.seek(offset); - const buf = this.dst.take(bufSize); - this.dst.seek(-offset - bufSize); - - for (let i = 0; i < Math.floor(length / bufSize); ++i) { - this.dst.writeCursor(buf); - } - - this.dst.writeCursor(buf.take(length % bufSize)); - } -} diff --git a/src/bin-data/compression/prs/index.test.ts b/src/bin-data/compression/prs/index.test.ts deleted file mode 100644 index cd398114..00000000 --- a/src/bin-data/compression/prs/index.test.ts +++ /dev/null @@ -1,83 +0,0 @@ -import { ArrayBufferCursor } from '../../ArrayBufferCursor'; -import { compress, decompress } from '../prs'; - -function testWithBytes(bytes: number[], expectedCompressedSize: number) { - const cursor = new ArrayBufferCursor(new Uint8Array(bytes).buffer, true); - - for (const byte of bytes) { - cursor.writeU8(byte); - } - - cursor.seekStart(0); - const compressedCursor = compress(cursor); - - expect(compressedCursor.size).toBe(expectedCompressedSize); - - const testCursor = decompress(compressedCursor); - cursor.seekStart(0); - - expect(testCursor.size).toBe(cursor.size); - - while (cursor.bytesLeft) { - if (cursor.u8() !== testCursor.u8()) { - cursor.seek(-1); - testCursor.seek(-1); - break; - } - } - - expect(testCursor.position).toBe(testCursor.size); -} - -test('PRS compression and decompression, best case', () => { - // Compression factor: 0.018 - testWithBytes(new Array(1000).fill(128), 18); -}); - -test('PRS compression and decompression, worst case', () => { - const prng = new Prng(); - - // Compression factor: 1.124 - testWithBytes(new Array(1000).fill(0).map(_ => prng.nextInteger(0, 255)), 1124); -}); - -test('PRS compression and decompression, typical case', () => { - const prng = new Prng(); - const pattern = [0, 0, 2, 0, 3, 0, 5, 0, 0, 0, 7, 9, 11, 13, 0, 0]; - const arrays = new Array(100) - .fill(pattern) - .map(array => array.map((e: number) => e + prng.nextInteger(0, 10))); - const flattenedArray = [].concat.apply([], arrays); - - // Compression factor: 0.834 - testWithBytes(flattenedArray, 1335); -}); - -test('PRS compression and decompression, 0 bytes', () => { - testWithBytes([], 3); -}); - -test('PRS compression and decompression, 1 byte', () => { - testWithBytes([111], 4); -}); - -test('PRS compression and decompression, 2 bytes', () => { - testWithBytes([111, 224], 5); -}); - -test('PRS compression and decompression, 3 bytes', () => { - testWithBytes([56, 237, 158], 6); -}); - -class Prng { - seed = 1; - - next(): number { - const x = Math.sin(this.seed++) * 10000; - return x - Math.floor(x); - } - - nextInteger(min: number, max: number): number { - return Math.floor(this.next() * (max + 1 - min)) + min; - } -} diff --git a/src/bin-data/compression/prs/index.ts b/src/bin-data/compression/prs/index.ts deleted file mode 100644 index 13e08870..00000000 --- a/src/bin-data/compression/prs/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export { compress } from './compress'; -export { decompress } from './decompress'; \ No newline at end of file diff --git a/src/bin-data/loading/areas.ts b/src/bin-data/loading/areas.ts deleted file mode 100644 index 5cb1e469..00000000 --- a/src/bin-data/loading/areas.ts +++ /dev/null @@ -1,79 +0,0 @@ -import { Object3D } from 'three'; -import { Section } from '../../domain'; -import { getAreaRenderData, getAreaCollisionData } from './binaryAssets'; -import { parseCRel, parseNRel } from '../parsing/geometry'; - -// -// Caches -// -const sectionsCache: Map> = new Map(); -const renderGeometryCache: Map> = new Map(); -const collisionGeometryCache: Map> = new Map(); - -export function getAreaSections( - episode: number, - areaId: number, - areaVariant: number -): Promise { - const sections = sectionsCache.get(`${episode}-${areaId}-${areaVariant}`); - - if (sections) { - return sections; - } else { - return getAreaSectionsAndRenderGeometry( - episode, areaId, areaVariant).then(({sections}) => sections); - } -} - -export function getAreaRenderGeometry( - episode: number, - areaId: number, - areaVariant: number -): Promise { - const object3d = renderGeometryCache.get(`${episode}-${areaId}-${areaVariant}`); - - if (object3d) { - return object3d; - } else { - return getAreaSectionsAndRenderGeometry( - episode, areaId, areaVariant).then(({object3d}) => object3d); - } -} - -export function getAreaCollisionGeometry( - episode: number, - areaId: number, - areaVariant: number -): Promise { - const object3d = collisionGeometryCache.get(`${episode}-${areaId}-${areaVariant}`); - - if (object3d) { - return object3d; - } else { - const object3d = getAreaCollisionData( - episode, areaId, areaVariant).then(parseCRel); - collisionGeometryCache.set(`${areaId}-${areaVariant}`, object3d); - return object3d; - } -} - -function getAreaSectionsAndRenderGeometry( - episode: number, - areaId: number, - areaVariant: number -): Promise<{ sections: Section[], object3d: Object3D }> { - const promise = getAreaRenderData( - episode, areaId, areaVariant).then(parseNRel); - - const sections = new Promise((resolve, reject) => { - promise.then(({sections}) => resolve(sections)).catch(reject); - }); - const object3d = new Promise((resolve, reject) => { - promise.then(({object3d}) => resolve(object3d)).catch(reject); - }); - - sectionsCache.set(`${episode}-${areaId}-${areaVariant}`, sections); - renderGeometryCache.set(`${episode}-${areaId}-${areaVariant}`, object3d); - - return promise; -} diff --git a/src/bin-data/loading/binaryAssets.ts b/src/bin-data/loading/binaryAssets.ts deleted file mode 100644 index 95448d1e..00000000 --- a/src/bin-data/loading/binaryAssets.ts +++ /dev/null @@ -1,208 +0,0 @@ -import { NpcType, ObjectType } from '../../domain'; - -export function getAreaRenderData( - episode: number, - areaId: number, - areaVersion: number -): Promise { - return getAreaAsset(episode, areaId, areaVersion, 'render'); -} - -export function getAreaCollisionData( - episode: number, - areaId: number, - areaVersion: number -): Promise { - return getAreaAsset(episode, areaId, areaVersion, 'collision'); -} - -export async function getNpcData(npcType: NpcType): Promise<{ url: string, data: ArrayBuffer }> { - const url = npcTypeToUrl(npcType); - const data = await getAsset(url); - return ({ url, data }); -} - -export async function getObjectData(objectType: ObjectType): Promise<{ url: string, data: ArrayBuffer }> { - const url = objectTypeToUrl(objectType); - const data = await getAsset(url); - return ({ url, data }); -} - -/** - * Cache for the binary data. - */ -const bufferCache: Map> = new Map(); - -function getAsset(url: string): Promise { - const promise = bufferCache.get(url); - - if (promise) { - return promise; - } else { - const baseUrl = process.env.PUBLIC_URL; - const promise = fetch(baseUrl + url).then(r => r.arrayBuffer()); - bufferCache.set(url, promise); - return promise; - } -} - -const areaBaseNames = [ - [ - ['city00_00', 1], - ['forest01', 1], - ['forest02', 1], - ['cave01_', 6], - ['cave02_', 5], - ['cave03_', 6], - ['machine01_', 6], - ['machine02_', 6], - ['ancient01_', 5], - ['ancient02_', 5], - ['ancient03_', 5], - ['boss01', 1], - ['boss02', 1], - ['boss03', 1], - ['darkfalz00', 1] - ], - [ - ['labo00_00', 1], - ['ruins01_', 3], - ['ruins02_', 3], - ['space01_', 3], - ['space02_', 3], - ['jungle01_00', 1], - ['jungle02_00', 1], - ['jungle03_00', 1], - ['jungle04_', 3], - ['jungle05_00', 1], - ['seabed01_', 3], - ['seabed02_', 3], - ['boss05', 1], - ['boss06', 1], - ['boss07', 1], - ['boss08', 1], - ['jungle06_00', 1], - ['jungle07_', 5] - ], - [ - // Don't remove this empty array, see usage of areaBaseNames in areaVersionToBaseUrl. - ], - [ - ['city02_00', 1], - ['wilds01_00', 1], - ['wilds01_01', 1], - ['wilds01_02', 1], - ['wilds01_03', 1], - ['crater01_00', 1], - ['desert01_', 3], - ['desert02_', 3], - ['desert03_', 3], - ['boss09_00', 1] - ] -]; - -function areaVersionToBaseUrl( - episode: number, - areaId: number, - areaVariant: number -): string { - const episodeBaseNames = areaBaseNames[episode - 1]; - - if (0 <= areaId && areaId < episodeBaseNames.length) { - const [baseName, variants] = episodeBaseNames[areaId]; - - if (0 <= areaVariant && areaVariant < variants) { - let variant: string; - - if (variants === 1) { - variant = ''; - } else { - variant = String(areaVariant); - while (variant.length < 2) variant = '0' + variant; - } - - return `/maps/map_${baseName}${variant}`; - } else { - throw new Error(`Unknown variant ${areaVariant} of area ${areaId} in episode ${episode}.`); - } - } else { - throw new Error(`Unknown episode ${episode} area ${areaId}.`); - } -} - -type AreaAssetType = 'render' | 'collision'; - -function getAreaAsset( - episode: number, - areaId: number, - areaVariant: number, - type: AreaAssetType -): Promise { - try { - const baseUrl = areaVersionToBaseUrl(episode, areaId, areaVariant); - const suffix = type === 'render' ? 'n.rel' : 'c.rel'; - return getAsset(baseUrl + suffix); - } catch (e) { - return Promise.reject(e); - } -} - -function npcTypeToUrl(npcType: NpcType): string { - switch (npcType) { - // The dubswitch model in in XJ format. - case NpcType.Dubswitch: return `/npcs/${npcType.code}.xj`; - - // Episode II VR Temple - - case NpcType.Hildebear2: return npcTypeToUrl(NpcType.Hildebear); - case NpcType.Hildeblue2: return npcTypeToUrl(NpcType.Hildeblue); - case NpcType.RagRappy2: return npcTypeToUrl(NpcType.RagRappy); - case NpcType.Monest2: return npcTypeToUrl(NpcType.Monest); - case NpcType.PoisonLily2: return npcTypeToUrl(NpcType.PoisonLily); - case NpcType.NarLily2: return npcTypeToUrl(NpcType.NarLily); - case NpcType.GrassAssassin2: return npcTypeToUrl(NpcType.GrassAssassin); - case NpcType.Dimenian2: return npcTypeToUrl(NpcType.Dimenian); - case NpcType.LaDimenian2: return npcTypeToUrl(NpcType.LaDimenian); - case NpcType.SoDimenian2: return npcTypeToUrl(NpcType.SoDimenian); - case NpcType.DarkBelra2: return npcTypeToUrl(NpcType.DarkBelra); - - // Episode II VR Spaceship - - case NpcType.SavageWolf2: return npcTypeToUrl(NpcType.SavageWolf); - case NpcType.BarbarousWolf2: return npcTypeToUrl(NpcType.BarbarousWolf); - case NpcType.PanArms2: return npcTypeToUrl(NpcType.PanArms); - case NpcType.Dubchic2: return npcTypeToUrl(NpcType.Dubchic); - case NpcType.Gilchic2: return npcTypeToUrl(NpcType.Gilchic); - case NpcType.Garanz2: return npcTypeToUrl(NpcType.Garanz); - case NpcType.Dubswitch2: return npcTypeToUrl(NpcType.Dubswitch); - case NpcType.Delsaber2: return npcTypeToUrl(NpcType.Delsaber); - case NpcType.ChaosSorcerer2: return npcTypeToUrl(NpcType.ChaosSorcerer); - - default: return `/npcs/${npcType.code}.nj`; - } -} - -function objectTypeToUrl(objectType: ObjectType): string { - switch (objectType) { - case ObjectType.EasterEgg: - case ObjectType.ChristmasTree: - case ObjectType.ChristmasWreath: - case ObjectType.TwentyFirstCentury: - case ObjectType.Sonic: - case ObjectType.WelcomeBoard: - case ObjectType.FloatingJelifish: - case ObjectType.RuinsSeal: - case ObjectType.Dolphin: - case ObjectType.Cacti: - case ObjectType.BigBrownRock: - case ObjectType.PoisonPlant: - case ObjectType.BigBlackRocks: - case ObjectType.FallingRock: - case ObjectType.DesertFixedTypeBoxBreakableCrystals: - case ObjectType.BeeHive: - return `/objects/${String(objectType.psoId)}.nj`; - - default: - return `/objects/${String(objectType.psoId)}.xj`; - } -} diff --git a/src/bin-data/loading/entities.ts b/src/bin-data/loading/entities.ts deleted file mode 100644 index 886c7126..00000000 --- a/src/bin-data/loading/entities.ts +++ /dev/null @@ -1,52 +0,0 @@ -import { BufferGeometry } from 'three'; -import { NpcType, ObjectType } from '../../domain'; -import { getNpcData, getObjectData } from './binaryAssets'; -import { ArrayBufferCursor } from '../ArrayBufferCursor'; -import { parseNj, parseXj } from '../parsing/ninja'; - -const npcCache: Map> = new Map(); -const objectCache: Map> = new Map(); - -export function getNpcGeometry(npcType: NpcType): Promise { - let geometry = npcCache.get(String(npcType.id)); - - if (geometry) { - return geometry; - } else { - geometry = getNpcData(npcType).then(({ url, data }) => { - const cursor = new ArrayBufferCursor(data, true); - const object3d = url.endsWith('.nj') ? parseNj(cursor) : parseXj(cursor); - - if (object3d) { - return object3d; - } else { - throw new Error('File could not be parsed into a BufferGeometry.'); - } - }); - - npcCache.set(String(npcType.id), geometry); - return geometry; - } -} - -export function getObjectGeometry(objectType: ObjectType): Promise { - let geometry = objectCache.get(String(objectType.id)); - - if (geometry) { - return geometry; - } else { - geometry = getObjectData(objectType).then(({ url, data }) => { - const cursor = new ArrayBufferCursor(data, true); - const object3d = url.endsWith('.nj') ? parseNj(cursor) : parseXj(cursor); - - if (object3d) { - return object3d; - } else { - throw new Error('File could not be parsed into a BufferGeometry.'); - } - }); - - objectCache.set(String(objectType.id), geometry); - return geometry; - } -} diff --git a/src/bin-data/parsing/bin.test.ts b/src/bin-data/parsing/bin.test.ts deleted file mode 100644 index 450fb126..00000000 --- a/src/bin-data/parsing/bin.test.ts +++ /dev/null @@ -1,27 +0,0 @@ -import * as fs from 'fs'; -import { ArrayBufferCursor } from '../ArrayBufferCursor'; -import * as prs from '../compression/prs'; -import { parseBin, writeBin } from './bin'; - -/** - * Parse a file, convert the resulting structure to BIN again and check whether the end result is equal to the original. - */ -test('parseBin and writeBin', () => { - const origBuffer = fs.readFileSync('test/resources/quest118_e.bin').buffer; - const origBin = prs.decompress(new ArrayBufferCursor(origBuffer, true)); - const testBin = writeBin(parseBin(origBin)); - origBin.seekStart(0); - - expect(testBin.size).toBe(origBin.size); - - let match = true; - - while (origBin.bytesLeft) { - if (testBin.u8() !== origBin.u8()) { - match = false; - break; - } - } - - expect(match).toBe(true); -}); diff --git a/src/bin-data/parsing/bin.ts b/src/bin-data/parsing/bin.ts deleted file mode 100644 index 9f993fad..00000000 --- a/src/bin-data/parsing/bin.ts +++ /dev/null @@ -1,1007 +0,0 @@ -import { ArrayBufferCursor } from '../ArrayBufferCursor'; - -export interface BinFile { - questNumber: number; - language: number; - questName: string; - shortDescription: string; - longDescription: string; - functionOffsets: number[]; - instructions: Instruction[]; - data: ArrayBufferCursor; -} - -export function parseBin(cursor: ArrayBufferCursor): BinFile { - const objectCodeOffset = cursor.u32(); - const functionOffsetTableOffset = cursor.u32(); // Relative offsets - const size = cursor.u32(); - cursor.seek(4); // Always seems to be 0xFFFFFFFF - const questNumber = cursor.u32(); - const language = cursor.u32(); - const questName = cursor.stringUtf16(64, true, true); - const shortDescription = cursor.stringUtf16(256, true, true); - const longDescription = cursor.stringUtf16(576, true, true); - - if (size !== cursor.size) { - console.warn(`Value ${size} in bin size field does not match actual size ${cursor.size}.`); - } - - const functionOffsetCount = Math.floor( - (cursor.size - functionOffsetTableOffset) / 4); - - cursor.seekStart(functionOffsetTableOffset); - const functionOffsets = []; - - for (let i = 0; i < functionOffsetCount; ++i) { - functionOffsets.push(cursor.i32()); - } - - const instructions = parseObjectCode( - cursor.seekStart(objectCodeOffset).take(functionOffsetTableOffset - objectCodeOffset) - ); - - return { - questNumber, - language, - questName, - shortDescription, - longDescription, - functionOffsets, - instructions, - data: cursor.seekStart(0).take(cursor.size) - }; -} - -export function writeBin({ data }: { data: ArrayBufferCursor }): ArrayBufferCursor { - return data.seekStart(0); -} - -export interface Instruction { - opcode: number; - mnemonic: string; - args: any[]; - size: number; -} - -function parseObjectCode(cursor: ArrayBufferCursor): Instruction[] { - const instructions = []; - - while (cursor.bytesLeft) { - const mainOpcode = cursor.u8(); - let opcode; - let opsize; - let list; - - switch (mainOpcode) { - case 0xF8: - opcode = cursor.u8(); - opsize = 2; - list = F8opcodeList; - break; - case 0xF9: - opcode = cursor.u8(); - opsize = 2; - list = F9opcodeList; - break; - default: - opcode = mainOpcode; - opsize = 1; - list = opcodeList; - break; - } - - const [, mnemonic, mask] = list[opcode]; - const opargs = parseInstructionArguments(cursor, mask); - - if (!opargs) { - console.error(`Parameters unknown for opcode 0x${opcode.toString(16).toUpperCase()}.`); - break; - } - - instructions.push({ - opcode, - mnemonic, - args: opargs.args, - size: opsize + opargs.size - }); - } - - return instructions; -} - -function parseInstructionArguments(cursor: ArrayBufferCursor, mask: string | null) { - if (mask == null) { - return; - } - - const oldPos = cursor.position; - const args = []; - let size = 0; - - outer: - for (let i = 0; i < mask.length; ++i) { - switch (mask.charAt(i)) { - // Pushes something on the stack - case 'p': - break; - // Pops the stack (no increments) - case 'a': - break outer; - - // Unsigned integers - case 'B': - args.push(cursor.u8()); - size += 1; - break; - case 'W': - args.push(cursor.u16()); - size += 2; - break; - case 'L': - args.push(cursor.u32()); - size += 4; - break; - - // Signed integers - case 'I': - args.push(cursor.i32()); - size += 4; - break; - - // Floats - case 'f': - case 'F': - args.push(cursor.f32()); - size += 4; - break; - - // Registers? - case 'R': - case 'r': - size += 1; - break; - - // Pointers to unsigned integers? - case 'b': - args.push(cursor.u8()); - size += 1; - break; - case 'w': - args.push(cursor.u16()); - size += 2; - break; - case 'l': - args.push(cursor.u32()); - size += 4; - break; - - // Pointers to signed integers? - case 'i': - args.push(cursor.i32()); - size += 4; - break; - - // Variably sized data (e.g. strings)? - case 'j': - case 'J': - size += 1 + cursor.seek(size).u8() * 2; - break; - case 't': - case 'T': - size += 1 + cursor.seek(size).u8(); - break; - case 's': - case 'S': - while (cursor.u16()) { - size += 2; - } - size += 2; - break; - - default: - throw new Error(`Unknown mask part ${mask.charAt(i)}.`); - } - } - - cursor.seekStart(oldPos + size); - return { args, size }; -} - -const opcodeList: Array<[number, string, string | null]> = [ - [0x00, 'nop', ''], - [0x01, 'ret', ''], - [0x02, 'sync', ''], - [0x03, 'exit', 'aL'], - [0x04, 'thread', 'W'], - [0x05, 'va_start', ''], - [0x06, 'va_end', ''], - [0x07, 'va_call', 'W'], - [0x08, 'let', 'RR'], - [0x09, 'leti', 'RI'], - [0x0A, 'unknown', null], - [0x0B, 'unknown', null], - [0x0C, 'unknown', null], - [0x0D, 'unknown', null], - [0x0E, 'unknown', null], - [0x0F, 'unknown', null], - [0x10, 'set', 'R'], - [0x11, 'clear', 'R'], - [0x12, 'rev', 'R'], - [0x13, 'gset', 'w'], - [0x14, 'gclear', 'w'], - [0x15, 'grev', 'w'], - [0x16, 'glet', 'w'], - [0x17, 'gget', 'wR'], - [0x18, 'add', 'RR'], - [0x19, 'addi', 'RI'], - [0x1A, 'sub', 'RR'], - [0x1B, 'subi', 'RI'], - [0x1C, 'mul', 'RR'], - [0x1D, 'muli', 'RI'], - [0x1E, 'div', 'RR'], - [0x1F, 'divi', 'RI'], - [0x20, 'and', 'RR'], - [0x21, 'andi', 'RI'], - [0x22, 'or', 'RR'], - [0x23, 'ori', 'RI'], - [0x24, 'xor', 'RR'], - [0x25, 'xori', 'RI'], - [0x26, 'mod', 'RR'], - [0x27, 'modi', 'RI'], - [0x28, 'jmp', 'W'], - [0x29, 'call', 'W'], - [0x2A, 'jmp_on', 'Wt'], - [0x2B, 'jmp_off', 'Wt'], - [0x2C, 'jmp_=', 'RRW'], - [0x2D, 'jmpi_=', 'RIW'], - [0x2E, 'jmp_!=', 'RRW'], - [0x2F, 'jmpi_!=', 'RIW'], - [0x30, 'ujmp_>', 'RRW'], - [0x31, 'ujmpi_>', 'RLW'], - [0x32, 'jmp_>', 'RRW'], - [0x33, 'jmpi_>', 'RIW'], - [0x34, 'ujmp_<', 'RRW'], - [0x35, 'ujmpi_<', 'RLW'], - [0x36, 'jmp_<', 'RRW'], - [0x37, 'jmpi_<', 'RIW'], - [0x38, 'ujmp_>=', 'RRW'], - [0x39, 'ujmpi_>=', 'RLW'], - [0x3A, 'jmp_>=', 'RRW'], - [0x3B, 'jmpi_>=', 'RIW'], - [0x3C, 'ujmp_<=', 'RRW'], - [0x3D, 'ujmpi_<=', 'RLW'], - [0x3E, 'jmp_<=', 'RRW'], - [0x3F, 'jmpi_<=', 'RIW'], - [0x40, 'switch_jmp', 'Rj'], - [0x41, 'switch_call', 'Rj'], - [0x42, 'stack_push', 'R'], - [0x43, 'stack_pop', 'R'], - [0x44, 'stack_pushm', 'RL'], - [0x45, 'stack_popm', 'RL'], - [0x46, 'unknown', null], - [0x47, 'unknown', null], - [0x48, 'arg_pushr', 'pR'], - [0x49, 'arg_pushl', 'pI'], - [0x4A, 'arg_pushb', 'pB'], - [0x4B, 'arg_pushw', 'pW'], - [0x4C, 'unknown', null], - [0x4D, 'unknown', null], - [0x4E, 'arg_pushs', 'ps'], - [0x4F, 'unknown4F', 'RR'], - [0x50, 'message', 'aLs'], - [0x51, 'list', 'aRs'], - [0x52, 'fadein', ''], - [0x53, 'fadeout', ''], - [0x54, 'se', 'aL'], - [0x55, 'bgm', 'aL'], - [0x56, 'unknown', null], - [0x57, 'unknown', null], - [0x58, 'enable', 'aL'], - [0x59, 'disable', 'aL'], - [0x5A, 'window_msg', 'as'], - [0x5B, 'add_msg', 'as'], - [0x5C, 'mesend', ''], - [0x5D, 'gettime', 'R'], - [0x5E, 'winend', ''], - [0x5F, 'unknown', null], - //[ 0x60, 'npc_crt_V1', null ], - [0x60, 'npc_crt_V3', 'R'], - [0x61, 'npc_stop', 'aR'], - [0x62, 'npc_play', 'aL'], - [0x63, 'npc_kill', 'aR'], - [0x64, 'npc_nont', ''], - [0x65, 'npc_talk', ''], - //[ 0x66, 'npc_crp_V1', null ], - [0x66, 'npc_crp_V3', 'R'], - [0x67, 'unknown', null], - [0x68, 'create_pipe', 'aL'], - //[ 0x69, 'p_hpstat_V1', null ], - [0x69, 'p_hpstat_V3', 'aRL'], - //[ 0x6A, 'p_dead_V1', null ], - [0x6A, 'p_dead_V3', 'aRL'], - [0x6B, 'p_disablewarp', ''], - [0x6C, 'p_enablewarp', ''], - //[ 0x6D, 'p_move_V1', null ], - [0x6D, 'p_move_V3', 'R'], - [0x6E, 'p_look', 'aL'], - [0x6F, 'unknown', null], - [0x70, 'p_action_disable', ''], - [0x71, 'p_action_enable', ''], - [0x72, 'disable_movement1', 'aR'], - [0x73, 'enable_movement1', 'aR'], - [0x74, 'p_noncol', ''], - [0x75, 'p_col', ''], - [0x76, 'p_setpos', 'aRR'], - [0x77, 'p_return_guild', ''], - [0x78, 'p_talk_guild', 'aL'], - //[ 0x79, 'npc_talk_pl_V1', null ], - [0x79, 'npc_talk_pl_V3', 'R'], - [0x7A, 'npc_talk_kill', 'aL'], - //[ 0x7B, 'npc_crtpk_V1', null ], - [0x7B, 'npc_crtpk_V3', 'R'], - //[ 0x7C, 'npc_crppk_V1', null ], - [0x7C, 'npc_crppk_V3', 'R'], - //[ 0x7D, 'npc_crptalk_v1', null ], - [0x7D, 'npc_crptalk_v3', 'R'], - [0x7E, 'p_look_at_V1', 'aLL'], - //[ 0x7F, 'npc_crp_id_V1', null ], - [0x7F, 'npc_crp_id_V3', 'R'], - [0x80, 'cam_quake', ''], - [0x81, 'cam_adj', ''], - [0x82, 'cam_zmin', ''], - [0x83, 'cam_zmout', ''], - //[ 0x84, 'cam_pan_V1', null ], - [0x84, 'cam_pan_V3', 'R'], - [0x85, 'game_lev_super', ''], - [0x86, 'game_lev_reset', ''], - //[ 0x87, 'pos_pipe_V1', null ], - [0x87, 'pos_pipe_V3', 'R'], - [0x88, 'if_zone_clear', 'RR'], - [0x89, 'chk_ene_num', 'R'], - [0x8A, 'unhide_obj', 'R'], - [0x8B, 'unhide_ene', 'R'], - [0x8C, 'at_coords_call', 'R'], - [0x8D, 'at_coords_talk', 'R'], - [0x8E, 'col_npcin', 'R'], - [0x8F, 'col_npcinr', 'R'], - [0x90, 'switch_on', 'aL'], - [0x91, 'switch_off', 'aL'], - [0x92, 'playbgm_epi', 'aL'], - [0x93, 'set_mainwarp', 'aL'], - [0x94, 'set_obj_param', 'RR'], - [0x95, 'set_floor_handler', 'aLW'], - [0x96, 'clr_floor_handler', 'aL'], - [0x97, 'col_plinaw', 'R'], - [0x98, 'hud_hide', ''], - [0x99, 'hud_show', ''], - [0x9A, 'cine_enable', ''], - [0x9B, 'cine_disable', ''], - [0x9C, 'unknown', null], - [0x9D, 'unknown', null], - [0x9E, 'unknown', null], - [0x9F, 'unknown', null], - [0xA0, 'unknown', null], - [0xA1, 'set_qt_failure', 'W'], - [0xA2, 'set_qt_success', 'W'], - [0xA3, 'clr_qt_failure', ''], - [0xA4, 'clr_qt_success', ''], - [0xA5, 'set_qt_cancel', 'W'], - [0xA6, 'clr_qt_cancel', ''], - [0xA7, 'unknown', null], - //[ 0xA8, 'pl_walk_V1', null ], - [0xA8, 'pl_walk_V3', 'R'], - [0xA9, 'unknown', null], - [0xAA, 'unknown', null], - [0xAB, 'unknown', null], - [0xAC, 'unknown', null], - [0xAD, 'unknown', null], - [0xAE, 'unknown', null], - [0xAF, 'unknown', null], - [0xB0, 'pl_add_meseta', 'aLL'], - [0xB1, 'thread_stg', 'W'], - [0xB2, 'del_obj_param', 'R'], - [0xB3, 'item_create', 'RR'], - [0xB4, 'item_create2', 'RR'], - [0xB5, 'item_delete', 'RR'], - [0xB6, 'item_delete2', 'RR'], - [0xB7, 'item_check', 'RR'], - [0xB8, 'setevt', 'aL'], - [0xB9, 'get_difflvl', 'R'], - [0xBA, 'set_qt_exit', 'W'], - [0xBB, 'clr_qt_exit', ''], - [0xBC, 'unknown', null], - [0xBD, 'unknown', null], - [0xBE, 'unknown', null], - [0xBF, 'unknown', null], - //[ 0xC0, 'particle_V1', null ], - [0xC0, 'particle_V3', 'R'], - [0xC1, 'npc_text', 'aLs'], - [0xC2, 'npc_chkwarp', ''], - [0xC3, 'pl_pkoff', ''], - [0xC4, 'map_designate', 'R'], - [0xC5, 'masterkey_on', ''], - [0xC6, 'masterkey_off', ''], - [0xC7, 'window_time', ''], - [0xC8, 'winend_time', ''], - [0xC9, 'winset_time', 'R'], - [0xCA, 'getmtime', 'R'], - [0xCB, 'set_quest_board_handler', 'aLWs'], - [0xCC, 'clear_quest_board_handler', 'aL'], - //[ 0xCD, 'particle_id_V1', null ], - [0xCD, 'particle_id_V3', 'R'], - //[ 0xCE, 'npc_crptalk_id_V1', null ], - [0xCE, 'npc_crptalk_id_V3', 'R'], - [0xCF, 'npc_lang_clean', ''], - [0xD0, 'pl_pkon', ''], - [0xD1, 'pl_chk_item2', 'RR'], - [0xD2, 'enable_mainmenu', ''], - [0xD3, 'disable_mainmenu', ''], - [0xD4, 'start_battlebgm', ''], - [0xD5, 'end_battlebgm', ''], - [0xD6, 'disp_msg_qb', 'as'], - [0xD7, 'close_msg_qb', ''], - //[ 0xD8, 'set_eventflag_v1', null ], - [0xD8, 'set_eventflag_v3', 'aLL'], - [0xD9, 'sync_leti', null], - [0xDA, 'set_returnhunter', ''], - [0xDB, 'set_returncity', ''], - [0xDC, 'load_pvr', ''], - [0xDD, 'load_midi', ''], - [0xDE, 'unknown', null], - //[ 0xDF, 'npc_param_V1', null ], - [0xDF, 'npc_param_V3', 'aRL'], - [0xE0, 'pad_dragon', ''], - [0xE1, 'clear_mainwarp', 'aL'], - //[ 0xE2, 'pcam_param_V1', null ], - [0xE2, 'pcam_param_V3', 'R'], - //[ 0xE3, 'start_setevt_v1', null ], - [0xE3, 'start_setevt_v3', 'aRL'], - [0xE4, 'warp_on', ''], - [0xE5, 'warp_off', ''], - [0xE6, 'get_slotnumber', 'R'], - [0xE7, 'get_servernumber', 'R'], - [0xE8, 'set_eventflag2', 'aLR'], - [0xE9, 'res', 'RR'], - [0xEA, 'unknownEA', 'RL'], - [0xEB, 'enable_bgmctrl', 'aL'], - [0xEC, 'sw_send', 'R'], - [0xED, 'create_bgmctrl', ''], - [0xEE, 'pl_add_meseta2', 'aL'], - //[ 0xEF, 'sync_let', null ], - [0xEF, 'sync_register', 'aRL'], - [0xF0, 'send_regwork', null], - //[ 0xF1, 'leti_fixed_camera_V1', null ], - [0xF1, 'leti_fixed_camera_V3', 'R'], - [0xF2, 'default_camera_pos1', ''], - [0xF3, 'unknown', null], - [0xF4, 'unknown', null], - [0xF5, 'unknown', null], - [0xF6, 'unknown', null], - [0xF7, 'unknown', null], - [0xF8, 'unknownF8', 'R'], - [0xF9, 'unknown', null], - [0xFA, 'get_gc_number', 'R'], - [0xFB, 'unknownFB', 'W'], - [0xFC, 'unknown', null], - [0xFD, 'unknown', null], - [0xFE, 'unknown', null], - [0xFF, 'unknownFF', ''], -]; - -const F8opcodeList: Array<[number, string, string | null]> = [ - [0x00, 'unknown', null], - [0x01, 'set_chat_callback?', 'aRs'], - [0x02, 'unknown', null], - [0x03, 'unknown', null], - [0x04, 'unknown', null], - [0x05, 'unknown', null], - [0x06, 'unknown', null], - [0x07, 'unknown', null], - [0x08, 'get_difficulty_level2', 'R'], - [0x09, 'get_number_of_player1', 'R'], - [0x0A, 'get_coord_of_player', 'RR'], - [0x0B, 'unknownF80B', ''], - [0x0C, 'unknownF80C', ''], - [0x0D, 'map_designate_ex', 'R'], - [0x0E, 'unknownF80E', 'aL'], - [0x0F, 'unknownF80F', 'aL'], - [0x10, 'ba_initial_floor', 'aL'], - [0x11, 'set_ba_rules', ''], - [0x12, 'unknownF812', 'aL'], - [0x13, 'unknownF813', 'aL'], - [0x14, 'unknownF814', 'aL'], - [0x15, 'unknownF815', 'aL'], - [0x16, 'unknownF816', 'aL'], - [0x17, 'unknownF817', 'aL'], - [0x18, 'unknownF818', 'aL'], - [0x19, 'unknownF819', 'aL'], - [0x1A, 'unknownF81A', 'aL'], - [0x1B, 'unknownF81B', 'aL'], - [0x1C, 'ba_disp_msg', 'as'], - [0x1D, 'death_lvl_up', 'aL'], - [0x1E, 'death_tech_lvl_up', 'aL'], - [0x1F, 'unknown', null], - [0x20, 'cmode_stage', 'aL'], - [0x21, 'unknown', null], - [0x22, 'unknown', null], - [0x23, 'unknownF823', 'aL'], - [0x24, 'unknownF824', 'aL'], - [0x25, 'exp_multiplication', 'R'], - [0x26, 'exp_division?', 'R'], - [0x27, 'get_user_is_dead?', 'R'], - [0x28, 'go_floor', 'RR'], - [0x29, 'unknown', null], - [0x2A, 'unknown', null], - [0x2B, 'unlock_door2', 'aLL'], - [0x2C, 'lock_door2', 'aLL'], - [0x2D, 'if_switch_not_pressed', 'R'], - [0x2E, 'if_switch_pressed', 'R'], - [0x2F, 'unknownF82F', 'aLL'], - [0x30, 'control_dragon', 'R'], - [0x31, 'release_dragon', ''], - [0x32, 'unknown', null], - [0x33, 'unknown', null], - [0x34, 'unknown', null], - [0x35, 'unknown', null], - [0x36, 'unknown', null], - [0x37, 'unknown', null], - [0x38, 'shrink', 'R'], - [0x39, 'unshrink', 'R'], - [0x3A, 'unknown', null], - [0x3B, 'unknown', null], - [0x3C, 'display_clock2?', 'R'], - [0x3D, 'unknownF83D', 'aL'], - [0x3E, 'delete_area_title?', 'aL'], - [0x3F, 'unknown', null], - [0x40, 'load_npc_data', ''], - [0x41, 'get_npc_data', 'W'], - [0x42, 'unknown', null], - [0x43, 'unknown', null], - [0x44, 'unknown', null], - [0x45, 'unknown', null], - [0x46, 'unknown', null], - [0x47, 'unknown', null], - [0x48, 'give_damage_score', 'R'], - [0x49, 'take_damage_score', 'R'], - [0x4A, 'unk_score_F84A', 'R'], - [0x4B, 'unk_score_F84B', 'R'], - [0x4C, 'kill_score', 'R'], - [0x4D, 'death_score', 'R'], - [0x4E, 'unk_score_F84E', 'R'], - [0x4F, 'enemy_death_score', 'R'], - [0x50, 'meseta_score', 'R'], - [0x51, 'unknownF851', 'R'], - [0x52, 'unknownF852', 'aL'], - [0x53, 'reverse_warps', ''], - [0x54, 'unreverse_warps', ''], - [0x55, 'set_ult_map', ''], - [0x56, 'unset_ult_map', ''], - [0x57, 'set_area_title', 'as'], - [0x58, 'unknownF858', ''], - [0x59, 'unknown', null], - [0x5A, 'equip_item', 'R'], - [0x5B, 'unequip_item', 'aLL'], - [0x5C, 'unknown', null], - [0x5D, 'unknown', null], - [0x5E, 'unknownF85E', 'aL'], - [0x5F, 'unknownF85F', 'aL'], - [0x60, 'unknownF860', ''], - [0x61, 'unknownF861', 'aL'], - [0x62, 'unknown', null], - [0x63, 'unknown', null], - [0x64, 'cmode_rank', 'aLs'], - [0x65, 'award_item_name?', ''], - [0x66, 'award_item_select?', ''], - [0x67, 'award_item_give_to?', 'R'], - [0x68, 'unknownF868', 'RR'], - [0x69, 'unknownF869', 'RR'], - [0x6A, 'item_create_cmode', 'RR'], - [0x6B, 'unknownF86B', 'R'], - [0x6C, 'award_item_ok?', 'R'], - [0x6D, 'unknownF86D', ''], - [0x6E, 'unknownF86E', ''], - [0x6F, 'ba_set_lives', 'aL'], - [0x70, 'ba_set_tech_lvl', 'aL'], - [0x71, 'ba_set_lvl', 'aL'], - [0x72, 'ba_set_time_limit', 'aL'], - [0x73, 'boss_is_dead?', 'R'], - [0x74, 'unknown', null], - [0x75, 'unknown', null], - [0x76, 'unknown', null], - [0x77, 'enable_techs', 'R'], - [0x78, 'disable_techs', 'R'], - [0x79, 'get_gender', 'RR'], - [0x7A, 'get_chara_class', 'RR'], - [0x7B, 'take_slot_meseta', 'RR'], - [0x7C, 'unknown', null], - [0x7D, 'unknown', null], - [0x7E, 'unknown', null], - [0x7F, 'read_guildcard_flag', 'RR'], - [0x80, 'unknownF880', 'R'], - [0x81, 'get_pl_name?', 'R'], - [0x82, 'unknown', null], - [0x83, 'unknownF883', 'RR'], - [0x84, 'unknown', null], - [0x85, 'unknown', null], - [0x86, 'unknown', null], - [0x87, 'unknown', null], - [0x88, 'ba_close_msg', ''], - [0x89, 'unknown', null], - [0x8A, 'get_player_status', 'RR'], - [0x8B, 'send_mail', 'aRs'], - [0x8C, 'online_check', 'R'], - [0x8D, 'chl_set_timerecord?', 'R'], - [0x8E, 'chl_get_timerecord?', 'R'], - [0x8F, 'unknownF88F', 'R'], - [0x90, 'unknownF890', ''], - [0x91, 'load_enemy_data', 'aL'], - [0x92, 'get_physical_data', 'W'], - [0x93, 'get_attack_data', 'W'], - [0x94, 'get_resist_data', 'W'], - [0x95, 'get_movement_data', 'W'], - [0x96, 'unknown', null], - [0x97, 'unknown', null], - [0x98, 'shift_left', 'RR'], - [0x99, 'shift_right', 'RR'], - [0x9A, 'get_random', 'RR'], - [0x9B, 'reset_map', ''], - [0x9C, 'disp_chl_retry_menu', 'R'], - [0x9D, 'chl_reverser?', ''], - [0x9E, 'unknownF89E', 'aL'], - [0x9F, 'unknownF89F', 'R'], - [0xA0, 'unknownF8A0', ''], - [0xA1, 'unknownF8A1', ''], - [0xA2, 'unknown', null], - [0xA3, 'unknown', null], - [0xA4, 'unknown', null], - [0xA5, 'unknown', null], - [0xA6, 'unknown', null], - [0xA7, 'unknown', null], - [0xA8, 'unknownF8A8', 'aL'], - [0xA9, 'unknownF8A9', 'R'], - [0xAA, 'unknown', null], - [0xAB, 'unknown', null], - [0xAC, 'unknown', null], - [0xAD, 'get_number_of_player2', 'R'], - [0xAE, 'unknown', null], - [0xAF, 'unknown', null], - [0xB0, 'unknown', null], - [0xB1, 'unknown', null], - [0xB2, 'unknown', null], - [0xB3, 'unknown', null], - [0xB4, 'unknown', null], - [0xB5, 'unknown', null], - [0xB6, 'unknown', null], - [0xB7, 'unknown', null], - [0xB8, 'unknownF8B8', ''], - [0xB9, 'chl_recovery?', ''], - [0xBA, 'unknown', null], - [0xBB, 'unknown', null], - [0xBC, 'set_episode', 'L'], - [0xBD, 'unknown', null], - [0xBE, 'unknown', null], - [0xBF, 'unknown', null], - [0xC0, 'file_dl_req', 'aLs'], - [0xC1, 'get_dl_status', 'R'], - [0xC2, 'gba_unknown4?', ''], - [0xC3, 'get_gba_state?', 'R'], - [0xC4, 'unknownF8C4', 'R'], - [0xC5, 'unknownF8C5', 'R'], - [0xC6, 'QEXIT', ''], - [0xC7, 'use_animation', 'RR'], - [0xC8, 'stop_animation', 'R'], - [0xC9, 'run_to_coord', 'RR'], - [0xCA, 'set_slot_invincible', 'RR'], - [0xCB, 'unknownF8CB', 'R'], - [0xCC, 'set_slot_poison', 'R'], - [0xCD, 'set_slot_paralyze', 'R'], - [0xCE, 'set_slot_shock', 'R'], - [0xCF, 'set_slot_freeze', 'R'], - [0xD0, 'set_slot_slow', 'R'], - [0xD1, 'set_slot_confuse', 'R'], - [0xD2, 'set_slot_shifta', 'R'], - [0xD3, 'set_slot_deband', 'R'], - [0xD4, 'set_slot_jellen', 'R'], - [0xD5, 'set_slot_zalure', 'R'], - [0xD6, 'fleti_fixed_camera', 'aR'], - [0xD7, 'fleti_locked_camera', 'aLR'], - [0xD8, 'default_camera_pos2', ''], - [0xD9, 'set_motion_blur', ''], - [0xDA, 'set_screen_b&w', ''], - [0xDB, 'unknownF8DB', 'aLLLLRW'], - [0xDC, 'NPC_action_string', 'RRW'], - [0xDD, 'get_pad_cond', 'RR'], - [0xDE, 'get_button_cond', 'RR'], - [0xDF, 'freeze_enemies', ''], - [0xE0, 'unfreeze_enemies', ''], - [0xE1, 'freeze_everything', ''], - [0xE2, 'unfreeze_everything', ''], - [0xE3, 'restore_hp', 'R'], - [0xE4, 'restore_tp', 'R'], - [0xE5, 'close_chat_bubble', 'R'], - [0xE6, 'unknownF8E6', 'RR'], - [0xE7, 'unknownF8E7', 'RR'], - [0xE8, 'unknownF8E8', 'RR'], - [0xE9, 'unknownF8E9', 'RR'], - [0xEA, 'unknownF8EA', 'RR'], - [0xEB, 'unknownF8EB', 'RR'], - [0xEC, 'unknownF8EC', 'RR'], - [0xED, 'animation_check', 'RR'], - [0xEE, 'call_image_data', 'aLW'], - [0xEF, 'unknownF8EF', ''], - [0xF0, 'turn_off_bgm_p2', ''], - [0xF1, 'turn_on_bgm_p2', ''], - [0xF2, 'load_unk_data', 'aLLLLRW'], - [0xF3, 'particle2', 'aRLf'], - [0xF4, 'unknown', null], - [0xF5, 'unknown', null], - [0xF6, 'unknown', null], - [0xF7, 'unknown', null], - [0xF8, 'unknown', null], - [0xF9, 'unknown', null], - [0xFA, 'unknown', null], - [0xFB, 'unknown', null], - [0xFC, 'unknown', null], - [0xFD, 'unknown', null], - [0xFE, 'unknown', null], - [0xFF, 'unknown', null], -]; - -const F9opcodeList: Array<[number, string, string | null]> = [ - [0x00, 'unknown', null], - [0x01, 'dec2float', 'RR'], - [0x02, 'float2dec', 'RR'], - [0x03, 'flet', 'RR'], - [0x04, 'fleti', 'RF'], - [0x05, 'unknown', null], - [0x06, 'unknown', null], - [0x07, 'unknown', null], - [0x08, 'fadd', 'RR'], - [0x09, 'faddi', 'RF'], - [0x0A, 'fsub', 'RR'], - [0x0B, 'fsubi', 'RF'], - [0x0C, 'fmul', 'RR'], - [0x0D, 'fmuli', 'RF'], - [0x0E, 'fdiv', 'RR'], - [0x0F, 'fdivi', 'RF'], - [0x10, 'get_unknown_count?', 'aLR'], - [0x11, 'get_stackable_item_count', 'RR'], - [0x12, 'freeze_and_hide_equip', ''], - [0x13, 'thaw_and_show_equip', ''], - [0x14, 'set_paletteX_callback', 'aRW'], - [0x15, 'activate_paletteX', 'aR'], - [0x16, 'enable_paletteX', 'aR'], - [0x17, 'restore_paletteX', 'aL'], - [0x18, 'disable_paletteX', 'aL'], - [0x19, 'get_paletteX_activated', 'aLR'], - [0x1A, 'get_unknown_paletteX_status?', 'aLR'], - [0x1B, 'disable_movement2', 'aR'], - [0x1C, 'enable_movement2', 'aR'], - [0x1D, 'get_time_played', 'R'], - [0x1E, 'get_guildcard_total', 'R'], - [0x1F, 'get_slot_meseta', 'R'], - [0x20, 'get_player_level', 'aLR'], - [0x21, 'get_Section_ID', 'aLR'], - [0x22, 'get_player_hp', 'aRR'], - [0x23, 'get_floor_number', 'aRR'], - [0x24, 'get_coord_player_detect', 'RR'], - [0x25, 'read_global_flag', 'abR'], - [0x26, 'write_global_flag', 'abR'], - [0x27, 'unknownF927', 'RR'], - [0x28, 'floor_player_detect', 'R'], - [0x29, 'read_disk_file?', 'as'], - [0x2A, 'open_pack_select', ''], - [0x2B, 'item_select', 'R'], - [0x2C, 'get_item_id', 'R'], - [0x2D, 'color_change', 'aRRRRR'], - [0x2E, 'send_statistic?', 'aLLLLLLLL'], - [0x2F, 'unknownF92F', 'aLL'], - [0x30, 'chat_box', 'aLLLLLs'], - [0x31, 'chat_bubble', 'aLs'], - [0x32, 'unknown', null], - [0x33, 'unknownF933', 'R'], - [0x34, 'scroll_text', 'aLLLLLfRs'], - [0x35, 'gba_unknown1', ''], - [0x36, 'gba_unknown2', ''], - [0x37, 'gba_unknown3', ''], - [0x38, 'add_damage_to?', 'aLL'], - [0x39, 'item_delete2', 'aL'], - [0x3A, 'get_item_info', 'aLR'], - [0x3B, 'item_packing1', 'aL'], - [0x3C, 'item_packing2', 'aLL'], - [0x3D, 'get_lang_setting?', 'aR'], - [0x3E, 'prepare_statistic?', 'aLWW'], - [0x3F, 'keyword_detect', ''], - [0x40, 'Keyword', 'aRLs'], - [0x41, 'get_guildcard_num', 'aLR'], - [0x42, 'unknown', null], - [0x43, 'unknown', null], - [0x44, 'get_wrap_status', 'aLR'], - [0x45, 'initial_floor', 'aL'], - [0x46, 'sin', 'aRL'], - [0x47, 'cos', 'aRL'], - [0x48, 'unknown', null], - [0x49, 'unknown', null], - [0x4A, 'boss_is_dead2?', 'R'], - [0x4B, 'unknownF94B', 'R'], - [0x4C, 'unknownF94C', 'R'], - [0x4D, 'is_there_cardbattle', 'R'], - [0x4E, 'unknown', null], - [0x4F, 'unknown', null], - [0x50, 'BB_p2_menu', 'aL'], - [0x51, 'BB_Map_Designate', 'BWBB'], - [0x52, 'BB_get_number_in_pack', 'R'], - [0x53, 'BB_swap_item', 'aLLLLLLWW'], - [0x54, 'BB_check_wrap', 'aRR'], - [0x55, 'BB_exchange_PD_item', 'aRRRWW'], - [0x56, 'BB_exchange_PD_srank', 'aRRRRRWW'], - [0x57, 'BB_exchange_PD_special', 'aRRRRRLWW'], - [0x58, 'BB_exchange_PD_percent', 'aRRRRRLWW'], - [0x59, 'unknownF959', 'aL'], - [0x5A, 'unknown', null], - [0x5B, 'unknown', null], - [0x5C, 'BB_exchange_SLT', 'aLRWW'], - [0x5D, 'BB_exchange_PC', ''], - [0x5E, 'BB_box_create_BP', 'aLff'], - [0x5F, 'BB_exchange_PT', 'aRRLWW'], - [0x60, 'unknownF960', 'aL'], - [0x61, 'unknownF961', ''], - [0x62, 'unknown', null], - [0x63, 'unknown', null], - [0x64, 'unknown', null], - [0x65, 'unknown', null], - [0x66, 'unknown', null], - [0x67, 'unknown', null], - [0x68, 'unknown', null], - [0x69, 'unknown', null], - [0x6A, 'unknown', null], - [0x6B, 'unknown', null], - [0x6C, 'unknown', null], - [0x6D, 'unknown', null], - [0x6E, 'unknown', null], - [0x6F, 'unknown', null], - [0x70, 'unknown', null], - [0x71, 'unknown', null], - [0x72, 'unknown', null], - [0x73, 'unknown', null], - [0x74, 'unknown', null], - [0x75, 'unknown', null], - [0x76, 'unknown', null], - [0x77, 'unknown', null], - [0x78, 'unknown', null], - [0x79, 'unknown', null], - [0x7A, 'unknown', null], - [0x7B, 'unknown', null], - [0x7C, 'unknown', null], - [0x7D, 'unknown', null], - [0x7E, 'unknown', null], - [0x7F, 'unknown', null], - [0x80, 'unknown', null], - [0x81, 'unknown', null], - [0x82, 'unknown', null], - [0x83, 'unknown', null], - [0x84, 'unknown', null], - [0x85, 'unknown', null], - [0x86, 'unknown', null], - [0x87, 'unknown', null], - [0x88, 'unknown', null], - [0x89, 'unknown', null], - [0x8A, 'unknown', null], - [0x8B, 'unknown', null], - [0x8C, 'unknown', null], - [0x8D, 'unknown', null], - [0x8E, 'unknown', null], - [0x8F, 'unknown', null], - [0x90, 'unknown', null], - [0x91, 'unknown', null], - [0x92, 'unknown', null], - [0x93, 'unknown', null], - [0x94, 'unknown', null], - [0x95, 'unknown', null], - [0x96, 'unknown', null], - [0x97, 'unknown', null], - [0x98, 'unknown', null], - [0x99, 'unknown', null], - [0x9A, 'unknown', null], - [0x9B, 'unknown', null], - [0x9C, 'unknown', null], - [0x9D, 'unknown', null], - [0x9E, 'unknown', null], - [0x9F, 'unknown', null], - [0xA0, 'unknown', null], - [0xA1, 'unknown', null], - [0xA2, 'unknown', null], - [0xA3, 'unknown', null], - [0xA4, 'unknown', null], - [0xA5, 'unknown', null], - [0xA6, 'unknown', null], - [0xA7, 'unknown', null], - [0xA8, 'unknown', null], - [0xA9, 'unknown', null], - [0xAA, 'unknown', null], - [0xAB, 'unknown', null], - [0xAC, 'unknown', null], - [0xAD, 'unknown', null], - [0xAE, 'unknown', null], - [0xAF, 'unknown', null], - [0xB0, 'unknown', null], - [0xB1, 'unknown', null], - [0xB2, 'unknown', null], - [0xB3, 'unknown', null], - [0xB4, 'unknown', null], - [0xB5, 'unknown', null], - [0xB6, 'unknown', null], - [0xB7, 'unknown', null], - [0xB8, 'unknown', null], - [0xB9, 'unknown', null], - [0xBA, 'unknown', null], - [0xBB, 'unknown', null], - [0xBC, 'unknown', null], - [0xBD, 'unknown', null], - [0xBE, 'unknown', null], - [0xBF, 'unknown', null], - [0xC0, 'unknown', null], - [0xC1, 'unknown', null], - [0xC2, 'unknown', null], - [0xC3, 'unknown', null], - [0xC4, 'unknown', null], - [0xC5, 'unknown', null], - [0xC6, 'unknown', null], - [0xC7, 'unknown', null], - [0xC8, 'unknown', null], - [0xC9, 'unknown', null], - [0xCA, 'unknown', null], - [0xCB, 'unknown', null], - [0xCC, 'unknown', null], - [0xCD, 'unknown', null], - [0xCE, 'unknown', null], - [0xCF, 'unknown', null], - [0xD0, 'unknown', null], - [0xD1, 'unknown', null], - [0xD2, 'unknown', null], - [0xD3, 'unknown', null], - [0xD4, 'unknown', null], - [0xD5, 'unknown', null], - [0xD6, 'unknown', null], - [0xD7, 'unknown', null], - [0xD8, 'unknown', null], - [0xD9, 'unknown', null], - [0xDA, 'unknown', null], - [0xDB, 'unknown', null], - [0xDC, 'unknown', null], - [0xDD, 'unknown', null], - [0xDE, 'unknown', null], - [0xDF, 'unknown', null], - [0xE0, 'unknown', null], - [0xE1, 'unknown', null], - [0xE2, 'unknown', null], - [0xE3, 'unknown', null], - [0xE4, 'unknown', null], - [0xE5, 'unknown', null], - [0xE6, 'unknown', null], - [0xE7, 'unknown', null], - [0xE8, 'unknown', null], - [0xE9, 'unknown', null], - [0xEA, 'unknown', null], - [0xEB, 'unknown', null], - [0xEC, 'unknown', null], - [0xED, 'unknown', null], - [0xEE, 'unknown', null], - [0xEF, 'unknown', null], - [0xF0, 'unknown', null], - [0xF1, 'unknown', null], - [0xF2, 'unknown', null], - [0xF3, 'unknown', null], - [0xF4, 'unknown', null], - [0xF5, 'unknown', null], - [0xF6, 'unknown', null], - [0xF7, 'unknown', null], - [0xF8, 'unknown', null], - [0xF9, 'unknown', null], - [0xFA, 'unknown', null], - [0xFB, 'unknown', null], - [0xFC, 'unknown', null], - [0xFD, 'unknown', null], - [0xFE, 'unknown', null], - [0xFF, 'unknown', null], -]; \ No newline at end of file diff --git a/src/bin-data/parsing/dat.test.ts b/src/bin-data/parsing/dat.test.ts deleted file mode 100644 index b9ed6383..00000000 --- a/src/bin-data/parsing/dat.test.ts +++ /dev/null @@ -1,62 +0,0 @@ -import * as fs from 'fs'; -import { ArrayBufferCursor } from '../ArrayBufferCursor'; -import * as prs from '../compression/prs'; -import { parseDat, writeDat } from './dat'; - -/** - * Parse a file, convert the resulting structure to DAT again and check whether the end result is equal to the original. - */ -test('parseDat and writeDat', () => { - const origBuffer = fs.readFileSync('test/resources/quest118_e.dat').buffer; - const origDat = prs.decompress(new ArrayBufferCursor(origBuffer, true)); - const testDat = writeDat(parseDat(origDat)); - origDat.seekStart(0); - - expect(testDat.size).toBe(origDat.size); - - let match = true; - - while (origDat.bytesLeft) { - if (testDat.u8() !== origDat.u8()) { - match = false; - break; - } - } - - expect(match).toBe(true); -}); - -/** - * Parse a file, modify the resulting structure, convert it to DAT again and check whether the end result is equal to the original except for the bytes that should be changed. - */ -test('parse, modify and write DAT', () => { - const origBuffer = fs.readFileSync('./test/resources/quest118_e.dat').buffer; - const origDat = prs.decompress(new ArrayBufferCursor(origBuffer, true)); - const testParsed = parseDat(origDat); - origDat.seekStart(0); - - testParsed.objs[9].position.x = 13; - testParsed.objs[9].position.y = 17; - testParsed.objs[9].position.z = 19; - - const testDat = writeDat(testParsed); - - expect(testDat.size).toBe(origDat.size); - - let match = true; - - while (origDat.bytesLeft) { - if (origDat.position === 16 + 9 * 68 + 16) { - origDat.seek(12); - - expect(testDat.f32()).toBe(13); - expect(testDat.f32()).toBe(17); - expect(testDat.f32()).toBe(19); - } else if (testDat.u8() !== origDat.u8()) { - match = false; - break; - } - } - - expect(match).toBe(true); -}); diff --git a/src/bin-data/parsing/dat.ts b/src/bin-data/parsing/dat.ts deleted file mode 100644 index 3d5024b3..00000000 --- a/src/bin-data/parsing/dat.ts +++ /dev/null @@ -1,220 +0,0 @@ -import { groupBy } from 'lodash'; -import { ArrayBufferCursor } from '../ArrayBufferCursor'; - -const OBJECT_SIZE = 68; -const NPC_SIZE = 72; - -export interface DatFile { - objs: DatObject[]; - npcs: DatNpc[]; - unknowns: DatUnknown[]; -} - -interface DatEntity { - typeId: number; - sectionId: number; - position: { x: number, y: number, z: number }; - rotation: { x: number, y: number, z: number }; - areaId: number; - unknown: number[][]; -} - -export interface DatObject extends DatEntity { -} - -export interface DatNpc extends DatEntity { - skin: number; -} - -export interface DatUnknown { - entityType: number; - totalSize: number; - areaId: number; - entitiesSize: number; - data: number[]; -} - -export function parseDat(cursor: ArrayBufferCursor): DatFile { - const objs: DatObject[] = []; - const npcs: DatNpc[] = []; - const unknowns: DatUnknown[] = []; - - while (cursor.bytesLeft) { - const entityType = cursor.u32(); - const totalSize = cursor.u32(); - const areaId = cursor.u32(); - const entitiesSize = cursor.u32(); - - if (entityType === 0) { - break; - } else { - if (entitiesSize !== totalSize - 16) { - throw Error(`Malformed DAT file. Expected an entities size of ${totalSize - 16}, got ${entitiesSize}.`); - } - - if (entityType === 1) { // Objects - const objectCount = Math.floor(entitiesSize / OBJECT_SIZE); - const startPosition = cursor.position; - - for (let i = 0; i < objectCount; ++i) { - const typeId = cursor.u16(); - const unknown1 = cursor.u8Array(10); - const sectionId = cursor.u16(); - const unknown2 = cursor.u8Array(2); - const x = cursor.f32(); - const y = cursor.f32(); - const z = cursor.f32(); - const rotationX = cursor.i32() / 0xFFFF * 2 * Math.PI; - const rotationY = cursor.i32() / 0xFFFF * 2 * Math.PI; - const rotationZ = cursor.i32() / 0xFFFF * 2 * Math.PI; - // The next 3 floats seem to be scale values. - const unknown3 = cursor.u8Array(28); - - objs.push({ - typeId, - sectionId, - position: { x, y, z }, - rotation: { x: rotationX, y: rotationY, z: rotationZ }, - areaId, - unknown: [unknown1, unknown2, unknown3] - }); - } - - const bytesRead = cursor.position - startPosition; - - if (bytesRead !== entitiesSize) { - console.warn(`Read ${bytesRead} bytes instead of expected ${entitiesSize} for entity type ${entityType} (Object).`); - cursor.seek(entitiesSize - bytesRead); - } - } else if (entityType === 2) { // NPCs - const npcCount = Math.floor(entitiesSize / NPC_SIZE); - const startPosition = cursor.position; - - for (let i = 0; i < npcCount; ++i) { - const typeId = cursor.u16(); - const unknown1 = cursor.u8Array(10); - const sectionId = cursor.u16(); - const unknown2 = cursor.u8Array(6); - const x = cursor.f32(); - const y = cursor.f32(); - const z = cursor.f32(); - const rotationX = cursor.i32() / 0xFFFF * 2 * Math.PI; - const rotationY = cursor.i32() / 0xFFFF * 2 * Math.PI; - const rotationZ = cursor.i32() / 0xFFFF * 2 * Math.PI; - const unknown3 = cursor.u8Array(20); - const skin = cursor.u32(); - const unknown4 = cursor.u8Array(4); - - npcs.push({ - typeId, - sectionId, - position: { x, y, z }, - rotation: { x: rotationX, y: rotationY, z: rotationZ }, - skin, - areaId, - unknown: [unknown1, unknown2, unknown3, unknown4] - }); - } - - const bytesRead = cursor.position - startPosition; - - if (bytesRead !== entitiesSize) { - console.warn(`Read ${bytesRead} bytes instead of expected ${entitiesSize} for entity type ${entityType} (NPC).`); - cursor.seek(entitiesSize - bytesRead); - } - } else { - // There are also waves (type 3) and unknown entity types 4 and 5. - unknowns.push({ - entityType, - totalSize, - areaId, - entitiesSize, - data: cursor.u8Array(entitiesSize) - }); - } - } - } - - return { objs, npcs, unknowns }; -} - -export function writeDat({ objs, npcs, unknowns }: DatFile): ArrayBufferCursor { - const cursor = new ArrayBufferCursor( - objs.length * OBJECT_SIZE + npcs.length * NPC_SIZE + unknowns.length * 1000, true); - - const groupedObjs = groupBy(objs, obj => obj.areaId); - const objAreaIds = Object.keys(groupedObjs) - .map(key => parseInt(key, 10)) - .sort((a, b) => a - b); - - for (const areaId of objAreaIds) { - const areaObjs = groupedObjs[areaId]; - const entitiesSize = areaObjs.length * OBJECT_SIZE; - cursor.writeU32(1); // Entity type - cursor.writeU32(entitiesSize + 16); - cursor.writeU32(areaId); - cursor.writeU32(entitiesSize); - - for (const obj of areaObjs) { - cursor.writeU16(obj.typeId); - cursor.writeU8Array(obj.unknown[0]); - cursor.writeU16(obj.sectionId); - cursor.writeU8Array(obj.unknown[1]); - cursor.writeF32(obj.position.x); - cursor.writeF32(obj.position.y); - cursor.writeF32(obj.position.z); - cursor.writeI32(Math.round(obj.rotation.x / (2 * Math.PI) * 0xFFFF)); - cursor.writeI32(Math.round(obj.rotation.y / (2 * Math.PI) * 0xFFFF)); - cursor.writeI32(Math.round(obj.rotation.z / (2 * Math.PI) * 0xFFFF)); - cursor.writeU8Array(obj.unknown[2]); - } - } - - const groupedNpcs = groupBy(npcs, npc => npc.areaId); - const npcAreaIds = Object.keys(groupedNpcs) - .map(key => parseInt(key, 10)) - .sort((a, b) => a - b); - - for (const areaId of npcAreaIds) { - const areaNpcs = groupedNpcs[areaId]; - const entitiesSize = areaNpcs.length * NPC_SIZE; - cursor.writeU32(2); // Entity type - cursor.writeU32(entitiesSize + 16); - cursor.writeU32(areaId); - cursor.writeU32(entitiesSize); - - for (const npc of areaNpcs) { - cursor.writeU16(npc.typeId); - cursor.writeU8Array(npc.unknown[0]); - cursor.writeU16(npc.sectionId); - cursor.writeU8Array(npc.unknown[1]); - cursor.writeF32(npc.position.x); - cursor.writeF32(npc.position.y); - cursor.writeF32(npc.position.z); - cursor.writeI32(Math.round(npc.rotation.x / (2 * Math.PI) * 0xFFFF)); - cursor.writeI32(Math.round(npc.rotation.y / (2 * Math.PI) * 0xFFFF)); - cursor.writeI32(Math.round(npc.rotation.z / (2 * Math.PI) * 0xFFFF)); - cursor.writeU8Array(npc.unknown[2]); - cursor.writeU32(npc.skin); - cursor.writeU8Array(npc.unknown[3]); - } - } - - for (const unknown of unknowns) { - cursor.writeU32(unknown.entityType); - cursor.writeU32(unknown.totalSize); - cursor.writeU32(unknown.areaId); - cursor.writeU32(unknown.entitiesSize); - cursor.writeU8Array(unknown.data); - } - - // Final header. - cursor.writeU32(0); - cursor.writeU32(0); - cursor.writeU32(0); - cursor.writeU32(0); - - cursor.seekStart(0); - - return cursor; -} diff --git a/src/bin-data/parsing/geometry.ts b/src/bin-data/parsing/geometry.ts deleted file mode 100644 index 1b86bfba..00000000 --- a/src/bin-data/parsing/geometry.ts +++ /dev/null @@ -1,374 +0,0 @@ -import { - BufferAttribute, - BufferGeometry, - DoubleSide, - Face3, - Geometry, - Mesh, - MeshBasicMaterial, - MeshLambertMaterial, - Object3D, - TriangleStripDrawMode, - Vector3 -} from 'three'; -import { Vec3, Section } from '../../domain'; - -export function parseCRel(arrayBuffer: ArrayBuffer): Object3D { - const dv = new DataView(arrayBuffer); - - const object = new Object3D(); - const materials = [ - // Wall - new MeshBasicMaterial({ - color: 0x80C0D0, - transparent: true, - opacity: 0.25 - }), - // Ground - new MeshLambertMaterial({ - color: 0x50D0D0, - side: DoubleSide - }), - // Vegetation - new MeshLambertMaterial({ - color: 0x50B070, - side: DoubleSide - }), - // Section transition zone - new MeshLambertMaterial({ - color: 0x604080, - side: DoubleSide - }) - ]; - const wireframeMaterials = [ - // Wall - new MeshBasicMaterial({ - color: 0x90D0E0, - wireframe: true, - transparent: true, - opacity: 0.3, - }), - // Ground - new MeshBasicMaterial({ - color: 0x60F0F0, - wireframe: true - }), - // Vegetation - new MeshBasicMaterial({ - color: 0x60C080, - wireframe: true - }), - // Section transition zone - new MeshBasicMaterial({ - color: 0x705090, - wireframe: true - }) - ]; - - const mainBlockOffset = dv.getUint32(dv.byteLength - 16, true); - const mainOffsetTableOffset = dv.getUint32(mainBlockOffset, true); - - for ( - let i = mainOffsetTableOffset; - i === mainOffsetTableOffset || dv.getUint32(i) !== 0; - i += 24 - ) { - const blockGeometry = new Geometry(); - - const blockTrailerOffset = dv.getUint32(i, true); - const vertexCount = dv.getUint32(blockTrailerOffset, true); - const vertexTableOffset = dv.getUint32(blockTrailerOffset + 4, true); - const vertexTableEnd = vertexTableOffset + 12 * vertexCount; - const triangleCount = dv.getUint32(blockTrailerOffset + 8, true); - const triangleTableOffset = dv.getUint32(blockTrailerOffset + 12, true); - const triangleTableEnd = triangleTableOffset + 36 * triangleCount; - - for (let j = vertexTableOffset; j < vertexTableEnd; j += 12) { - const x = dv.getFloat32(j, true); - const y = dv.getFloat32(j + 4, true); - const z = dv.getFloat32(j + 8, true); - - blockGeometry.vertices.push(new Vector3(x, y, z)); - } - - for (let j = triangleTableOffset; j < triangleTableEnd; j += 36) { - const v1 = dv.getUint16(j, true); - const v2 = dv.getUint16(j + 2, true); - const v3 = dv.getUint16(j + 4, true); - const flags = dv.getUint16(j + 6, true); - const n = new Vector3( - dv.getFloat32(j + 8, true), - dv.getFloat32(j + 12, true), - dv.getFloat32(j + 16, true) - ); - const isSectionTransition = flags & 0b1000000; - const isVegetation = flags & 0b10000; - const isGround = flags & 0b1; - const colorIndex = isSectionTransition ? 3 : (isVegetation ? 2 : (isGround ? 1 : 0)); - - blockGeometry.faces.push(new Face3(v1, v2, v3, n, undefined, colorIndex)); - } - - const mesh = new Mesh(blockGeometry, materials); - mesh.renderOrder = 1; - object.add(mesh); - - const wireframeMesh = new Mesh(blockGeometry, wireframeMaterials); - wireframeMesh.renderOrder = 2; - object.add(wireframeMesh); - } - - return object; -} - -export function parseNRel( - arrayBuffer: ArrayBuffer -): { sections: Section[], object3d: Object3D } { - const dv = new DataView(arrayBuffer); - const sections = new Map(); - - const object = new Object3D(); - - const mainBlockOffset = dv.getUint32(dv.byteLength - 16, true); - const sectionCount = dv.getUint32(mainBlockOffset + 8, true); - const sectionTableOffset = dv.getUint32(mainBlockOffset + 16, true); - // const textureNameOffset = dv.getUint32(mainBlockOffset + 20, true); - - for ( - let i = sectionTableOffset; - i < sectionTableOffset + sectionCount * 52; - i += 52 - ) { - const sectionId = dv.getInt32(i, true); - const sectionX = dv.getFloat32(i + 4, true); - const sectionY = dv.getFloat32(i + 8, true); - const sectionZ = dv.getFloat32(i + 12, true); - const sectionRotation = dv.getInt32(i + 20, true) / 0xFFFF * 2 * Math.PI; - const section = new Section( - sectionId, - new Vec3(sectionX, sectionY, sectionZ), - sectionRotation - ); - sections.set(sectionId, section); - - const indexListsList = []; - const positionListsList = []; - const normalListsList = []; - - const simpleGeometryOffsetTableOffset = dv.getUint32(i + 32, true); - // const complexGeometryOffsetTableOffset = dv.getUint32(i + 36, true); - const simpleGeometryOffsetCount = dv.getUint32(i + 40, true); - // const complexGeometryOffsetCount = dv.getUint32(i + 44, true); - - // console.log(`section id: ${sectionId}, section rotation: ${sectionRotation}, simple vertices: ${simpleGeometryOffsetCount}, complex vertices: ${complexGeometryOffsetCount}`); - - for ( - let j = simpleGeometryOffsetTableOffset; - j < simpleGeometryOffsetTableOffset + simpleGeometryOffsetCount * 16; - j += 16 - ) { - let offset = dv.getUint32(j, true); - const flags = dv.getUint32(j + 12, true); - - if (flags & 0b100) { - offset = dv.getUint32(offset, true); - } - - const geometryOffset = dv.getUint32(offset + 4, true); - - if (geometryOffset > 0) { - const vertexInfoTableOffset = dv.getUint32(geometryOffset + 4, true); - const vertexInfoCount = dv.getUint32(geometryOffset + 8, true); - const triangleStripTableOffset = dv.getUint32(geometryOffset + 12, true); - const triangleStripCount = dv.getUint32(geometryOffset + 16, true); - // const transparentObjectTableOffset = dv.getUint32(blockOffset + 20, true); - // const transparentObjectCount = dv.getUint32(blockOffset + 24, true); - - // console.log(`block offset: ${blockOffset}, vertex info count: ${vertexInfoCount}, object table offset ${objectTableOffset}, object count: ${objectCount}, transparent object count: ${transparentObjectCount}`); - - const geomIndexLists = []; - - for ( - let k = triangleStripTableOffset; - k < triangleStripTableOffset + triangleStripCount * 20; - k += 20 - ) { - // const flagAndTextureIdOffset = dv.getUint32(k, true); - // const dataType = dv.getUint32(k + 4, true); - const triangleStripIndexTableOffset = dv.getUint32(k + 8, true); - const triangleStripIndexCount = dv.getUint32(k + 12, true); - - const triangleStripIndices = []; - - for ( - let l = triangleStripIndexTableOffset; - l < triangleStripIndexTableOffset + triangleStripIndexCount * 2; - l += 2 - ) { - triangleStripIndices.push(dv.getUint16(l, true)); - } - - geomIndexLists.push(triangleStripIndices); - - // TODO: Read texture info. - } - - // TODO: Do the previous for the transparent index table. - - // Assume vertexInfoCount == 1. TODO: Does that make sense? - if (vertexInfoCount > 1) { - console.warn(`Vertex info count of ${vertexInfoCount} was larger than expected.`); - } - - // const vertexType = dv.getUint32(vertexInfoTableOffset, true); - const vertexTableOffset = dv.getUint32(vertexInfoTableOffset + 4, true); - const vertexSize = dv.getUint32(vertexInfoTableOffset + 8, true); - const vertexCount = dv.getUint32(vertexInfoTableOffset + 12, true); - - // console.log(`vertex type: ${vertexType}, vertex size: ${vertexSize}, vertex count: ${vertexCount}`); - - const geomPositions = []; - const geomNormals = []; - - for ( - let k = vertexTableOffset; - k < vertexTableOffset + vertexCount * vertexSize; - k += vertexSize - ) { - let nX, nY, nZ; - - switch (vertexSize) { - case 16: - case 24: - // TODO: are these values sensible? - nX = 0; - nY = 1; - nZ = 0; - break; - case 28: - case 36: - nX = dv.getFloat32(k + 12, true); - nY = dv.getFloat32(k + 16, true); - nZ = dv.getFloat32(k + 20, true); - // TODO: color, texture coords. - break; - default: - console.error(`Unexpected vertex size of ${vertexSize}.`); - continue; - } - - const x = dv.getFloat32(k, true); - const y = dv.getFloat32(k + 4, true); - const z = dv.getFloat32(k + 8, true); - const rotatedX = section.cosYAxisRotation * x + section.sinYAxisRotation * z; - const rotatedZ = -section.sinYAxisRotation * x + section.cosYAxisRotation * z; - - geomPositions.push(sectionX + rotatedX); - geomPositions.push(sectionY + y); - geomPositions.push(sectionZ + rotatedZ); - geomNormals.push(nX); - geomNormals.push(nY); - geomNormals.push(nZ); - } - - indexListsList.push(geomIndexLists); - positionListsList.push(geomPositions); - normalListsList.push(geomNormals); - } else { - // console.error(`Block offset at ${offset + 4} was ${blockOffset}.`); - } - } - - // function vEqual(v, w) { - // return v[0] === w[0] && v[1] === w[1] && v[2] === w[2]; - // } - - for (let i = 0; i < positionListsList.length; ++i) { - const positions = positionListsList[i]; - const normals = normalListsList[i]; - const geomIndexLists = indexListsList[i]; - // const indices = []; - - geomIndexLists.forEach(objectIndices => { - // for (let j = 2; j < objectIndices.length; ++j) { - // const a = objectIndices[j - 2]; - // const b = objectIndices[j - 1]; - // const c = objectIndices[j]; - - // if (a !== b && a !== c && b !== c) { - // const ap = positions.slice(3 * a, 3 * a + 3); - // const bp = positions.slice(3 * b, 3 * b + 3); - // const cp = positions.slice(3 * c, 3 * c + 3); - - // if (!vEqual(ap, bp) && !vEqual(ap, cp) && !vEqual(bp, cp)) { - // if (j % 2 === 0) { - // indices.push(a); - // indices.push(b); - // indices.push(c); - // } else { - // indices.push(b); - // indices.push(a); - // indices.push(c); - // } - // } - // } - // } - - const geometry = new BufferGeometry(); - geometry.addAttribute( - 'position', new BufferAttribute(new Float32Array(positions), 3)); - geometry.addAttribute( - 'normal', new BufferAttribute(new Float32Array(normals), 3)); - geometry.setIndex(new BufferAttribute(new Uint16Array(objectIndices), 1)); - - const mesh = new Mesh( - geometry, - new MeshLambertMaterial({ - color: 0x44aaff, - // transparent: true, - opacity: 0.25, - side: DoubleSide - }) - ); - mesh.setDrawMode(TriangleStripDrawMode); - mesh.userData.section = section; - object.add(mesh); - }); - - // const geometry = new BufferGeometry(); - // geometry.addAttribute( - // 'position', new BufferAttribute(new Float32Array(positions), 3)); - // geometry.addAttribute( - // 'normal', new BufferAttribute(new Float32Array(normals), 3)); - // geometry.setIndex(new BufferAttribute(new Uint16Array(indices), 1)); - - // const mesh = new Mesh( - // geometry, - // new MeshLambertMaterial({ - // color: 0x44aaff, - // transparent: true, - // opacity: 0.25, - // side: DoubleSide - // }) - // ); - // object.add(mesh); - - // const wireframeMesh = new Mesh( - // geometry, - // new MeshBasicMaterial({ - // color: 0x88ccff, - // wireframe: true, - // transparent: true, - // opacity: 0.75, - // }) - // ); - // wireframeMesh.setDrawMode(THREE.TriangleStripDrawMode); - // object.add(wireframeMesh); - } - } - - return { - sections: [...sections.values()].sort((a, b) => a.id - b.id), - object3d: object - }; -} diff --git a/src/bin-data/parsing/ninja/index.ts b/src/bin-data/parsing/ninja/index.ts deleted file mode 100644 index 4feb687f..00000000 --- a/src/bin-data/parsing/ninja/index.ts +++ /dev/null @@ -1,139 +0,0 @@ -import { - BufferAttribute, - BufferGeometry, - Euler, - Matrix4, - Quaternion, - Vector3 -} from 'three'; -import { ArrayBufferCursor } from '../../ArrayBufferCursor'; -import { parseNjModel, NjContext } from './nj'; -import { parseXjModel, XjContext } from './xj'; - -// TODO: -// - deal with multiple NJCM chunks -// - deal with other types of chunks - -export function parseNj(cursor: ArrayBufferCursor): BufferGeometry | undefined { - return parseNinja(cursor, 'nj'); -} - -export function parseXj(cursor: ArrayBufferCursor): BufferGeometry | undefined { - return parseNinja(cursor, 'xj'); -} - -type Format = 'nj' | 'xj'; -type Context = NjContext | XjContext; - -function parseNinja(cursor: ArrayBufferCursor, format: Format): BufferGeometry | undefined { - while (cursor.bytesLeft) { - // Ninja uses a little endian variant of the IFF format. - // IFF files contain chunks preceded by an 8-byte header. - // The header consists of 4 ASCII characters for the "Type ID" and a 32-bit integer specifying the chunk size. - const iffTypeId = cursor.stringAscii(4, false, false); - const iffChunkSize = cursor.u32(); - - if (iffTypeId === 'NJCM') { - return parseNjcm(cursor.take(iffChunkSize), format); - } else { - cursor.seek(iffChunkSize); - } - } -} - -function parseNjcm(cursor: ArrayBufferCursor, format: Format): BufferGeometry | undefined { - if (cursor.bytesLeft) { - let context: Context; - - if (format === 'nj') { - context = { - format, - positions: [], - normals: [], - cachedChunkOffsets: [], - vertices: [] - }; - } else { - context = { - format, - positions: [], - normals: [], - indices: [] - }; - } - - parseSiblingObjects(cursor, new Matrix4(), context); - return createBufferGeometry(context); - } -} - -function parseSiblingObjects( - cursor: ArrayBufferCursor, - parentMatrix: Matrix4, - context: Context -): void { - const evalFlags = cursor.u32(); - const noTranslate = (evalFlags & 0b1) !== 0; - const noRotate = (evalFlags & 0b10) !== 0; - const noScale = (evalFlags & 0b100) !== 0; - const hidden = (evalFlags & 0b1000) !== 0; - const breakChildTrace = (evalFlags & 0b10000) !== 0; - const zxyRotationOrder = (evalFlags & 0b100000) !== 0; - - const modelOffset = cursor.u32(); - const posX = cursor.f32(); - const posY = cursor.f32(); - const posZ = cursor.f32(); - const rotationX = cursor.i32() * (2 * Math.PI / 0xFFFF); - const rotationY = cursor.i32() * (2 * Math.PI / 0xFFFF); - const rotationZ = cursor.i32() * (2 * Math.PI / 0xFFFF); - const scaleX = cursor.f32(); - const scaleY = cursor.f32(); - const scaleZ = cursor.f32(); - const childOffset = cursor.u32(); - const siblingOffset = cursor.u32(); - - const rotation = new Euler(rotationX, rotationY, rotationZ, zxyRotationOrder ? 'ZXY' : 'ZYX'); - const matrix = new Matrix4() - .compose( - noTranslate ? new Vector3() : new Vector3(posX, posY, posZ), - noRotate ? new Quaternion(0, 0, 0, 1) : new Quaternion().setFromEuler(rotation), - noScale ? new Vector3(1, 1, 1) : new Vector3(scaleX, scaleY, scaleZ) - ) - .premultiply(parentMatrix); - - if (modelOffset && !hidden) { - cursor.seekStart(modelOffset); - parseModel(cursor, matrix, context); - } - - if (childOffset && !breakChildTrace) { - cursor.seekStart(childOffset); - parseSiblingObjects(cursor, matrix, context); - } - - if (siblingOffset) { - cursor.seekStart(siblingOffset); - parseSiblingObjects(cursor, parentMatrix, context); - } -} - -function createBufferGeometry(context: Context): BufferGeometry { - const geometry = new BufferGeometry(); - geometry.addAttribute('position', new BufferAttribute(new Float32Array(context.positions), 3)); - geometry.addAttribute('normal', new BufferAttribute(new Float32Array(context.normals), 3)); - - if ('indices' in context) { - geometry.setIndex(new BufferAttribute(new Uint16Array(context.indices), 1)); - } - - return geometry; -} - -function parseModel(cursor: ArrayBufferCursor, matrix: Matrix4, context: Context): void { - if (context.format === 'nj') { - parseNjModel(cursor, matrix, context); - } else { - parseXjModel(cursor, matrix, context); - } -} diff --git a/src/bin-data/parsing/ninja/nj.ts b/src/bin-data/parsing/ninja/nj.ts deleted file mode 100644 index 4eb305c2..00000000 --- a/src/bin-data/parsing/ninja/nj.ts +++ /dev/null @@ -1,316 +0,0 @@ -import { Matrix3, Matrix4, Vector3 } from 'three'; -import { ArrayBufferCursor } from '../../ArrayBufferCursor'; - -// TODO: -// - deal with multiple NJCM chunks -// - deal with other types of chunks -// - textures -// - colors -// - bump maps -// - animation -// - deal with vertex information contained in triangle strips - -export interface NjContext { - format: 'nj'; - positions: number[]; - normals: number[]; - cachedChunkOffsets: number[]; - vertices: { position: Vector3, normal: Vector3 }[]; -} - -interface Node { - vertices: { position: Vector3, normal: Vector3 }[]; - indices: number[]; - parent?: Node; - children: Node[]; -} - -interface ChunkVertex { - index: number; - position: [number, number, number]; - normal?: [number, number, number]; -} - -interface ChunkTriangleStrip { - clockwiseWinding: boolean; - indices: number[]; -} - -export function parseNjModel(cursor: ArrayBufferCursor, matrix: Matrix4, context: NjContext): void { - const { positions, normals, cachedChunkOffsets, vertices } = context; - - const vlistOffset = cursor.u32(); // Vertex list - const plistOffset = cursor.u32(); // Triangle strip index list - - const normalMatrix = new Matrix3().getNormalMatrix(matrix); - - if (vlistOffset) { - cursor.seekStart(vlistOffset); - - for (const chunk of parseChunks(cursor, cachedChunkOffsets, true)) { - if (chunk.chunkType === 'VERTEX') { - const chunkVertices: ChunkVertex[] = chunk.data; - - for (const vertex of chunkVertices) { - const position = new Vector3(...vertex.position).applyMatrix4(matrix); - const normal = vertex.normal ? new Vector3(...vertex.normal).applyMatrix3(normalMatrix) : new Vector3(0, 1, 0); - vertices[vertex.index] = { position, normal }; - } - } - } - } - - if (plistOffset) { - cursor.seekStart(plistOffset); - - for (const chunk of parseChunks(cursor, cachedChunkOffsets, false)) { - if (chunk.chunkType === 'STRIP') { - for (const { clockwiseWinding, indices: stripIndices } of chunk.data) { - for (let j = 2; j < stripIndices.length; ++j) { - const a = vertices[stripIndices[j - 2]]; - const b = vertices[stripIndices[j - 1]]; - const c = vertices[stripIndices[j]]; - - if (a && b && c) { - if (j % 2 === (clockwiseWinding ? 1 : 0)) { - positions.splice(positions.length, 0, a.position.x, a.position.y, a.position.z); - positions.splice(positions.length, 0, b.position.x, b.position.y, b.position.z); - positions.splice(positions.length, 0, c.position.x, c.position.y, c.position.z); - normals.splice(normals.length, 0, a.normal.x, a.normal.y, a.normal.z); - normals.splice(normals.length, 0, b.normal.x, b.normal.y, b.normal.z); - normals.splice(normals.length, 0, c.normal.x, c.normal.y, c.normal.z); - } else { - positions.splice(positions.length, 0, b.position.x, b.position.y, b.position.z); - positions.splice(positions.length, 0, a.position.x, a.position.y, a.position.z); - positions.splice(positions.length, 0, c.position.x, c.position.y, c.position.z); - normals.splice(normals.length, 0, b.normal.x, b.normal.y, b.normal.z); - normals.splice(normals.length, 0, a.normal.x, a.normal.y, a.normal.z); - normals.splice(normals.length, 0, c.normal.x, c.normal.y, c.normal.z); - } - } - } - } - } - } - } -} - -function parseChunks(cursor: ArrayBufferCursor, cachedChunkOffsets: number[], wideEndChunks: boolean): Array<{ - chunkType: string, - chunkSubType: string | null, - chunkTypeId: number, - data: any -}> { - const chunks = []; - let loop = true; - - while (loop) { - const chunkTypeId = cursor.u8(); - const flags = cursor.u8(); - const chunkStartPosition = cursor.position; - let chunkType = 'UNKOWN'; - let chunkSubType = null; - let data = null; - let size = 0; - - if (chunkTypeId === 0) { - chunkType = 'NULL'; - } else if (1 <= chunkTypeId && chunkTypeId <= 5) { - chunkType = 'BITS'; - - if (chunkTypeId === 4) { - chunkSubType = 'CACHE_POLYGON_LIST'; - data = { - storeIndex: flags, - offset: cursor.position - }; - cachedChunkOffsets[data.storeIndex] = data.offset; - loop = false; - } else if (chunkTypeId === 5) { - chunkSubType = 'DRAW_POLYGON_LIST'; - data = { - storeIndex: flags - }; - cursor.seekStart(cachedChunkOffsets[data.storeIndex]); - chunks.splice(chunks.length, 0, ...parseChunks(cursor, cachedChunkOffsets, wideEndChunks)); - } - } else if (8 <= chunkTypeId && chunkTypeId <= 9) { - chunkType = 'TINY'; - size = 2; - } else if (17 <= chunkTypeId && chunkTypeId <= 31) { - chunkType = 'MATERIAL'; - size = 2 + 2 * cursor.u16(); - } else if (32 <= chunkTypeId && chunkTypeId <= 50) { - chunkType = 'VERTEX'; - size = 2 + 4 * cursor.u16(); - data = parseChunkVertex(cursor, chunkTypeId, flags); - } else if (56 <= chunkTypeId && chunkTypeId <= 58) { - chunkType = 'VOLUME'; - size = 2 + 2 * cursor.u16(); - } else if (64 <= chunkTypeId && chunkTypeId <= 75) { - chunkType = 'STRIP'; - size = 2 + 2 * cursor.u16(); - data = parseChunkTriangleStrip(cursor, chunkTypeId); - } else if (chunkTypeId === 255) { - chunkType = 'END'; - size = wideEndChunks ? 2 : 0; - loop = false; - } else { - // Ignore unknown chunks. - console.warn(`Unknown chunk type: ${chunkTypeId}.`); - size = 2 + 2 * cursor.u16(); - } - - cursor.seekStart(chunkStartPosition + size); - - chunks.push({ - chunkType, - chunkSubType, - chunkTypeId, - data - }); - } - - return chunks; -} - -function parseChunkVertex(cursor: ArrayBufferCursor, chunkTypeId: number, flags: number): ChunkVertex[] { - // There are apparently 4 different sets of vertices, ignore all but set 0. - if ((flags & 0b11) !== 0) { - return []; - } - - const index = cursor.u16(); - const vertexCount = cursor.u16(); - - const vertices: ChunkVertex[] = []; - - for (let i = 0; i < vertexCount; ++i) { - const vertex: ChunkVertex = { - index: index + i, - position: [ - cursor.f32(), // x - cursor.f32(), // y - cursor.f32(), // z - ] - }; - - if (chunkTypeId === 32) { - cursor.seek(4); // Always 1.0 - } else if (chunkTypeId === 33) { - cursor.seek(4); // Always 1.0 - vertex.normal = [ - cursor.f32(), // x - cursor.f32(), // y - cursor.f32(), // z - ]; - cursor.seek(4); // Always 0.0 - } else if (35 <= chunkTypeId && chunkTypeId <= 40) { - if (chunkTypeId === 37) { - // Ninja flags - vertex.index = index + cursor.u16(); - cursor.seek(2); - } else { - // Skip user flags and material information. - cursor.seek(4); - } - } else if (41 <= chunkTypeId && chunkTypeId <= 47) { - vertex.normal = [ - cursor.f32(), // x - cursor.f32(), // y - cursor.f32(), // z - ]; - - if (chunkTypeId >= 42) { - if (chunkTypeId === 44) { - // Ninja flags - vertex.index = index + cursor.u16(); - cursor.seek(2); - } else { - // Skip user flags and material information. - cursor.seek(4); - } - } - } else if (chunkTypeId >= 48) { - // Skip 32-bit vertex normal in format: reserved(2)|x(10)|y(10)|z(10) - cursor.seek(4); - - if (chunkTypeId >= 49) { - // Skip user flags and material information. - cursor.seek(4); - } - } - - vertices.push(vertex); - } - - return vertices; -} - -function parseChunkTriangleStrip(cursor: ArrayBufferCursor, chunkTypeId: number): ChunkTriangleStrip[] { - const userOffsetAndStripCount = cursor.u16(); - const userFlagsSize = userOffsetAndStripCount >>> 14; - const stripCount = userOffsetAndStripCount & 0x3FFF; - let options; - - switch (chunkTypeId) { - case 64: options = [false, false, false, false]; break; - case 65: options = [true, false, false, false]; break; - case 66: options = [true, false, false, false]; break; - case 67: options = [false, false, true, false]; break; - case 68: options = [true, false, true, false]; break; - case 69: options = [true, false, true, false]; break; - case 70: options = [false, true, false, false]; break; - case 71: options = [true, true, false, false]; break; - case 72: options = [true, true, false, false]; break; - case 73: options = [false, false, false, false]; break; - case 74: options = [true, false, false, true]; break; - case 75: options = [true, false, false, true]; break; - default: throw new Error(`Unexpected chunk type ID: ${chunkTypeId}.`); - } - - const [ - parseTextureCoords, - parseColor, - parseNormal, - parseTextureCoordsHires - ] = options; - - const strips = []; - - for (let i = 0; i < stripCount; ++i) { - const windingFlagAndIndexCount = cursor.i16(); - const clockwiseWinding = windingFlagAndIndexCount < 1; - const indexCount = Math.abs(windingFlagAndIndexCount); - - const indices = []; - - for (let j = 0; j < indexCount; ++j) { - indices.push(cursor.u16()); - - if (parseTextureCoords) { - cursor.seek(4); - } - - if (parseColor) { - cursor.seek(4); - } - - if (parseNormal) { - cursor.seek(6); - } - - if (parseTextureCoordsHires) { - cursor.seek(8); - } - - if (j >= 2) { - cursor.seek(2 * userFlagsSize); - } - } - - strips.push({ clockwiseWinding, indices }); - } - - return strips; -} diff --git a/src/bin-data/parsing/ninja/xj.ts b/src/bin-data/parsing/ninja/xj.ts deleted file mode 100644 index e863e818..00000000 --- a/src/bin-data/parsing/ninja/xj.ts +++ /dev/null @@ -1,172 +0,0 @@ -import { Matrix3, Matrix4, Vector3 } from 'three'; -import { ArrayBufferCursor } from '../../ArrayBufferCursor'; - -// TODO: -// - textures -// - colors -// - bump maps -// - animation - -export interface XjContext { - format: 'xj'; - positions: number[]; - normals: number[]; - indices: number[]; -} - -export function parseXjModel(cursor: ArrayBufferCursor, matrix: Matrix4, context: XjContext): void { - const { positions, normals, indices } = context; - - cursor.seek(4); // Flags according to QEdit, seemingly always 0. - const vertexInfoListOffset = cursor.u32(); - cursor.seek(4); // Seems to be the vertexInfoCount, always 1. - const triangleStripListAOffset = cursor.u32(); - const triangleStripACount = cursor.u32(); - const triangleStripListBOffset = cursor.u32(); - const triangleStripBCount = cursor.u32(); - cursor.seek(16); // Bounding sphere position and radius in floats. - - const normalMatrix = new Matrix3().getNormalMatrix(matrix); - const indexOffset = positions.length / 3; - - if (vertexInfoListOffset) { - cursor.seekStart(vertexInfoListOffset); - cursor.seek(4); // Possibly the vertex type. - const vertexListOffset = cursor.u32(); - const vertexSize = cursor.u32(); - const vertexCount = cursor.u32(); - - for (let i = 0; i < vertexCount; ++i) { - cursor.seekStart(vertexListOffset + i * vertexSize); - const position = new Vector3( - cursor.f32(), - cursor.f32(), - cursor.f32() - ).applyMatrix4(matrix); - let normal; - - if (vertexSize === 28 || vertexSize === 32 || vertexSize === 36) { - normal = new Vector3( - cursor.f32(), - cursor.f32(), - cursor.f32() - ).applyMatrix3(normalMatrix); - } else { - normal = new Vector3(0, 1, 0); - } - - positions.push(position.x); - positions.push(position.y); - positions.push(position.z); - normals.push(normal.x); - normals.push(normal.y); - normals.push(normal.z); - } - } - - if (triangleStripListAOffset) { - parseTriangleStripList( - cursor, - triangleStripListAOffset, - triangleStripACount, - positions, - normals, - indices, - indexOffset - ); - } - - if (triangleStripListBOffset) { - parseTriangleStripList( - cursor, - triangleStripListBOffset, - triangleStripBCount, - positions, - normals, - indices, - indexOffset - ); - } -} - -function parseTriangleStripList( - cursor: ArrayBufferCursor, - triangleStripListOffset: number, - triangleStripCount: number, - positions: number[], - normals: number[], - indices: number[], - indexOffset: number -): void { - for (let i = 0; i < triangleStripCount; ++i) { - cursor.seekStart(triangleStripListOffset + i * 20); - cursor.seek(8); // Skip material information. - const indexListOffset = cursor.u32(); - const indexCount = cursor.u32(); - // Ignoring 4 bytes. - - cursor.seekStart(indexListOffset); - const stripIndices = cursor.u16Array(indexCount); - let clockwise = true; - - for (let j = 2; j < stripIndices.length; ++j) { - const a = indexOffset + stripIndices[j - 2]; - const b = indexOffset + stripIndices[j - 1]; - const c = indexOffset + stripIndices[j]; - const pa = new Vector3(positions[3 * a], positions[3 * a + 1], positions[3 * a + 2]); - const pb = new Vector3(positions[3 * b], positions[3 * b + 1], positions[3 * b + 2]); - const pc = new Vector3(positions[3 * c], positions[3 * c + 1], positions[3 * c + 2]); - const na = new Vector3(normals[3 * a], normals[3 * a + 1], normals[3 * a + 2]); - const nb = new Vector3(normals[3 * a], normals[3 * a + 1], normals[3 * a + 2]); - const nc = new Vector3(normals[3 * a], normals[3 * a + 1], normals[3 * a + 2]); - - // Calculate a surface normal and reverse the vertex winding if at least 2 of the vertex normals point in the opposite direction. - // This hack fixes the winding for most models. - const normal = pb.clone().sub(pa).cross(pc.clone().sub(pa)); - - if (clockwise) { - normal.negate(); - } - - const oppositeCount = - (normal.dot(na) < 0 ? 1 : 0) + - (normal.dot(nb) < 0 ? 1 : 0) + - (normal.dot(nc) < 0 ? 1 : 0); - - if (oppositeCount >= 2) { - clockwise = !clockwise; - } - - if (clockwise) { - indices.push(b); - indices.push(a); - indices.push(c); - } else { - indices.push(a); - indices.push(b); - indices.push(c); - } - - clockwise = !clockwise; - - // The following switch statement fixes model 180.xj (zanba). - // switch (j) { - // case 17: - // case 52: - // case 70: - // case 92: - // case 97: - // case 126: - // case 140: - // case 148: - // case 187: - // case 200: - // console.warn(`swapping winding at: ${j}, (${a}, ${b}, ${c})`); - // break; - // default: - // ccw = !ccw; - // break; - // } - } - } -} diff --git a/src/bin-data/parsing/qst.test.ts b/src/bin-data/parsing/qst.test.ts deleted file mode 100644 index 3381b9a9..00000000 --- a/src/bin-data/parsing/qst.test.ts +++ /dev/null @@ -1,31 +0,0 @@ -import { ArrayBufferCursor } from '../ArrayBufferCursor'; -import { parseQst, writeQst } from './qst'; -import { walkQstFiles } from '../../../test/src/utils'; - -/** - * Parse a file, convert the resulting structure to QST again and check whether the end result is equal to the original. - */ -test('parseQst and writeQst', () => { - walkQstFiles((_filePath, _fileName, fileContent) => { - const origQst = new ArrayBufferCursor(fileContent.buffer, true); - const origQuest = parseQst(origQst); - - if (origQuest) { - const testQst = writeQst(origQuest); - origQst.seekStart(0); - - expect(testQst.size).toBe(origQst.size); - - let match = true; - - while (origQst.bytesLeft) { - if (testQst.u8() !== origQst.u8()) { - match = false; - break; - } - } - - expect(match).toBe(true); - } - }); -}); diff --git a/src/bin-data/parsing/qst.ts b/src/bin-data/parsing/qst.ts deleted file mode 100644 index 119d279d..00000000 --- a/src/bin-data/parsing/qst.ts +++ /dev/null @@ -1,291 +0,0 @@ -import { ArrayBufferCursor } from '../ArrayBufferCursor'; - -interface QstContainedFile { - name: string; - name2?: string; // Unsure what this is - questNo?: number; - expectedSize?: number; - data: ArrayBufferCursor; - chunkNos: Set; -} - -interface ParseQstResult { - version: string; - files: QstContainedFile[]; -} - -/** - * Low level parsing function for .qst files. - * Can only read the Blue Burst format. - */ -export function parseQst(cursor: ArrayBufferCursor): ParseQstResult | undefined { - // A .qst file contains two 88-byte headers that describe the embedded .dat and .bin files. - let version = 'PC'; - - // Detect version. - const versionA = cursor.u8(); - cursor.seek(1); - const versionB = cursor.u8(); - - if (versionA === 0x44) { - version = 'Dreamcast/GameCube'; - } else if (versionA === 0x58) { - if (versionB === 0x44) { - version = 'Blue Burst'; - } - } else if (versionA === 0xA6) { - version = 'Dreamcast download'; - } - - if (version === 'Blue Burst') { - // Read headers and contained files. - cursor.seekStart(0); - - const headers = parseHeaders(cursor); - - const files = parseFiles( - cursor, new Map(headers.map(h => [h.fileName, h.size]))); - - for (const file of files) { - const header = headers.find(h => h.fileName === file.name); - - if (header) { - file.questNo = header.questNo; - file.name2 = header.fileName2; - } - } - - return { - version, - files - }; - } else { - console.error(`Can't parse ${version} QST files.`); - } -} - -interface SimpleQstContainedFile { - name: string; - name2?: string; - questNo?: number; - data: ArrayBufferCursor; -} - -interface WriteQstParams { - version?: string; - files: SimpleQstContainedFile[]; -} - -/** - * Always writes in Blue Burst format. - */ -export function writeQst(params: WriteQstParams): ArrayBufferCursor { - const files = params.files; - const totalSize = files - .map(f => 88 + Math.ceil(f.data.size / 1024) * 1056) - .reduce((a, b) => a + b); - const cursor = new ArrayBufferCursor(totalSize, true); - - writeFileHeaders(cursor, files); - writeFileChunks(cursor, files); - - if (cursor.size !== totalSize) { - throw new Error(`Expected a final file size of ${totalSize}, but got ${cursor.size}.`); - } - - return cursor.seekStart(0); -} - -interface QstHeader { - questNo: number; - fileName: string; - fileName2: string; - size: number; -} - -/** - * TODO: Read all headers instead of just the first 2. - */ -function parseHeaders(cursor: ArrayBufferCursor): QstHeader[] { - const headers = []; - - for (let i = 0; i < 2; ++i) { - cursor.seek(4); - const questNo = cursor.u16(); - cursor.seek(38); - const fileName = cursor.stringAscii(16, true, true); - const size = cursor.u32(); - // Not sure what this is: - const fileName2 = cursor.stringAscii(24, true, true); - - headers.push({ - questNo, - fileName, - fileName2, - size - }); - } - - return headers; -} - -function parseFiles(cursor: ArrayBufferCursor, expectedSizes: Map): QstContainedFile[] { - // Files are interleaved in 1056 byte chunks. - // Each chunk has a 24 byte header, 1024 byte data segment and an 8 byte trailer. - const files = new Map(); - - while (cursor.bytesLeft) { - const startPosition = cursor.position; - - // Read meta data. - const chunkNo = cursor.seek(4).u8(); - const fileName = cursor.seek(3).stringAscii(16, true, true); - - let file = files.get(fileName); - - if (!file) { - const expectedSize = expectedSizes.get(fileName); - files.set(fileName, file = { - name: fileName, - expectedSize, - data: new ArrayBufferCursor(expectedSize || (10 * 1024), true), - chunkNos: new Set() - }); - } - - if (file.chunkNos.has(chunkNo)) { - console.warn(`File chunk number ${chunkNo} of file ${fileName} was already encountered, overwriting previous chunk.`); - } else { - file.chunkNos.add(chunkNo); - } - - // Read file data. - let size = cursor.seek(1024).u32(); - cursor.seek(-1028); - - if (size > 1024) { - console.warn(`Data segment size of ${size} is larger than expected maximum size, reading just 1024 bytes.`); - size = 1024; - } - - const data = cursor.take(size); - const chunkPosition = chunkNo * 1024; - file.data.size = Math.max(chunkPosition + size, file.data.size); - file.data.seekStart(chunkPosition).writeCursor(data); - - // Skip the padding and the trailer. - cursor.seek(1032 - data.size); - - if (cursor.position !== startPosition + 1056) { - throw new Error(`Read ${cursor.position - startPosition} file chunk message bytes instead of expected 1056.`); - } - } - - for (const file of files.values()) { - // Clean up file properties. - file.data.seekStart(0); - file.chunkNos = new Set(Array.from(file.chunkNos.values()).sort((a, b) => a - b)); - - // Check whether the expected size was correct. - if (file.expectedSize != null && file.data.size !== file.expectedSize) { - console.warn(`File ${file.name} has an actual size of ${file.data.size} instead of the expected size ${file.expectedSize}.`); - } - - // Detect missing file chunks. - const actualSize = Math.max(file.data.size, file.expectedSize || 0); - - for (let chunkNo = 0; chunkNo < Math.ceil(actualSize / 1024); ++chunkNo) { - if (!file.chunkNos.has(chunkNo)) { - console.warn(`File ${file.name} is missing chunk ${chunkNo}.`); - } - } - } - - return Array.from(files.values()); -} - -function writeFileHeaders(cursor: ArrayBufferCursor, files: SimpleQstContainedFile[]): void { - for (const file of files) { - if (file.name.length > 16) { - throw Error(`File ${file.name} has a name longer than 16 characters.`); - } - - cursor.writeU16(88); // Header size. - cursor.writeU16(0x44); // Magic number. - cursor.writeU16(file.questNo || 0); - - for (let i = 0; i < 38; ++i) { - cursor.writeU8(0); - } - - cursor.writeStringAscii(file.name, 16); - cursor.writeU32(file.data.size); - - let fileName2: string; - - if (file.name2 == null) { - // Not sure this makes sense. - const dotPos = file.name.lastIndexOf('.'); - fileName2 = dotPos === -1 - ? file.name + '_j' - : file.name.slice(0, dotPos) + '_j' + file.name.slice(dotPos); - } else { - fileName2 = file.name2; - } - - if (fileName2.length > 24) { - throw Error(`File ${file.name} has a fileName2 length (${fileName2}) longer than 24 characters.`); - } - - cursor.writeStringAscii(fileName2, 24); - } -} - -function writeFileChunks(cursor: ArrayBufferCursor, files: SimpleQstContainedFile[]): void { - // Files are interleaved in 1056 byte chunks. - // Each chunk has a 24 byte header, 1024 byte data segment and an 8 byte trailer. - files = files.slice(); - const chunkNos = new Array(files.length).fill(0); - - while (files.length) { - let i = 0; - - while (i < files.length) { - if (!writeFileChunk(cursor, files[i].data, chunkNos[i]++, files[i].name)) { - // Remove if there are no more chunks to write. - files.splice(i, 1); - chunkNos.splice(i, 1); - } else { - ++i; - } - } - } -} - -/** - * @returns true if there are bytes left to write in data, false otherwise. - */ -function writeFileChunk( - cursor: ArrayBufferCursor, - data: ArrayBufferCursor, - chunkNo: number, - name: string -): boolean { - cursor.writeU8Array([28, 4, 19, 0]); - cursor.writeU8(chunkNo); - cursor.writeU8Array([0, 0, 0]); - cursor.writeStringAscii(name, 16); - - const size = Math.min(1024, data.bytesLeft); - cursor.writeCursor(data.take(size)); - - // Padding. - for (let i = size; i < 1024; ++i) { - cursor.writeU8(0); - } - - cursor.writeU32(size); - cursor.writeU32(0); - - return !!data.bytesLeft; -} diff --git a/src/bin-data/parsing/quest.test.ts b/src/bin-data/parsing/quest.test.ts deleted file mode 100644 index 1e330550..00000000 --- a/src/bin-data/parsing/quest.test.ts +++ /dev/null @@ -1,66 +0,0 @@ -import * as fs from 'fs'; -import { ArrayBufferCursor } from '../ArrayBufferCursor'; -import { parseQuest, writeQuestQst } from './quest'; -import { ObjectType, Quest } from '../../domain'; - -test('parse Towards the Future', () => { - const buffer = fs.readFileSync('test/resources/quest118_e.qst').buffer; - const cursor = new ArrayBufferCursor(buffer, true); - const quest = parseQuest(cursor)!; - - expect(quest.name).toBe('Towards the Future'); - expect(quest.shortDescription).toBe('Challenge the\nnew simulator.'); - expect(quest.longDescription).toBe('Client: Principal\nQuest: Wishes to have\nhunters challenge the\nnew simulator\nReward: ??? Meseta'); - expect(quest.episode).toBe(1); - expect(quest.objects.length).toBe(277); - expect(quest.objects[0].type).toBe(ObjectType.MenuActivation); - expect(quest.objects[4].type).toBe(ObjectType.PlayerSet); - expect(quest.npcs.length).toBe(216); - expect(testableAreaVariants(quest)).toEqual([ - [0, 0], [2, 0], [11, 0], [5, 4], [12, 0], [7, 4], [13, 0], [8, 4], [10, 4], [14, 0] - ]); -}); - -/** - * Parse a QST file, write the resulting Quest object to QST again, then parse that again. - * Then check whether the two Quest objects are equal. - */ -test('parseQuest and writeQuestQst', () => { - const buffer = fs.readFileSync('test/resources/tethealla_v0.143_quests/solo/ep1/02.qst').buffer; - const cursor = new ArrayBufferCursor(buffer, true); - const origQuest = parseQuest(cursor)!; - const testQuest = parseQuest(writeQuestQst(origQuest, '02.qst'))!; - - expect(testQuest.name).toBe(origQuest.name); - expect(testQuest.shortDescription).toBe(origQuest.shortDescription); - expect(testQuest.longDescription).toBe(origQuest.longDescription); - expect(testQuest.episode).toBe(origQuest.episode); - expect(testableObjects(testQuest)) - .toEqual(testableObjects(origQuest)); - expect(testableNpcs(testQuest)) - .toEqual(testableNpcs(origQuest)); - expect(testableAreaVariants(testQuest)) - .toEqual(testableAreaVariants(origQuest)); -}); - -function testableObjects(quest: Quest) { - return quest.objects.map(object => [ - object.areaId, - object.sectionId, - object.position, - object.type - ]); -} - -function testableNpcs(quest: Quest) { - return quest.npcs.map(npc => [ - npc.areaId, - npc.sectionId, - npc.position, - npc.type - ]); -} - -function testableAreaVariants(quest: Quest) { - return quest.areaVariants.map(av => [av.area.id, av.id]); -} diff --git a/src/bin-data/parsing/quest.ts b/src/bin-data/parsing/quest.ts deleted file mode 100644 index e9d3d959..00000000 --- a/src/bin-data/parsing/quest.ts +++ /dev/null @@ -1,538 +0,0 @@ -import { ArrayBufferCursor } from '../ArrayBufferCursor'; -import * as prs from '../compression/prs'; -import { parseDat, writeDat, DatObject, DatNpc } from './dat'; -import { parseBin, writeBin, Instruction } from './bin'; -import { parseQst, writeQst } from './qst'; -import { - Vec3, - AreaVariant, - QuestNpc, - QuestObject, - Quest, - ObjectType, - NpcType -} from '../../domain'; -import { areaStore } from '../../stores/AreaStore'; - -/** - * High level parsing function that delegates to lower level parsing functions. - * - * Always delegates to parseQst at the moment. - */ -export function parseQuest(cursor: ArrayBufferCursor): Quest | undefined { - const qst = parseQst(cursor); - - if (!qst) { - return; - } - - let datFile = null; - let binFile = null; - - for (const file of qst.files) { - if (file.name.endsWith('.dat')) { - datFile = file; - } else if (file.name.endsWith('.bin')) { - binFile = file; - } - } - - // TODO: deal with missing/multiple DAT or BIN file. - - if (!datFile) { - console.error('File contains no DAT file.'); - return; - } - - if (!binFile) { - console.error('File contains no BIN file.'); - return; - } - - const dat = parseDat(prs.decompress(datFile.data)); - const bin = parseBin(prs.decompress(binFile.data)); - let episode = 1; - let areaVariants: AreaVariant[] = []; - - if (bin.functionOffsets.length) { - const func0Ops = getFuncOperations(bin.instructions, bin.functionOffsets[0]); - - if (func0Ops) { - episode = getEpisode(func0Ops); - areaVariants = getAreaVariants(episode, func0Ops); - } else { - console.warn(`Function 0 offset ${bin.functionOffsets[0]} is invalid.`); - } - } else { - console.warn('File contains no functions.'); - } - - return new Quest( - bin.questName, - bin.shortDescription, - bin.longDescription, - datFile.questNo, - episode, - areaVariants, - parseObjData(dat.objs), - parseNpcData(episode, dat.npcs), - dat.unknowns, - bin.data - ); -} - -export function writeQuestQst(quest: Quest, fileName: string): ArrayBufferCursor { - const dat = writeDat({ - objs: objectsToDatData(quest.objects), - npcs: npcsToDatData(quest.npcs), - unknowns: quest.datUnkowns - }); - const bin = writeBin({ data: quest.binData }); - const extStart = fileName.lastIndexOf('.'); - const baseFileName = extStart === -1 ? fileName : fileName.slice(0, extStart); - - return writeQst({ - files: [ - { - name: baseFileName + '.dat', - questNo: quest.questNo, - data: prs.compress(dat) - }, - { - name: baseFileName + '.bin', - questNo: quest.questNo, - data: prs.compress(bin) - } - ] - }); -} - -/** - * Defaults to episode I. - */ -function getEpisode(func0Ops: Instruction[]): number { - const setEpisode = func0Ops.find(op => op.mnemonic === 'set_episode'); - - if (setEpisode) { - switch (setEpisode.args[0]) { - default: - case 0: return 1; - case 1: return 2; - case 2: return 4; - } - } else { - console.warn('Function 0 has no set_episode instruction.'); - return 1; - } -} - -function getAreaVariants(episode: number, func0Ops: Instruction[]): AreaVariant[] { - const areaVariants = new Map(); - const bbMaps = func0Ops.filter(op => op.mnemonic === 'BB_Map_Designate'); - - for (const bbMap of bbMaps) { - const areaId = bbMap.args[0]; - const variantId = bbMap.args[2]; - areaVariants.set(areaId, variantId); - } - - // Sort by area order and then variant id. - return ( - Array.from(areaVariants) - .map(([areaId, variantId]) => - areaStore.getVariant(episode, areaId, variantId)) - .sort((a, b) => a.area.order - b.area.order || a.id - b.id) - ); -} - -function getFuncOperations(operations: Instruction[], funcOffset: number) { - let position = 0; - let funcFound = false; - const funcOps: Instruction[] = []; - - for (const operation of operations) { - if (position === funcOffset) { - funcFound = true; - } - - if (funcFound) { - funcOps.push(operation); - - // Break when ret is encountered. - if (operation.opcode === 1) { - break; - } - } - - position += operation.size; - } - - return funcFound ? funcOps : null; -} - -function parseObjData(objs: DatObject[]): QuestObject[] { - return objs.map(objData => { - const { x, y, z } = objData.position; - const rot = objData.rotation; - return new QuestObject( - objData.areaId, - objData.sectionId, - new Vec3(x, y, z), - new Vec3(rot.x, rot.y, rot.z), - ObjectType.fromPsoId(objData.typeId), - objData - ); - }); -} - -function parseNpcData(episode: number, npcs: DatNpc[]): QuestNpc[] { - return npcs.map(npcData => { - const { x, y, z } = npcData.position; - const rot = npcData.rotation; - return new QuestNpc( - npcData.areaId, - npcData.sectionId, - new Vec3(x, y, z), - new Vec3(rot.x, rot.y, rot.z), - getNpcType(episode, npcData), - npcData - ); - }); -} - -// TODO: detect Mothmant, St. Rappy, Hallo Rappy, Egg Rappy, Death Gunner, Bulk and Recon. -function getNpcType(episode: number, { typeId, unknown, skin, areaId }: DatNpc): NpcType { - const regular = (unknown[2][18] & 0x80) === 0; - - switch (`${typeId}, ${skin % 3}, ${episode}`) { - case `${0x044}, 0, 1`: return NpcType.Booma; - case `${0x044}, 1, 1`: return NpcType.Gobooma; - case `${0x044}, 2, 1`: return NpcType.Gigobooma; - - case `${0x063}, 0, 1`: return NpcType.EvilShark; - case `${0x063}, 1, 1`: return NpcType.PalShark; - case `${0x063}, 2, 1`: return NpcType.GuilShark; - - case `${0x0A6}, 0, 1`: return NpcType.Dimenian; - case `${0x0A6}, 0, 2`: return NpcType.Dimenian2; - case `${0x0A6}, 1, 1`: return NpcType.LaDimenian; - case `${0x0A6}, 1, 2`: return NpcType.LaDimenian2; - case `${0x0A6}, 2, 1`: return NpcType.SoDimenian; - case `${0x0A6}, 2, 2`: return NpcType.SoDimenian2; - - case `${0x0D6}, 0, 2`: return NpcType.Mericarol; - case `${0x0D6}, 1, 2`: return NpcType.Mericus; - case `${0x0D6}, 2, 2`: return NpcType.Merikle; - - case `${0x115}, 0, 4`: return NpcType.Boota; - case `${0x115}, 1, 4`: return NpcType.ZeBoota; - case `${0x115}, 2, 4`: return NpcType.BaBoota; - case `${0x117}, 0, 4`: return NpcType.Goran; - case `${0x117}, 1, 4`: return NpcType.PyroGoran; - case `${0x117}, 2, 4`: return NpcType.GoranDetonator; - } - - switch (`${typeId}, ${skin % 2}, ${episode}`) { - case `${0x040}, 0, 1`: return NpcType.Hildebear; - case `${0x040}, 0, 2`: return NpcType.Hildebear2; - case `${0x040}, 1, 1`: return NpcType.Hildeblue; - case `${0x040}, 1, 2`: return NpcType.Hildeblue2; - case `${0x041}, 0, 1`: return NpcType.RagRappy; - case `${0x041}, 0, 2`: return NpcType.RagRappy2; - case `${0x041}, 0, 4`: return NpcType.SandRappy; - case `${0x041}, 1, 1`: return NpcType.AlRappy; - case `${0x041}, 1, 2`: return NpcType.LoveRappy; - case `${0x041}, 1, 4`: return NpcType.DelRappy; - - case `${0x061}, 0, 1`: return areaId > 15 ? NpcType.DelLily : NpcType.PoisonLily; - case `${0x061}, 0, 2`: return areaId > 15 ? NpcType.DelLily : NpcType.PoisonLily2; - case `${0x061}, 1, 1`: return areaId > 15 ? NpcType.DelLily : NpcType.NarLily; - case `${0x061}, 1, 2`: return areaId > 15 ? NpcType.DelLily : NpcType.NarLily2; - - case `${0x080}, 0, 1`: return NpcType.Dubchic; - case `${0x080}, 0, 2`: return NpcType.Dubchic2; - case `${0x080}, 1, 1`: return NpcType.Gilchic; - case `${0x080}, 1, 2`: return NpcType.Gilchic2; - - case `${0x0D4}, 0, 2`: return NpcType.SinowBerill; - case `${0x0D4}, 1, 2`: return NpcType.SinowSpigell; - case `${0x0D5}, 0, 2`: return NpcType.Merillia; - case `${0x0D5}, 1, 2`: return NpcType.Meriltas; - case `${0x0D7}, 0, 2`: return NpcType.UlGibbon; - case `${0x0D7}, 1, 2`: return NpcType.ZolGibbon; - - case `${0x0DD}, 0, 2`: return NpcType.Dolmolm; - case `${0x0DD}, 1, 2`: return NpcType.Dolmdarl; - case `${0x0E0}, 0, 2`: return areaId > 15 ? NpcType.Epsilon : NpcType.SinowZoa; - case `${0x0E0}, 1, 2`: return areaId > 15 ? NpcType.Epsilon : NpcType.SinowZele; - - case `${0x112}, 0, 4`: return NpcType.MerissaA; - case `${0x112}, 1, 4`: return NpcType.MerissaAA; - case `${0x114}, 0, 4`: return NpcType.Zu; - case `${0x114}, 1, 4`: return NpcType.Pazuzu; - case `${0x116}, 0, 4`: return NpcType.Dorphon; - case `${0x116}, 1, 4`: return NpcType.DorphonEclair; - case `${0x119}, 0, 4`: return regular ? NpcType.SaintMilion : NpcType.Kondrieu; - case `${0x119}, 1, 4`: return regular ? NpcType.Shambertin : NpcType.Kondrieu; - } - - switch (`${typeId}, ${episode}`) { - case `${0x042}, 1`: return NpcType.Monest; - case `${0x042}, 2`: return NpcType.Monest2; - case `${0x043}, 1`: return regular ? NpcType.SavageWolf : NpcType.BarbarousWolf; - case `${0x043}, 2`: return regular ? NpcType.SavageWolf2 : NpcType.BarbarousWolf2; - - case `${0x060}, 1`: return NpcType.GrassAssassin; - case `${0x060}, 2`: return NpcType.GrassAssassin2; - case `${0x062}, 1`: return NpcType.NanoDragon; - case `${0x064}, 1`: return regular ? NpcType.PofuillySlime : NpcType.PouillySlime; - case `${0x065}, 1`: return NpcType.PanArms; - case `${0x065}, 2`: return NpcType.PanArms2; - - case `${0x081}, 1`: return NpcType.Garanz; - case `${0x081}, 2`: return NpcType.Garanz2; - case `${0x082}, 1`: return regular ? NpcType.SinowBeat : NpcType.SinowGold; - case `${0x083}, 1`: return NpcType.Canadine; - case `${0x084}, 1`: return NpcType.Canane; - case `${0x085}, 1`: return NpcType.Dubswitch; - case `${0x085}, 2`: return NpcType.Dubswitch2; - - case `${0x0A0}, 1`: return NpcType.Delsaber; - case `${0x0A0}, 2`: return NpcType.Delsaber2; - case `${0x0A1}, 1`: return NpcType.ChaosSorcerer; - case `${0x0A1}, 2`: return NpcType.ChaosSorcerer2; - case `${0x0A2}, 1`: return NpcType.DarkGunner; - case `${0x0A4}, 1`: return NpcType.ChaosBringer; - case `${0x0A5}, 1`: return NpcType.DarkBelra; - case `${0x0A5}, 2`: return NpcType.DarkBelra2; - case `${0x0A7}, 1`: return NpcType.Bulclaw; - case `${0x0A8}, 1`: return NpcType.Claw; - - case `${0x0C0}, 1`: return NpcType.Dragon; - case `${0x0C0}, 2`: return NpcType.GalGryphon; - case `${0x0C1}, 1`: return NpcType.DeRolLe; - // TODO: - // case `${0x0C2}, 1`: return NpcType.VolOptPart1; - case `${0x0C5}, 1`: return NpcType.VolOpt; - case `${0x0C8}, 1`: return NpcType.DarkFalz; - case `${0x0CA}, 2`: return NpcType.OlgaFlow; - case `${0x0CB}, 2`: return NpcType.BarbaRay; - case `${0x0CC}, 2`: return NpcType.GolDragon; - - case `${0x0D8}, 2`: return NpcType.Gibbles; - case `${0x0D9}, 2`: return NpcType.Gee; - case `${0x0DA}, 2`: return NpcType.GiGue; - - case `${0x0DB}, 2`: return NpcType.Deldepth; - case `${0x0DC}, 2`: return NpcType.Delbiter; - case `${0x0DE}, 2`: return NpcType.Morfos; - case `${0x0DF}, 2`: return NpcType.Recobox; - case `${0x0E1}, 2`: return NpcType.IllGill; - - case `${0x110}, 4`: return NpcType.Astark; - case `${0x111}, 4`: return regular ? NpcType.SatelliteLizard : NpcType.Yowie; - case `${0x113}, 4`: return NpcType.Girtablulu; - } - - switch (typeId) { - case 0x004: return NpcType.FemaleFat; - case 0x005: return NpcType.FemaleMacho; - case 0x007: return NpcType.FemaleTall; - case 0x00A: return NpcType.MaleDwarf; - case 0x00B: return NpcType.MaleFat; - case 0x00C: return NpcType.MaleMacho; - case 0x00D: return NpcType.MaleOld; - case 0x019: return NpcType.BlueSoldier; - case 0x01A: return NpcType.RedSoldier; - case 0x01B: return NpcType.Principal; - case 0x01C: return NpcType.Tekker; - case 0x01D: return NpcType.GuildLady; - case 0x01E: return NpcType.Scientist; - case 0x01F: return NpcType.Nurse; - case 0x020: return NpcType.Irene; - case 0x0F1: return NpcType.ItemShop; - case 0x0FE: return NpcType.Nurse2; - } - - return NpcType.Unknown; -} - -function objectsToDatData(objects: QuestObject[]): DatObject[] { - return objects.map(object => ({ - typeId: object.type.psoId!, - sectionId: object.sectionId, - position: object.sectionPosition, - rotation: object.rotation, - areaId: object.areaId, - unknown: object.dat.unknown - })); -} - -function npcsToDatData(npcs: QuestNpc[]): DatNpc[] { - return npcs.map(npc => { - // If the type is unknown, typeData will be null and we use the raw data from the DAT file. - const typeData = npcTypeToDatData(npc.type); - - if (typeData) { - npc.dat.unknown[2][18] = (npc.dat.unknown[2][18] & ~0x80) | (typeData.regular ? 0 : 0x80); - } - - return { - typeId: typeData ? typeData.typeId : npc.dat.typeId, - sectionId: npc.sectionId, - position: npc.sectionPosition, - rotation: npc.rotation, - skin: typeData ? typeData.skin : npc.dat.skin, - areaId: npc.areaId, - unknown: npc.dat.unknown - }; - }); -} - -function npcTypeToDatData( - type: NpcType -): { typeId: number, skin: number, regular: boolean } | null { - switch (type) { - default: throw new Error(`Unexpected type ${type.code}.`); - - case NpcType.Unknown: return null; - - case NpcType.FemaleFat: return { typeId: 0x004, skin: 0, regular: true }; - case NpcType.FemaleMacho: return { typeId: 0x005, skin: 0, regular: true }; - case NpcType.FemaleTall: return { typeId: 0x007, skin: 0, regular: true }; - case NpcType.MaleDwarf: return { typeId: 0x00A, skin: 0, regular: true }; - case NpcType.MaleFat: return { typeId: 0x00B, skin: 0, regular: true }; - case NpcType.MaleMacho: return { typeId: 0x00C, skin: 0, regular: true }; - case NpcType.MaleOld: return { typeId: 0x00D, skin: 0, regular: true }; - case NpcType.BlueSoldier: return { typeId: 0x019, skin: 0, regular: true }; - case NpcType.RedSoldier: return { typeId: 0x01A, skin: 0, regular: true }; - case NpcType.Principal: return { typeId: 0x01B, skin: 0, regular: true }; - case NpcType.Tekker: return { typeId: 0x01C, skin: 0, regular: true }; - case NpcType.GuildLady: return { typeId: 0x01D, skin: 0, regular: true }; - case NpcType.Scientist: return { typeId: 0x01E, skin: 0, regular: true }; - case NpcType.Nurse: return { typeId: 0x01F, skin: 0, regular: true }; - case NpcType.Irene: return { typeId: 0x020, skin: 0, regular: true }; - case NpcType.ItemShop: return { typeId: 0x0F1, skin: 0, regular: true }; - case NpcType.Nurse2: return { typeId: 0x0FE, skin: 0, regular: true }; - - case NpcType.Hildebear: return { typeId: 0x040, skin: 0, regular: true }; - case NpcType.Hildeblue: return { typeId: 0x040, skin: 1, regular: true }; - case NpcType.RagRappy: return { typeId: 0x041, skin: 0, regular: true }; - case NpcType.AlRappy: return { typeId: 0x041, skin: 1, regular: true }; - case NpcType.Monest: return { typeId: 0x042, skin: 0, regular: true }; - case NpcType.SavageWolf: return { typeId: 0x043, skin: 0, regular: true }; - case NpcType.BarbarousWolf: return { typeId: 0x043, skin: 0, regular: false }; - case NpcType.Booma: return { typeId: 0x044, skin: 0, regular: true }; - case NpcType.Gobooma: return { typeId: 0x044, skin: 1, regular: true }; - case NpcType.Gigobooma: return { typeId: 0x044, skin: 2, regular: true }; - case NpcType.Dragon: return { typeId: 0x0C0, skin: 0, regular: true }; - - case NpcType.GrassAssassin: return { typeId: 0x060, skin: 0, regular: true }; - case NpcType.PoisonLily: return { typeId: 0x061, skin: 0, regular: true }; - case NpcType.NarLily: return { typeId: 0x061, skin: 1, regular: true }; - case NpcType.NanoDragon: return { typeId: 0x062, skin: 0, regular: true }; - case NpcType.EvilShark: return { typeId: 0x063, skin: 0, regular: true }; - case NpcType.PalShark: return { typeId: 0x063, skin: 1, regular: true }; - case NpcType.GuilShark: return { typeId: 0x063, skin: 2, regular: true }; - case NpcType.PofuillySlime: return { typeId: 0x064, skin: 0, regular: true }; - case NpcType.PouillySlime: return { typeId: 0x064, skin: 0, regular: false }; - case NpcType.PanArms: return { typeId: 0x065, skin: 0, regular: true }; - case NpcType.DeRolLe: return { typeId: 0x0C1, skin: 0, regular: true }; - - case NpcType.Dubchic: return { typeId: 0x080, skin: 0, regular: true }; - case NpcType.Gilchic: return { typeId: 0x080, skin: 1, regular: true }; - case NpcType.Garanz: return { typeId: 0x081, skin: 0, regular: true }; - case NpcType.SinowBeat: return { typeId: 0x082, skin: 0, regular: true }; - case NpcType.SinowGold: return { typeId: 0x082, skin: 0, regular: false }; - case NpcType.Canadine: return { typeId: 0x083, skin: 0, regular: true }; - case NpcType.Canane: return { typeId: 0x084, skin: 0, regular: true }; - case NpcType.Dubswitch: return { typeId: 0x085, skin: 0, regular: true }; - case NpcType.VolOpt: return { typeId: 0x0C5, skin: 0, regular: true }; - - case NpcType.Delsaber: return { typeId: 0x0A0, skin: 0, regular: true }; - case NpcType.ChaosSorcerer: return { typeId: 0x0A1, skin: 0, regular: true }; - case NpcType.DarkGunner: return { typeId: 0x0A2, skin: 0, regular: true }; - case NpcType.ChaosBringer: return { typeId: 0x0A4, skin: 0, regular: true }; - case NpcType.DarkBelra: return { typeId: 0x0A5, skin: 0, regular: true }; - case NpcType.Dimenian: return { typeId: 0x0A6, skin: 0, regular: true }; - case NpcType.LaDimenian: return { typeId: 0x0A6, skin: 1, regular: true }; - case NpcType.SoDimenian: return { typeId: 0x0A6, skin: 2, regular: true }; - case NpcType.Bulclaw: return { typeId: 0x0A7, skin: 0, regular: true }; - case NpcType.Claw: return { typeId: 0x0A8, skin: 0, regular: true }; - case NpcType.DarkFalz: return { typeId: 0x0C8, skin: 0, regular: true }; - - case NpcType.Hildebear2: return { typeId: 0x040, skin: 0, regular: true }; - case NpcType.Hildeblue2: return { typeId: 0x040, skin: 1, regular: true }; - case NpcType.RagRappy2: return { typeId: 0x041, skin: 0, regular: true }; - case NpcType.LoveRappy: return { typeId: 0x041, skin: 1, regular: true }; - case NpcType.Monest2: return { typeId: 0x042, skin: 0, regular: true }; - case NpcType.PoisonLily2: return { typeId: 0x061, skin: 0, regular: true }; - case NpcType.NarLily2: return { typeId: 0x061, skin: 1, regular: true }; - case NpcType.GrassAssassin2: return { typeId: 0x060, skin: 0, regular: true }; - case NpcType.Dimenian2: return { typeId: 0x0A6, skin: 0, regular: true }; - case NpcType.LaDimenian2: return { typeId: 0x0A6, skin: 1, regular: true }; - case NpcType.SoDimenian2: return { typeId: 0x0A6, skin: 2, regular: true }; - case NpcType.DarkBelra2: return { typeId: 0x0A5, skin: 0, regular: true }; - case NpcType.BarbaRay: return { typeId: 0x0CB, skin: 0, regular: true }; - - case NpcType.SavageWolf2: return { typeId: 0x043, skin: 0, regular: true }; - case NpcType.BarbarousWolf2: return { typeId: 0x043, skin: 0, regular: false }; - case NpcType.PanArms2: return { typeId: 0x065, skin: 0, regular: true }; - case NpcType.Dubchic2: return { typeId: 0x080, skin: 0, regular: true }; - case NpcType.Gilchic2: return { typeId: 0x080, skin: 1, regular: true }; - case NpcType.Garanz2: return { typeId: 0x081, skin: 0, regular: true }; - case NpcType.Dubswitch2: return { typeId: 0x085, skin: 0, regular: true }; - case NpcType.Delsaber2: return { typeId: 0x0A0, skin: 0, regular: true }; - case NpcType.ChaosSorcerer2: return { typeId: 0x0A1, skin: 0, regular: true }; - case NpcType.GolDragon: return { typeId: 0x0CC, skin: 0, regular: true }; - - case NpcType.SinowBerill: return { typeId: 0x0D4, skin: 0, regular: true }; - case NpcType.SinowSpigell: return { typeId: 0x0D4, skin: 1, regular: true }; - case NpcType.Merillia: return { typeId: 0x0D5, skin: 0, regular: true }; - case NpcType.Meriltas: return { typeId: 0x0D5, skin: 1, regular: true }; - case NpcType.Mericarol: return { typeId: 0x0D6, skin: 0, regular: true }; - case NpcType.Mericus: return { typeId: 0x0D6, skin: 1, regular: true }; - case NpcType.Merikle: return { typeId: 0x0D6, skin: 2, regular: true }; - case NpcType.UlGibbon: return { typeId: 0x0D7, skin: 0, regular: true }; - case NpcType.ZolGibbon: return { typeId: 0x0D7, skin: 1, regular: true }; - case NpcType.Gibbles: return { typeId: 0x0D8, skin: 0, regular: true }; - case NpcType.Gee: return { typeId: 0x0D9, skin: 0, regular: true }; - case NpcType.GiGue: return { typeId: 0x0DA, skin: 0, regular: true }; - case NpcType.GalGryphon: return { typeId: 0x0C0, skin: 0, regular: true }; - - case NpcType.Deldepth: return { typeId: 0x0DB, skin: 0, regular: true }; - case NpcType.Delbiter: return { typeId: 0x0DC, skin: 0, regular: true }; - case NpcType.Dolmolm: return { typeId: 0x0DD, skin: 0, regular: true }; - case NpcType.Dolmdarl: return { typeId: 0x0DD, skin: 1, regular: true }; - case NpcType.Morfos: return { typeId: 0x0DE, skin: 0, regular: true }; - case NpcType.Recobox: return { typeId: 0x0DF, skin: 0, regular: true }; - case NpcType.Epsilon: return { typeId: 0x0E0, skin: 0, regular: true }; - case NpcType.SinowZoa: return { typeId: 0x0E0, skin: 0, regular: true }; - case NpcType.SinowZele: return { typeId: 0x0E0, skin: 1, regular: true }; - case NpcType.IllGill: return { typeId: 0x0E1, skin: 0, regular: true }; - case NpcType.DelLily: return { typeId: 0x061, skin: 0, regular: true }; - case NpcType.OlgaFlow: return { typeId: 0x0CA, skin: 0, regular: true }; - - case NpcType.SandRappy: return { typeId: 0x041, skin: 0, regular: true }; - case NpcType.DelRappy: return { typeId: 0x041, skin: 1, regular: true }; - case NpcType.Astark: return { typeId: 0x110, skin: 0, regular: true }; - case NpcType.SatelliteLizard: return { typeId: 0x111, skin: 0, regular: true }; - case NpcType.Yowie: return { typeId: 0x111, skin: 0, regular: false }; - case NpcType.MerissaA: return { typeId: 0x112, skin: 0, regular: true }; - case NpcType.MerissaAA: return { typeId: 0x112, skin: 1, regular: true }; - case NpcType.Girtablulu: return { typeId: 0x113, skin: 0, regular: true }; - case NpcType.Zu: return { typeId: 0x114, skin: 0, regular: true }; - case NpcType.Pazuzu: return { typeId: 0x114, skin: 1, regular: true }; - case NpcType.Boota: return { typeId: 0x115, skin: 0, regular: true }; - case NpcType.ZeBoota: return { typeId: 0x115, skin: 1, regular: true }; - case NpcType.BaBoota: return { typeId: 0x115, skin: 2, regular: true }; - case NpcType.Dorphon: return { typeId: 0x116, skin: 0, regular: true }; - case NpcType.DorphonEclair: return { typeId: 0x116, skin: 1, regular: true }; - case NpcType.Goran: return { typeId: 0x117, skin: 0, regular: true }; - case NpcType.PyroGoran: return { typeId: 0x117, skin: 1, regular: true }; - case NpcType.GoranDetonator: return { typeId: 0x117, skin: 2, regular: true }; - case NpcType.SaintMilion: return { typeId: 0x119, skin: 0, regular: true }; - case NpcType.Shambertin: return { typeId: 0x119, skin: 1, regular: true }; - case NpcType.Kondrieu: return { typeId: 0x119, skin: 0, regular: false }; - } -} diff --git a/src/domain/NpcType.ts b/src/domain/NpcType.ts deleted file mode 100644 index 6f50028b..00000000 --- a/src/domain/NpcType.ts +++ /dev/null @@ -1,636 +0,0 @@ -import { Episode, checkEpisode } from "."; - -export class NpcType { - readonly id: number; - readonly code: string; - /** - * Unique name. E.g. a Delsaber would have (Ep. II) appended to its name. - */ - readonly name: string; - /** - * Name used in the game. - * Might conflict with other NPC names (e.g. Delsaber from ep. I and ep. II). - */ - readonly simpleName: string; - readonly ultimateName: string; - readonly episode?: number; - readonly enemy: boolean; - rareType?: NpcType; - - constructor( - id: number, - code: string, - name: string, - simpleName: string, - ultimateName: string, - episode: number | undefined, - enemy: boolean - ) { - if (!Number.isInteger(id) || id < 1) - throw new Error(`Expected id to be an integer greater than or equal to 1, got ${id}.`); - if (!code) throw new Error('code is required.'); - if (!name) throw new Error('name is required.'); - if (!simpleName) throw new Error('simpleName is required.'); - if (!ultimateName) throw new Error('ultimateName is required.'); - if (episode != null && episode !== 1 && episode !== 2 && episode !== 4) - throw new Error(`episode should be undefined, 1, 2 or 4, got ${episode}.`); - if (typeof enemy !== 'boolean') throw new Error('enemy is required.'); - - this.id = id; - this.code = code; - this.simpleName = simpleName; - this.ultimateName = ultimateName; - this.name = name; - this.episode = episode; - this.enemy = enemy; - - if (episode) { - const map = NpcType.byEpAndName[episode]; - - if (map) { - map.set(simpleName, this); - map.set(ultimateName, this); - } - } - } - - private static byEpAndName = [ - undefined, new Map(), new Map(), undefined, new Map() - ]; - - /** - * Uniquely identifies an NPC. Tries to match on simpleName and ultimateName. - */ - static byNameAndEpisode(name: string, episode: Episode): NpcType | undefined { - checkEpisode(episode); - return this.byEpAndName[episode]!.get(name); - } - - // - // Unknown NPCs - // - - static Unknown: NpcType; - - // - // Friendly NPCs - // - - static FemaleFat: NpcType; - static FemaleMacho: NpcType; - static FemaleTall: NpcType; - static MaleDwarf: NpcType; - static MaleFat: NpcType; - static MaleMacho: NpcType; - static MaleOld: NpcType; - static BlueSoldier: NpcType; - static RedSoldier: NpcType; - static Principal: NpcType; - static Tekker: NpcType; - static GuildLady: NpcType; - static Scientist: NpcType; - static Nurse: NpcType; - static Irene: NpcType; - static ItemShop: NpcType; - static Nurse2: NpcType; - - // - // Enemy NPCs - // - - // Episode I Forest - - static Hildebear: NpcType; - static Hildeblue: NpcType; - static RagRappy: NpcType; - static AlRappy: NpcType; - static Monest: NpcType; - static Mothmant: NpcType; - static SavageWolf: NpcType; - static BarbarousWolf: NpcType; - static Booma: NpcType; - static Gobooma: NpcType; - static Gigobooma: NpcType; - static Dragon: NpcType; - - // Episode I Caves - - static GrassAssassin: NpcType; - static PoisonLily: NpcType; - static NarLily: NpcType; - static NanoDragon: NpcType; - static EvilShark: NpcType; - static PalShark: NpcType; - static GuilShark: NpcType; - static PofuillySlime: NpcType; - static PouillySlime: NpcType; - static PanArms: NpcType; - static Migium: NpcType; - static Hidoom: NpcType; - static DeRolLe: NpcType; - - // Episode I Mines - - static Dubchic: NpcType; - static Gilchic: NpcType; - static Garanz: NpcType; - static SinowBeat: NpcType; - static SinowGold: NpcType; - static Canadine: NpcType; - static Canane: NpcType; - static Dubswitch: NpcType; - static VolOpt: NpcType; - - // Episode I Ruins - - static Delsaber: NpcType; - static ChaosSorcerer: NpcType; - static DarkGunner: NpcType; - static DeathGunner: NpcType; - static ChaosBringer: NpcType; - static DarkBelra: NpcType; - static Dimenian: NpcType; - static LaDimenian: NpcType; - static SoDimenian: NpcType; - static Bulclaw: NpcType; - static Bulk: NpcType; - static Claw: NpcType; - static DarkFalz: NpcType; - - // Episode II VR Temple - - static Hildebear2: NpcType; - static Hildeblue2: NpcType; - static RagRappy2: NpcType; - static LoveRappy: NpcType; - static StRappy: NpcType; - static HalloRappy: NpcType; - static EggRappy: NpcType; - static Monest2: NpcType; - static Mothmant2: NpcType; - static PoisonLily2: NpcType; - static NarLily2: NpcType; - static GrassAssassin2: NpcType; - static Dimenian2: NpcType; - static LaDimenian2: NpcType; - static SoDimenian2: NpcType; - static DarkBelra2: NpcType; - static BarbaRay: NpcType; - - // Episode II VR Spaceship - - static SavageWolf2: NpcType; - static BarbarousWolf2: NpcType; - static PanArms2: NpcType; - static Migium2: NpcType; - static Hidoom2: NpcType; - static Dubchic2: NpcType; - static Gilchic2: NpcType; - static Garanz2: NpcType; - static Dubswitch2: NpcType; - static Delsaber2: NpcType; - static ChaosSorcerer2: NpcType; - static GolDragon: NpcType; - - // Episode II Central Control Area - - static SinowBerill: NpcType; - static SinowSpigell: NpcType; - static Merillia: NpcType; - static Meriltas: NpcType; - static Mericarol: NpcType; - static Mericus: NpcType; - static Merikle: NpcType; - static UlGibbon: NpcType; - static ZolGibbon: NpcType; - static Gibbles: NpcType; - static Gee: NpcType; - static GiGue: NpcType; - static GalGryphon: NpcType; - - // Episode II Seabed - - static Deldepth: NpcType; - static Delbiter: NpcType; - static Dolmolm: NpcType; - static Dolmdarl: NpcType; - static Morfos: NpcType; - static Recobox: NpcType; - static Recon: NpcType; - static Epsilon: NpcType; - static SinowZoa: NpcType; - static SinowZele: NpcType; - static IllGill: NpcType; - static DelLily: NpcType; - static OlgaFlow: NpcType; - - // Episode IV - - static SandRappy: NpcType; - static DelRappy: NpcType; - static Astark: NpcType; - static SatelliteLizard: NpcType; - static Yowie: NpcType; - static MerissaA: NpcType; - static MerissaAA: NpcType; - static Girtablulu: NpcType; - static Zu: NpcType; - static Pazuzu: NpcType; - static Boota: NpcType; - static ZeBoota: NpcType; - static BaBoota: NpcType; - static Dorphon: NpcType; - static DorphonEclair: NpcType; - static Goran: NpcType; - static PyroGoran: NpcType; - static GoranDetonator: NpcType; - static SaintMilion: NpcType; - static Shambertin: NpcType; - static Kondrieu: NpcType; -} - -(function () { - let id = 1; - - // - // Unknown NPCs - // - - NpcType.Unknown = new NpcType(id++, 'Unknown', 'Unknown', 'Unknown', 'Unknown', undefined, false); - - // - // Friendly NPCs - // - - NpcType.FemaleFat = new NpcType(id++, 'FemaleFat', 'Female Fat', 'Female Fat', 'Female Fat', undefined, false); - NpcType.FemaleMacho = new NpcType(id++, 'FemaleMacho', 'Female Macho', 'Female Macho', 'Female Macho', undefined, false); - NpcType.FemaleTall = new NpcType(id++, 'FemaleTall', 'Female Tall', 'Female Tall', 'Female Tall', undefined, false); - NpcType.MaleDwarf = new NpcType(id++, 'MaleDwarf', 'Male Dwarf', 'Male Dwarf', 'Male Dwarf', undefined, false); - NpcType.MaleFat = new NpcType(id++, 'MaleFat', 'Male Fat', 'Male Fat', 'Male Fat', undefined, false); - NpcType.MaleMacho = new NpcType(id++, 'MaleMacho', 'Male Macho', 'Male Macho', 'Male Macho', undefined, false); - NpcType.MaleOld = new NpcType(id++, 'MaleOld', 'Male Old', 'Male Old', 'Male Old', undefined, false); - NpcType.BlueSoldier = new NpcType(id++, 'BlueSoldier', 'Blue Soldier', 'Blue Soldier', 'Blue Soldier', undefined, false); - NpcType.RedSoldier = new NpcType(id++, 'RedSoldier', 'Red Soldier', 'Red Soldier', 'Red Soldier', undefined, false); - NpcType.Principal = new NpcType(id++, 'Principal', 'Principal', 'Principal', 'Principal', undefined, false); - NpcType.Tekker = new NpcType(id++, 'Tekker', 'Tekker', 'Tekker', 'Tekker', undefined, false); - NpcType.GuildLady = new NpcType(id++, 'GuildLady', 'Guild Lady', 'Guild Lady', 'Guild Lady', undefined, false); - NpcType.Scientist = new NpcType(id++, 'Scientist', 'Scientist', 'Scientist', 'Scientist', undefined, false); - NpcType.Nurse = new NpcType(id++, 'Nurse', 'Nurse', 'Nurse', 'Nurse', undefined, false); - NpcType.Irene = new NpcType(id++, 'Irene', 'Irene', 'Irene', 'Irene', undefined, false); - NpcType.ItemShop = new NpcType(id++, 'ItemShop', 'Item Shop', 'Item Shop', 'Item Shop', undefined, false); - NpcType.Nurse2 = new NpcType(id++, 'Nurse2', 'Nurse (Ep. II)', 'Nurse', 'Nurse', 2, false); - - // - // Enemy NPCs - // - - // Episode I Forest - - NpcType.Hildebear = new NpcType(id++, 'Hildebear', 'Hildebear', 'Hildebear', 'Hildelt', 1, true); - NpcType.Hildeblue = new NpcType(id++, 'Hildeblue', 'Hildeblue', 'Hildeblue', 'Hildetorr', 1, true); - NpcType.Hildebear.rareType = NpcType.Hildeblue; - NpcType.RagRappy = new NpcType(id++, 'RagRappy', 'Rag Rappy', 'Rag Rappy', 'El Rappy', 1, true); - NpcType.AlRappy = new NpcType(id++, 'AlRappy', 'Al Rappy', 'Al Rappy', 'Pal Rappy', 1, true); - NpcType.RagRappy.rareType = NpcType.AlRappy; - NpcType.Monest = new NpcType(id++, 'Monest', 'Monest', 'Monest', 'Mothvist', 1, true); - NpcType.Mothmant = new NpcType(id++, 'Mothmant', 'Mothmant', 'Mothmant', 'Mothvert', 1, true); - NpcType.SavageWolf = new NpcType(id++, 'SavageWolf', 'Savage Wolf', 'Savage Wolf', 'Gulgus', 1, true); - NpcType.BarbarousWolf = new NpcType(id++, 'BarbarousWolf', 'Barbarous Wolf', 'Barbarous Wolf', 'Gulgus-Gue', 1, true); - NpcType.Booma = new NpcType(id++, 'Booma', 'Booma', 'Booma', 'Bartle', 1, true); - NpcType.Gobooma = new NpcType(id++, 'Gobooma', 'Gobooma', 'Gobooma', 'Barble', 1, true); - NpcType.Gigobooma = new NpcType(id++, 'Gigobooma', 'Gigobooma', 'Gigobooma', 'Tollaw', 1, true); - NpcType.Dragon = new NpcType(id++, 'Dragon', 'Dragon', 'Dragon', 'Sil Dragon', 1, true); - - // Episode I Caves - - NpcType.GrassAssassin = new NpcType(id++, 'GrassAssassin', 'Grass Assassin', 'Grass Assassin', 'Crimson Assassin', 1, true); - NpcType.PoisonLily = new NpcType(id++, 'PoisonLily', 'Poison Lily', 'Poison Lily', 'Ob Lily', 1, true); - NpcType.NarLily = new NpcType(id++, 'NarLily', 'Nar Lily', 'Nar Lily', 'Mil Lily', 1, true); - NpcType.PoisonLily.rareType = NpcType.NarLily; - NpcType.NanoDragon = new NpcType(id++, 'NanoDragon', 'Nano Dragon', 'Nano Dragon', 'Nano Dragon', 1, true); - NpcType.EvilShark = new NpcType(id++, 'EvilShark', 'Evil Shark', 'Evil Shark', 'Vulmer', 1, true); - NpcType.PalShark = new NpcType(id++, 'PalShark', 'Pal Shark', 'Pal Shark', 'Govulmer', 1, true); - NpcType.GuilShark = new NpcType(id++, 'GuilShark', 'Guil Shark', 'Guil Shark', 'Melqueek', 1, true); - NpcType.PofuillySlime = new NpcType(id++, 'PofuillySlime', 'Pofuilly Slime', 'Pofuilly Slime', 'Pofuilly Slime', 1, true); - NpcType.PouillySlime = new NpcType(id++, 'PouillySlime', 'Pouilly Slime', 'Pouilly Slime', 'Pouilly Slime', 1, true); - NpcType.PofuillySlime.rareType = NpcType.PouillySlime; - NpcType.PanArms = new NpcType(id++, 'PanArms', 'Pan Arms', 'Pan Arms', 'Pan Arms', 1, true); - NpcType.Migium = new NpcType(id++, 'Migium', 'Migium', 'Migium', 'Migium', 1, true); - NpcType.Hidoom = new NpcType(id++, 'Hidoom', 'Hidoom', 'Hidoom', 'Hidoom', 1, true); - NpcType.DeRolLe = new NpcType(id++, 'DeRolLe', 'De Rol Le', 'De Rol Le', 'Dal Ra Lie', 1, true); - - // Episode I Mines - - NpcType.Dubchic = new NpcType(id++, 'Dubchic', 'Dubchic', 'Dubchic', 'Dubchich', 1, true); - NpcType.Gilchic = new NpcType(id++, 'Gilchic', 'Gilchic', 'Gilchic', 'Gilchich', 1, true); - NpcType.Garanz = new NpcType(id++, 'Garanz', 'Garanz', 'Garanz', 'Baranz', 1, true); - NpcType.SinowBeat = new NpcType(id++, 'SinowBeat', 'Sinow Beat', 'Sinow Beat', 'Sinow Blue', 1, true); - NpcType.SinowGold = new NpcType(id++, 'SinowGold', 'Sinow Gold', 'Sinow Gold', 'Sinow Red', 1, true); - NpcType.Canadine = new NpcType(id++, 'Canadine', 'Canadine', 'Canadine', 'Canabin', 1, true); - NpcType.Canane = new NpcType(id++, 'Canane', 'Canane', 'Canane', 'Canune', 1, true); - NpcType.Dubswitch = new NpcType(id++, 'Dubswitch', 'Dubswitch', 'Dubswitch', 'Dubswitch', 1, true); - NpcType.VolOpt = new NpcType(id++, 'VolOpt', 'Vol Opt', 'Vol Opt', 'Vol Opt ver.2', 1, true); - - // Episode I Ruins - - NpcType.Delsaber = new NpcType(id++, 'Delsaber', 'Delsaber', 'Delsaber', 'Delsaber', 1, true); - NpcType.ChaosSorcerer = new NpcType(id++, 'ChaosSorcerer', 'Chaos Sorcerer', 'Chaos Sorcerer', 'Gran Sorcerer', 1, true); - NpcType.DarkGunner = new NpcType(id++, 'DarkGunner', 'Dark Gunner', 'Dark Gunner', 'Dark Gunner', 1, true); - NpcType.DeathGunner = new NpcType(id++, 'DeathGunner', 'Death Gunner', 'Death Gunner', 'Death Gunner', 1, true); - NpcType.ChaosBringer = new NpcType(id++, 'ChaosBringer', 'Chaos Bringer', 'Chaos Bringer', 'Dark Bringer', 1, true); - NpcType.DarkBelra = new NpcType(id++, 'DarkBelra', 'Dark Belra', 'Dark Belra', 'Indi Belra', 1, true); - NpcType.Dimenian = new NpcType(id++, 'Dimenian', 'Dimenian', 'Dimenian', 'Arlan', 1, true); - NpcType.LaDimenian = new NpcType(id++, 'LaDimenian', 'La Dimenian', 'La Dimenian', 'Merlan', 1, true); - NpcType.SoDimenian = new NpcType(id++, 'SoDimenian', 'So Dimenian', 'So Dimenian', 'Del-D', 1, true); - NpcType.Bulclaw = new NpcType(id++, 'Bulclaw', 'Bulclaw', 'Bulclaw', 'Bulclaw', 1, true); - NpcType.Bulk = new NpcType(id++, 'Bulk', 'Bulk', 'Bulk', 'Bulk', 1, true); - NpcType.Claw = new NpcType(id++, 'Claw', 'Claw', 'Claw', 'Claw', 1, true); - NpcType.DarkFalz = new NpcType(id++, 'DarkFalz', 'Dark Falz', 'Dark Falz', 'Dark Falz', 1, true); - - // Episode II VR Temple - - NpcType.Hildebear2 = new NpcType(id++, 'Hildebear2', 'Hildebear (Ep. II)', 'Hildebear', 'Hildelt', 2, true); - NpcType.Hildeblue2 = new NpcType(id++, 'Hildeblue2', 'Hildeblue (Ep. II)', 'Hildeblue', 'Hildetorr', 2, true); - NpcType.Hildebear2.rareType = NpcType.Hildeblue2; - NpcType.RagRappy2 = new NpcType(id++, 'RagRappy2', 'Rag Rappy (Ep. II)', 'Rag Rappy', 'El Rappy', 2, true); - NpcType.LoveRappy = new NpcType(id++, 'LoveRappy', 'Love Rappy', 'Love Rappy', 'Love Rappy', 2, true); - NpcType.RagRappy2.rareType = NpcType.LoveRappy; - NpcType.StRappy = new NpcType(id++, 'StRappy', 'St. Rappy', 'St. Rappy', 'St. Rappy', 2, true); - NpcType.HalloRappy = new NpcType(id++, 'HalloRappy', 'Hallo Rappy', 'Hallo Rappy', 'Hallo Rappy', 2, true); - NpcType.EggRappy = new NpcType(id++, 'EggRappy', 'Egg Rappy', 'Egg Rappy', 'Egg Rappy', 2, true); - NpcType.Monest2 = new NpcType(id++, 'Monest2', 'Monest (Ep. II)', 'Monest', 'Mothvist', 2, true); - NpcType.Mothmant2 = new NpcType(id++, 'Mothmant2', 'Mothmant', 'Mothmant', 'Mothvert', 2, true); - NpcType.PoisonLily2 = new NpcType(id++, 'PoisonLily2', 'Poison Lily (Ep. II)', 'Poison Lily', 'Ob Lily', 2, true); - NpcType.NarLily2 = new NpcType(id++, 'NarLily2', 'Nar Lily (Ep. II)', 'Nar Lily', 'Mil Lily', 2, true); - NpcType.PoisonLily2.rareType = NpcType.NarLily2; - NpcType.GrassAssassin2 = new NpcType(id++, 'GrassAssassin2', 'Grass Assassin (Ep. II)', 'Grass Assassin', 'Crimson Assassin', 2, true); - NpcType.Dimenian2 = new NpcType(id++, 'Dimenian2', 'Dimenian (Ep. II)', 'Dimenian', 'Arlan', 2, true); - NpcType.LaDimenian2 = new NpcType(id++, 'LaDimenian2', 'La Dimenian (Ep. II)', 'La Dimenian', 'Merlan', 2, true); - NpcType.SoDimenian2 = new NpcType(id++, 'SoDimenian2', 'So Dimenian (Ep. II)', 'So Dimenian', 'Del-D', 2, true); - NpcType.DarkBelra2 = new NpcType(id++, 'DarkBelra2', 'Dark Belra (Ep. II)', 'Dark Belra', 'Indi Belra', 2, true); - NpcType.BarbaRay = new NpcType(id++, 'BarbaRay', 'Barba Ray', 'Barba Ray', 'Barba Ray', 2, true); - - // Episode II VR Spaceship - - NpcType.SavageWolf2 = new NpcType(id++, 'SavageWolf2', 'Savage Wolf (Ep. II)', 'Savage Wolf', 'Gulgus', 2, true); - NpcType.BarbarousWolf2 = new NpcType(id++, 'BarbarousWolf2', 'Barbarous Wolf (Ep. II)', 'Barbarous Wolf', 'Gulgus-Gue', 2, true); - NpcType.PanArms2 = new NpcType(id++, 'PanArms2', 'Pan Arms (Ep. II)', 'Pan Arms', 'Pan Arms', 2, true); - NpcType.Migium2 = new NpcType(id++, 'Migium2', 'Migium (Ep. II)', 'Migium', 'Migium', 2, true); - NpcType.Hidoom2 = new NpcType(id++, 'Hidoom2', 'Hidoom (Ep. II)', 'Hidoom', 'Hidoom', 2, true); - NpcType.Dubchic2 = new NpcType(id++, 'Dubchic2', 'Dubchic (Ep. II)', 'Dubchic', 'Dubchich', 2, true); - NpcType.Gilchic2 = new NpcType(id++, 'Gilchic2', 'Gilchic (Ep. II)', 'Gilchic', 'Gilchich', 2, true); - NpcType.Garanz2 = new NpcType(id++, 'Garanz2', 'Garanz (Ep. II)', 'Garanz', 'Baranz', 2, true); - NpcType.Dubswitch2 = new NpcType(id++, 'Dubswitch2', 'Dubswitch (Ep. II)', 'Dubswitch', 'Dubswitch', 2, true); - NpcType.Delsaber2 = new NpcType(id++, 'Delsaber2', 'Delsaber (Ep. II)', 'Delsaber', 'Delsaber', 2, true); - NpcType.ChaosSorcerer2 = new NpcType(id++, 'ChaosSorcerer2', 'Chaos Sorcerer (Ep. II)', 'Chaos Sorcerer', 'Gran Sorcerer', 2, true); - NpcType.GolDragon = new NpcType(id++, 'GolDragon', 'Gol Dragon', 'Gol Dragon', 'Gol Dragon', 2, true); - - // Episode II Central Control Area - - NpcType.SinowBerill = new NpcType(id++, 'SinowBerill', 'Sinow Berill', 'Sinow Berill', 'Sinow Berill', 2, true); - NpcType.SinowSpigell = new NpcType(id++, 'SinowSpigell', 'Sinow Spigell', 'Sinow Spigell', 'Sinow Spigell', 2, true); - NpcType.Merillia = new NpcType(id++, 'Merillia', 'Merillia', 'Merillia', 'Merillia', 2, true); - NpcType.Meriltas = new NpcType(id++, 'Meriltas', 'Meriltas', 'Meriltas', 'Meriltas', 2, true); - NpcType.Mericarol = new NpcType(id++, 'Mericarol', 'Mericarol', 'Mericarol', 'Mericarol', 2, true); - NpcType.Mericus = new NpcType(id++, 'Mericus', 'Mericus', 'Mericus', 'Mericus', 2, true); - NpcType.Merikle = new NpcType(id++, 'Merikle', 'Merikle', 'Merikle', 'Merikle', 2, true); - NpcType.UlGibbon = new NpcType(id++, 'UlGibbon', 'Ul Gibbon', 'Ul Gibbon', 'Ul Gibbon', 2, true); - NpcType.ZolGibbon = new NpcType(id++, 'ZolGibbon', 'Zol Gibbon', 'Zol Gibbon', 'Zol Gibbon', 2, true); - NpcType.Gibbles = new NpcType(id++, 'Gibbles', 'Gibbles', 'Gibbles', 'Gibbles', 2, true); - NpcType.Gee = new NpcType(id++, 'Gee', 'Gee', 'Gee', 'Gee', 2, true); - NpcType.GiGue = new NpcType(id++, 'GiGue', 'Gi Gue', 'Gi Gue', 'Gi Gue', 2, true); - NpcType.GalGryphon = new NpcType(id++, 'GalGryphon', 'Gal Gryphon', 'Gal Gryphon', 'Gal Gryphon', 2, true); - - // Episode II Seabed - - NpcType.Deldepth = new NpcType(id++, 'Deldepth', 'Deldepth', 'Deldepth', 'Deldepth', 2, true); - NpcType.Delbiter = new NpcType(id++, 'Delbiter', 'Delbiter', 'Delbiter', 'Delbiter', 2, true); - NpcType.Dolmolm = new NpcType(id++, 'Dolmolm', 'Dolmolm', 'Dolmolm', 'Dolmolm', 2, true); - NpcType.Dolmdarl = new NpcType(id++, 'Dolmdarl', 'Dolmdarl', 'Dolmdarl', 'Dolmdarl', 2, true); - NpcType.Morfos = new NpcType(id++, 'Morfos', 'Morfos', 'Morfos', 'Morfos', 2, true); - NpcType.Recobox = new NpcType(id++, 'Recobox', 'Recobox', 'Recobox', 'Recobox', 2, true); - NpcType.Recon = new NpcType(id++, 'Recon', 'Recon', 'Recon', 'Recon', 2, true); - NpcType.Epsilon = new NpcType(id++, 'Epsilon', 'Epsilon', 'Epsilon', 'Epsilon', 2, true); - NpcType.SinowZoa = new NpcType(id++, 'SinowZoa', 'Sinow Zoa', 'Sinow Zoa', 'Sinow Zoa', 2, true); - NpcType.SinowZele = new NpcType(id++, 'SinowZele', 'Sinow Zele', 'Sinow Zele', 'Sinow Zele', 2, true); - NpcType.IllGill = new NpcType(id++, 'IllGill', 'Ill Gill', 'Ill Gill', 'Ill Gill', 2, true); - NpcType.DelLily = new NpcType(id++, 'DelLily', 'Del Lily', 'Del Lily', 'Del Lily', 2, true); - NpcType.OlgaFlow = new NpcType(id++, 'OlgaFlow', 'Olga Flow', 'Olga Flow', 'Olga Flow', 2, true); - - // Episode IV - - NpcType.SandRappy = new NpcType(id++, 'SandRappy', 'Sand Rappy', 'Sand Rappy', 'Sand Rappy', 4, true); - NpcType.DelRappy = new NpcType(id++, 'DelRappy', 'Del Rappy', 'Del Rappy', 'Del Rappy', 4, true); - NpcType.SandRappy.rareType = NpcType.DelRappy; - NpcType.Astark = new NpcType(id++, 'Astark', 'Astark', 'Astark', 'Astark', 4, true); - NpcType.SatelliteLizard = new NpcType(id++, 'SatelliteLizard', 'Satellite Lizard', 'Satellite Lizard', 'Satellite Lizard', 4, true); - NpcType.Yowie = new NpcType(id++, 'Yowie', 'Yowie', 'Yowie', 'Yowie', 4, true); - NpcType.MerissaA = new NpcType(id++, 'MerissaA', 'Merissa A', 'Merissa A', 'Merissa A', 4, true); - NpcType.MerissaAA = new NpcType(id++, 'MerissaAA', 'Merissa AA', 'Merissa AA', 'Merissa AA', 4, true); - NpcType.MerissaA.rareType = NpcType.MerissaAA; - NpcType.Girtablulu = new NpcType(id++, 'Girtablulu', 'Girtablulu', 'Girtablulu', 'Girtablulu', 4, true); - NpcType.Zu = new NpcType(id++, 'Zu', 'Zu', 'Zu', 'Zu', 4, true); - NpcType.Pazuzu = new NpcType(id++, 'Pazuzu', 'Pazuzu', 'Pazuzu', 'Pazuzu', 4, true); - NpcType.Zu.rareType = NpcType.Pazuzu; - NpcType.Boota = new NpcType(id++, 'Boota', 'Boota', 'Boota', 'Boota', 4, true); - NpcType.ZeBoota = new NpcType(id++, 'ZeBoota', 'Ze Boota', 'Ze Boota', 'Ze Boota', 4, true); - NpcType.BaBoota = new NpcType(id++, 'BaBoota', 'Ba Boota', 'Ba Boota', 'Ba Boota', 4, true); - NpcType.Dorphon = new NpcType(id++, 'Dorphon', 'Dorphon', 'Dorphon', 'Dorphon', 4, true); - NpcType.DorphonEclair = new NpcType(id++, 'DorphonEclair', 'Dorphon Eclair', 'Dorphon Eclair', 'Dorphon Eclair', 4, true); - NpcType.Dorphon.rareType = NpcType.DorphonEclair; - NpcType.Goran = new NpcType(id++, 'Goran', 'Goran', 'Goran', 'Goran', 4, true); - NpcType.PyroGoran = new NpcType(id++, 'PyroGoran', 'Pyro Goran', 'Pyro Goran', 'Pyro Goran', 4, true); - NpcType.GoranDetonator = new NpcType(id++, 'GoranDetonator', 'Goran Detonator', 'Goran Detonator', 'Goran Detonator', 4, true); - NpcType.SaintMilion = new NpcType(id++, 'SaintMilion', 'Saint-Milion', 'Saint-Milion', 'Saint-Milion', 4, true); - NpcType.Shambertin = new NpcType(id++, 'Shambertin', 'Shambertin', 'Shambertin', 'Shambertin', 4, true); - NpcType.Kondrieu = new NpcType(id++, 'Kondrieu', 'Kondrieu', 'Kondrieu', 'Kondrieu', 4, true); - NpcType.SaintMilion.rareType = NpcType.Kondrieu; - NpcType.Shambertin.rareType = NpcType.Kondrieu; -}()); - -export const NpcTypes: Array = [ - - // - // Unknown NPCs - // - - NpcType.Unknown, - - // - // Friendly NPCs - // - - NpcType.FemaleFat, - NpcType.FemaleMacho, - NpcType.FemaleTall, - NpcType.MaleDwarf, - NpcType.MaleFat, - NpcType.MaleMacho, - NpcType.MaleOld, - NpcType.BlueSoldier, - NpcType.RedSoldier, - NpcType.Principal, - NpcType.Tekker, - NpcType.GuildLady, - NpcType.Scientist, - NpcType.Nurse, - NpcType.Irene, - NpcType.ItemShop, - NpcType.Nurse2, - - // - // Enemy NPCs - // - - // Episode I Forest - - NpcType.Hildebear, - NpcType.Hildeblue, - NpcType.RagRappy, - NpcType.AlRappy, - NpcType.Monest, - NpcType.Mothmant, - NpcType.SavageWolf, - NpcType.BarbarousWolf, - NpcType.Booma, - NpcType.Gobooma, - NpcType.Gigobooma, - NpcType.Dragon, - - // Episode I Caves - - NpcType.GrassAssassin, - NpcType.PoisonLily, - NpcType.NarLily, - NpcType.NanoDragon, - NpcType.EvilShark, - NpcType.PalShark, - NpcType.GuilShark, - NpcType.PofuillySlime, - NpcType.PouillySlime, - NpcType.PanArms, - NpcType.Migium, - NpcType.Hidoom, - NpcType.DeRolLe, - - // Episode I Mines - - NpcType.Dubchic, - NpcType.Gilchic, - NpcType.Garanz, - NpcType.SinowBeat, - NpcType.SinowGold, - NpcType.Canadine, - NpcType.Canane, - NpcType.Dubswitch, - NpcType.VolOpt, - - // Episode I Ruins - - NpcType.Delsaber, - NpcType.ChaosSorcerer, - NpcType.DarkGunner, - NpcType.DeathGunner, - NpcType.ChaosBringer, - NpcType.DarkBelra, - NpcType.Dimenian, - NpcType.LaDimenian, - NpcType.SoDimenian, - NpcType.Bulclaw, - NpcType.Bulk, - NpcType.Claw, - NpcType.DarkFalz, - - // Episode II VR Temple - - NpcType.Hildebear2, - NpcType.Hildeblue2, - NpcType.RagRappy2, - NpcType.LoveRappy, - NpcType.StRappy, - NpcType.HalloRappy, - NpcType.EggRappy, - NpcType.Monest2, - NpcType.Mothmant2, - NpcType.PoisonLily2, - NpcType.NarLily2, - NpcType.GrassAssassin2, - NpcType.Dimenian2, - NpcType.LaDimenian2, - NpcType.SoDimenian2, - NpcType.DarkBelra2, - NpcType.BarbaRay, - - // Episode II VR Spaceship - - NpcType.SavageWolf2, - NpcType.BarbarousWolf2, - NpcType.PanArms2, - NpcType.Migium2, - NpcType.Hidoom2, - NpcType.Dubchic2, - NpcType.Gilchic2, - NpcType.Garanz2, - NpcType.Dubswitch2, - NpcType.Delsaber2, - NpcType.ChaosSorcerer2, - NpcType.GolDragon, - - // Episode II Central Control Area - - NpcType.SinowBerill, - NpcType.SinowSpigell, - NpcType.Merillia, - NpcType.Meriltas, - NpcType.Mericarol, - NpcType.Mericus, - NpcType.Merikle, - NpcType.UlGibbon, - NpcType.ZolGibbon, - NpcType.Gibbles, - NpcType.Gee, - NpcType.GiGue, - NpcType.GalGryphon, - - // Episode II Seabed - - NpcType.Deldepth, - NpcType.Delbiter, - NpcType.Dolmolm, - NpcType.Dolmdarl, - NpcType.Morfos, - NpcType.Recobox, - NpcType.Recon, - NpcType.Epsilon, - NpcType.SinowZoa, - NpcType.SinowZele, - NpcType.IllGill, - NpcType.DelLily, - NpcType.OlgaFlow, - - // Episode IV - - NpcType.SandRappy, - NpcType.DelRappy, - NpcType.Astark, - NpcType.SatelliteLizard, - NpcType.Yowie, - NpcType.MerissaA, - NpcType.MerissaAA, - NpcType.Girtablulu, - NpcType.Zu, - NpcType.Pazuzu, - NpcType.Boota, - NpcType.ZeBoota, - NpcType.BaBoota, - NpcType.Dorphon, - NpcType.DorphonEclair, - NpcType.Goran, - NpcType.PyroGoran, - NpcType.GoranDetonator, - NpcType.SaintMilion, - NpcType.Shambertin, - NpcType.Kondrieu, -]; - -export const EnemyNpcTypes = NpcTypes.filter(type => type.enemy); diff --git a/src/domain/ObjectType.ts b/src/domain/ObjectType.ts deleted file mode 100644 index f59190ad..00000000 --- a/src/domain/ObjectType.ts +++ /dev/null @@ -1,870 +0,0 @@ -export class ObjectType { - id: number; - psoId?: number; - name: string; - - constructor(id: number, psoId: number | undefined, name: string) { - if (!Number.isInteger(id) || id < 1) - throw new Error(`Expected id to be an integer greater than or equal to 1, got ${id}.`); - if (psoId != null && (!Number.isInteger(psoId) || psoId < 0)) - throw new Error(`Expected psoId to be null or an integer greater than or equal to 0, got ${psoId}.`); - if (!name) throw new Error('name is required.'); - - this.id = id; - this.psoId = psoId; - this.name = name; - } - - static Unknown: ObjectType; - static PlayerSet: ObjectType; - static Particle: ObjectType; - static Teleporter: ObjectType; - static Warp: ObjectType; - static LightCollision: ObjectType; - static Item: ObjectType; - static EnvSound: ObjectType; - static FogCollision: ObjectType; - static EventCollision: ObjectType; - static CharaCollision: ObjectType; - static ElementalTrap: ObjectType; - static StatusTrap: ObjectType; - static HealTrap: ObjectType; - static LargeElementalTrap: ObjectType; - static ObjRoomID: ObjectType; - static Sensor: ObjectType; - static UnknownItem16: ObjectType; - static Lensflare: ObjectType; - static ScriptCollision: ObjectType; - static HealRing: ObjectType; - static MapCollision: ObjectType; - static ScriptCollisionA: ObjectType; - static ItemLight: ObjectType; - static RadarCollision: ObjectType; - static FogCollisionSW: ObjectType; - static BossTeleporter: ObjectType; - static ImageBoard: ObjectType; - static QuestWarp: ObjectType; - static Epilogue: ObjectType; - static UnknownItem29: ObjectType; - static UnknownItem30: ObjectType; - static UnknownItem31: ObjectType; - static BoxDetectObject: ObjectType; - static SymbolChatObject: ObjectType; - static TouchPlateObject: ObjectType; - static TargetableObject: ObjectType; - static EffectObject: ObjectType; - static CountDownObject: ObjectType; - static UnknownItem38: ObjectType; - static UnknownItem39: ObjectType; - static UnknownItem40: ObjectType; - static UnknownItem41: ObjectType; - static MenuActivation: ObjectType; - static TelepipeLocation: ObjectType; - static BGMCollision: ObjectType; - static MainRagolTeleporter: ObjectType; - static LobbyTeleporter: ObjectType; - static PrincipalWarp: ObjectType; - static ShopDoor: ObjectType; - static HuntersGuildDoor: ObjectType; - static TeleporterDoor: ObjectType; - static MedicalCenterDoor: ObjectType; - static Elevator: ObjectType; - static EasterEgg: ObjectType; - static ValentinesHeart: ObjectType; - static ChristmasTree: ObjectType; - static ChristmasWreath: ObjectType; - static HalloweenPumpkin: ObjectType; - static TwentyFirstCentury: ObjectType; - static Sonic: ObjectType; - static WelcomeBoard: ObjectType; - static Firework: ObjectType; - static LobbyScreenDoor: ObjectType; - static MainRagolTeleporterBattleInNextArea: ObjectType; - static LabTeleporterDoor: ObjectType; - static Pioneer2InvisibleTouchplate: ObjectType; - static ForestDoor: ObjectType; - static ForestSwitch: ObjectType; - static LaserFence: ObjectType; - static LaserSquareFence: ObjectType; - static ForestLaserFenceSwitch: ObjectType; - static LightRays: ObjectType; - static BlueButterfly: ObjectType; - static Probe: ObjectType; - static RandomTypeBox1: ObjectType; - static ForestWeatherStation: ObjectType; - static Battery: ObjectType; - static ForestConsole: ObjectType; - static BlackSlidingDoor: ObjectType; - static RicoMessagePod: ObjectType; - static EnergyBarrier: ObjectType; - static ForestRisingBridge: ObjectType; - static SwitchNoneDoor: ObjectType; - static EnemyBoxGrey: ObjectType; - static FixedTypeBox: ObjectType; - static EnemyBoxBrown: ObjectType; - static EmptyTypeBox: ObjectType; - static LaserFenseEx: ObjectType; - static LaserSquareFenceEx: ObjectType; - static FloorPanel1: ObjectType; - static Caves4ButtonDoor: ObjectType; - static CavesNormalDoor: ObjectType; - static CavesSmashingPillar: ObjectType; - static CavesSign1: ObjectType; - static CavesSign2: ObjectType; - static CavesSign3: ObjectType; - static HexagalTank: ObjectType; - static BrownPlatform: ObjectType; - static WarningLightObject: ObjectType; - static Rainbow: ObjectType; - static FloatingJelifish: ObjectType; - static FloatingDragonfly: ObjectType; - static CavesSwitchDoor: ObjectType; - static RobotRechargeStation: ObjectType; - static CavesCakeShop: ObjectType; - static Caves1SmallRedRock: ObjectType; - static Caves1MediumRedRock: ObjectType; - static Caves1LargeRedRock: ObjectType; - static Caves2SmallRock1: ObjectType; - static Caves2MediumRock1: ObjectType; - static Caves2LargeRock1: ObjectType; - static Caves2SmallRock2: ObjectType; - static Caves2MediumRock2: ObjectType; - static Caves2LargeRock2: ObjectType; - static Caves3SmallRock: ObjectType; - static Caves3MediumRock: ObjectType; - static Caves3LargeRock: ObjectType; - static FloorPanel2: ObjectType; - static DestructableRockCaves1: ObjectType; - static DestructableRockCaves2: ObjectType; - static DestructableRockCaves3: ObjectType; - static MinesDoor: ObjectType; - static FloorPanel3: ObjectType; - static MinesSwitchDoor: ObjectType; - static LargeCryoTube: ObjectType; - static ComputerLikeCalus: ObjectType; - static GreenScreenOpeningAndClosing: ObjectType; - static FloatingRobot: ObjectType; - static FloatingBlueLight: ObjectType; - static SelfDestructingObject1: ObjectType; - static SelfDestructingObject2: ObjectType; - static SelfDestructingObject3: ObjectType; - static SparkMachine: ObjectType; - static MinesLargeFlashingCrate: ObjectType; - static RuinsSeal: ObjectType; - static RuinsTeleporter: ObjectType; - static RuinsWarpSiteToSite: ObjectType; - static RuinsSwitch: ObjectType; - static FloorPanel4: ObjectType; - static Ruins1Door: ObjectType; - static Ruins3Door: ObjectType; - static Ruins2Door: ObjectType; - static Ruins11ButtonDoor: ObjectType; - static Ruins21ButtonDoor: ObjectType; - static Ruins31ButtonDoor: ObjectType; - static Ruins4ButtonDoor: ObjectType; - static Ruins2ButtonDoor: ObjectType; - static RuinsSensor: ObjectType; - static RuinsFenceSwitch: ObjectType; - static RuinsLaserFence4x2: ObjectType; - static RuinsLaserFence6x2: ObjectType; - static RuinsLaserFence4x4: ObjectType; - static RuinsLaserFence6x4: ObjectType; - static RuinsPoisonBlob: ObjectType; - static RuinsPilarTrap: ObjectType; - static PopupTrapNoTech: ObjectType; - static RuinsCrystal: ObjectType; - static Monument: ObjectType; - static RuinsRock1: ObjectType; - static RuinsRock2: ObjectType; - static RuinsRock3: ObjectType; - static RuinsRock4: ObjectType; - static RuinsRock5: ObjectType; - static RuinsRock6: ObjectType; - static RuinsRock7: ObjectType; - static Poison: ObjectType; - static FixedBoxTypeRuins: ObjectType; - static RandomBoxTypeRuins: ObjectType; - static EnemyTypeBoxYellow: ObjectType; - static EnemyTypeBoxBlue: ObjectType; - static EmptyTypeBoxBlue: ObjectType; - static DestructableRock: ObjectType; - static PopupTrapsTechs: ObjectType; - static FlyingWhiteBird: ObjectType; - static Tower: ObjectType; - static FloatingRocks: ObjectType; - static FloatingSoul: ObjectType; - static Butterfly: ObjectType; - static LobbyGameMenu: ObjectType; - static LobbyWarpObject: ObjectType; - static Lobby1EventObjectDefaultTree: ObjectType; - static UnknownItem387: ObjectType; - static UnknownItem388: ObjectType; - static UnknownItem389: ObjectType; - static LobbyEventObjectStaticPumpkin: ObjectType; - static LobbyEventObject3ChristmasWindows: ObjectType; - static LobbyEventObjectRedAndWhiteCurtain: ObjectType; - static UnknownItem393: ObjectType; - static UnknownItem394: ObjectType; - static LobbyFishTank: ObjectType; - static LobbyEventObjectButterflies: ObjectType; - static UnknownItem400: ObjectType; - static GreyWallLow: ObjectType; - static SpaceshipDoor: ObjectType; - static GreyWallHigh: ObjectType; - static TempleNormalDoor: ObjectType; - static BreakableWallWallButUnbreakable: ObjectType; - static BrokenCilinderAndRubble: ObjectType; - static ThreeBrokenWallPiecesOnFloor: ObjectType; - static HighBrickCilinder: ObjectType; - static LyingCilinder: ObjectType; - static BrickConeWithFlatTop: ObjectType; - static BreakableTempleWall: ObjectType; - static TempleMapDetect: ObjectType; - static SmallBrownBrickRisingBridge: ObjectType; - static LongRisingBridgeWithPinkHighEdges: ObjectType; - static FourSwitchTempleDoor: ObjectType; - static FourButtonSpaceshipDoor: ObjectType; - static ItemBoxCca: ObjectType; - static TeleporterEp2: ObjectType; - static CCADoor: ObjectType; - static SpecialBoxCCA: ObjectType; - static BigCCADoor: ObjectType; - static BigCCADoorSwitch: ObjectType; - static LittleRock: ObjectType; - static Little3StoneWall: ObjectType; - static Medium3StoneWall: ObjectType; - static SpiderPlant: ObjectType; - static CCAAreaTeleporter: ObjectType; - static UnknownItem523: ObjectType; - static WhiteBird: ObjectType; - static OrangeBird: ObjectType; - static Saw: ObjectType; - static LaserDetect: ObjectType; - static UnknownItem529: ObjectType; - static UnknownItem530: ObjectType; - static Seagull: ObjectType; - static Fish: ObjectType; - static SeabedDoorWithBlueEdges: ObjectType; - static SeabedDoorAlwaysOpenNonTriggerable: ObjectType; - static LittleCryotube: ObjectType; - static WideGlassWallBreakable: ObjectType; - static BlueFloatingRobot: ObjectType; - static RedFloatingRobot: ObjectType; - static Dolphin: ObjectType; - static CaptureTrap: ObjectType; - static VRLink: ObjectType; - static UnknownItem576: ObjectType; - static WarpInBarbaRayRoom: ObjectType; - static UnknownItem672: ObjectType; - static GeeNest: ObjectType; - static LabComputerConsole: ObjectType; - static LabComputerConsoleGreenScreen: ObjectType; - static ChairYelllowPillow: ObjectType; - static OrangeWallWithHoleInMiddle: ObjectType; - static GreyWallWithHoleInMiddle: ObjectType; - static LongTable: ObjectType; - static GBAStation: ObjectType; - static TalkLinkToSupport: ObjectType; - static InstaWarp: ObjectType; - static LabInvisibleObject: ObjectType; - static LabGlassWindowDoor: ObjectType; - static UnknownItem700: ObjectType; - static LabCelingWarp: ObjectType; - static Ep4LightSource: ObjectType; - static Cacti: ObjectType; - static BigBrownRock: ObjectType; - static BreakableBrownRock: ObjectType; - static UnknownItem832: ObjectType; - static UnknownItem833: ObjectType; - static PoisonPlant: ObjectType; - static UnknownItem897: ObjectType; - static UnknownItem898: ObjectType; - static OozingDesertPlant: ObjectType; - static UnknownItem901: ObjectType; - static BigBlackRocks: ObjectType; - static UnknownItem903: ObjectType; - static UnknownItem904: ObjectType; - static UnknownItem905: ObjectType; - static UnknownItem906: ObjectType; - static FallingRock: ObjectType; - static DesertPlantHasCollision: ObjectType; - static DesertFixedTypeBoxBreakableCrystals: ObjectType; - static UnknownItem910: ObjectType; - static BeeHive: ObjectType; - static UnknownItem912: ObjectType; - static Heat: ObjectType; - static TopOfSaintMillionEgg: ObjectType; - static UnknownItem961: ObjectType; - - static fromPsoId(psoId: number): ObjectType { - switch (psoId) { - default: return ObjectType.Unknown; - - case 0: return ObjectType.PlayerSet; - case 1: return ObjectType.Particle; - case 2: return ObjectType.Teleporter; - case 3: return ObjectType.Warp; - case 4: return ObjectType.LightCollision; - case 5: return ObjectType.Item; - case 6: return ObjectType.EnvSound; - case 7: return ObjectType.FogCollision; - case 8: return ObjectType.EventCollision; - case 9: return ObjectType.CharaCollision; - case 10: return ObjectType.ElementalTrap; - case 11: return ObjectType.StatusTrap; - case 12: return ObjectType.HealTrap; - case 13: return ObjectType.LargeElementalTrap; - case 14: return ObjectType.ObjRoomID; - case 15: return ObjectType.Sensor; - case 16: return ObjectType.UnknownItem16; - case 17: return ObjectType.Lensflare; - case 18: return ObjectType.ScriptCollision; - case 19: return ObjectType.HealRing; - case 20: return ObjectType.MapCollision; - case 21: return ObjectType.ScriptCollisionA; - case 22: return ObjectType.ItemLight; - case 23: return ObjectType.RadarCollision; - case 24: return ObjectType.FogCollisionSW; - case 25: return ObjectType.BossTeleporter; - case 26: return ObjectType.ImageBoard; - case 27: return ObjectType.QuestWarp; - case 28: return ObjectType.Epilogue; - case 29: return ObjectType.UnknownItem29; - case 30: return ObjectType.UnknownItem30; - case 31: return ObjectType.UnknownItem31; - case 32: return ObjectType.BoxDetectObject; - case 33: return ObjectType.SymbolChatObject; - case 34: return ObjectType.TouchPlateObject; - case 35: return ObjectType.TargetableObject; - case 36: return ObjectType.EffectObject; - case 37: return ObjectType.CountDownObject; - case 38: return ObjectType.UnknownItem38; - case 39: return ObjectType.UnknownItem39; - case 40: return ObjectType.UnknownItem40; - case 41: return ObjectType.UnknownItem41; - case 64: return ObjectType.MenuActivation; - case 65: return ObjectType.TelepipeLocation; - case 66: return ObjectType.BGMCollision; - case 67: return ObjectType.MainRagolTeleporter; - case 68: return ObjectType.LobbyTeleporter; - case 69: return ObjectType.PrincipalWarp; - case 70: return ObjectType.ShopDoor; - case 71: return ObjectType.HuntersGuildDoor; - case 72: return ObjectType.TeleporterDoor; - case 73: return ObjectType.MedicalCenterDoor; - case 74: return ObjectType.Elevator; - case 75: return ObjectType.EasterEgg; - case 76: return ObjectType.ValentinesHeart; - case 77: return ObjectType.ChristmasTree; - case 78: return ObjectType.ChristmasWreath; - case 79: return ObjectType.HalloweenPumpkin; - case 80: return ObjectType.TwentyFirstCentury; - case 81: return ObjectType.Sonic; - case 82: return ObjectType.WelcomeBoard; - case 83: return ObjectType.Firework; - case 84: return ObjectType.LobbyScreenDoor; - case 85: return ObjectType.MainRagolTeleporterBattleInNextArea; - case 86: return ObjectType.LabTeleporterDoor; - case 87: return ObjectType.Pioneer2InvisibleTouchplate; - case 128: return ObjectType.ForestDoor; - case 129: return ObjectType.ForestSwitch; - case 130: return ObjectType.LaserFence; - case 131: return ObjectType.LaserSquareFence; - case 132: return ObjectType.ForestLaserFenceSwitch; - case 133: return ObjectType.LightRays; - case 134: return ObjectType.BlueButterfly; - case 135: return ObjectType.Probe; - case 136: return ObjectType.RandomTypeBox1; - case 137: return ObjectType.ForestWeatherStation; - case 138: return ObjectType.Battery; - case 139: return ObjectType.ForestConsole; - case 140: return ObjectType.BlackSlidingDoor; - case 141: return ObjectType.RicoMessagePod; - case 142: return ObjectType.EnergyBarrier; - case 143: return ObjectType.ForestRisingBridge; - case 144: return ObjectType.SwitchNoneDoor; - case 145: return ObjectType.EnemyBoxGrey; - case 146: return ObjectType.FixedTypeBox; - case 147: return ObjectType.EnemyBoxBrown; - case 149: return ObjectType.EmptyTypeBox; - case 150: return ObjectType.LaserFenseEx; - case 151: return ObjectType.LaserSquareFenceEx; - case 192: return ObjectType.FloorPanel1; - case 193: return ObjectType.Caves4ButtonDoor; - case 194: return ObjectType.CavesNormalDoor; - case 195: return ObjectType.CavesSmashingPillar; - case 196: return ObjectType.CavesSign1; - case 197: return ObjectType.CavesSign2; - case 198: return ObjectType.CavesSign3; - case 199: return ObjectType.HexagalTank; - case 200: return ObjectType.BrownPlatform; - case 201: return ObjectType.WarningLightObject; - case 203: return ObjectType.Rainbow; - case 204: return ObjectType.FloatingJelifish; - case 205: return ObjectType.FloatingDragonfly; - case 206: return ObjectType.CavesSwitchDoor; - case 207: return ObjectType.RobotRechargeStation; - case 208: return ObjectType.CavesCakeShop; - case 209: return ObjectType.Caves1SmallRedRock; - case 210: return ObjectType.Caves1MediumRedRock; - case 211: return ObjectType.Caves1LargeRedRock; - case 212: return ObjectType.Caves2SmallRock1; - case 213: return ObjectType.Caves2MediumRock1; - case 214: return ObjectType.Caves2LargeRock1; - case 215: return ObjectType.Caves2SmallRock2; - case 216: return ObjectType.Caves2MediumRock2; - case 217: return ObjectType.Caves2LargeRock2; - case 218: return ObjectType.Caves3SmallRock; - case 219: return ObjectType.Caves3MediumRock; - case 220: return ObjectType.Caves3LargeRock; - case 222: return ObjectType.FloorPanel2; - case 223: return ObjectType.DestructableRockCaves1; - case 224: return ObjectType.DestructableRockCaves2; - case 225: return ObjectType.DestructableRockCaves3; - case 256: return ObjectType.MinesDoor; - case 257: return ObjectType.FloorPanel3; - case 258: return ObjectType.MinesSwitchDoor; - case 259: return ObjectType.LargeCryoTube; - case 260: return ObjectType.ComputerLikeCalus; - case 261: return ObjectType.GreenScreenOpeningAndClosing; - case 262: return ObjectType.FloatingRobot; - case 263: return ObjectType.FloatingBlueLight; - case 264: return ObjectType.SelfDestructingObject1; - case 265: return ObjectType.SelfDestructingObject2; - case 266: return ObjectType.SelfDestructingObject3; - case 267: return ObjectType.SparkMachine; - case 268: return ObjectType.MinesLargeFlashingCrate; - case 304: return ObjectType.RuinsSeal; - case 320: return ObjectType.RuinsTeleporter; - case 321: return ObjectType.RuinsWarpSiteToSite; - case 322: return ObjectType.RuinsSwitch; - case 323: return ObjectType.FloorPanel4; - case 324: return ObjectType.Ruins1Door; - case 325: return ObjectType.Ruins3Door; - case 326: return ObjectType.Ruins2Door; - case 327: return ObjectType.Ruins11ButtonDoor; - case 328: return ObjectType.Ruins21ButtonDoor; - case 329: return ObjectType.Ruins31ButtonDoor; - case 330: return ObjectType.Ruins4ButtonDoor; - case 331: return ObjectType.Ruins2ButtonDoor; - case 332: return ObjectType.RuinsSensor; - case 333: return ObjectType.RuinsFenceSwitch; - case 334: return ObjectType.RuinsLaserFence4x2; - case 335: return ObjectType.RuinsLaserFence6x2; - case 336: return ObjectType.RuinsLaserFence4x4; - case 337: return ObjectType.RuinsLaserFence6x4; - case 338: return ObjectType.RuinsPoisonBlob; - case 339: return ObjectType.RuinsPilarTrap; - case 340: return ObjectType.PopupTrapNoTech; - case 341: return ObjectType.RuinsCrystal; - case 342: return ObjectType.Monument; - case 345: return ObjectType.RuinsRock1; - case 346: return ObjectType.RuinsRock2; - case 347: return ObjectType.RuinsRock3; - case 348: return ObjectType.RuinsRock4; - case 349: return ObjectType.RuinsRock5; - case 350: return ObjectType.RuinsRock6; - case 351: return ObjectType.RuinsRock7; - case 352: return ObjectType.Poison; - case 353: return ObjectType.FixedBoxTypeRuins; - case 354: return ObjectType.RandomBoxTypeRuins; - case 355: return ObjectType.EnemyTypeBoxYellow; - case 356: return ObjectType.EnemyTypeBoxBlue; - case 357: return ObjectType.EmptyTypeBoxBlue; - case 358: return ObjectType.DestructableRock; - case 359: return ObjectType.PopupTrapsTechs; - case 368: return ObjectType.FlyingWhiteBird; - case 369: return ObjectType.Tower; - case 370: return ObjectType.FloatingRocks; - case 371: return ObjectType.FloatingSoul; - case 372: return ObjectType.Butterfly; - case 384: return ObjectType.LobbyGameMenu; - case 385: return ObjectType.LobbyWarpObject; - case 386: return ObjectType.Lobby1EventObjectDefaultTree; - case 387: return ObjectType.UnknownItem387; - case 388: return ObjectType.UnknownItem388; - case 389: return ObjectType.UnknownItem389; - case 390: return ObjectType.LobbyEventObjectStaticPumpkin; - case 391: return ObjectType.LobbyEventObject3ChristmasWindows; - case 392: return ObjectType.LobbyEventObjectRedAndWhiteCurtain; - case 393: return ObjectType.UnknownItem393; - case 394: return ObjectType.UnknownItem394; - case 395: return ObjectType.LobbyFishTank; - case 396: return ObjectType.LobbyEventObjectButterflies; - case 400: return ObjectType.UnknownItem400; - case 401: return ObjectType.GreyWallLow; - case 402: return ObjectType.SpaceshipDoor; - case 403: return ObjectType.GreyWallHigh; - case 416: return ObjectType.TempleNormalDoor; - case 417: return ObjectType.BreakableWallWallButUnbreakable; - case 418: return ObjectType.BrokenCilinderAndRubble; - case 419: return ObjectType.ThreeBrokenWallPiecesOnFloor; - case 420: return ObjectType.HighBrickCilinder; - case 421: return ObjectType.LyingCilinder; - case 422: return ObjectType.BrickConeWithFlatTop; - case 423: return ObjectType.BreakableTempleWall; - case 424: return ObjectType.TempleMapDetect; - case 425: return ObjectType.SmallBrownBrickRisingBridge; - case 426: return ObjectType.LongRisingBridgeWithPinkHighEdges; - case 427: return ObjectType.FourSwitchTempleDoor; - case 448: return ObjectType.FourButtonSpaceshipDoor; - case 512: return ObjectType.ItemBoxCca; - case 513: return ObjectType.TeleporterEp2; - case 514: return ObjectType.CCADoor; - case 515: return ObjectType.SpecialBoxCCA; - case 516: return ObjectType.BigCCADoor; - case 517: return ObjectType.BigCCADoorSwitch; - case 518: return ObjectType.LittleRock; - case 519: return ObjectType.Little3StoneWall; - case 520: return ObjectType.Medium3StoneWall; - case 521: return ObjectType.SpiderPlant; - case 522: return ObjectType.CCAAreaTeleporter; - case 523: return ObjectType.UnknownItem523; - case 524: return ObjectType.WhiteBird; - case 525: return ObjectType.OrangeBird; - case 527: return ObjectType.Saw; - case 528: return ObjectType.LaserDetect; - case 529: return ObjectType.UnknownItem529; - case 530: return ObjectType.UnknownItem530; - case 531: return ObjectType.Seagull; - case 544: return ObjectType.Fish; - case 545: return ObjectType.SeabedDoorWithBlueEdges; - case 546: return ObjectType.SeabedDoorAlwaysOpenNonTriggerable; - case 547: return ObjectType.LittleCryotube; - case 548: return ObjectType.WideGlassWallBreakable; - case 549: return ObjectType.BlueFloatingRobot; - case 550: return ObjectType.RedFloatingRobot; - case 551: return ObjectType.Dolphin; - case 552: return ObjectType.CaptureTrap; - case 553: return ObjectType.VRLink; - case 576: return ObjectType.UnknownItem576; - case 640: return ObjectType.WarpInBarbaRayRoom; - case 672: return ObjectType.UnknownItem672; - case 688: return ObjectType.GeeNest; - case 689: return ObjectType.LabComputerConsole; - case 690: return ObjectType.LabComputerConsoleGreenScreen; - case 691: return ObjectType.ChairYelllowPillow; - case 692: return ObjectType.OrangeWallWithHoleInMiddle; - case 693: return ObjectType.GreyWallWithHoleInMiddle; - case 694: return ObjectType.LongTable; - case 695: return ObjectType.GBAStation; - case 696: return ObjectType.TalkLinkToSupport; - case 697: return ObjectType.InstaWarp; - case 698: return ObjectType.LabInvisibleObject; - case 699: return ObjectType.LabGlassWindowDoor; - case 700: return ObjectType.UnknownItem700; - case 701: return ObjectType.LabCelingWarp; - case 768: return ObjectType.Ep4LightSource; - case 769: return ObjectType.Cacti; - case 770: return ObjectType.BigBrownRock; - case 771: return ObjectType.BreakableBrownRock; - case 832: return ObjectType.UnknownItem832; - case 833: return ObjectType.UnknownItem833; - case 896: return ObjectType.PoisonPlant; - case 897: return ObjectType.UnknownItem897; - case 898: return ObjectType.UnknownItem898; - case 899: return ObjectType.OozingDesertPlant; - case 901: return ObjectType.UnknownItem901; - case 902: return ObjectType.BigBlackRocks; - case 903: return ObjectType.UnknownItem903; - case 904: return ObjectType.UnknownItem904; - case 905: return ObjectType.UnknownItem905; - case 906: return ObjectType.UnknownItem906; - case 907: return ObjectType.FallingRock; - case 908: return ObjectType.DesertPlantHasCollision; - case 909: return ObjectType.DesertFixedTypeBoxBreakableCrystals; - case 910: return ObjectType.UnknownItem910; - case 911: return ObjectType.BeeHive; - case 912: return ObjectType.UnknownItem912; - case 913: return ObjectType.Heat; - case 960: return ObjectType.TopOfSaintMillionEgg; - case 961: return ObjectType.UnknownItem961; - } - } -} - -(function () { - let id = 1; - - ObjectType.Unknown = new ObjectType(id++, undefined, 'Unknown'); - - ObjectType.PlayerSet = new ObjectType(id++, 0, 'Player Set'); - ObjectType.Particle = new ObjectType(id++, 1, 'Particle'); - ObjectType.Teleporter = new ObjectType(id++, 2, 'Teleporter'); - ObjectType.Warp = new ObjectType(id++, 3, 'Warp'); - ObjectType.LightCollision = new ObjectType(id++, 4, 'Light Collision'); - ObjectType.Item = new ObjectType(id++, 5, 'Item'); - ObjectType.EnvSound = new ObjectType(id++, 6, 'Env Sound'); - ObjectType.FogCollision = new ObjectType(id++, 7, 'Fog Collision'); - ObjectType.EventCollision = new ObjectType(id++, 8, 'Event Collision'); - ObjectType.CharaCollision = new ObjectType(id++, 9, 'Chara Collision'); - ObjectType.ElementalTrap = new ObjectType(id++, 10, 'Elemental Trap'); - ObjectType.StatusTrap = new ObjectType(id++, 11, 'Status Trap'); - ObjectType.HealTrap = new ObjectType(id++, 12, 'Heal Trap'); - ObjectType.LargeElementalTrap = new ObjectType(id++, 13, 'Large Elemental Trap'); - ObjectType.ObjRoomID = new ObjectType(id++, 14, 'Obj Room ID'); - ObjectType.Sensor = new ObjectType(id++, 15, 'Sensor'); - ObjectType.UnknownItem16 = new ObjectType(id++, 16, 'Unknown Item (16)'); - ObjectType.Lensflare = new ObjectType(id++, 17, 'Lensflare'); - ObjectType.ScriptCollision = new ObjectType(id++, 18, 'Script Collision'); - ObjectType.HealRing = new ObjectType(id++, 19, 'Heal Ring'); - ObjectType.MapCollision = new ObjectType(id++, 20, 'Map Collision'); - ObjectType.ScriptCollisionA = new ObjectType(id++, 21, 'Script Collision A'); - ObjectType.ItemLight = new ObjectType(id++, 22, 'Item Light'); - ObjectType.RadarCollision = new ObjectType(id++, 23, 'Radar Collision'); - ObjectType.FogCollisionSW = new ObjectType(id++, 24, 'Fog Collision SW'); - ObjectType.BossTeleporter = new ObjectType(id++, 25, 'Boss Teleporter'); - ObjectType.ImageBoard = new ObjectType(id++, 26, 'Image Board'); - ObjectType.QuestWarp = new ObjectType(id++, 27, 'Quest Warp'); - ObjectType.Epilogue = new ObjectType(id++, 28, 'Epilogue'); - ObjectType.UnknownItem29 = new ObjectType(id++, 29, 'Unknown Item (29)'); - ObjectType.UnknownItem30 = new ObjectType(id++, 30, 'Unknown Item (30)'); - ObjectType.UnknownItem31 = new ObjectType(id++, 31, 'Unknown Item (31)'); - ObjectType.BoxDetectObject = new ObjectType(id++, 32, 'Box Detect Object'); - ObjectType.SymbolChatObject = new ObjectType(id++, 33, 'Symbol Chat Object'); - ObjectType.TouchPlateObject = new ObjectType(id++, 34, 'Touch plate Object'); - ObjectType.TargetableObject = new ObjectType(id++, 35, 'Targetable Object'); - ObjectType.EffectObject = new ObjectType(id++, 36, 'Effect object'); - ObjectType.CountDownObject = new ObjectType(id++, 37, 'Count Down Object'); - ObjectType.UnknownItem38 = new ObjectType(id++, 38, 'Unknown Item (38)'); - ObjectType.UnknownItem39 = new ObjectType(id++, 39, 'Unknown Item (39)'); - ObjectType.UnknownItem40 = new ObjectType(id++, 40, 'Unknown Item (40)'); - ObjectType.UnknownItem41 = new ObjectType(id++, 41, 'Unknown Item (41)'); - ObjectType.MenuActivation = new ObjectType(id++, 64, 'Menu activation'); - ObjectType.TelepipeLocation = new ObjectType(id++, 65, 'Telepipe Location'); - ObjectType.BGMCollision = new ObjectType(id++, 66, 'BGM Collision'); - ObjectType.MainRagolTeleporter = new ObjectType(id++, 67, 'Main Ragol Teleporter'); - ObjectType.LobbyTeleporter = new ObjectType(id++, 68, 'Lobby Teleporter'); - ObjectType.PrincipalWarp = new ObjectType(id++, 69, 'Principal warp'); - ObjectType.ShopDoor = new ObjectType(id++, 70, 'Shop Door'); - ObjectType.HuntersGuildDoor = new ObjectType(id++, 71, 'Hunter\'s Guild Door'); - ObjectType.TeleporterDoor = new ObjectType(id++, 72, 'Teleporter Door'); - ObjectType.MedicalCenterDoor = new ObjectType(id++, 73, 'Medical Center Door'); - ObjectType.Elevator = new ObjectType(id++, 74, 'Elevator'); - ObjectType.EasterEgg = new ObjectType(id++, 75, 'Easter Egg'); - ObjectType.ValentinesHeart = new ObjectType(id++, 76, 'Valentines Heart'); - ObjectType.ChristmasTree = new ObjectType(id++, 77, 'Christmas Tree'); - ObjectType.ChristmasWreath = new ObjectType(id++, 78, 'Christmas Wreath'); - ObjectType.HalloweenPumpkin = new ObjectType(id++, 79, 'Halloween Pumpkin'); - ObjectType.TwentyFirstCentury = new ObjectType(id++, 80, '21st Century'); - ObjectType.Sonic = new ObjectType(id++, 81, 'Sonic'); - ObjectType.WelcomeBoard = new ObjectType(id++, 82, 'Welcome Board'); - ObjectType.Firework = new ObjectType(id++, 83, 'Firework'); - ObjectType.LobbyScreenDoor = new ObjectType(id++, 84, 'Lobby Screen Door'); - ObjectType.MainRagolTeleporterBattleInNextArea = new ObjectType(id++, 85, 'Main Ragol Teleporter (Battle in next area?)'); - ObjectType.LabTeleporterDoor = new ObjectType(id++, 86, 'Lab Teleporter Door'); - ObjectType.Pioneer2InvisibleTouchplate = new ObjectType(id++, 87, 'Pioneer 2 Invisible Touchplate'); - ObjectType.ForestDoor = new ObjectType(id++, 128, 'Forest Door'); - ObjectType.ForestSwitch = new ObjectType(id++, 129, 'Forest Switch'); - ObjectType.LaserFence = new ObjectType(id++, 130, 'Laser Fence'); - ObjectType.LaserSquareFence = new ObjectType(id++, 131, 'Laser Square Fence'); - ObjectType.ForestLaserFenceSwitch = new ObjectType(id++, 132, 'Forest Laser Fence Switch'); - ObjectType.LightRays = new ObjectType(id++, 133, 'Light rays'); - ObjectType.BlueButterfly = new ObjectType(id++, 134, 'Blue Butterfly'); - ObjectType.Probe = new ObjectType(id++, 135, 'Probe'); - ObjectType.RandomTypeBox1 = new ObjectType(id++, 136, 'Random Type Box 1'); - ObjectType.ForestWeatherStation = new ObjectType(id++, 137, 'Forest Weather Station'); - ObjectType.Battery = new ObjectType(id++, 138, 'Battery'); - ObjectType.ForestConsole = new ObjectType(id++, 139, 'Forest Console'); - ObjectType.BlackSlidingDoor = new ObjectType(id++, 140, 'Black Sliding Door'); - ObjectType.RicoMessagePod = new ObjectType(id++, 141, 'Rico Message Pod'); - ObjectType.EnergyBarrier = new ObjectType(id++, 142, 'Energy Barrier'); - ObjectType.ForestRisingBridge = new ObjectType(id++, 143, 'Forest Rising Bridge'); - ObjectType.SwitchNoneDoor = new ObjectType(id++, 144, 'Switch (none door)'); - ObjectType.EnemyBoxGrey = new ObjectType(id++, 145, 'Enemy Box (Grey)'); - ObjectType.FixedTypeBox = new ObjectType(id++, 146, 'Fixed Type Box'); - ObjectType.EnemyBoxBrown = new ObjectType(id++, 147, 'Enemy Box (Brown)'); - ObjectType.EmptyTypeBox = new ObjectType(id++, 149, 'Empty Type Box'); - ObjectType.LaserFenseEx = new ObjectType(id++, 150, 'Laser Fense Ex'); - ObjectType.LaserSquareFenceEx = new ObjectType(id++, 151, 'Laser Square Fence Ex'); - ObjectType.FloorPanel1 = new ObjectType(id++, 192, 'Floor Panel 1'); - ObjectType.Caves4ButtonDoor = new ObjectType(id++, 193, 'Caves 4 Button door'); - ObjectType.CavesNormalDoor = new ObjectType(id++, 194, 'Caves Normal door'); - ObjectType.CavesSmashingPillar = new ObjectType(id++, 195, 'Caves Smashing Pillar'); - ObjectType.CavesSign1 = new ObjectType(id++, 196, 'Caves Sign 1'); - ObjectType.CavesSign2 = new ObjectType(id++, 197, 'Caves Sign 2'); - ObjectType.CavesSign3 = new ObjectType(id++, 198, 'Caves Sign 3'); - ObjectType.HexagalTank = new ObjectType(id++, 199, 'Hexagal Tank'); - ObjectType.BrownPlatform = new ObjectType(id++, 200, 'Brown Platform'); - ObjectType.WarningLightObject = new ObjectType(id++, 201, 'Warning Light Object'); - ObjectType.Rainbow = new ObjectType(id++, 203, 'Rainbow'); - ObjectType.FloatingJelifish = new ObjectType(id++, 204, 'Floating Jelifish'); - ObjectType.FloatingDragonfly = new ObjectType(id++, 205, 'Floating Dragonfly'); - ObjectType.CavesSwitchDoor = new ObjectType(id++, 206, 'Caves Switch Door'); - ObjectType.RobotRechargeStation = new ObjectType(id++, 207, 'Robot Recharge Station'); - ObjectType.CavesCakeShop = new ObjectType(id++, 208, 'Caves Cake Shop'); - ObjectType.Caves1SmallRedRock = new ObjectType(id++, 209, 'Caves 1 Small Red Rock'); - ObjectType.Caves1MediumRedRock = new ObjectType(id++, 210, 'Caves 1 Medium Red Rock'); - ObjectType.Caves1LargeRedRock = new ObjectType(id++, 211, 'Caves 1 Large Red Rock'); - ObjectType.Caves2SmallRock1 = new ObjectType(id++, 212, 'Caves 2 Small Rock 1'); - ObjectType.Caves2MediumRock1 = new ObjectType(id++, 213, 'Caves 2 Medium Rock 1'); - ObjectType.Caves2LargeRock1 = new ObjectType(id++, 214, 'Caves 2 Large Rock 1'); - ObjectType.Caves2SmallRock2 = new ObjectType(id++, 215, 'Caves 2 Small Rock 2'); - ObjectType.Caves2MediumRock2 = new ObjectType(id++, 216, 'Caves 2 Medium Rock 2'); - ObjectType.Caves2LargeRock2 = new ObjectType(id++, 217, 'Caves 2 Large Rock 2'); - ObjectType.Caves3SmallRock = new ObjectType(id++, 218, 'Caves 3 Small Rock'); - ObjectType.Caves3MediumRock = new ObjectType(id++, 219, 'Caves 3 Medium Rock'); - ObjectType.Caves3LargeRock = new ObjectType(id++, 220, 'Caves 3 Large Rock'); - ObjectType.FloorPanel2 = new ObjectType(id++, 222, 'Floor Panel 2'); - ObjectType.DestructableRockCaves1 = new ObjectType(id++, 223, 'Destructable Rock (Caves 1)'); - ObjectType.DestructableRockCaves2 = new ObjectType(id++, 224, 'Destructable Rock (Caves 2)'); - ObjectType.DestructableRockCaves3 = new ObjectType(id++, 225, 'Destructable Rock (Caves 3)'); - ObjectType.MinesDoor = new ObjectType(id++, 256, 'Mines Door'); - ObjectType.FloorPanel3 = new ObjectType(id++, 257, 'Floor Panel 3'); - ObjectType.MinesSwitchDoor = new ObjectType(id++, 258, 'Mines Switch Door'); - ObjectType.LargeCryoTube = new ObjectType(id++, 259, 'Large Cryo-Tube'); - ObjectType.ComputerLikeCalus = new ObjectType(id++, 260, 'Computer (like calus)'); - ObjectType.GreenScreenOpeningAndClosing = new ObjectType(id++, 261, 'Green Screen opening and closing'); - ObjectType.FloatingRobot = new ObjectType(id++, 262, 'Floating Robot'); - ObjectType.FloatingBlueLight = new ObjectType(id++, 263, 'Floating Blue Light'); - ObjectType.SelfDestructingObject1 = new ObjectType(id++, 264, 'Self Destructing Object 1'); - ObjectType.SelfDestructingObject2 = new ObjectType(id++, 265, 'Self Destructing Object 2'); - ObjectType.SelfDestructingObject3 = new ObjectType(id++, 266, 'Self Destructing Object 3'); - ObjectType.SparkMachine = new ObjectType(id++, 267, 'Spark Machine'); - ObjectType.MinesLargeFlashingCrate = new ObjectType(id++, 268, 'Mines Large Flashing Crate'); - ObjectType.RuinsSeal = new ObjectType(id++, 304, 'Ruins Seal'); - ObjectType.RuinsTeleporter = new ObjectType(id++, 320, 'Ruins Teleporter'); - ObjectType.RuinsWarpSiteToSite = new ObjectType(id++, 321, 'Ruins Warp (Site to site)'); - ObjectType.RuinsSwitch = new ObjectType(id++, 322, 'Ruins Switch'); - ObjectType.FloorPanel4 = new ObjectType(id++, 323, 'Floor Panel 4'); - ObjectType.Ruins1Door = new ObjectType(id++, 324, 'Ruins 1 Door'); - ObjectType.Ruins3Door = new ObjectType(id++, 325, 'Ruins 3 Door'); - ObjectType.Ruins2Door = new ObjectType(id++, 326, 'Ruins 2 Door'); - ObjectType.Ruins11ButtonDoor = new ObjectType(id++, 327, 'Ruins 1-1 Button Door'); - ObjectType.Ruins21ButtonDoor = new ObjectType(id++, 328, 'Ruins 2-1 Button Door'); - ObjectType.Ruins31ButtonDoor = new ObjectType(id++, 329, 'Ruins 3-1 Button Door'); - ObjectType.Ruins4ButtonDoor = new ObjectType(id++, 330, 'Ruins 4-Button Door'); - ObjectType.Ruins2ButtonDoor = new ObjectType(id++, 331, 'Ruins 2-Button Door'); - ObjectType.RuinsSensor = new ObjectType(id++, 332, 'Ruins Sensor'); - ObjectType.RuinsFenceSwitch = new ObjectType(id++, 333, 'Ruins Fence Switch'); - ObjectType.RuinsLaserFence4x2 = new ObjectType(id++, 334, 'Ruins Laser Fence 4x2'); - ObjectType.RuinsLaserFence6x2 = new ObjectType(id++, 335, 'Ruins Laser Fence 6x2'); - ObjectType.RuinsLaserFence4x4 = new ObjectType(id++, 336, 'Ruins Laser Fence 4x4'); - ObjectType.RuinsLaserFence6x4 = new ObjectType(id++, 337, 'Ruins Laser Fence 6x4'); - ObjectType.RuinsPoisonBlob = new ObjectType(id++, 338, 'Ruins poison Blob'); - ObjectType.RuinsPilarTrap = new ObjectType(id++, 339, 'Ruins Pilar Trap'); - ObjectType.PopupTrapNoTech = new ObjectType(id++, 340, 'Popup Trap (No Tech)'); - ObjectType.RuinsCrystal = new ObjectType(id++, 341, 'Ruins Crystal'); - ObjectType.Monument = new ObjectType(id++, 342, 'Monument'); - ObjectType.RuinsRock1 = new ObjectType(id++, 345, 'Ruins Rock 1'); - ObjectType.RuinsRock2 = new ObjectType(id++, 346, 'Ruins Rock 2'); - ObjectType.RuinsRock3 = new ObjectType(id++, 347, 'Ruins Rock 3'); - ObjectType.RuinsRock4 = new ObjectType(id++, 348, 'Ruins Rock 4'); - ObjectType.RuinsRock5 = new ObjectType(id++, 349, 'Ruins Rock 5'); - ObjectType.RuinsRock6 = new ObjectType(id++, 350, 'Ruins Rock 6'); - ObjectType.RuinsRock7 = new ObjectType(id++, 351, 'Ruins Rock 7'); - ObjectType.Poison = new ObjectType(id++, 352, 'Poison'); - ObjectType.FixedBoxTypeRuins = new ObjectType(id++, 353, 'Fixed Box Type (Ruins)'); - ObjectType.RandomBoxTypeRuins = new ObjectType(id++, 354, 'Random Box Type (Ruins)'); - ObjectType.EnemyTypeBoxYellow = new ObjectType(id++, 355, 'Enemy Type Box (Yellow)'); - ObjectType.EnemyTypeBoxBlue = new ObjectType(id++, 356, 'Enemy Type Box (Blue)'); - ObjectType.EmptyTypeBoxBlue = new ObjectType(id++, 357, 'Empty Type Box (Blue)'); - ObjectType.DestructableRock = new ObjectType(id++, 358, 'Destructable Rock'); - ObjectType.PopupTrapsTechs = new ObjectType(id++, 359, 'Popup Traps (techs)'); - ObjectType.FlyingWhiteBird = new ObjectType(id++, 368, 'Flying White Bird'); - ObjectType.Tower = new ObjectType(id++, 369, 'Tower'); - ObjectType.FloatingRocks = new ObjectType(id++, 370, 'Floating Rocks'); - ObjectType.FloatingSoul = new ObjectType(id++, 371, 'Floating Soul'); - ObjectType.Butterfly = new ObjectType(id++, 372, 'Butterfly'); - ObjectType.LobbyGameMenu = new ObjectType(id++, 384, 'Lobby Game menu'); - ObjectType.LobbyWarpObject = new ObjectType(id++, 385, 'Lobby Warp Object'); - ObjectType.Lobby1EventObjectDefaultTree = new ObjectType(id++, 386, 'Lobby 1 Event Object (Default Tree)'); - ObjectType.UnknownItem387 = new ObjectType(id++, 387, 'Unknown Item (387)'); - ObjectType.UnknownItem388 = new ObjectType(id++, 388, 'Unknown Item (388)'); - ObjectType.UnknownItem389 = new ObjectType(id++, 389, 'Unknown Item (389)'); - ObjectType.LobbyEventObjectStaticPumpkin = new ObjectType(id++, 390, 'Lobby Event Object (Static Pumpkin)'); - ObjectType.LobbyEventObject3ChristmasWindows = new ObjectType(id++, 391, 'Lobby Event Object (3 Christmas Windows)'); - ObjectType.LobbyEventObjectRedAndWhiteCurtain = new ObjectType(id++, 392, 'Lobby Event Object (Red and White Curtain)'); - ObjectType.UnknownItem393 = new ObjectType(id++, 393, 'Unknown Item (393)'); - ObjectType.UnknownItem394 = new ObjectType(id++, 394, 'Unknown Item (394)'); - ObjectType.LobbyFishTank = new ObjectType(id++, 395, 'Lobby Fish Tank'); - ObjectType.LobbyEventObjectButterflies = new ObjectType(id++, 396, 'Lobby Event Object (Butterflies)'); - ObjectType.UnknownItem400 = new ObjectType(id++, 400, 'Unknown Item (400)'); - ObjectType.GreyWallLow = new ObjectType(id++, 401, 'grey wall low'); - ObjectType.SpaceshipDoor = new ObjectType(id++, 402, 'Spaceship Door'); - ObjectType.GreyWallHigh = new ObjectType(id++, 403, 'grey wall high'); - ObjectType.TempleNormalDoor = new ObjectType(id++, 416, 'Temple Normal Door'); - ObjectType.BreakableWallWallButUnbreakable = new ObjectType(id++, 417, '"breakable wall wall, but unbreakable"'); - ObjectType.BrokenCilinderAndRubble = new ObjectType(id++, 418, 'Broken cilinder and rubble'); - ObjectType.ThreeBrokenWallPiecesOnFloor = new ObjectType(id++, 419, '3 broken wall pieces on floor'); - ObjectType.HighBrickCilinder = new ObjectType(id++, 420, 'high brick cilinder'); - ObjectType.LyingCilinder = new ObjectType(id++, 421, 'lying cilinder'); - ObjectType.BrickConeWithFlatTop = new ObjectType(id++, 422, 'brick cone with flat top'); - ObjectType.BreakableTempleWall = new ObjectType(id++, 423, 'breakable temple wall'); - ObjectType.TempleMapDetect = new ObjectType(id++, 424, 'Temple Map Detect'); - ObjectType.SmallBrownBrickRisingBridge = new ObjectType(id++, 425, 'small brown brick rising bridge'); - ObjectType.LongRisingBridgeWithPinkHighEdges = new ObjectType(id++, 426, 'long rising bridge (with pink high edges)'); - ObjectType.FourSwitchTempleDoor = new ObjectType(id++, 427, '4 switch temple door'); - ObjectType.FourButtonSpaceshipDoor = new ObjectType(id++, 448, '4 button spaceship door'); - ObjectType.ItemBoxCca = new ObjectType(id++, 512, 'item box cca'); - ObjectType.TeleporterEp2 = new ObjectType(id++, 513, 'Teleporter (Ep 2)'); - ObjectType.CCADoor = new ObjectType(id++, 514, 'CCA Door'); - ObjectType.SpecialBoxCCA = new ObjectType(id++, 515, 'Special Box CCA'); - ObjectType.BigCCADoor = new ObjectType(id++, 516, 'Big CCA Door'); - ObjectType.BigCCADoorSwitch = new ObjectType(id++, 517, 'Big CCA Door Switch'); - ObjectType.LittleRock = new ObjectType(id++, 518, 'Little Rock'); - ObjectType.Little3StoneWall = new ObjectType(id++, 519, 'Little 3 Stone Wall'); - ObjectType.Medium3StoneWall = new ObjectType(id++, 520, 'Medium 3 stone wall'); - ObjectType.SpiderPlant = new ObjectType(id++, 521, 'Spider Plant'); - ObjectType.CCAAreaTeleporter = new ObjectType(id++, 522, 'CCA Area Teleporter'); - ObjectType.UnknownItem523 = new ObjectType(id++, 523, 'Unknown Item (523)'); - ObjectType.WhiteBird = new ObjectType(id++, 524, 'White Bird'); - ObjectType.OrangeBird = new ObjectType(id++, 525, 'Orange Bird'); - ObjectType.Saw = new ObjectType(id++, 527, 'Saw'); - ObjectType.LaserDetect = new ObjectType(id++, 528, 'Laser Detect'); - ObjectType.UnknownItem529 = new ObjectType(id++, 529, 'Unknown Item (529)'); - ObjectType.UnknownItem530 = new ObjectType(id++, 530, 'Unknown Item (530)'); - ObjectType.Seagull = new ObjectType(id++, 531, 'Seagull'); - ObjectType.Fish = new ObjectType(id++, 544, 'Fish'); - ObjectType.SeabedDoorWithBlueEdges = new ObjectType(id++, 545, 'Seabed Door (with blue edges)'); - ObjectType.SeabedDoorAlwaysOpenNonTriggerable = new ObjectType(id++, 546, 'Seabed door (always open, non-triggerable)'); - ObjectType.LittleCryotube = new ObjectType(id++, 547, 'Little Cryotube'); - ObjectType.WideGlassWallBreakable = new ObjectType(id++, 548, 'Wide Glass Wall (breakable)'); - ObjectType.BlueFloatingRobot = new ObjectType(id++, 549, 'Blue floating robot'); - ObjectType.RedFloatingRobot = new ObjectType(id++, 550, 'Red floating robot'); - ObjectType.Dolphin = new ObjectType(id++, 551, 'Dolphin'); - ObjectType.CaptureTrap = new ObjectType(id++, 552, 'Capture Trap'); - ObjectType.VRLink = new ObjectType(id++, 553, 'VR link'); - ObjectType.UnknownItem576 = new ObjectType(id++, 576, 'Unknown Item (576)'); - ObjectType.WarpInBarbaRayRoom = new ObjectType(id++, 640, 'Warp in Barba Ray Room'); - ObjectType.UnknownItem672 = new ObjectType(id++, 672, 'Unknown Item (672)'); - ObjectType.GeeNest = new ObjectType(id++, 688, 'Gee Nest'); - ObjectType.LabComputerConsole = new ObjectType(id++, 689, 'Lab Computer Console'); - ObjectType.LabComputerConsoleGreenScreen = new ObjectType(id++, 690, 'Lab Computer Console (Green Screen)'); - ObjectType.ChairYelllowPillow = new ObjectType(id++, 691, 'Chair, Yelllow Pillow'); - ObjectType.OrangeWallWithHoleInMiddle = new ObjectType(id++, 692, 'orange wall with hole in middle'); - ObjectType.GreyWallWithHoleInMiddle = new ObjectType(id++, 693, 'grey wall with hole in middle'); - ObjectType.LongTable = new ObjectType(id++, 694, 'long table'); - ObjectType.GBAStation = new ObjectType(id++, 695, 'GBA Station'); - ObjectType.TalkLinkToSupport = new ObjectType(id++, 696, 'Talk (link to support)'); - ObjectType.InstaWarp = new ObjectType(id++, 697, 'insta-warp'); - ObjectType.LabInvisibleObject = new ObjectType(id++, 698, 'Lab Invisible Object'); - ObjectType.LabGlassWindowDoor = new ObjectType(id++, 699, 'Lab Glass window Door'); - ObjectType.UnknownItem700 = new ObjectType(id++, 700, 'Unknown Item (700)'); - ObjectType.LabCelingWarp = new ObjectType(id++, 701, 'Lab Celing Warp'); - ObjectType.Ep4LightSource = new ObjectType(id++, 768, 'Ep4 Light Source'); - ObjectType.Cacti = new ObjectType(id++, 769, 'cacti'); - ObjectType.BigBrownRock = new ObjectType(id++, 770, 'Big Brown Rock'); - ObjectType.BreakableBrownRock = new ObjectType(id++, 771, 'Breakable Brown Rock'); - ObjectType.UnknownItem832 = new ObjectType(id++, 832, 'Unknown Item (832)'); - ObjectType.UnknownItem833 = new ObjectType(id++, 833, 'Unknown Item (833)'); - ObjectType.PoisonPlant = new ObjectType(id++, 896, 'Poison Plant'); - ObjectType.UnknownItem897 = new ObjectType(id++, 897, 'Unknown Item (897)'); - ObjectType.UnknownItem898 = new ObjectType(id++, 898, 'Unknown Item (898)'); - ObjectType.OozingDesertPlant = new ObjectType(id++, 899, 'Oozing Desert Plant'); - ObjectType.UnknownItem901 = new ObjectType(id++, 901, 'Unknown Item (901)'); - ObjectType.BigBlackRocks = new ObjectType(id++, 902, 'big black rocks'); - ObjectType.UnknownItem903 = new ObjectType(id++, 903, 'Unknown Item (903)'); - ObjectType.UnknownItem904 = new ObjectType(id++, 904, 'Unknown Item (904)'); - ObjectType.UnknownItem905 = new ObjectType(id++, 905, 'Unknown Item (905)'); - ObjectType.UnknownItem906 = new ObjectType(id++, 906, 'Unknown Item (906)'); - ObjectType.FallingRock = new ObjectType(id++, 907, 'Falling Rock'); - ObjectType.DesertPlantHasCollision = new ObjectType(id++, 908, 'Desert Plant (has collision)'); - ObjectType.DesertFixedTypeBoxBreakableCrystals = new ObjectType(id++, 909, 'Desert Fixed Type Box (Breakable Crystals)'); - ObjectType.UnknownItem910 = new ObjectType(id++, 910, 'Unknown Item (910)'); - ObjectType.BeeHive = new ObjectType(id++, 911, 'Bee Hive'); - ObjectType.UnknownItem912 = new ObjectType(id++, 912, 'Unknown Item (912)'); - ObjectType.Heat = new ObjectType(id++, 913, 'Heat'); - ObjectType.TopOfSaintMillionEgg = new ObjectType(id++, 960, 'Top of saint million egg'); - ObjectType.UnknownItem961 = new ObjectType(id++, 961, 'Unknown Item (961)'); -}()); diff --git a/src/domain/index.ts b/src/domain/index.ts deleted file mode 100644 index 670fa845..00000000 --- a/src/domain/index.ts +++ /dev/null @@ -1,381 +0,0 @@ -import { computed, observable } from 'mobx'; -import { Object3D } from 'three'; -import { ArrayBufferCursor } from '../bin-data/ArrayBufferCursor'; -import { DatNpc, DatObject, DatUnknown } from '../bin-data/parsing/dat'; -import { NpcType } from './NpcType'; -import { ObjectType } from './ObjectType'; -import { enumValues } from '../enums'; - -export { NpcType } from './NpcType'; -export { ObjectType } from './ObjectType'; - -export const RARE_ENEMY_PROB = 1 / 512; -export const KONDRIEU_PROB = 1 / 512; - -export enum Server { - Ephinea = 'Ephinea' -} - -export const Servers: Server[] = enumValues(Server); - -export enum Episode { - I = 1, - II = 2, - IV = 4 -} - -export function checkEpisode(episode: Episode) { - if (!Episode[episode]) { - throw new Error(`Invalid episode ${episode}.`); - } -} - -export enum SectionId { - Viridia = 'Viridia', - Greenill = 'Greenill', - Skyly = 'Skyly', - Bluefull = 'Bluefull', - Purplenum = 'Purplenum', - Pinkal = 'Pinkal', - Redria = 'Redria', - Oran = 'Oran', - Yellowboze = 'Yellowboze', - Whitill = 'Whitill', -} - -export const SectionIds: SectionId[] = enumValues(SectionId); - -export enum Difficulty { - Normal = 'Normal', - Hard = 'Hard', - VHard = 'VHard', - Ultimate = 'Ultimate' -} - -export const Difficulties: Difficulty[] = enumValues(Difficulty); - -export class Vec3 { - x: number; - y: number; - z: number; - - constructor(x?: number, y?: number, z?: number) { - this.x = x || 0; - this.y = y || 0; - this.z = z || 0; - } - - add(v: Vec3): Vec3 { - this.x += v.x; - this.y += v.y; - this.z += v.z; - return this; - } - - clone(x?: number, y?: number, z?: number) { - return new Vec3( - typeof x === 'number' ? x : this.x, - typeof y === 'number' ? y : this.y, - typeof z === 'number' ? z : this.z); - } -}; - -export class Section { - id: number; - @observable position: Vec3; - @observable yAxisRotation: number; - - @computed get sinYAxisRotation(): number { - return Math.sin(this.yAxisRotation); - } - - @computed get cosYAxisRotation(): number { - return Math.cos(this.yAxisRotation); - } - - constructor( - id: number, - position: Vec3, - yAxisRotation: number - ) { - if (!Number.isInteger(id) || id < -1) - throw new Error(`Expected id to be an integer greater than or equal to -1, got ${id}.`); - if (!position) throw new Error('position is required.'); - if (typeof yAxisRotation !== 'number') throw new Error('yAxisRotation is required.'); - - this.id = id; - this.position = position; - this.yAxisRotation = yAxisRotation; - } -} - -export class Quest { - @observable name: string; - @observable shortDescription: string; - @observable longDescription: string; - @observable questNo?: number; - @observable episode: Episode; - @observable areaVariants: AreaVariant[]; - @observable objects: QuestObject[]; - @observable npcs: QuestNpc[]; - /** - * (Partial) raw DAT data that can't be parsed yet by Phantasmal. - */ - datUnkowns: DatUnknown[]; - /** - * (Partial) raw BIN data that can't be parsed yet by Phantasmal. - */ - binData: ArrayBufferCursor; - - constructor( - name: string, - shortDescription: string, - longDescription: string, - questNo: number | undefined, - episode: Episode, - areaVariants: AreaVariant[], - objects: QuestObject[], - npcs: QuestNpc[], - datUnknowns: DatUnknown[], - binData: ArrayBufferCursor - ) { - if (questNo != null && (!Number.isInteger(questNo) || questNo < 0)) throw new Error('questNo should be null or a non-negative integer.'); - checkEpisode(episode); - if (!objects || !(objects instanceof Array)) throw new Error('objs is required.'); - if (!npcs || !(npcs instanceof Array)) throw new Error('npcs is required.'); - - this.name = name; - this.shortDescription = shortDescription; - this.longDescription = longDescription; - this.questNo = questNo; - this.episode = episode; - this.areaVariants = areaVariants; - this.objects = objects; - this.npcs = npcs; - this.datUnkowns = datUnknowns; - this.binData = binData; - } -} - -/** - * Abstract class from which QuestNpc and QuestObject derive. - */ -export class QuestEntity { - @observable areaId: number; - - private _sectionId: number; - - @computed get sectionId(): number { - return this.section ? this.section.id : this._sectionId; - } - - @observable section?: Section; - - /** - * World position - */ - @observable position: Vec3; - - @observable rotation: Vec3; - - /** - * Section-relative position - */ - @computed get sectionPosition(): Vec3 { - let { x, y, z } = this.position; - - if (this.section) { - const relX = x - this.section.position.x; - const relY = y - this.section.position.y; - const relZ = z - this.section.position.z; - const sin = -this.section.sinYAxisRotation; - const cos = this.section.cosYAxisRotation; - const rotX = cos * relX + sin * relZ; - const rotZ = -sin * relX + cos * relZ; - x = rotX; - y = relY; - z = rotZ; - } - - return new Vec3(x, y, z); - } - - set sectionPosition(sectPos: Vec3) { - let { x: relX, y: relY, z: relZ } = sectPos; - - if (this.section) { - const sin = -this.section.sinYAxisRotation; - const cos = this.section.cosYAxisRotation; - const rotX = cos * relX - sin * relZ; - const rotZ = sin * relX + cos * relZ; - const x = rotX + this.section.position.x; - const y = relY + this.section.position.y; - const z = rotZ + this.section.position.z; - this.position = new Vec3(x, y, z); - } - } - - object3d?: Object3D; - - constructor( - areaId: number, - sectionId: number, - position: Vec3, - rotation: Vec3 - ) { - if (Object.getPrototypeOf(this) === Object.getPrototypeOf(QuestEntity)) - throw new Error('Abstract class should not be instantiated directly.'); - if (!Number.isInteger(areaId) || areaId < 0) - throw new Error(`Expected areaId to be a non-negative integer, got ${areaId}.`); - if (!Number.isInteger(sectionId) || sectionId < 0) - throw new Error(`Expected sectionId to be a non-negative integer, got ${sectionId}.`); - if (!position) throw new Error('position is required.'); - if (!rotation) throw new Error('rotation is required.'); - - this.areaId = areaId; - this._sectionId = sectionId; - this.position = position; - this.rotation = rotation; - } -} - -export class QuestObject extends QuestEntity { - @observable type: ObjectType; - /** - * The raw data from a DAT file. - */ - dat: DatObject; - - constructor( - areaId: number, - sectionId: number, - position: Vec3, - rotation: Vec3, - type: ObjectType, - dat: DatObject - ) { - super(areaId, sectionId, position, rotation); - - if (!type) throw new Error('type is required.'); - - this.type = type; - this.dat = dat; - } -} - -export class QuestNpc extends QuestEntity { - @observable type: NpcType; - /** - * The raw data from a DAT file. - */ - dat: DatNpc; - - constructor( - areaId: number, - sectionId: number, - position: Vec3, - rotation: Vec3, - type: NpcType, - dat: DatNpc - ) { - super(areaId, sectionId, position, rotation); - - if (!type) throw new Error('type is required.'); - - this.type = type; - this.dat = dat; - } -} - -export class Area { - id: number; - name: string; - order: number; - areaVariants: AreaVariant[]; - - constructor(id: number, name: string, order: number, areaVariants: AreaVariant[]) { - if (!Number.isInteger(id) || id < 0) - throw new Error(`Expected id to be a non-negative integer, got ${id}.`); - if (!name) throw new Error('name is required.'); - if (!areaVariants) throw new Error('areaVariants is required.'); - - this.id = id; - this.name = name; - this.order = order; - this.areaVariants = areaVariants; - } -} - -export class AreaVariant { - @observable sections: Section[] = []; - - constructor(public id: number, public area: Area) { - if (!Number.isInteger(id) || id < 0) - throw new Error(`Expected id to be a non-negative integer, got ${id}.`); - } -} - -export class Item { - constructor(public name: string) { } -} - -type ItemDrop = { - item: Item, - anythingRate: number, - rareRate: number -} - -export class EnemyDrop implements ItemDrop { - rate: number; - - constructor( - public item: Item, - public anythingRate: number, - public rareRate: number - ) { - this.rate = anythingRate * rareRate; - } -} - -export class HuntMethod { - readonly npcs: Array; - readonly enemies: Array; - readonly enemyCounts: Map; - - constructor( - /** - * The time it takes to complete the quest in hours. - */ - public readonly time: number, - public readonly name: string, - public readonly quest: SimpleQuest - ) { - if (time <= 0) throw new Error('time must be greater than zero.'); - - this.npcs = this.quest.npcs; - this.enemies = this.npcs.filter(npc => npc.type.enemy); - this.enemyCounts = new Map(); - - for (const npc of this.enemies) { - this.enemyCounts.set(npc.type, (this.enemyCounts.get(npc.type) || 0) + 1); - } - } -} - -export class SimpleQuest { - constructor( - public readonly name: string, - public readonly npcs: SimpleNpc[] - ) { - if (!name) throw new Error('name is required.'); - if (!npcs) throw new Error('npcs is required.'); - } -} - -export class SimpleNpc { - constructor( - public type: NpcType - ) { - if (!type) throw new Error('type is required.'); - } -} diff --git a/src/dto.ts b/src/dto.ts deleted file mode 100644 index 212d6800..00000000 --- a/src/dto.ts +++ /dev/null @@ -1,24 +0,0 @@ -import { Difficulty, SectionId } from "./domain"; - -export type ItemDto = { - name: string, -} - -export type EnemyDropDto = { - difficulty: Difficulty, - episode: number, - sectionId: SectionId, - enemy: string, - item: string, - dropRate: number, - rareRate: number, -} - -export type BoxDropDto = { - difficulty: Difficulty, - episode: number, - sectionId: SectionId, - box: string, - item: string, - dropRate: number, -} diff --git a/src/enums.ts b/src/enums.ts deleted file mode 100644 index ba634ed7..00000000 --- a/src/enums.ts +++ /dev/null @@ -1,34 +0,0 @@ -export function enumValues(e: any): E[] { - const values = Object.values(e); - const numberValues = values.filter(v => typeof v === 'number'); - - if (numberValues.length) { - return numberValues as any as E[]; - } else { - return values as any as E[]; - } -} - -export function enumNames(e: any): string[] { - return Object.keys(e).filter(k => typeof (e as any)[k] === 'string'); -} - -/** - * Map with a guaranteed value per enum key. - */ -export class EnumMap { - private keys: K[]; - private values = new Map(); - - constructor(enum_: any, initialValue: (key: K) => V) { - this.keys = enumValues(enum_); - - for (const key of this.keys) { - this.values.set(key, initialValue(key)); - } - } - - get(key: K): V { - return this.values.get(key)!; - } -} diff --git a/src/index.less b/src/index.less deleted file mode 100644 index da1c2519..00000000 --- a/src/index.less +++ /dev/null @@ -1,40 +0,0 @@ -@import '~antd/dist/antd.less'; -@import 'ui/theme.less'; - -#phantasmal-world-root { - position: absolute; - top: 0; - bottom: 0; - left: 0; - right: 0; -} - -* { - scrollbar-color: @scrollbar-thumb-color @scrollbar-color; -} - -::-webkit-scrollbar { - background-color: @scrollbar-color; -} - -::-webkit-scrollbar-track { - background-color: @scrollbar-color; -} - -::-webkit-scrollbar-thumb { - background-color: @scrollbar-thumb-color; -} - -::-webkit-scrollbar-corner { - background-color: @scrollbar-color; -} - -#phantasmal-world-root { - & .ReactVirtualized__Grid { - outline: none; - } - - & .ReactVirtualized__Table__headerRow { - text-transform: none; - } -} diff --git a/src/index.tsx b/src/index.tsx deleted file mode 100644 index 073373f1..00000000 --- a/src/index.tsx +++ /dev/null @@ -1,10 +0,0 @@ -import React from 'react'; -import ReactDOM from 'react-dom'; -import './index.less'; -import { ApplicationComponent } from './ui/ApplicationComponent'; -import 'react-virtualized/styles.css'; - -ReactDOM.render( - , - document.getElementById('phantasmal-world-root') -); diff --git a/src/javascript-lp-solver.d.ts b/src/javascript-lp-solver.d.ts deleted file mode 100644 index 012468a7..00000000 --- a/src/javascript-lp-solver.d.ts +++ /dev/null @@ -1 +0,0 @@ -declare module 'javascript-lp-solver'; \ No newline at end of file diff --git a/src/react-app-env.d.ts b/src/react-app-env.d.ts deleted file mode 100644 index 6431bc5f..00000000 --- a/src/react-app-env.d.ts +++ /dev/null @@ -1 +0,0 @@ -/// diff --git a/src/rendering/Renderer.ts b/src/rendering/Renderer.ts deleted file mode 100644 index a825e902..00000000 --- a/src/rendering/Renderer.ts +++ /dev/null @@ -1,455 +0,0 @@ -import * as THREE from 'three'; -import { Color, HemisphereLight, Intersection, Mesh, MeshLambertMaterial, MOUSE, Object3D, PerspectiveCamera, Plane, Raycaster, Scene, Vector2, Vector3, WebGLRenderer } from 'three'; -import OrbitControlsCreator from 'three-orbit-controls'; -import { getAreaCollisionGeometry, getAreaRenderGeometry } from '../bin-data/loading/areas'; -import { Area, Quest, QuestEntity, QuestNpc, QuestObject, Section, Vec3 } from '../domain'; -import { questEditorStore } from '../stores/QuestEditorStore'; -import { NPC_COLOR, NPC_HOVER_COLOR, NPC_SELECTED_COLOR, OBJECT_COLOR, OBJECT_HOVER_COLOR, OBJECT_SELECTED_COLOR } from './entities'; - -const OrbitControls = OrbitControlsCreator(THREE); - -interface PickEntityResult { - object: Mesh; - entity: QuestEntity; - grabOffset: Vector3; - dragAdjust: Vector3; - dragY: number; - manipulating: boolean; -} - -let renderer: Renderer | undefined; - -export function getRenderer(): Renderer { - if (!renderer) { - renderer = new Renderer(); - } - - return renderer; -} - -/** - * Renders a quest area or an NJ/XJ model. - */ -export class Renderer { - private renderer = new WebGLRenderer({ antialias: true }); - private camera: PerspectiveCamera; - private controls: any; - private raycaster = new Raycaster(); - private scene = new Scene(); - private quest?: Quest; - private questEntitiesLoaded = false; - private area?: Area; - private objs: Map = new Map(); // Objs grouped by area id - private npcs: Map = new Map(); // Npcs grouped by area id - private collisionGeometry = new Object3D(); - private renderGeometry = new Object3D(); - private objGeometry = new Object3D(); - private npcGeometry = new Object3D(); - private hoveredData?: PickEntityResult; - private selectedData?: PickEntityResult; - private model?: Object3D; - - constructor() { - this.renderer.domElement.addEventListener( - 'mousedown', this.onMouseDown); - this.renderer.domElement.addEventListener( - 'mouseup', this.onMouseUp); - this.renderer.domElement.addEventListener( - 'mousemove', this.onMouseMove); - - this.camera = new PerspectiveCamera(75, 1, 0.1, 5000); - this.controls = new OrbitControls( - this.camera, this.renderer.domElement); - this.controls.mouseButtons.ORBIT = MOUSE.RIGHT; - this.controls.mouseButtons.PAN = MOUSE.LEFT; - - this.scene.background = new Color(0x151C21); - this.scene.add(new HemisphereLight(0xffffff, 0x505050, 1)); - this.scene.add(this.objGeometry); - this.scene.add(this.npcGeometry); - - requestAnimationFrame(this.renderLoop); - } - - get domElement(): HTMLElement { - return this.renderer.domElement; - } - - setSize(width: number, height: number) { - this.renderer.setSize(width, height); - this.camera.aspect = width / height; - this.camera.updateProjectionMatrix(); - } - - setQuestAndArea(quest?: Quest, area?: Area) { - let update = false; - - if (this.area !== area) { - this.area = area; - update = true; - } - - if (this.quest !== quest) { - this.quest = quest; - - this.objs.clear(); - this.npcs.clear(); - - if (quest) { - for (const obj of quest.objects) { - const array = this.objs.get(obj.areaId) || []; - array.push(obj); - this.objs.set(obj.areaId, array); - } - - for (const npc of quest.npcs) { - const array = this.npcs.get(npc.areaId) || []; - array.push(npc); - this.npcs.set(npc.areaId, array); - } - } - - update = true; - } - - if (update) { - this.updateGeometry(); - } - } - - /** - * Renders a generic Object3D. - */ - setModel(model?: Object3D) { - if (this.model !== model) { - if (this.model) { - this.scene.remove(this.model); - } - - if (model) { - this.setQuestAndArea(undefined, undefined); - this.scene.add(model); - this.resetCamera(); - } - - this.model = model; - } - } - - private updateGeometry() { - this.scene.remove(this.objGeometry); - this.scene.remove(this.npcGeometry); - this.objGeometry = new Object3D(); - this.npcGeometry = new Object3D(); - this.scene.add(this.objGeometry); - this.scene.add(this.npcGeometry); - this.questEntitiesLoaded = false; - - this.scene.remove(this.collisionGeometry); - - if (this.quest && this.area) { - const episode = this.quest.episode; - const areaId = this.area.id; - const variant = this.quest.areaVariants.find(v => v.area.id === areaId); - const variantId = (variant && variant.id) || 0; - - getAreaCollisionGeometry(episode, areaId, variantId).then(geometry => { - if (this.quest && this.area) { - this.setModel(undefined); - this.scene.remove(this.collisionGeometry); - - this.resetCamera(); - - this.collisionGeometry = geometry; - this.scene.add(geometry); - } - }); - - getAreaRenderGeometry(episode, areaId, variantId).then(geometry => { - if (this.quest && this.area) { - this.renderGeometry = geometry; - } - }); - } - } - - private resetCamera() { - this.controls.reset(); - this.camera.position.set(0, 800, 700); - this.camera.lookAt(new Vector3(0, 0, 0)); - } - - private renderLoop = () => { - this.controls.update(); - this.addLoadedEntities(); - this.renderer.render(this.scene, this.camera); - requestAnimationFrame(this.renderLoop); - } - - private addLoadedEntities() { - if (this.quest && this.area && !this.questEntitiesLoaded) { - let loaded = true; - - for (const object of this.quest.objects) { - if (object.areaId === this.area.id) { - if (object.object3d) { - this.objGeometry.add(object.object3d); - } else { - loaded = false; - } - } - } - - for (const npc of this.quest.npcs) { - if (npc.areaId === this.area.id) { - if (npc.object3d) { - this.npcGeometry.add(npc.object3d); - } else { - loaded = false; - } - } - } - - this.questEntitiesLoaded = loaded; - } - } - - private onMouseDown = (e: MouseEvent) => { - const oldSelectedData = this.selectedData; - const data = this.pickEntity( - this.pointerPosToDeviceCoords(e)); - - // Did we pick a different object than the previously hovered over 3D object? - if (this.hoveredData && (!data || data.object !== this.hoveredData.object)) { - (this.hoveredData.object.material as MeshLambertMaterial).color.set( - this.getColor(this.hoveredData.entity, 'normal')); - } - - // Did we pick a different object than the previously selected 3D object? - if (this.selectedData && (!data || data.object !== this.selectedData.object)) { - (this.selectedData.object.material as MeshLambertMaterial).color.set( - this.getColor(this.selectedData.entity, 'normal')); - this.selectedData.manipulating = false; - } - - if (data) { - // User selected an entity. - (data.object.material as MeshLambertMaterial).color.set(this.getColor(data.entity, 'selected')); - data.manipulating = true; - this.hoveredData = data; - this.selectedData = data; - this.controls.enabled = false; - } else { - // User clicked on terrain or outside of area. - this.hoveredData = undefined; - this.selectedData = undefined; - this.controls.enabled = true; - } - - const selectionChanged = oldSelectedData && data - ? oldSelectedData.object !== data.object - : oldSelectedData !== data; - - if (selectionChanged) { - questEditorStore.setSelectedEntity(data && data.entity); - } - } - - private onMouseUp = () => { - if (this.selectedData) { - this.selectedData.manipulating = false; - this.controls.enabled = true; - } - } - - private onMouseMove = (e: MouseEvent) => { - const pointerPos = this.pointerPosToDeviceCoords(e); - - if (this.selectedData && this.selectedData.manipulating) { - if (e.buttons === 1) { - // User is dragging a selected entity. - const data = this.selectedData; - - if (e.shiftKey) { - // Vertical movement. - // We intersect with a plane that's oriented toward the camera and that's coplanar with the point where the entity was grabbed. - this.raycaster.setFromCamera(pointerPos, this.camera); - const ray = this.raycaster.ray; - const negativeWorldDir = this.camera.getWorldDirection(new Vector3()).negate(); - const plane = new Plane().setFromNormalAndCoplanarPoint( - new Vector3(negativeWorldDir.x, 0, negativeWorldDir.z).normalize(), - data.object.position.sub(data.grabOffset)); - const intersectionPoint = new Vector3(); - - if (ray.intersectPlane(plane, intersectionPoint)) { - const y = intersectionPoint.y + data.grabOffset.y; - const yDelta = y - data.entity.position.y; - data.dragY += yDelta; - data.dragAdjust.y -= yDelta; - data.entity.position = new Vec3( - data.entity.position.x, - y, - data.entity.position.z - ); - } - } else { - // Horizontal movement accross terrain. - // Cast ray adjusted for dragging entities. - const { intersection, section } = this.pickTerrain(pointerPos, data); - - if (intersection) { - data.entity.position = new Vec3( - intersection.point.x, - intersection.point.y + data.dragY, - intersection.point.z - ); - data.entity.section = section; - } else { - // If the cursor is not over any terrain, we translate the entity accross the horizontal plane in which the entity's origin lies. - this.raycaster.setFromCamera(pointerPos, this.camera); - const ray = this.raycaster.ray; - // ray.origin.add(data.dragAdjust); - const plane = new Plane( - new Vector3(0, 1, 0), - -data.entity.position.y + data.grabOffset.y); - const intersectionPoint = new Vector3(); - - if (ray.intersectPlane(plane, intersectionPoint)) { - data.entity.position = new Vec3( - intersectionPoint.x + data.grabOffset.x, - data.entity.position.y, - intersectionPoint.z + data.grabOffset.z - ); - } - } - } - } - } else { - // User is hovering. - const oldData = this.hoveredData; - const data = this.pickEntity(pointerPos); - - if (oldData && (!data || data.object !== oldData.object)) { - if (!this.selectedData || oldData.object !== this.selectedData.object) { - (oldData.object.material as MeshLambertMaterial).color.set( - this.getColor(oldData.entity, 'normal')); - } - - this.hoveredData = undefined; - } - - if (data && (!oldData || data.object !== oldData.object)) { - if (!this.selectedData || data.object !== this.selectedData.object) { - (data.object.material as MeshLambertMaterial).color.set( - this.getColor(data.entity, 'hover')); - } - - this.hoveredData = data; - } - } - } - - private pointerPosToDeviceCoords(e: MouseEvent) { - const coords = new Vector2(); - this.renderer.getSize(coords); - coords.width = e.offsetX / coords.width * 2 - 1; - coords.height = e.offsetY / coords.height * -2 + 1; - return coords; - } - - /** - * @param pointerPos - pointer coordinates in normalized device space - */ - private pickEntity(pointerPos: Vector2): PickEntityResult | undefined { - // Find the nearest object and NPC under the pointer. - this.raycaster.setFromCamera(pointerPos, this.camera); - const [nearestObject] = this.raycaster.intersectObjects( - this.objGeometry.children - ); - const [nearestNpc] = this.raycaster.intersectObjects( - this.npcGeometry.children - ); - - if (!nearestObject && !nearestNpc) { - return; - } - - const objectDist = nearestObject ? nearestObject.distance : Infinity; - const npcDist = nearestNpc ? nearestNpc.distance : Infinity; - const intersection = objectDist < npcDist ? nearestObject : nearestNpc; - - const entity = intersection.object.userData.entity; - // Vector that points from the grabbing point to the model's origin. - const grabOffset = intersection.object.position - .clone() - .sub(intersection.point); - // Vector that points from the grabbing point to the terrain point directly under the model's origin. - const dragAdjust = grabOffset.clone(); - // Distance to terrain. - let dragY = 0; - - // Find vertical distance to terrain. - this.raycaster.set( - intersection.object.position, new Vector3(0, -1, 0) - ); - const [terrain] = this.raycaster.intersectObjects( - this.collisionGeometry.children, true - ); - - if (terrain) { - dragAdjust.sub(new Vector3(0, terrain.distance, 0)); - dragY += terrain.distance; - } - - return { - object: intersection.object as Mesh, - entity, - grabOffset, - dragAdjust, - dragY, - manipulating: false - }; - } - - /** - * @param pointerPos - pointer coordinates in normalized device space - */ - private pickTerrain(pointerPos: Vector2, data: PickEntityResult): { intersection?: Intersection, section?: Section } { - this.raycaster.setFromCamera(pointerPos, this.camera); - this.raycaster.ray.origin.add(data.dragAdjust); - const terrains = this.raycaster.intersectObjects( - this.collisionGeometry.children, true); - - // Don't allow entities to be placed on very steep terrain. - // E.g. walls. - // TODO: make use of the flags field in the collision data. - for (const terrain of terrains) { - if (terrain.face!.normal.y > 0.75) { - // Find section ID. - this.raycaster.set( - terrain.point.clone().setY(1000), new Vector3(0, -1, 0)); - const renderTerrains = this.raycaster - .intersectObjects(this.renderGeometry.children, true) - .filter(rt => rt.object.userData.section.id >= 0); - - return { - intersection: terrain, - section: renderTerrains[0] && renderTerrains[0].object.userData.section - }; - } - } - - return {}; - } - - private getColor(entity: QuestEntity, type: 'normal' | 'hover' | 'selected') { - const isNpc = entity instanceof QuestNpc; - - switch (type) { - default: - case 'normal': return isNpc ? NPC_COLOR : OBJECT_COLOR; - case 'hover': return isNpc ? NPC_HOVER_COLOR : OBJECT_HOVER_COLOR; - case 'selected': return isNpc ? NPC_SELECTED_COLOR : OBJECT_SELECTED_COLOR; - } - } -} diff --git a/src/rendering/entities.test.ts b/src/rendering/entities.test.ts deleted file mode 100644 index e49fd2d9..00000000 --- a/src/rendering/entities.test.ts +++ /dev/null @@ -1,48 +0,0 @@ -import { CylinderBufferGeometry, MeshLambertMaterial, Object3D, Vector3 } from 'three'; -import { DatNpc, DatObject } from '../bin-data/parsing/dat'; -import { NpcType, ObjectType, QuestNpc, QuestObject, Vec3 } from '../domain'; -import { createNpcMesh, createObjectMesh, NPC_COLOR, OBJECT_COLOR } from './entities'; - -const cylinder = new CylinderBufferGeometry(3, 3, 20).translate(0, 10, 0); - -test('create geometry for quest objects', () => { - const object = new QuestObject(7, 13, new Vec3(17, 19, 23), new Vec3(), ObjectType.PrincipalWarp, {} as DatObject); - const geometry = createObjectMesh(object, cylinder); - - expect(geometry).toBeInstanceOf(Object3D); - expect(geometry.name).toBe('Object'); - expect(geometry.userData.entity).toBe(object); - expect(geometry.position.x).toBe(17); - expect(geometry.position.y).toBe(19); - expect(geometry.position.z).toBe(23); - expect((geometry.material as MeshLambertMaterial).color.getHex()).toBe(OBJECT_COLOR); -}); - -test('create geometry for quest NPCs', () => { - const npc = new QuestNpc(7, 13, new Vec3(17, 19, 23), new Vec3(), NpcType.Booma, {} as DatNpc); - const geometry = createNpcMesh(npc, cylinder); - - expect(geometry).toBeInstanceOf(Object3D); - expect(geometry.name).toBe('NPC'); - expect(geometry.userData.entity).toBe(npc); - expect(geometry.position.x).toBe(17); - expect(geometry.position.y).toBe(19); - expect(geometry.position.z).toBe(23); - expect((geometry.material as MeshLambertMaterial).color.getHex()).toBe(NPC_COLOR); -}); - -test('geometry position changes when entity position changes element-wise', () => { - const npc = new QuestNpc(7, 13, new Vec3(17, 19, 23), new Vec3(), NpcType.Booma, {} as DatNpc); - const geometry = createNpcMesh(npc, cylinder); - npc.position = new Vec3(2, 3, 5).add(npc.position); - - expect(geometry.position).toEqual(new Vector3(19, 22, 28)); -}); - -test('geometry position changes when entire entity position changes', () => { - const npc = new QuestNpc(7, 13, new Vec3(17, 19, 23), new Vec3(), NpcType.Booma, {} as DatNpc); - const geometry = createNpcMesh(npc, cylinder); - npc.position = new Vec3(2, 3, 5); - - expect(geometry.position).toEqual(new Vector3(2, 3, 5)); -}); diff --git a/src/rendering/entities.ts b/src/rendering/entities.ts deleted file mode 100644 index e970991c..00000000 --- a/src/rendering/entities.ts +++ /dev/null @@ -1,45 +0,0 @@ -import { autorun } from 'mobx'; -import { BufferGeometry, DoubleSide, Mesh, MeshLambertMaterial } from 'three'; -import { QuestNpc, QuestObject, QuestEntity } from '../domain'; - -export const OBJECT_COLOR = 0xFFFF00; -export const OBJECT_HOVER_COLOR = 0xFFDF3F; -export const OBJECT_SELECTED_COLOR = 0xFFAA00; -export const NPC_COLOR = 0xFF0000; -export const NPC_HOVER_COLOR = 0xFF3F5F; -export const NPC_SELECTED_COLOR = 0xFF0054; - -export function createObjectMesh(object: QuestObject, geometry: BufferGeometry): Mesh { - return createMesh(object, geometry, OBJECT_COLOR, 'Object'); -} - -export function createNpcMesh(npc: QuestNpc, geometry: BufferGeometry): Mesh { - return createMesh(npc, geometry, NPC_COLOR, 'NPC'); -} - -function createMesh( - entity: QuestEntity, - geometry: BufferGeometry, - color: number, - type: string -): Mesh { - const object3d = new Mesh( - geometry, - new MeshLambertMaterial({ - color, - side: DoubleSide - }) - ); - object3d.name = type; - object3d.userData.entity = entity; - - // TODO: dispose autorun? - autorun(() => { - const { x, y, z } = entity.position; - object3d.position.set(x, y, z); - const rot = entity.rotation; - object3d.rotation.set(rot.x, rot.y, rot.z); - }); - - return object3d; -} diff --git a/src/rendering/models.ts b/src/rendering/models.ts deleted file mode 100644 index 856dff11..00000000 --- a/src/rendering/models.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { BufferGeometry, DoubleSide, Mesh, MeshLambertMaterial } from 'three'; - -export function createModelMesh(geometry?: BufferGeometry): Mesh | undefined { - return geometry && new Mesh( - geometry, - new MeshLambertMaterial({ - color: 0xFF00FF, - side: DoubleSide - }) - ); -} diff --git a/src/stores/ApplicationStore.ts b/src/stores/ApplicationStore.ts deleted file mode 100644 index be837fff..00000000 --- a/src/stores/ApplicationStore.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { observable } from "mobx"; -import { Server } from "../domain"; - -class ApplicationStore { - @observable currentServer: Server = Server.Ephinea; -} - -export const applicationStore = new ApplicationStore(); diff --git a/src/stores/AreaStore.ts b/src/stores/AreaStore.ts deleted file mode 100644 index 66974fbf..00000000 --- a/src/stores/AreaStore.ts +++ /dev/null @@ -1,86 +0,0 @@ -import { Area, AreaVariant } from '../domain'; - -function area(id: number, name: string, order: number, variants: number) { - const area = new Area(id, name, order, []); - const varis = Array(variants).fill(null).map((_, i) => new AreaVariant(i, area)); - area.areaVariants.splice(0, 0, ...varis); - return area; -} - -class AreaStore { - areas: Area[][]; - - constructor() { - // The IDs match the PSO IDs for areas. - this.areas = []; - let order = 0; - this.areas[1] = [ - area(0, 'Pioneer II', order++, 1), - area(1, 'Forest 1', order++, 1), - area(2, 'Forest 2', order++, 1), - area(11, 'Under the Dome', order++, 1), - area(3, 'Cave 1', order++, 6), - area(4, 'Cave 2', order++, 5), - area(5, 'Cave 3', order++, 6), - area(12, 'Underground Channel', order++, 1), - area(6, 'Mine 1', order++, 6), - area(7, 'Mine 2', order++, 6), - area(13, 'Monitor Room', order++, 1), - area(8, 'Ruins 1', order++, 5), - area(9, 'Ruins 2', order++, 5), - area(10, 'Ruins 3', order++, 5), - area(14, 'Dark Falz', order++, 1) - ]; - order = 0; - this.areas[2] = [ - area(0, 'Lab', order++, 1), - area(1, 'VR Temple Alpha', order++, 3), - area(2, 'VR Temple Beta', order++, 3), - area(14, 'VR Temple Final', order++, 1), - area(3, 'VR Spaceship Alpha', order++, 3), - area(4, 'VR Spaceship Beta', order++, 3), - area(15, 'VR Spaceship Final', order++, 1), - area(5, 'Central Control Area', order++, 1), - area(6, 'Jungle Area East', order++, 1), - area(7, 'Jungle Area North', order++, 1), - area(8, 'Mountain Area', order++, 3), - area(9, 'Seaside Area', order++, 1), - area(12, 'Cliffs of Gal Da Val', order++, 1), - area(10, 'Seabed Upper Levels', order++, 3), - area(11, 'Seabed Lower Levels', order++, 3), - area(13, 'Test Subject Disposal Area', order++, 1), - area(16, 'Seaside Area at Night', order++, 1), - area(17, 'Control Tower', order++, 5) - ]; - order = 0; - this.areas[4] = [ - area(0, 'Pioneer II (Ep. IV)', order++, 1), - area(1, 'Crater Route 1', order++, 1), - area(2, 'Crater Route 2', order++, 1), - area(3, 'Crater Route 3', order++, 1), - area(4, 'Crater Route 4', order++, 1), - area(5, 'Crater Interior', order++, 1), - area(6, 'Subterranean Desert 1', order++, 3), - area(7, 'Subterranean Desert 2', order++, 3), - area(8, 'Subterranean Desert 3', order++, 3), - area(9, 'Meteor Impact Site', order++, 1) - ]; - } - - getVariant(episode: number, areaId: number, variantId: number) { - if (episode !== 1 && episode !== 2 && episode !== 4) - throw new Error(`Expected episode to be 1, 2 or 4, got ${episode}.`); - - const area = this.areas[episode].find(a => a.id === areaId); - if (!area) - throw new Error(`Area id ${areaId} for episode ${episode} is invalid.`); - - const areaVariant = area.areaVariants[variantId]; - if (!areaVariant) - throw new Error(`Area variant id ${variantId} for area ${areaId} of episode ${episode} is invalid.`); - - return areaVariant; - } -} - -export const areaStore = new AreaStore(); diff --git a/src/stores/HuntMethodStore.ts b/src/stores/HuntMethodStore.ts deleted file mode 100644 index 956c8c33..00000000 --- a/src/stores/HuntMethodStore.ts +++ /dev/null @@ -1,55 +0,0 @@ -import { observable } from "mobx"; -import { HuntMethod, NpcType, Server, SimpleNpc, SimpleQuest } from "../domain"; -import { Loadable } from "../Loadable"; -import { ServerMap } from "./ServerMap"; - -class HuntMethodStore { - @observable methods: ServerMap>> = new ServerMap(server => - new Loadable([], () => this.loadHuntMethods(server)) - ); - - private async loadHuntMethods(server: Server): Promise { - const response = await fetch( - `${process.env.PUBLIC_URL}/quests.${Server[server].toLowerCase()}.tsv` - ); - const data = await response.text(); - const rows = data.split('\n').map(line => line.split('\t')); - - const npcTypeByIndex = rows[0].slice(2, -2).map((episode, i) => { - const enemy = rows[1][i + 2]; - return NpcType.byNameAndEpisode(enemy, parseInt(episode, 10))!; - }); - - return rows.slice(2).map(row => { - const questName = row[0]; - const time = parseFloat(row[1]); - - const npcs = row.slice(2, -2).flatMap((cell, cellI) => { - const amount = parseInt(cell, 10); - const type = npcTypeByIndex[cellI]; - const enemies = []; - - if (type) { - for (let i = 0; i < amount; i++) { - enemies.push(new SimpleNpc(type)); - } - } else { - console.error(`Couldn't get type for cellI ${cellI}.`); - } - - return enemies; - }); - - return new HuntMethod( - time, - questName, - new SimpleQuest( - questName, - npcs - ) - ); - }); - } -} - -export const huntMethodStore = new HuntMethodStore(); diff --git a/src/stores/HuntOptimizerStore.ts b/src/stores/HuntOptimizerStore.ts deleted file mode 100644 index 8b12f076..00000000 --- a/src/stores/HuntOptimizerStore.ts +++ /dev/null @@ -1,274 +0,0 @@ -import solver from 'javascript-lp-solver'; -import { autorun, IObservableArray, observable, runInAction } from "mobx"; -import { Difficulties, Difficulty, HuntMethod, Item, KONDRIEU_PROB, NpcType, RARE_ENEMY_PROB, SectionId, SectionIds } from "../domain"; -import { applicationStore } from './ApplicationStore'; -import { huntMethodStore } from "./HuntMethodStore"; -import { itemDropStore } from './ItemDropStore'; -import { itemStore } from './ItemStore'; - -export class WantedItem { - @observable readonly item: Item; - @observable amount: number; - - constructor(item: Item, amount: number) { - this.item = item; - this.amount = amount; - } -} - -export class OptimizationResult { - public readonly totalTime: number; - - constructor( - public readonly difficulty: Difficulty, - public readonly sectionId: SectionId, - public readonly methodName: string, - public readonly methodTime: number, - public readonly runs: number, - public readonly itemCounts: Map - ) { - this.totalTime = runs * methodTime; - } -} - -// TODO: Prefer methods that don't split pan arms over methods that do. -// For some reason this doesn't actually seem to be a problem, should probably investigate. -// TODO: group similar methods (e.g. same difficulty, same quest and similar ID). -// This way people can choose their preferred section ID. -// TODO: order of items in results table should match order in wanted table. -// TODO: boxes. -class HuntOptimizerStore { - @observable readonly wantedItems: IObservableArray = observable.array(); - @observable readonly results: IObservableArray = observable.array(); - - constructor() { - this.initialize(); - } - - initialize = async () => { - try { - await this.loadFromLocalStorage(); - autorun(this.storeInLocalStorage); - } catch (e) { - console.error(e); - } - } - - loadFromLocalStorage = async () => { - const wantedItemsJson = localStorage.getItem( - `HuntOptimizerStore.wantedItems.${applicationStore.currentServer}` - ); - - if (wantedItemsJson) { - const items = await itemStore.items.current.promise; - const wi = JSON.parse(wantedItemsJson); - - const wantedItems: WantedItem[] = []; - - for (const { itemName, amount } of wi) { - const item = items.find(item => item.name === itemName); - - if (item) { - wantedItems.push(new WantedItem(item, amount)); - } - } - - this.wantedItems.replace(wantedItems); - } - } - - storeInLocalStorage = () => { - try { - localStorage.setItem( - `HuntOptimizerStore.wantedItems.${applicationStore.currentServer}`, - JSON.stringify( - this.wantedItems.map(({ item, amount }) => ({ itemName: item.name, amount })) - ) - ); - } catch (e) { - console.error(e); - } - } - - optimize = async () => { - if (!this.wantedItems.length) { - this.results.splice(0); - return; - } - - const methods = await huntMethodStore.methods.current.promise; - const dropTable = await itemDropStore.enemyDrops.current.promise; - - // Add a constraint per wanted item. - const constraints: { [itemName: string]: { min: number } } = {}; - - for (const wanted of this.wantedItems) { - constraints[wanted.item.name] = { min: wanted.amount }; - } - - // Add a variable to the LP model per method per difficulty per section ID. - // When a method with pan arms is encountered, two variables are added. One for the method - // with migiums and hidooms and one with pan arms. - // Each variable has a time property to minimize and a property per item with the number - // of enemies that drop the item multiplied by the corresponding drop rate as its value. - type Variable = { - time: number, - [itemName: string]: number, - } - const variables: { [methodName: string]: Variable } = {}; - - type VariableDetails = { - method: HuntMethod, - difficulty: Difficulty, - sectionId: SectionId, - splitPanArms: boolean, - } - const variableDetails: Map = new Map(); - - const wantedItems = new Set(this.wantedItems.filter(w => w.amount > 0).map(w => w.item)); - - for (const method of methods) { - // Counts include rare enemies, so they are fractional. - const counts = new Map(); - - for (const enemy of method.enemies) { - const count = counts.get(enemy.type); - - if (enemy.type.rareType == null) { - counts.set(enemy.type, (count || 0) + 1); - } else { - let rate, rareRate; - - if (enemy.type.rareType === NpcType.Kondrieu) { - rate = 1 - KONDRIEU_PROB; - rareRate = KONDRIEU_PROB; - } else { - rate = 1 - RARE_ENEMY_PROB; - rareRate = RARE_ENEMY_PROB; - } - - counts.set(enemy.type, (count || 0) + rate); - - const rareCount = counts.get(enemy.type.rareType); - counts.set(enemy.type.rareType, (rareCount || 0) + rareRate); - } - } - - // Create a secondary counts map if there are any pan arms that can be split into - // migiums and hidooms. - const countsList: Array> = [counts]; - const panArmsCount = counts.get(NpcType.PanArms); - const panArms2Count = counts.get(NpcType.PanArms2); - - if (panArmsCount || panArms2Count) { - const splitCounts = new Map(counts); - - if (panArmsCount) { - splitCounts.delete(NpcType.PanArms); - splitCounts.set(NpcType.Migium, panArmsCount); - splitCounts.set(NpcType.Hidoom, panArmsCount); - } - - if (panArms2Count) { - splitCounts.delete(NpcType.PanArms2); - splitCounts.set(NpcType.Migium2, panArms2Count); - splitCounts.set(NpcType.Hidoom2, panArms2Count); - } - - countsList.push(splitCounts); - } - - for (let i = 0; i < countsList.length; i++) { - const counts = countsList[i]; - const splitPanArms = i === 1; - - for (const diff of Difficulties) { - for (const sectionId of SectionIds) { - const variable: Variable = { - time: method.time - }; - let addVariable = false; - - for (const [npcType, count] of counts.entries()) { - const drop = dropTable.getDrop(diff, sectionId, npcType); - - if (drop && wantedItems.has(drop.item)) { - const value = variable[drop.item.name] || 0; - variable[drop.item.name] = value + count * drop.rate; - addVariable = true; - } - } - - if (addVariable) { - let name = `${diff}\t${sectionId}\t${method.name}`; - - if (splitPanArms) { - name += ' (Split Pan Arms)'; - } - - variables[name] = variable; - variableDetails.set(name, { - method, - difficulty: diff, - sectionId, - splitPanArms - }); - } - } - } - } - } - - const result: { - feasible: boolean, - bounded: boolean, - result: number, - [method: string]: number | boolean - } = solver.Solve({ - optimize: 'time', - opType: 'min', - constraints, - variables - }); - - runInAction(() => { - this.results.splice(0); - - if (!result.feasible) { - return; - } - - for (const [variableName, runsOrOther] of Object.entries(result)) { - const details = variableDetails.get(variableName); - - if (details) { - const { method, difficulty, sectionId, splitPanArms } = details; - const runs = runsOrOther as number; - const variable = variables[variableName]; - - const items = new Map(); - - for (const [itemName, expectedValue] of Object.entries(variable)) { - for (const item of wantedItems) { - if (itemName === item.name) { - items.set(item, runs * expectedValue); - break; - } - } - } - - this.results.push(new OptimizationResult( - difficulty, - sectionId, - method.name + (splitPanArms ? ' (Split Pan Arms)' : ''), - method.time, - runs, - items - )); - } - } - }); - } -} - -export const huntOptimizerStore = new HuntOptimizerStore(); diff --git a/src/stores/ItemDropStore.ts b/src/stores/ItemDropStore.ts deleted file mode 100644 index 85e788b2..00000000 --- a/src/stores/ItemDropStore.ts +++ /dev/null @@ -1,54 +0,0 @@ -import { observable } from "mobx"; -import { Difficulty, EnemyDrop, NpcType, SectionId, Server } from "../domain"; -import { EnumMap } from "../enums"; -import { Loadable } from "../Loadable"; -import { itemStore } from "./ItemStore"; -import { ServerMap } from "./ServerMap"; -import { EnemyDropDto } from "../dto"; - -class EnemyDropTable { - private map: EnumMap>> = - new EnumMap(Difficulty, () => new EnumMap(SectionId, () => new Map())); - - getDrop(difficulty: Difficulty, sectionId: SectionId, npcType: NpcType): EnemyDrop | undefined { - return this.map.get(difficulty).get(sectionId).get(npcType); - } - - setDrop(difficulty: Difficulty, sectionId: SectionId, npcType: NpcType, drop: EnemyDrop) { - this.map.get(difficulty).get(sectionId).set(npcType, drop); - } -} - -class ItemDropStore { - @observable enemyDrops: ServerMap> = new ServerMap(server => - new Loadable(new EnemyDropTable(), () => this.loadEnemyDrops(server)) - ); - - private loadEnemyDrops = async (server: Server): Promise => { - const response = await fetch( - `${process.env.PUBLIC_URL}/enemyDrops.${Server[server].toLowerCase()}.json` - ); - const data: Array = await response.json(); - - const drops = new EnemyDropTable(); - - for (const dropDto of data) { - const npcType = NpcType.byNameAndEpisode(dropDto.enemy, dropDto.episode); - - if (!npcType) { - console.error(`Couldn't determine NpcType of episode ${dropDto.episode} ${dropDto.enemy}.`); - continue; - } - - drops.setDrop(dropDto.difficulty, dropDto.sectionId, npcType, new EnemyDrop( - itemStore.dedupItem(dropDto.item), - dropDto.dropRate, - dropDto.rareRate - )); - } - - return drops; - } -} - -export const itemDropStore = new ItemDropStore(); diff --git a/src/stores/ItemStore.ts b/src/stores/ItemStore.ts deleted file mode 100644 index 35e5e6b6..00000000 --- a/src/stores/ItemStore.ts +++ /dev/null @@ -1,33 +0,0 @@ -import { observable } from "mobx"; -import { Item, Server } from "../domain"; -import { Loadable } from "../Loadable"; -import { ServerMap } from "./ServerMap"; -import { ItemDto } from "../dto"; - -class ItemStore { - private itemMap = new Map(); - - @observable items: ServerMap>> = new ServerMap(server => - new Loadable([], () => this.loadItems(server)) - ); - - dedupItem = (name: string): Item => { - let item = this.itemMap.get(name); - - if (!item) { - this.itemMap.set(name, item = new Item(name)); - } - - return item; - } - - private async loadItems(server: Server): Promise { - const response = await fetch( - `${process.env.PUBLIC_URL}/items.${Server[server].toLowerCase()}.json` - ); - const data: Array = await response.json(); - return data.map(({ name }) => this.dedupItem(name)); - } -} - -export const itemStore = new ItemStore(); diff --git a/src/stores/QuestEditorStore.ts b/src/stores/QuestEditorStore.ts deleted file mode 100644 index 3f649337..00000000 --- a/src/stores/QuestEditorStore.ts +++ /dev/null @@ -1,150 +0,0 @@ -import { observable, action } from 'mobx'; -import { Object3D } from 'three'; -import { ArrayBufferCursor } from '../bin-data/ArrayBufferCursor'; -import { getAreaSections } from '../bin-data/loading/areas'; -import { getNpcGeometry, getObjectGeometry } from '../bin-data/loading/entities'; -import { parseNj, parseXj } from '../bin-data/parsing/ninja'; -import { parseQuest, writeQuestQst } from '../bin-data/parsing/quest'; -import { Area, Quest, QuestEntity, Section, Vec3 } from '../domain'; -import { createNpcMesh, createObjectMesh } from '../rendering/entities'; -import { createModelMesh } from '../rendering/models'; - -class QuestEditorStore { - @observable currentModel?: Object3D; - @observable currentQuest?: Quest; - @observable currentArea?: Area; - @observable selectedEntity?: QuestEntity; - - setModel = action('setModel', (model?: Object3D) => { - this.resetModelAndQuestState(); - this.currentModel = model; - }) - - setQuest = action('setQuest', (quest?: Quest) => { - this.resetModelAndQuestState(); - this.currentQuest = quest; - - if (quest && quest.areaVariants.length) { - this.currentArea = quest.areaVariants[0].area; - } - }) - - private resetModelAndQuestState() { - this.currentQuest = undefined; - this.currentArea = undefined; - this.selectedEntity = undefined; - this.currentModel = undefined; - } - - setSelectedEntity = (entity?: QuestEntity) => { - this.selectedEntity = entity; - } - - setCurrentAreaId = action('setCurrentAreaId', (areaId?: number) => { - this.selectedEntity = undefined; - - if (areaId == null) { - this.currentArea = undefined; - } else if (this.currentQuest) { - const areaVariant = this.currentQuest.areaVariants.find( - variant => variant.area.id === areaId - ); - this.currentArea = areaVariant && areaVariant.area; - } - }) - - loadFile = (file: File) => { - const reader = new FileReader(); - reader.addEventListener('loadend', () => { this.loadend(file, reader) }); - reader.readAsArrayBuffer(file); - } - - // TODO: notify user of problems. - private loadend = async (file: File, reader: FileReader) => { - if (!(reader.result instanceof ArrayBuffer)) { - console.error('Couldn\'t read file.'); - return; - } - - if (file.name.endsWith('.nj')) { - this.setModel(createModelMesh(parseNj(new ArrayBufferCursor(reader.result, true)))); - } else if (file.name.endsWith('.xj')) { - this.setModel(createModelMesh(parseXj(new ArrayBufferCursor(reader.result, true)))); - } else { - const quest = parseQuest(new ArrayBufferCursor(reader.result, true)); - this.setQuest(quest); - - if (quest) { - // Load section data. - for (const variant of quest.areaVariants) { - const sections = await getAreaSections(quest.episode, variant.area.id, variant.id); - variant.sections = sections; - - // Generate object geometry. - for (const object of quest.objects.filter(o => o.areaId === variant.area.id)) { - try { - const geometry = await getObjectGeometry(object.type); - this.setSectionOnVisibleQuestEntity(object, sections); - object.object3d = createObjectMesh(object, geometry); - } catch (e) { - console.error(e); - } - } - - // Generate NPC geometry. - for (const npc of quest.npcs.filter(npc => npc.areaId === variant.area.id)) { - try { - const geometry = await getNpcGeometry(npc.type); - this.setSectionOnVisibleQuestEntity(npc, sections); - npc.object3d = createNpcMesh(npc, geometry); - } catch (e) { - console.error(e); - } - } - } - } else { - console.error('Couldn\'t parse quest file.'); - } - } - } - - private setSectionOnVisibleQuestEntity = async (entity: QuestEntity, sections: Section[]) => { - let { x, y, z } = entity.position; - - const section = sections.find(s => s.id === entity.sectionId); - entity.section = section; - - if (section) { - const { x: secX, y: secY, z: secZ } = section.position; - const rotX = section.cosYAxisRotation * x + section.sinYAxisRotation * z; - const rotZ = -section.sinYAxisRotation * x + section.cosYAxisRotation * z; - x = rotX + secX; - y += secY; - z = rotZ + secZ; - } else { - console.warn(`Section ${entity.sectionId} not found.`); - } - - entity.position = new Vec3(x, y, z); - } - - saveCurrentQuestToFile = (fileName: string) => { - if (this.currentQuest) { - const cursor = writeQuestQst(this.currentQuest, fileName); - - if (!fileName.endsWith('.qst')) { - fileName += '.qst'; - } - - const a = document.createElement('a'); - a.href = URL.createObjectURL(new Blob([cursor.buffer])); - a.download = fileName; - document.body.appendChild(a); - a.click(); - URL.revokeObjectURL(a.href); - document.body.removeChild(a); - } - } -} - -export const questEditorStore = new QuestEditorStore(); diff --git a/src/stores/ServerMap.ts b/src/stores/ServerMap.ts deleted file mode 100644 index d4f444bb..00000000 --- a/src/stores/ServerMap.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { computed } from "mobx"; -import { Server } from "../domain"; -import { applicationStore } from "./ApplicationStore"; -import { EnumMap } from "../enums"; - -export class ServerMap extends EnumMap { - constructor(initialValue: (server: Server) => V) { - super(Server, initialValue) - } - - /** - * @returns the value for the current server as set in {@link applicationStore}. - */ - @computed get current(): V { - return this.get(applicationStore.currentServer); - } -} diff --git a/src/three-orbit-controls.d.ts b/src/three-orbit-controls.d.ts deleted file mode 100644 index e82fb35f..00000000 --- a/src/three-orbit-controls.d.ts +++ /dev/null @@ -1 +0,0 @@ -declare module 'three-orbit-controls'; \ No newline at end of file diff --git a/src/ui/ApplicationComponent.css b/src/ui/ApplicationComponent.css deleted file mode 100644 index b15df593..00000000 --- a/src/ui/ApplicationComponent.css +++ /dev/null @@ -1,34 +0,0 @@ -.ApplicationComponent { - display: flex; - flex-direction: column; - position: absolute; - top: 0; - bottom: 0; - left: 0; - right: 0; -} - -.ApplicationComponent-navbar { - display: flex; -} - -.ApplicationComponent-heading { - margin: 5px 10px 0 10px; -} - -.ApplicationComponent-beta { - color: #f55656; - font-weight: bold; -} - -.ApplicationComponent-main { - flex: 1; - display: flex; - flex-direction: column; - align-items: stretch; - overflow: hidden; -} - -.ApplicationComponent-main > * { - flex: 1; -} diff --git a/src/ui/ApplicationComponent.tsx b/src/ui/ApplicationComponent.tsx deleted file mode 100644 index 1c0b147d..00000000 --- a/src/ui/ApplicationComponent.tsx +++ /dev/null @@ -1,58 +0,0 @@ -import { Menu } from 'antd'; -import { ClickParam } from 'antd/lib/menu'; -import { observer } from 'mobx-react'; -import React from 'react'; -import './ApplicationComponent.css'; -import { withErrorBoundary } from './ErrorBoundary'; -import { HuntOptimizerComponent } from './hunt-optimizer/HuntOptimizerComponent'; -import { QuestEditorComponent } from './quest-editor/QuestEditorComponent'; - -const QuestEditor = withErrorBoundary(QuestEditorComponent); -const HuntOptimizer = withErrorBoundary(HuntOptimizerComponent); - -@observer -export class ApplicationComponent extends React.Component { - state = { tool: 'questEditor' } - - render() { - let toolComponent; - - switch (this.state.tool) { - case 'questEditor': - toolComponent = ; - break; - case 'huntOptimizer': - toolComponent = ; - break; - } - - return ( -
-
-

- Phantasmal World -

- - - Quest Editor(Beta) - - - Hunt Optimizer - - -
-
- {toolComponent} -
-
- ); - } - - private menuClicked = (e: ClickParam) => { - this.setState({ tool: e.key }); - }; -} diff --git a/src/ui/ErrorBoundary.css b/src/ui/ErrorBoundary.css deleted file mode 100644 index 1fac8171..00000000 --- a/src/ui/ErrorBoundary.css +++ /dev/null @@ -1,10 +0,0 @@ -.ErrorBoundary-error { - display: flex; - flex-direction: column; - align-items: center; - overflow: hidden; -} - -.ErrorBoundary-error > * { - margin-top: 10%; -} diff --git a/src/ui/ErrorBoundary.tsx b/src/ui/ErrorBoundary.tsx deleted file mode 100644 index 19055abc..00000000 --- a/src/ui/ErrorBoundary.tsx +++ /dev/null @@ -1,31 +0,0 @@ -import { Alert } from 'antd'; -import React from 'react'; -import './ErrorBoundary.css'; - -export class ErrorBoundary extends React.Component { - state = { - hasError: false - }; - - render() { - if (this.state.hasError) { - return ( -
-
- -
-
- ); - } else { - return this.props.children; - } - } - - static getDerivedStateFromError(_error: any) { - return { hasError: true }; - } -} - -export function withErrorBoundary(Component: React.ComponentType) { - return () => ; -} diff --git a/src/ui/dataTable.less b/src/ui/dataTable.less deleted file mode 100644 index f2469320..00000000 --- a/src/ui/dataTable.less +++ /dev/null @@ -1,67 +0,0 @@ -@import "./theme.less"; - -.DataTable > * { - border: solid 1px @border-color-base; - background-color: lighten(@component-background, 3%); - - & * { - scrollbar-color: @table-scrollbar-thumb-color @table-scrollbar-color; - } - - & ::-webkit-scrollbar { - background-color: @table-scrollbar-color; - } - - & ::-webkit-scrollbar-track { - background-color: @table-scrollbar-color; - } - - & ::-webkit-scrollbar-thumb { - background-color: @table-scrollbar-thumb-color; - } - - & ::-webkit-scrollbar-corner { - background-color: @table-scrollbar-color; - } -} - -.DataTable-header { - background-color: lighten(@component-background, 12%); - font-weight: bold; - - & .DataTable-cell { - border-right: solid 1px @border-color-base; - } -} - -.DataTable-cell { - display: flex; - align-items: center; - box-sizing: border-box; - padding: 0 5px; - border-bottom: solid 1px @border-color-base; - border-right: solid 1px darken(@border-color-base, 11%); - - & > span { - white-space: nowrap; - overflow: hidden; - text-overflow: ellipsis; - } - - &.last-in-row { - border-right: solid 1px @border-color-base; - } - - &.number { - justify-content: flex-end; - } - - &.footer-cell { - font-weight: bold; - } -} - -.DataTable-no-result { - margin: 20px; - color: @text-color-secondary; -} \ No newline at end of file diff --git a/src/ui/dataTable.tsx b/src/ui/dataTable.tsx deleted file mode 100644 index 5a8debb2..00000000 --- a/src/ui/dataTable.tsx +++ /dev/null @@ -1,99 +0,0 @@ -import React from "react"; -import { Index, MultiGrid, GridCellRenderer } from "react-virtualized"; -import "./dataTable.less"; - -export type Column = { - name: string, - width: number, - cellValue: (record: T) => string, - tooltip?: (record: T) => string, - footerValue?: string, - footerTooltip?: string, - className?: string, -} - -/** - * A table with a fixed header. Optionally has fixed columns and a footer. - * TODO: no-content message. - */ -export class DataTable extends React.Component<{ - width: number, - height: number, - rowCount: number, - columns: Array>, - fixedColumnCount?: number, - record: (index: Index) => T, - footer?: boolean, -}> { - render() { - return ( -
- -
- ); - } - - private columnWidth = ({ index }: Index): number => { - return this.props.columns[index].width; - } - - private cellRenderer: GridCellRenderer = ({ columnIndex, rowIndex, style }) => { - const column = this.props.columns[columnIndex]; - let text: string; - let title: string | undefined; - const classes = ['DataTable-cell']; - - if (columnIndex === this.props.columns.length - 1) { - classes.push('last-in-row'); - } - - if (rowIndex === 0) { - // Header row - text = title = column.name; - } else { - // Record or footer row - if (column.className) { - classes.push(column.className); - } - - if (this.props.footer && rowIndex === 1 + this.props.rowCount) { - // Footer row - classes.push('footer-cell'); - text = column.footerValue == null ? '' : column.footerValue; - title = column.footerTooltip == null ? '' : column.footerTooltip; - } else { - // Record row - const result = this.props.record({ index: rowIndex - 1 }); - - text = column.cellValue(result); - - if (column.tooltip) { - title = column.tooltip(result); - } - } - } - - return ( -
- {text} -
- ); - } -} diff --git a/src/ui/hunt-optimizer/HuntOptimizerComponent.css b/src/ui/hunt-optimizer/HuntOptimizerComponent.css deleted file mode 100644 index 2dc3adbf..00000000 --- a/src/ui/hunt-optimizer/HuntOptimizerComponent.css +++ /dev/null @@ -1,27 +0,0 @@ -.ho-HuntOptimizerComponent { - display: flex; - align-items: stretch; - overflow: hidden; - margin-top: 10px; -} - -.ho-HuntOptimizerComponent > .ant-tabs { - flex: 1; - display: flex; - flex-direction: column; - align-items: stretch; -} - -.ho-HuntOptimizerComponent > .ant-tabs > .ant-tabs-content { - flex: 1; - display: flex; - flex-direction: column; - align-items: stretch; -} - -.ho-HuntOptimizerComponent > .ant-tabs > .ant-tabs-content > .ant-tabs-tabpane-active { - flex: 1; - display: flex; - flex-direction: column; - align-items: stretch; -} diff --git a/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx b/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx deleted file mode 100644 index e4ea802c..00000000 --- a/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx +++ /dev/null @@ -1,22 +0,0 @@ -import { Tabs } from "antd"; -import React from "react"; -import './HuntOptimizerComponent.css'; -import { OptimizerComponent } from "./OptimizerComponent"; -import { MethodsComponent } from "./MethodsComponent"; - -const TabPane = Tabs.TabPane; - -export function HuntOptimizerComponent() { - return ( -
- - - - - - - - -
- ); -} diff --git a/src/ui/hunt-optimizer/MethodsComponent.css b/src/ui/hunt-optimizer/MethodsComponent.css deleted file mode 100644 index 9c441aae..00000000 --- a/src/ui/hunt-optimizer/MethodsComponent.css +++ /dev/null @@ -1,3 +0,0 @@ -.ho-MethodsComponent { - flex: 1; -} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/MethodsComponent.tsx b/src/ui/hunt-optimizer/MethodsComponent.tsx deleted file mode 100644 index 17e83562..00000000 --- a/src/ui/hunt-optimizer/MethodsComponent.tsx +++ /dev/null @@ -1,69 +0,0 @@ -import { observer } from "mobx-react"; -import React from "react"; -import { AutoSizer, Index } from "react-virtualized"; -import { HuntMethod } from "../../domain"; -import { EnemyNpcTypes } from "../../domain/NpcType"; -import { huntMethodStore } from "../../stores/HuntMethodStore"; -import "./MethodsComponent.css"; -import { DataTable, Column } from "../dataTable"; -import { hoursToString } from "../time"; - -@observer -export class MethodsComponent extends React.Component { - static columns: Array> = (() => { - // Standard columns. - const columns: Column[] = [ - { - name: 'Method', - width: 250, - cellValue: (method) => method.name, - }, - { - name: 'Time', - width: 50, - cellValue: (method) => hoursToString(method.time), - className: 'number', - }, - ]; - - // One column per enemy type. - for (const enemy of EnemyNpcTypes) { - columns.push({ - name: enemy.name, - width: 75, - cellValue: (method) => { - const count = method.enemyCounts.get(enemy); - return count == null ? '' : count.toString(); - }, - className: 'number', - }); - } - - return columns; - })(); - - render() { - const methods = huntMethodStore.methods.current.value; - - return ( -
- - {({ width, height }) => ( - - width={width} - height={height} - rowCount={methods.length} - columns={MethodsComponent.columns} - fixedColumnCount={2} - record={this.record} - /> - )} - -
- ); - } - - private record = ({ index }: Index) => { - return huntMethodStore.methods.current.value[index]; - } -} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/OptimizationResultComponent.less b/src/ui/hunt-optimizer/OptimizationResultComponent.less deleted file mode 100644 index 3098a9a4..00000000 --- a/src/ui/hunt-optimizer/OptimizationResultComponent.less +++ /dev/null @@ -1,10 +0,0 @@ -@import "../theme.less"; - -.ho-OptimizationResultComponent { - display: flex; - flex-direction: column; -} - -.ho-OptimizationResultComponent-table { - flex: 1; -} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/OptimizationResultComponent.tsx b/src/ui/hunt-optimizer/OptimizationResultComponent.tsx deleted file mode 100644 index 78a32ae4..00000000 --- a/src/ui/hunt-optimizer/OptimizationResultComponent.tsx +++ /dev/null @@ -1,132 +0,0 @@ -import { computed } from "mobx"; -import { observer } from "mobx-react"; -import React from "react"; -import { AutoSizer, Index } from "react-virtualized"; -import { Item } from "../../domain"; -import { huntOptimizerStore, OptimizationResult } from "../../stores/HuntOptimizerStore"; -import { Column, DataTable } from "../dataTable"; -import "./OptimizationResultComponent.less"; -import { hoursToString } from "../time"; - -@observer -export class OptimizationResultComponent extends React.Component { - @computed private get columns(): Column[] { - // Standard columns. - const results = huntOptimizerStore.results; - let totalRuns = 0; - let totalTime = 0; - - for (const result of results) { - totalRuns += result.runs; - totalTime += result.totalTime; - } - - const columns: Column[] = [ - { - name: 'Difficulty', - width: 75, - cellValue: (result) => result.difficulty, - footerValue: 'Totals:', - }, - { - name: 'Method', - width: 200, - cellValue: (result) => result.methodName, - tooltip: (result) => result.methodName, - }, - { - name: 'Section ID', - width: 80, - cellValue: (result) => result.sectionId, - }, - { - name: 'Time/Run', - width: 80, - cellValue: (result) => hoursToString(result.methodTime), - className: 'number', - }, - { - name: 'Runs', - width: 60, - cellValue: (result) => result.runs.toFixed(1), - tooltip: (result) => result.runs.toString(), - footerValue: totalRuns.toFixed(1), - footerTooltip: totalRuns.toString(), - className: 'number', - }, - { - name: 'Total Hours', - width: 90, - cellValue: (result) => result.totalTime.toFixed(1), - tooltip: (result) => result.totalTime.toString(), - footerValue: totalTime.toFixed(1), - footerTooltip: totalTime.toString(), - className: 'number', - }, - ]; - - // Add one column per item. - const items = new Set(); - - for (const r of results) { - for (const i of r.itemCounts.keys()) { - items.add(i); - } - } - - for (const item of items) { - const totalCount = results.reduce( - (acc, r) => acc + (r.itemCounts.get(item) || 0), - 0 - ); - - columns.push({ - name: item.name, - width: 80, - cellValue: (result) => { - const count = result.itemCounts.get(item); - return count ? count.toFixed(2) : ''; - }, - tooltip: (result) => { - const count = result.itemCounts.get(item); - return count ? count.toString() : ''; - }, - className: 'number', - footerValue: totalCount.toFixed(2), - footerTooltip: totalCount.toString() - }); - } - - return columns; - } - - render() { - // Make sure render is called when result changes. - huntOptimizerStore.results.slice(0, 0); - - return ( -
-

Optimization Result

-
- - {({ width, height }) => - 0} - /> - } - -
-
- ); - } - - private record = ({ index }: Index): OptimizationResult => { - return huntOptimizerStore.results[index]; - } -} diff --git a/src/ui/hunt-optimizer/OptimizerComponent.css b/src/ui/hunt-optimizer/OptimizerComponent.css deleted file mode 100644 index 1c53b948..00000000 --- a/src/ui/hunt-optimizer/OptimizerComponent.css +++ /dev/null @@ -1,10 +0,0 @@ -.ho-OptimizerComponent { - flex: 1; - display: flex; - align-items: stretch; -} - -.ho-OptimizerComponent > *:nth-child(2) { - flex: 1; - overflow: hidden; -} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/OptimizerComponent.tsx b/src/ui/hunt-optimizer/OptimizerComponent.tsx deleted file mode 100644 index 994ebf74..00000000 --- a/src/ui/hunt-optimizer/OptimizerComponent.tsx +++ /dev/null @@ -1,13 +0,0 @@ -import React from "react"; -import { WantedItemsComponent } from "./WantedItemsComponent"; -import { OptimizationResultComponent } from "./OptimizationResultComponent"; -import "./OptimizerComponent.css"; - -export function OptimizerComponent() { - return ( -
- - -
- ); -} diff --git a/src/ui/hunt-optimizer/WantedItemsComponent.css b/src/ui/hunt-optimizer/WantedItemsComponent.css deleted file mode 100644 index 0f16843d..00000000 --- a/src/ui/hunt-optimizer/WantedItemsComponent.css +++ /dev/null @@ -1,11 +0,0 @@ -.ho-WantedItemsComponent { - display: flex; - flex-direction: column; - margin: 0 10px; -} - -.ho-WantedItemsComponent-table { - position: relative; - flex: 1; - margin: 10px 0 0 -10px; -} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/WantedItemsComponent.tsx b/src/ui/hunt-optimizer/WantedItemsComponent.tsx deleted file mode 100644 index 9eeec229..00000000 --- a/src/ui/hunt-optimizer/WantedItemsComponent.tsx +++ /dev/null @@ -1,124 +0,0 @@ -import { Button, InputNumber, Select } from "antd"; -import { observer } from "mobx-react"; -import React from "react"; -import { AutoSizer, Column, Table, TableCellRenderer } from "react-virtualized"; -import { huntOptimizerStore, WantedItem } from "../../stores/HuntOptimizerStore"; -import { itemStore } from "../../stores/ItemStore"; -import './WantedItemsComponent.css'; - -@observer -export class WantedItemsComponent extends React.Component { - render() { - // Make sure render is called on updates. - huntOptimizerStore.wantedItems.slice(0, 0); - - return ( -
-

Wanted Items

-
- - -
-
- - {({ width, height }) => ( - huntOptimizerStore.wantedItems[index]} - > - - - } - /> - rowData.item.name} - /> - -
- )} -
-
-
- ); - } - - private addWanted = (itemName: string) => { - let added = huntOptimizerStore.wantedItems.find(w => w.item.name === itemName); - - if (!added) { - const item = itemStore.items.current.value.find(i => i.name === itemName)!; - huntOptimizerStore.wantedItems.push(new WantedItem(item, 1)); - } - } - - private removeWanted = (wanted: WantedItem) => () => { - const i = huntOptimizerStore.wantedItems.findIndex(w => w === wanted); - - if (i !== -1) { - huntOptimizerStore.wantedItems.splice(i, 1); - } - } - - private tableRemoveCellRenderer: TableCellRenderer = ({ rowData }) => { - return - - {areas && ( - - )} - {quest && ( - - )} - - ); - } - - private setFilename = (info: UploadChangeParam) => { - if (info.file.originFileObj) { - this.setState({ filename: info.file.name }); - questEditorStore.loadFile(info.file.originFileObj); - } - } - - private saveAsClicked = () => { - this.props.onSaveAsClicked(this.state.filename); - } -} - -class SaveAsForm extends React.Component<{ - isOpen: boolean, - filename: string, - onFilenameChange: (name: string) => void, - onOk: () => void, - onCancel: () => void -}> { - render() { - return ( - Save as...} - visible={this.props.isOpen} - onOk={this.props.onOk} - onCancel={this.props.onCancel} - > -
- - - -
-
- ); - } - - private nameChanged = (e: ChangeEvent) => { - this.props.onFilenameChange(e.currentTarget.value); - } -} diff --git a/src/ui/quest-editor/QuestInfoComponent.css b/src/ui/quest-editor/QuestInfoComponent.css deleted file mode 100644 index 08aca7ce..00000000 --- a/src/ui/quest-editor/QuestInfoComponent.css +++ /dev/null @@ -1,26 +0,0 @@ -.qe-QuestInfoComponent { - width: 280px; - padding: 10px; - display: flex; - flex-direction: column; -} - -.qe-QuestInfoComponent table { - border-collapse: collapse; - width: 100%; -} - -.qe-QuestInfoComponent table tbody th { - text-align: right; - padding-right: 5px; -} - -.qe-QuestInfoComponent pre { - padding: 8px; - border: solid 1px hsl(200, 10%, 30%); - margin: 4px 0; -} - -.qe-QuestInfoComponent-npc-counts-container { - overflow: auto; -} \ No newline at end of file diff --git a/src/ui/quest-editor/QuestInfoComponent.tsx b/src/ui/quest-editor/QuestInfoComponent.tsx deleted file mode 100644 index 376e88fa..00000000 --- a/src/ui/quest-editor/QuestInfoComponent.tsx +++ /dev/null @@ -1,67 +0,0 @@ -import React from 'react'; -import { NpcType, Quest } from '../../domain'; -import './QuestInfoComponent.css'; - -export function QuestInfoComponent({ quest }: { quest?: Quest }) { - if (quest) { - const episode = quest.episode === 4 ? 'IV' : (quest.episode === 2 ? 'II' : 'I'); - const npcCounts = new Map(); - - for (const npc of quest.npcs) { - const val = npcCounts.get(npc.type) || 0; - npcCounts.set(npc.type, val + 1); - } - - const extraCanadines = (npcCounts.get(NpcType.Canane) || 0) * 8; - - // Sort by type ID. - const sortedNpcCounts = [...npcCounts].sort((a, b) => a[0].id - b[0].id); - - const npcCountRows = sortedNpcCounts.map(([npcType, count]) => { - const extra = npcType === NpcType.Canadine ? extraCanadines : 0; - return ( - - {npcType.name}: - {count + extra} - - ); - }); - - return ( -
- - - - - - - - - - - - - - - -
Name:{quest.name}
Episode:{episode}
-
{quest.shortDescription}
-
-
{quest.longDescription}
-
-
- - - - - - {npcCountRows} - -
NPC Counts
-
-
- ); - } else { - return
; - } -} diff --git a/src/ui/quest-editor/RendererComponent.tsx b/src/ui/quest-editor/RendererComponent.tsx deleted file mode 100644 index 9779037c..00000000 --- a/src/ui/quest-editor/RendererComponent.tsx +++ /dev/null @@ -1,50 +0,0 @@ -import React from 'react'; -import { Object3D } from 'three'; -import { Area, Quest } from '../../domain'; -import { getRenderer } from '../../rendering/Renderer'; - -interface Props { - quest?: Quest; - area?: Area; - model?: Object3D; -} - -export class RendererComponent extends React.Component { - private renderer = getRenderer(); - - render() { - return
; - } - - componentDidMount() { - window.addEventListener('resize', this.onResize); - } - - componentWillUnmount() { - window.removeEventListener('resize', this.onResize); - } - - componentWillReceiveProps({ quest, area, model }: Props) { - if (model) { - this.renderer.setModel(model); - } else { - this.renderer.setQuestAndArea(quest, area); - } - } - - shouldComponentUpdate() { - return false; - } - - private modifyDom = (div: HTMLDivElement | null) => { - if (div) { - this.renderer.setSize(div.clientWidth, div.clientHeight); - div.appendChild(this.renderer.domElement); - } - } - - private onResize = () => { - const wrapperDiv = this.renderer.domElement.parentNode as HTMLDivElement; - this.renderer.setSize(wrapperDiv.clientWidth, wrapperDiv.clientHeight); - } -} diff --git a/src/ui/theme.less b/src/ui/theme.less deleted file mode 100644 index b950e528..00000000 --- a/src/ui/theme.less +++ /dev/null @@ -1,5 +0,0 @@ -@scrollbar-color: darken(@component-background, 3%); -@scrollbar-thumb-color: lighten(@component-background, 3%); - -@table-scrollbar-color: lighten(@scrollbar-color, 1%); -@table-scrollbar-thumb-color: lighten(@scrollbar-thumb-color, 5%); diff --git a/src/ui/time.ts b/src/ui/time.ts deleted file mode 100644 index ad7f51af..00000000 --- a/src/ui/time.ts +++ /dev/null @@ -1,9 +0,0 @@ -/** - * @param hours can be fractional. - * @returns a string of the shape ##:##. - */ -export function hoursToString(hours: number): string { - const h = Math.floor(hours); - const m = Math.round(60 * (hours - h)); - return `${h.toString().padStart(2, '0')}:${m.toString().padStart(2, '0')}`; -} diff --git a/static/css/2.5fcf4e3c.chunk.css b/static/css/2.5fcf4e3c.chunk.css new file mode 100644 index 00000000..1bc3fb34 --- /dev/null +++ b/static/css/2.5fcf4e3c.chunk.css @@ -0,0 +1,2 @@ +body,html{width:100%;height:100%}input::-ms-clear,input::-ms-reveal{display:none}*,:after,:before{box-sizing:border-box}html{font-family:sans-serif;line-height:1.15;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%;-ms-overflow-style:scrollbar;-webkit-tap-highlight-color:rgba(255,255,255,0)}@-ms-viewport{width:device-width}article,aside,dialog,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}body{margin:0;color:#e3e6e8;font-size:14px;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,PingFang SC,Hiragino Sans GB,Microsoft YaHei,Helvetica Neue,Helvetica,Arial,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol;font-variant:tabular-nums;line-height:1.5;background-color:#2e3538;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum"}[tabindex="-1"]:focus{outline:none!important}hr{box-sizing:content-box;height:0;overflow:visible}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5em;color:hsla(0,0%,100%,.85);font-weight:500}p{margin-top:0;margin-bottom:1em}abbr[data-original-title],abbr[title]{text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted;border-bottom:0;cursor:help}address{margin-bottom:1em;font-style:normal;line-height:inherit}input[type=number],input[type=password],input[type=text],textarea{-webkit-appearance:none}dl,ol,ul{margin-top:0;margin-bottom:1em}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:500}dd{margin-bottom:.5em;margin-left:0}blockquote{margin:0 0 1em}dfn{font-style:italic}b,strong{font-weight:bolder}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#0af;text-decoration:none;background-color:transparent;outline:none;cursor:pointer;-webkit-transition:color .3s;transition:color .3s;-webkit-text-decoration-skip:objects}a:hover{color:#29bfff}a:active{color:#0089d9}a:active,a:hover{text-decoration:none;outline:0}a[disabled]{color:hsla(0,0%,100%,.5);cursor:not-allowed;pointer-events:none}code,kbd,pre,samp{font-size:1em;font-family:SFMono-Regular,Consolas,Liberation Mono,Menlo,Courier,monospace}pre{margin-top:0;margin-bottom:1em;overflow:auto}figure{margin:0 0 1em}img{vertical-align:middle;border-style:none}svg:not(:root){overflow:hidden}[role=button],a,area,button,input:not([type=range]),label,select,summary,textarea{touch-action:manipulation}table{border-collapse:collapse}caption{padding-top:.75em;padding-bottom:.3em;color:#c2cfd6;text-align:left;caption-side:bottom}th{text-align:inherit}button,input,optgroup,select,textarea{margin:0;color:inherit;font-size:inherit;font-family:inherit;line-height:inherit}button,input{overflow:visible}button,select{text-transform:none}[type=reset],[type=submit],button,html [type=button]{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{padding:0;border-style:none}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=date],input[type=datetime-local],input[type=month],input[type=time]{-webkit-appearance:listbox}textarea{overflow:auto;resize:vertical}fieldset{min-width:0;margin:0;padding:0;border:0}legend{display:block;width:100%;max-width:100%;margin-bottom:.5em;padding:0;color:inherit;font-size:1.5em;line-height:inherit;white-space:normal}progress{vertical-align:baseline}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:none}[type=search]::-webkit-search-cancel-button,[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}summary{display:list-item}template{display:none}[hidden]{display:none!important}mark{padding:.2em;background-color:#feffe6}::-moz-selection{color:#000;background:#0af}::selection{color:#000;background:#0af}.clearfix{zoom:1}.clearfix:after,.clearfix:before{display:table;content:""}.clearfix:after{clear:both}.anticon{display:inline-block;color:inherit;font-style:normal;line-height:0;text-align:center;text-transform:none;vertical-align:-.125em;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.anticon>*{line-height:1}.anticon svg{display:inline-block}.anticon:before{display:none}.anticon .anticon-icon{display:block}.anticon[tabindex]{cursor:pointer}.anticon-spin,.anticon-spin:before{display:inline-block;-webkit-animation:loadingCircle 1s linear infinite;animation:loadingCircle 1s linear infinite}.fade-appear,.fade-enter,.fade-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.fade-appear.fade-appear-active,.fade-enter.fade-enter-active{-webkit-animation-name:antFadeIn;animation-name:antFadeIn;-webkit-animation-play-state:running;animation-play-state:running}.fade-leave.fade-leave-active{-webkit-animation-name:antFadeOut;animation-name:antFadeOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.fade-appear,.fade-enter{opacity:0}.fade-appear,.fade-enter,.fade-leave{-webkit-animation-timing-function:linear;animation-timing-function:linear}@-webkit-keyframes antFadeIn{0%{opacity:0}to{opacity:1}}@keyframes antFadeIn{0%{opacity:0}to{opacity:1}}@-webkit-keyframes antFadeOut{0%{opacity:1}to{opacity:0}}@keyframes antFadeOut{0%{opacity:1}to{opacity:0}}.move-up-appear,.move-up-enter,.move-up-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.move-up-appear.move-up-appear-active,.move-up-enter.move-up-enter-active{-webkit-animation-name:antMoveUpIn;animation-name:antMoveUpIn;-webkit-animation-play-state:running;animation-play-state:running}.move-up-leave.move-up-leave-active{-webkit-animation-name:antMoveUpOut;animation-name:antMoveUpOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.move-up-appear,.move-up-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.move-up-leave{-webkit-animation-timing-function:cubic-bezier(.6,.04,.98,.34);animation-timing-function:cubic-bezier(.6,.04,.98,.34)}.move-down-appear,.move-down-enter,.move-down-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.move-down-appear.move-down-appear-active,.move-down-enter.move-down-enter-active{-webkit-animation-name:antMoveDownIn;animation-name:antMoveDownIn;-webkit-animation-play-state:running;animation-play-state:running}.move-down-leave.move-down-leave-active{-webkit-animation-name:antMoveDownOut;animation-name:antMoveDownOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.move-down-appear,.move-down-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.move-down-leave{-webkit-animation-timing-function:cubic-bezier(.6,.04,.98,.34);animation-timing-function:cubic-bezier(.6,.04,.98,.34)}.move-left-appear,.move-left-enter,.move-left-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.move-left-appear.move-left-appear-active,.move-left-enter.move-left-enter-active{-webkit-animation-name:antMoveLeftIn;animation-name:antMoveLeftIn;-webkit-animation-play-state:running;animation-play-state:running}.move-left-leave.move-left-leave-active{-webkit-animation-name:antMoveLeftOut;animation-name:antMoveLeftOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.move-left-appear,.move-left-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.move-left-leave{-webkit-animation-timing-function:cubic-bezier(.6,.04,.98,.34);animation-timing-function:cubic-bezier(.6,.04,.98,.34)}.move-right-appear,.move-right-enter,.move-right-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.move-right-appear.move-right-appear-active,.move-right-enter.move-right-enter-active{-webkit-animation-name:antMoveRightIn;animation-name:antMoveRightIn;-webkit-animation-play-state:running;animation-play-state:running}.move-right-leave.move-right-leave-active{-webkit-animation-name:antMoveRightOut;animation-name:antMoveRightOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.move-right-appear,.move-right-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.move-right-leave{-webkit-animation-timing-function:cubic-bezier(.6,.04,.98,.34);animation-timing-function:cubic-bezier(.6,.04,.98,.34)}@-webkit-keyframes antMoveDownIn{0%{-webkit-transform:translateY(100%);transform:translateY(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@keyframes antMoveDownIn{0%{-webkit-transform:translateY(100%);transform:translateY(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@-webkit-keyframes antMoveDownOut{0%{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateY(100%);transform:translateY(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@keyframes antMoveDownOut{0%{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateY(100%);transform:translateY(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@-webkit-keyframes antMoveLeftIn{0%{-webkit-transform:translateX(-100%);transform:translateX(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@keyframes antMoveLeftIn{0%{-webkit-transform:translateX(-100%);transform:translateX(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@-webkit-keyframes antMoveLeftOut{0%{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateX(-100%);transform:translateX(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@keyframes antMoveLeftOut{0%{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateX(-100%);transform:translateX(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@-webkit-keyframes antMoveRightIn{0%{-webkit-transform:translateX(100%);transform:translateX(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@keyframes antMoveRightIn{0%{-webkit-transform:translateX(100%);transform:translateX(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@-webkit-keyframes antMoveRightOut{0%{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateX(100%);transform:translateX(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@keyframes antMoveRightOut{0%{-webkit-transform:translateX(0);transform:translateX(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateX(100%);transform:translateX(100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@-webkit-keyframes antMoveUpIn{0%{-webkit-transform:translateY(-100%);transform:translateY(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@keyframes antMoveUpIn{0%{-webkit-transform:translateY(-100%);transform:translateY(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@-webkit-keyframes antMoveUpOut{0%{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateY(-100%);transform:translateY(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@keyframes antMoveUpOut{0%{-webkit-transform:translateY(0);transform:translateY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:translateY(-100%);transform:translateY(-100%);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@-webkit-keyframes loadingCircle{to{-webkit-transform:rotate(1turn);transform:rotate(1turn)}}@keyframes loadingCircle{to{-webkit-transform:rotate(1turn);transform:rotate(1turn)}}[ant-click-animating-without-extra-node=true],[ant-click-animating=true]{position:relative}html{--antd-wave-shadow-color:#0af}.ant-click-animating-node,[ant-click-animating-without-extra-node=true]:after{position:absolute;top:0;right:0;bottom:0;left:0;display:block;border-radius:inherit;box-shadow:0 0 0 0 #0af;box-shadow:0 0 0 0 var(--antd-wave-shadow-color);opacity:.2;-webkit-animation:fadeEffect 2s cubic-bezier(.08,.82,.17,1),waveEffect .4s cubic-bezier(.08,.82,.17,1);animation:fadeEffect 2s cubic-bezier(.08,.82,.17,1),waveEffect .4s cubic-bezier(.08,.82,.17,1);-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards;content:"";pointer-events:none}@-webkit-keyframes waveEffect{to{box-shadow:0 0 0 #0af;box-shadow:0 0 0 6px #0af;box-shadow:0 0 0 6px var(--antd-wave-shadow-color)}}@keyframes waveEffect{to{box-shadow:0 0 0 #0af;box-shadow:0 0 0 6px #0af;box-shadow:0 0 0 6px var(--antd-wave-shadow-color)}}@-webkit-keyframes fadeEffect{to{opacity:0}}@keyframes fadeEffect{to{opacity:0}}.slide-up-appear,.slide-up-enter,.slide-up-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.slide-up-appear.slide-up-appear-active,.slide-up-enter.slide-up-enter-active{-webkit-animation-name:antSlideUpIn;animation-name:antSlideUpIn;-webkit-animation-play-state:running;animation-play-state:running}.slide-up-leave.slide-up-leave-active{-webkit-animation-name:antSlideUpOut;animation-name:antSlideUpOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.slide-up-appear,.slide-up-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.23,1,.32,1);animation-timing-function:cubic-bezier(.23,1,.32,1)}.slide-up-leave{-webkit-animation-timing-function:cubic-bezier(.755,.05,.855,.06);animation-timing-function:cubic-bezier(.755,.05,.855,.06)}.slide-down-appear,.slide-down-enter,.slide-down-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.slide-down-appear.slide-down-appear-active,.slide-down-enter.slide-down-enter-active{-webkit-animation-name:antSlideDownIn;animation-name:antSlideDownIn;-webkit-animation-play-state:running;animation-play-state:running}.slide-down-leave.slide-down-leave-active{-webkit-animation-name:antSlideDownOut;animation-name:antSlideDownOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.slide-down-appear,.slide-down-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.23,1,.32,1);animation-timing-function:cubic-bezier(.23,1,.32,1)}.slide-down-leave{-webkit-animation-timing-function:cubic-bezier(.755,.05,.855,.06);animation-timing-function:cubic-bezier(.755,.05,.855,.06)}.slide-left-appear,.slide-left-enter,.slide-left-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.slide-left-appear.slide-left-appear-active,.slide-left-enter.slide-left-enter-active{-webkit-animation-name:antSlideLeftIn;animation-name:antSlideLeftIn;-webkit-animation-play-state:running;animation-play-state:running}.slide-left-leave.slide-left-leave-active{-webkit-animation-name:antSlideLeftOut;animation-name:antSlideLeftOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.slide-left-appear,.slide-left-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.23,1,.32,1);animation-timing-function:cubic-bezier(.23,1,.32,1)}.slide-left-leave{-webkit-animation-timing-function:cubic-bezier(.755,.05,.855,.06);animation-timing-function:cubic-bezier(.755,.05,.855,.06)}.slide-right-appear,.slide-right-enter,.slide-right-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.slide-right-appear.slide-right-appear-active,.slide-right-enter.slide-right-enter-active{-webkit-animation-name:antSlideRightIn;animation-name:antSlideRightIn;-webkit-animation-play-state:running;animation-play-state:running}.slide-right-leave.slide-right-leave-active{-webkit-animation-name:antSlideRightOut;animation-name:antSlideRightOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.slide-right-appear,.slide-right-enter{opacity:0;-webkit-animation-timing-function:cubic-bezier(.23,1,.32,1);animation-timing-function:cubic-bezier(.23,1,.32,1)}.slide-right-leave{-webkit-animation-timing-function:cubic-bezier(.755,.05,.855,.06);animation-timing-function:cubic-bezier(.755,.05,.855,.06)}@-webkit-keyframes antSlideUpIn{0%{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@keyframes antSlideUpIn{0%{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@-webkit-keyframes antSlideUpOut{0%{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@keyframes antSlideUpOut{0%{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@-webkit-keyframes antSlideDownIn{0%{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:0}to{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:1}}@keyframes antSlideDownIn{0%{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:0}to{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:1}}@-webkit-keyframes antSlideDownOut{0%{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:1}to{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:0}}@keyframes antSlideDownOut{0%{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:1}to{-webkit-transform:scaleY(.8);transform:scaleY(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%;opacity:0}}@-webkit-keyframes antSlideLeftIn{0%{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@keyframes antSlideLeftIn{0%{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@-webkit-keyframes antSlideLeftOut{0%{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@keyframes antSlideLeftOut{0%{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@-webkit-keyframes antSlideRightIn{0%{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:0}to{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:1}}@keyframes antSlideRightIn{0%{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:0}to{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:1}}@-webkit-keyframes antSlideRightOut{0%{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:1}to{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:0}}@keyframes antSlideRightOut{0%{-webkit-transform:scaleX(1);transform:scaleX(1);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:1}to{-webkit-transform:scaleX(.8);transform:scaleX(.8);-webkit-transform-origin:100% 0;transform-origin:100% 0;opacity:0}}.swing-appear,.swing-enter{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.swing-appear.swing-appear-active,.swing-enter.swing-enter-active{-webkit-animation-name:antSwingIn;animation-name:antSwingIn;-webkit-animation-play-state:running;animation-play-state:running}@-webkit-keyframes antSwingIn{0%,to{-webkit-transform:translateX(0);transform:translateX(0)}20%{-webkit-transform:translateX(-10px);transform:translateX(-10px)}40%{-webkit-transform:translateX(10px);transform:translateX(10px)}60%{-webkit-transform:translateX(-5px);transform:translateX(-5px)}80%{-webkit-transform:translateX(5px);transform:translateX(5px)}}@keyframes antSwingIn{0%,to{-webkit-transform:translateX(0);transform:translateX(0)}20%{-webkit-transform:translateX(-10px);transform:translateX(-10px)}40%{-webkit-transform:translateX(10px);transform:translateX(10px)}60%{-webkit-transform:translateX(-5px);transform:translateX(-5px)}80%{-webkit-transform:translateX(5px);transform:translateX(5px)}}.zoom-appear,.zoom-enter,.zoom-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.zoom-appear.zoom-appear-active,.zoom-enter.zoom-enter-active{-webkit-animation-name:antZoomIn;animation-name:antZoomIn;-webkit-animation-play-state:running;animation-play-state:running}.zoom-leave.zoom-leave-active{-webkit-animation-name:antZoomOut;animation-name:antZoomOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.zoom-appear,.zoom-enter{-webkit-transform:scale(0);transform:scale(0);opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.zoom-leave{-webkit-animation-timing-function:cubic-bezier(.78,.14,.15,.86);animation-timing-function:cubic-bezier(.78,.14,.15,.86)}.zoom-big-appear,.zoom-big-enter,.zoom-big-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.zoom-big-appear.zoom-big-appear-active,.zoom-big-enter.zoom-big-enter-active{-webkit-animation-name:antZoomBigIn;animation-name:antZoomBigIn;-webkit-animation-play-state:running;animation-play-state:running}.zoom-big-leave.zoom-big-leave-active{-webkit-animation-name:antZoomBigOut;animation-name:antZoomBigOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.zoom-big-appear,.zoom-big-enter{-webkit-transform:scale(0);transform:scale(0);opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.zoom-big-leave{-webkit-animation-timing-function:cubic-bezier(.78,.14,.15,.86);animation-timing-function:cubic-bezier(.78,.14,.15,.86)}.zoom-big-fast-appear,.zoom-big-fast-enter,.zoom-big-fast-leave{-webkit-animation-duration:33ms;animation-duration:33ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.zoom-big-fast-appear.zoom-big-fast-appear-active,.zoom-big-fast-enter.zoom-big-fast-enter-active{-webkit-animation-name:antZoomBigIn;animation-name:antZoomBigIn;-webkit-animation-play-state:running;animation-play-state:running}.zoom-big-fast-leave.zoom-big-fast-leave-active{-webkit-animation-name:antZoomBigOut;animation-name:antZoomBigOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.zoom-big-fast-appear,.zoom-big-fast-enter{-webkit-transform:scale(0);transform:scale(0);opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.zoom-big-fast-leave{-webkit-animation-timing-function:cubic-bezier(.78,.14,.15,.86);animation-timing-function:cubic-bezier(.78,.14,.15,.86)}.zoom-up-appear,.zoom-up-enter,.zoom-up-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.zoom-up-appear.zoom-up-appear-active,.zoom-up-enter.zoom-up-enter-active{-webkit-animation-name:antZoomUpIn;animation-name:antZoomUpIn;-webkit-animation-play-state:running;animation-play-state:running}.zoom-up-leave.zoom-up-leave-active{-webkit-animation-name:antZoomUpOut;animation-name:antZoomUpOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.zoom-up-appear,.zoom-up-enter{-webkit-transform:scale(0);transform:scale(0);opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.zoom-up-leave{-webkit-animation-timing-function:cubic-bezier(.78,.14,.15,.86);animation-timing-function:cubic-bezier(.78,.14,.15,.86)}.zoom-down-appear,.zoom-down-enter,.zoom-down-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.zoom-down-appear.zoom-down-appear-active,.zoom-down-enter.zoom-down-enter-active{-webkit-animation-name:antZoomDownIn;animation-name:antZoomDownIn;-webkit-animation-play-state:running;animation-play-state:running}.zoom-down-leave.zoom-down-leave-active{-webkit-animation-name:antZoomDownOut;animation-name:antZoomDownOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.zoom-down-appear,.zoom-down-enter{-webkit-transform:scale(0);transform:scale(0);opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.zoom-down-leave{-webkit-animation-timing-function:cubic-bezier(.78,.14,.15,.86);animation-timing-function:cubic-bezier(.78,.14,.15,.86)}.zoom-left-appear,.zoom-left-enter,.zoom-left-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.zoom-left-appear.zoom-left-appear-active,.zoom-left-enter.zoom-left-enter-active{-webkit-animation-name:antZoomLeftIn;animation-name:antZoomLeftIn;-webkit-animation-play-state:running;animation-play-state:running}.zoom-left-leave.zoom-left-leave-active{-webkit-animation-name:antZoomLeftOut;animation-name:antZoomLeftOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.zoom-left-appear,.zoom-left-enter{-webkit-transform:scale(0);transform:scale(0);opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.zoom-left-leave{-webkit-animation-timing-function:cubic-bezier(.78,.14,.15,.86);animation-timing-function:cubic-bezier(.78,.14,.15,.86)}.zoom-right-appear,.zoom-right-enter,.zoom-right-leave{-webkit-animation-duration:66ms;animation-duration:66ms;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.zoom-right-appear.zoom-right-appear-active,.zoom-right-enter.zoom-right-enter-active{-webkit-animation-name:antZoomRightIn;animation-name:antZoomRightIn;-webkit-animation-play-state:running;animation-play-state:running}.zoom-right-leave.zoom-right-leave-active{-webkit-animation-name:antZoomRightOut;animation-name:antZoomRightOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.zoom-right-appear,.zoom-right-enter{-webkit-transform:scale(0);transform:scale(0);opacity:0;-webkit-animation-timing-function:cubic-bezier(.08,.82,.17,1);animation-timing-function:cubic-bezier(.08,.82,.17,1)}.zoom-right-leave{-webkit-animation-timing-function:cubic-bezier(.78,.14,.15,.86);animation-timing-function:cubic-bezier(.78,.14,.15,.86)}@-webkit-keyframes antZoomIn{0%{-webkit-transform:scale(.2);transform:scale(.2);opacity:0}to{-webkit-transform:scale(1);transform:scale(1);opacity:1}}@keyframes antZoomIn{0%{-webkit-transform:scale(.2);transform:scale(.2);opacity:0}to{-webkit-transform:scale(1);transform:scale(1);opacity:1}}@-webkit-keyframes antZoomOut{0%{-webkit-transform:scale(1);transform:scale(1)}to{-webkit-transform:scale(.2);transform:scale(.2);opacity:0}}@keyframes antZoomOut{0%{-webkit-transform:scale(1);transform:scale(1)}to{-webkit-transform:scale(.2);transform:scale(.2);opacity:0}}@-webkit-keyframes antZoomBigIn{0%{-webkit-transform:scale(.8);transform:scale(.8);opacity:0}to{-webkit-transform:scale(1);transform:scale(1);opacity:1}}@keyframes antZoomBigIn{0%{-webkit-transform:scale(.8);transform:scale(.8);opacity:0}to{-webkit-transform:scale(1);transform:scale(1);opacity:1}}@-webkit-keyframes antZoomBigOut{0%{-webkit-transform:scale(1);transform:scale(1)}to{-webkit-transform:scale(.8);transform:scale(.8);opacity:0}}@keyframes antZoomBigOut{0%{-webkit-transform:scale(1);transform:scale(1)}to{-webkit-transform:scale(.8);transform:scale(.8);opacity:0}}@-webkit-keyframes antZoomUpIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 0;transform-origin:50% 0;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 0;transform-origin:50% 0}}@keyframes antZoomUpIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 0;transform-origin:50% 0;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 0;transform-origin:50% 0}}@-webkit-keyframes antZoomUpOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 0;transform-origin:50% 0}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 0;transform-origin:50% 0;opacity:0}}@keyframes antZoomUpOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 0;transform-origin:50% 0}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 0;transform-origin:50% 0;opacity:0}}@-webkit-keyframes antZoomLeftIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:0 50%;transform-origin:0 50%;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:0 50%;transform-origin:0 50%}}@keyframes antZoomLeftIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:0 50%;transform-origin:0 50%;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:0 50%;transform-origin:0 50%}}@-webkit-keyframes antZoomLeftOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:0 50%;transform-origin:0 50%}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:0 50%;transform-origin:0 50%;opacity:0}}@keyframes antZoomLeftOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:0 50%;transform-origin:0 50%}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:0 50%;transform-origin:0 50%;opacity:0}}@-webkit-keyframes antZoomRightIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:100% 50%;transform-origin:100% 50%;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:100% 50%;transform-origin:100% 50%}}@keyframes antZoomRightIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:100% 50%;transform-origin:100% 50%;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:100% 50%;transform-origin:100% 50%}}@-webkit-keyframes antZoomRightOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:100% 50%;transform-origin:100% 50%}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:100% 50%;transform-origin:100% 50%;opacity:0}}@keyframes antZoomRightOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:100% 50%;transform-origin:100% 50%}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:100% 50%;transform-origin:100% 50%;opacity:0}}@-webkit-keyframes antZoomDownIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 100%;transform-origin:50% 100%}}@keyframes antZoomDownIn{0%{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;opacity:0}to{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 100%;transform-origin:50% 100%}}@-webkit-keyframes antZoomDownOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 100%;transform-origin:50% 100%}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;opacity:0}}@keyframes antZoomDownOut{0%{-webkit-transform:scale(1);transform:scale(1);-webkit-transform-origin:50% 100%;transform-origin:50% 100%}to{-webkit-transform:scale(.8);transform:scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;opacity:0}}.ant-motion-collapse-legacy{overflow:hidden}.ant-motion-collapse,.ant-motion-collapse-legacy-active{-webkit-transition:height .15s cubic-bezier(.645,.045,.355,1),opacity .15s cubic-bezier(.645,.045,.355,1)!important;transition:height .15s cubic-bezier(.645,.045,.355,1),opacity .15s cubic-bezier(.645,.045,.355,1)!important}.ant-motion-collapse{overflow:hidden}.ant-menu{box-sizing:border-box;font-size:14px;font-variant:tabular-nums;line-height:1.5;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";margin:0;padding:0;color:#e3e6e8;line-height:0;list-style:none;background:#2e3538;outline:none;box-shadow:0 2px 8px rgba(0,0,0,.15);-webkit-transition:background .3s,width .2s;transition:background .3s,width .2s;zoom:1}.ant-menu:after,.ant-menu:before{display:table;content:""}.ant-menu:after{clear:both}.ant-menu ol,.ant-menu ul{margin:0;padding:0;list-style:none}.ant-menu-hidden{display:none}.ant-menu-item-group-title{padding:8px 16px;color:#c2cfd6;font-size:14px;line-height:1.5;-webkit-transition:all .3s;transition:all .3s}.ant-menu-submenu,.ant-menu-submenu-inline{-webkit-transition:border-color .3s cubic-bezier(.645,.045,.355,1),background .3s cubic-bezier(.645,.045,.355,1),padding .15s cubic-bezier(.645,.045,.355,1);transition:border-color .3s cubic-bezier(.645,.045,.355,1),background .3s cubic-bezier(.645,.045,.355,1),padding .15s cubic-bezier(.645,.045,.355,1)}.ant-menu-item:active,.ant-menu-submenu-title:active{background:rgba(0,170,255,.2)}.ant-menu-submenu .ant-menu-sub{cursor:auto;-webkit-transition:background .3s cubic-bezier(.645,.045,.355,1),padding .3s cubic-bezier(.645,.045,.355,1);transition:background .3s cubic-bezier(.645,.045,.355,1),padding .3s cubic-bezier(.645,.045,.355,1)}.ant-menu-item>a{display:block;color:#e3e6e8}.ant-menu-item>a:hover{color:#0af}.ant-menu-item>a:before{position:absolute;top:0;right:0;bottom:0;left:0;background-color:transparent;content:""}.ant-menu-item-divider{height:1px;overflow:hidden;line-height:0;background-color:#454f54}.ant-menu-item-active,.ant-menu-item:hover,.ant-menu-submenu-active,.ant-menu-submenu-title:hover,.ant-menu:not(.ant-menu-inline) .ant-menu-submenu-open{color:#0af}.ant-menu-horizontal .ant-menu-item,.ant-menu-horizontal .ant-menu-submenu{margin-top:-1px}.ant-menu-horizontal>.ant-menu-item-active,.ant-menu-horizontal>.ant-menu-item:hover,.ant-menu-horizontal>.ant-menu-submenu .ant-menu-submenu-title:hover{background-color:transparent}.ant-menu-item-selected,.ant-menu-item-selected>a,.ant-menu-item-selected>a:hover{color:#0af}.ant-menu:not(.ant-menu-horizontal) .ant-menu-item-selected{background-color:rgba(0,170,255,.2)}.ant-menu-inline,.ant-menu-vertical,.ant-menu-vertical-left{border-right:1px solid #454f54}.ant-menu-vertical-right{border-left:1px solid #454f54}.ant-menu-vertical-left.ant-menu-sub,.ant-menu-vertical-right.ant-menu-sub,.ant-menu-vertical.ant-menu-sub{min-width:160px;padding:0;border-right:0;-webkit-transform-origin:0 0;transform-origin:0 0}.ant-menu-vertical-left.ant-menu-sub .ant-menu-item,.ant-menu-vertical-right.ant-menu-sub .ant-menu-item,.ant-menu-vertical.ant-menu-sub .ant-menu-item{left:0;margin-left:0;border-right:0}.ant-menu-vertical-left.ant-menu-sub .ant-menu-item:after,.ant-menu-vertical-right.ant-menu-sub .ant-menu-item:after,.ant-menu-vertical.ant-menu-sub .ant-menu-item:after{border-right:0}.ant-menu-vertical-left.ant-menu-sub>.ant-menu-item,.ant-menu-vertical-left.ant-menu-sub>.ant-menu-submenu,.ant-menu-vertical-right.ant-menu-sub>.ant-menu-item,.ant-menu-vertical-right.ant-menu-sub>.ant-menu-submenu,.ant-menu-vertical.ant-menu-sub>.ant-menu-item,.ant-menu-vertical.ant-menu-sub>.ant-menu-submenu{-webkit-transform-origin:0 0;transform-origin:0 0}.ant-menu-horizontal.ant-menu-sub{min-width:114px}.ant-menu-item,.ant-menu-submenu-title{position:relative;display:block;margin:0;padding:0 20px;white-space:nowrap;cursor:pointer;-webkit-transition:color .3s cubic-bezier(.645,.045,.355,1),border-color .3s cubic-bezier(.645,.045,.355,1),background .3s cubic-bezier(.645,.045,.355,1),padding .15s cubic-bezier(.645,.045,.355,1);transition:color .3s cubic-bezier(.645,.045,.355,1),border-color .3s cubic-bezier(.645,.045,.355,1),background .3s cubic-bezier(.645,.045,.355,1),padding .15s cubic-bezier(.645,.045,.355,1)}.ant-menu-item .anticon,.ant-menu-submenu-title .anticon{min-width:14px;margin-right:10px;font-size:14px;-webkit-transition:font-size .15s cubic-bezier(.215,.61,.355,1),margin .3s cubic-bezier(.645,.045,.355,1);transition:font-size .15s cubic-bezier(.215,.61,.355,1),margin .3s cubic-bezier(.645,.045,.355,1)}.ant-menu-item .anticon+span,.ant-menu-submenu-title .anticon+span{opacity:1;-webkit-transition:opacity .3s cubic-bezier(.645,.045,.355,1),width .3s cubic-bezier(.645,.045,.355,1);transition:opacity .3s cubic-bezier(.645,.045,.355,1),width .3s cubic-bezier(.645,.045,.355,1)}.ant-menu>.ant-menu-item-divider{height:1px;margin:1px 0;padding:0;overflow:hidden;line-height:0;background-color:#454f54}.ant-menu-submenu-popup{position:absolute;z-index:1050;background:#2e3538;border-radius:2px}.ant-menu-submenu-popup .submenu-title-wrapper{padding-right:20px}.ant-menu-submenu-popup:before{position:absolute;top:-7px;right:0;bottom:0;left:0;opacity:.0001;content:" "}.ant-menu-submenu>.ant-menu{background-color:#2e3538;border-radius:2px}.ant-menu-submenu>.ant-menu-submenu-title:after{-webkit-transition:-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1)}.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow,.ant-menu-submenu-vertical-left>.ant-menu-submenu-title .ant-menu-submenu-arrow,.ant-menu-submenu-vertical-right>.ant-menu-submenu-title .ant-menu-submenu-arrow,.ant-menu-submenu-vertical>.ant-menu-submenu-title .ant-menu-submenu-arrow{position:absolute;top:50%;right:16px;width:10px;-webkit-transition:-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1)}.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical-left>.ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical-left>.ant-menu-submenu-title .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical-right>.ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical-right>.ant-menu-submenu-title .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical>.ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical>.ant-menu-submenu-title .ant-menu-submenu-arrow:before{position:absolute;width:6px;height:1.5px;background:#2e3538;background:#e3e6e8\9;background-image:-webkit-gradient(linear,left top,right top,from(#e3e6e8),to(#e3e6e8));background-image:-webkit-linear-gradient(left,#e3e6e8,#e3e6e8);background-image:linear-gradient(90deg,#e3e6e8,#e3e6e8);background-image:none\9;border-radius:2px;-webkit-transition:background .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:background .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:background .3s cubic-bezier(.645,.045,.355,1),transform .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1);transition:background .3s cubic-bezier(.645,.045,.355,1),transform .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1);content:""}.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical-left>.ant-menu-submenu-title .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical-right>.ant-menu-submenu-title .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical>.ant-menu-submenu-title .ant-menu-submenu-arrow:before{-webkit-transform:rotate(45deg) translateY(-2px);transform:rotate(45deg) translateY(-2px)}.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical-left>.ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical-right>.ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical>.ant-menu-submenu-title .ant-menu-submenu-arrow:after{-webkit-transform:rotate(-45deg) translateY(2px);transform:rotate(-45deg) translateY(2px)}.ant-menu-submenu-inline>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:after,.ant-menu-submenu-inline>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical-left>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical-left>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical-right>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical-right>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:before,.ant-menu-submenu-vertical>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:after,.ant-menu-submenu-vertical>.ant-menu-submenu-title:hover .ant-menu-submenu-arrow:before{background:-webkit-gradient(linear,left top,right top,from(#0af),to(#0af));background:-webkit-linear-gradient(left,#0af,#0af);background:linear-gradient(90deg,#0af,#0af)}.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:before{-webkit-transform:rotate(-45deg) translateX(2px);transform:rotate(-45deg) translateX(2px)}.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:after{-webkit-transform:rotate(45deg) translateX(-2px);transform:rotate(45deg) translateX(-2px)}.ant-menu-submenu-open.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow{-webkit-transform:translateY(-2px);transform:translateY(-2px)}.ant-menu-submenu-open.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:after{-webkit-transform:rotate(-45deg) translateX(-2px);transform:rotate(-45deg) translateX(-2px)}.ant-menu-submenu-open.ant-menu-submenu-inline>.ant-menu-submenu-title .ant-menu-submenu-arrow:before{-webkit-transform:rotate(45deg) translateX(2px);transform:rotate(45deg) translateX(2px)}.ant-menu-vertical-left .ant-menu-submenu-selected,.ant-menu-vertical-left .ant-menu-submenu-selected>a,.ant-menu-vertical-right .ant-menu-submenu-selected,.ant-menu-vertical-right .ant-menu-submenu-selected>a,.ant-menu-vertical .ant-menu-submenu-selected,.ant-menu-vertical .ant-menu-submenu-selected>a{color:#0af}.ant-menu-horizontal{line-height:46px;white-space:nowrap;border:0;border-bottom:1px solid #454f54;box-shadow:none}.ant-menu-horizontal>.ant-menu-item,.ant-menu-horizontal>.ant-menu-submenu{position:relative;top:1px;display:inline-block;vertical-align:bottom;border-bottom:2px solid transparent}.ant-menu-horizontal>.ant-menu-item-active,.ant-menu-horizontal>.ant-menu-item-open,.ant-menu-horizontal>.ant-menu-item-selected,.ant-menu-horizontal>.ant-menu-item:hover,.ant-menu-horizontal>.ant-menu-submenu-active,.ant-menu-horizontal>.ant-menu-submenu-open,.ant-menu-horizontal>.ant-menu-submenu-selected,.ant-menu-horizontal>.ant-menu-submenu:hover{color:#0af;border-bottom:2px solid #0af}.ant-menu-horizontal>.ant-menu-item>a{display:block;color:#e3e6e8}.ant-menu-horizontal>.ant-menu-item>a:hover{color:#0af}.ant-menu-horizontal>.ant-menu-item>a:before{bottom:-2px}.ant-menu-horizontal>.ant-menu-item-selected>a{color:#0af}.ant-menu-horizontal:after{display:block;clear:both;height:0;content:"\20"}.ant-menu-inline .ant-menu-item,.ant-menu-vertical-left .ant-menu-item,.ant-menu-vertical-right .ant-menu-item,.ant-menu-vertical .ant-menu-item{position:relative}.ant-menu-inline .ant-menu-item:after,.ant-menu-vertical-left .ant-menu-item:after,.ant-menu-vertical-right .ant-menu-item:after,.ant-menu-vertical .ant-menu-item:after{position:absolute;top:0;right:0;bottom:0;border-right:3px solid #0af;-webkit-transform:scaleY(.0001);transform:scaleY(.0001);opacity:0;-webkit-transition:opacity .15s cubic-bezier(.215,.61,.355,1),-webkit-transform .15s cubic-bezier(.215,.61,.355,1);transition:opacity .15s cubic-bezier(.215,.61,.355,1),-webkit-transform .15s cubic-bezier(.215,.61,.355,1);transition:transform .15s cubic-bezier(.215,.61,.355,1),opacity .15s cubic-bezier(.215,.61,.355,1);transition:transform .15s cubic-bezier(.215,.61,.355,1),opacity .15s cubic-bezier(.215,.61,.355,1),-webkit-transform .15s cubic-bezier(.215,.61,.355,1);content:""}.ant-menu-inline .ant-menu-item,.ant-menu-inline .ant-menu-submenu-title,.ant-menu-vertical-left .ant-menu-item,.ant-menu-vertical-left .ant-menu-submenu-title,.ant-menu-vertical-right .ant-menu-item,.ant-menu-vertical-right .ant-menu-submenu-title,.ant-menu-vertical .ant-menu-item,.ant-menu-vertical .ant-menu-submenu-title{height:40px;margin-top:4px;margin-bottom:4px;padding:0 16px;overflow:hidden;font-size:14px;line-height:40px;text-overflow:ellipsis}.ant-menu-inline .ant-menu-submenu,.ant-menu-vertical-left .ant-menu-submenu,.ant-menu-vertical-right .ant-menu-submenu,.ant-menu-vertical .ant-menu-submenu{padding-bottom:.01px}.ant-menu-inline .ant-menu-item:not(:last-child),.ant-menu-vertical-left .ant-menu-item:not(:last-child),.ant-menu-vertical-right .ant-menu-item:not(:last-child),.ant-menu-vertical .ant-menu-item:not(:last-child){margin-bottom:8px}.ant-menu-inline>.ant-menu-item,.ant-menu-inline>.ant-menu-submenu>.ant-menu-submenu-title,.ant-menu-vertical-left>.ant-menu-item,.ant-menu-vertical-left>.ant-menu-submenu>.ant-menu-submenu-title,.ant-menu-vertical-right>.ant-menu-item,.ant-menu-vertical-right>.ant-menu-submenu>.ant-menu-submenu-title,.ant-menu-vertical>.ant-menu-item,.ant-menu-vertical>.ant-menu-submenu>.ant-menu-submenu-title{height:40px;line-height:40px}.ant-menu-inline{width:100%}.ant-menu-inline .ant-menu-item-selected:after,.ant-menu-inline .ant-menu-selected:after{-webkit-transform:scaleY(1);transform:scaleY(1);opacity:1;-webkit-transition:opacity .15s cubic-bezier(.645,.045,.355,1),-webkit-transform .15s cubic-bezier(.645,.045,.355,1);transition:opacity .15s cubic-bezier(.645,.045,.355,1),-webkit-transform .15s cubic-bezier(.645,.045,.355,1);transition:transform .15s cubic-bezier(.645,.045,.355,1),opacity .15s cubic-bezier(.645,.045,.355,1);transition:transform .15s cubic-bezier(.645,.045,.355,1),opacity .15s cubic-bezier(.645,.045,.355,1),-webkit-transform .15s cubic-bezier(.645,.045,.355,1)}.ant-menu-inline .ant-menu-item,.ant-menu-inline .ant-menu-submenu-title{width:calc(100% + 1px)}.ant-menu-inline .ant-menu-submenu-title{padding-right:34px}.ant-menu-inline-collapsed{width:80px}.ant-menu-inline-collapsed>.ant-menu-item,.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-item,.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-submenu>.ant-menu-submenu-title,.ant-menu-inline-collapsed>.ant-menu-submenu>.ant-menu-submenu-title{left:0;padding:0 32px!important;text-overflow:clip}.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-item .ant-menu-submenu-arrow,.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-submenu>.ant-menu-submenu-title .ant-menu-submenu-arrow,.ant-menu-inline-collapsed>.ant-menu-item .ant-menu-submenu-arrow,.ant-menu-inline-collapsed>.ant-menu-submenu>.ant-menu-submenu-title .ant-menu-submenu-arrow{display:none}.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-item .anticon,.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-submenu>.ant-menu-submenu-title .anticon,.ant-menu-inline-collapsed>.ant-menu-item .anticon,.ant-menu-inline-collapsed>.ant-menu-submenu>.ant-menu-submenu-title .anticon{margin:0;font-size:16px;line-height:40px}.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-item .anticon+span,.ant-menu-inline-collapsed>.ant-menu-item-group>.ant-menu-item-group-list>.ant-menu-submenu>.ant-menu-submenu-title .anticon+span,.ant-menu-inline-collapsed>.ant-menu-item .anticon+span,.ant-menu-inline-collapsed>.ant-menu-submenu>.ant-menu-submenu-title .anticon+span{display:inline-block;max-width:0;opacity:0}.ant-menu-inline-collapsed-tooltip{pointer-events:none}.ant-menu-inline-collapsed-tooltip .anticon{display:none}.ant-menu-inline-collapsed-tooltip a{color:hsla(0,0%,100%,.85)}.ant-menu-inline-collapsed .ant-menu-item-group-title{padding-right:4px;padding-left:4px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.ant-menu-item-group-list{margin:0;padding:0}.ant-menu-item-group-list .ant-menu-item,.ant-menu-item-group-list .ant-menu-submenu-title{padding:0 16px 0 28px}.ant-menu-root.ant-menu-inline,.ant-menu-root.ant-menu-vertical,.ant-menu-root.ant-menu-vertical-left,.ant-menu-root.ant-menu-vertical-right{box-shadow:none}.ant-menu-sub.ant-menu-inline{padding:0;border:0;border-radius:0;box-shadow:none}.ant-menu-sub.ant-menu-inline>.ant-menu-item,.ant-menu-sub.ant-menu-inline>.ant-menu-submenu>.ant-menu-submenu-title{height:40px;line-height:40px;list-style-position:inside;list-style-type:disc}.ant-menu-sub.ant-menu-inline .ant-menu-item-group-title{padding-left:32px}.ant-menu-item-disabled,.ant-menu-submenu-disabled{color:hsla(0,0%,100%,.5)!important;background:none;border-color:transparent!important;cursor:not-allowed}.ant-menu-item-disabled>a,.ant-menu-submenu-disabled>a{color:hsla(0,0%,100%,.5)!important;pointer-events:none}.ant-menu-item-disabled>.ant-menu-submenu-title,.ant-menu-submenu-disabled>.ant-menu-submenu-title{color:hsla(0,0%,100%,.5)!important;cursor:not-allowed}.ant-menu-item-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-item-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before,.ant-menu-submenu-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-submenu-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before{background:hsla(0,0%,100%,.5)!important}.ant-menu-dark,.ant-menu-dark .ant-menu-sub{color:hsla(0,0%,100%,.65);background:#2e3538}.ant-menu-dark .ant-menu-sub .ant-menu-submenu-title .ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-title .ant-menu-submenu-arrow{opacity:.45;-webkit-transition:all .3s;transition:all .3s}.ant-menu-dark .ant-menu-sub .ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-sub .ant-menu-submenu-title .ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-title .ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-title .ant-menu-submenu-arrow:before{background:#fff}.ant-menu-dark.ant-menu-submenu-popup{background:transparent}.ant-menu-dark .ant-menu-inline.ant-menu-sub{background:#000c17;box-shadow:inset 0 2px 8px hsla(0,0%,100%,.45)}.ant-menu-dark.ant-menu-horizontal{border-bottom:0}.ant-menu-dark.ant-menu-horizontal>.ant-menu-item,.ant-menu-dark.ant-menu-horizontal>.ant-menu-submenu{top:0;margin-top:0;border-color:#2e3538;border-bottom:0}.ant-menu-dark.ant-menu-horizontal>.ant-menu-item>a:before{bottom:0}.ant-menu-dark .ant-menu-item,.ant-menu-dark .ant-menu-item-group-title,.ant-menu-dark .ant-menu-item>a{color:hsla(0,0%,100%,.65)}.ant-menu-dark.ant-menu-inline,.ant-menu-dark.ant-menu-vertical,.ant-menu-dark.ant-menu-vertical-left,.ant-menu-dark.ant-menu-vertical-right{border-right:0}.ant-menu-dark.ant-menu-inline .ant-menu-item,.ant-menu-dark.ant-menu-vertical-left .ant-menu-item,.ant-menu-dark.ant-menu-vertical-right .ant-menu-item,.ant-menu-dark.ant-menu-vertical .ant-menu-item{left:0;margin-left:0;border-right:0}.ant-menu-dark.ant-menu-inline .ant-menu-item:after,.ant-menu-dark.ant-menu-vertical-left .ant-menu-item:after,.ant-menu-dark.ant-menu-vertical-right .ant-menu-item:after,.ant-menu-dark.ant-menu-vertical .ant-menu-item:after{border-right:0}.ant-menu-dark.ant-menu-inline .ant-menu-item,.ant-menu-dark.ant-menu-inline .ant-menu-submenu-title{width:100%}.ant-menu-dark .ant-menu-item-active,.ant-menu-dark .ant-menu-item:hover,.ant-menu-dark .ant-menu-submenu-active,.ant-menu-dark .ant-menu-submenu-open,.ant-menu-dark .ant-menu-submenu-selected,.ant-menu-dark .ant-menu-submenu-title:hover{color:#fff;background-color:transparent}.ant-menu-dark .ant-menu-item-active>a,.ant-menu-dark .ant-menu-item:hover>a,.ant-menu-dark .ant-menu-submenu-active>a,.ant-menu-dark .ant-menu-submenu-open>a,.ant-menu-dark .ant-menu-submenu-selected>a,.ant-menu-dark .ant-menu-submenu-title:hover>a{color:#fff}.ant-menu-dark .ant-menu-item-active>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-item-active>.ant-menu-submenu-title>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-item:hover>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-item:hover>.ant-menu-submenu-title>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-active>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-active>.ant-menu-submenu-title>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-open>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-open>.ant-menu-submenu-title>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-selected>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-selected>.ant-menu-submenu-title>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-title:hover>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow,.ant-menu-dark .ant-menu-submenu-title:hover>.ant-menu-submenu-title>.ant-menu-submenu-arrow{opacity:1}.ant-menu-dark .ant-menu-item-active>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-item-active>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-item-active>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-item-active>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-item:hover>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-item:hover>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-item:hover>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-item:hover>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-active>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-active>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-active>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-active>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-open>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-open>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-open>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-open>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-selected>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-selected>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-selected>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-selected>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-title:hover>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-title:hover>.ant-menu-submenu-title:hover>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-title:hover>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-title:hover>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before{background:#fff}.ant-menu-dark .ant-menu-item-selected{color:#fff;border-right:0}.ant-menu-dark .ant-menu-item-selected:after{border-right:0}.ant-menu-dark .ant-menu-item-selected>a,.ant-menu-dark .ant-menu-item-selected>a:hover{color:#fff}.ant-menu-submenu-popup.ant-menu-dark .ant-menu-item-selected,.ant-menu.ant-menu-dark .ant-menu-item-selected{background-color:#0af}.ant-menu-dark .ant-menu-item-disabled,.ant-menu-dark .ant-menu-item-disabled>a,.ant-menu-dark .ant-menu-submenu-disabled,.ant-menu-dark .ant-menu-submenu-disabled>a{color:hsla(0,0%,100%,.35)!important;opacity:.8}.ant-menu-dark .ant-menu-item-disabled>.ant-menu-submenu-title,.ant-menu-dark .ant-menu-submenu-disabled>.ant-menu-submenu-title{color:hsla(0,0%,100%,.35)!important}.ant-menu-dark .ant-menu-item-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-item-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before,.ant-menu-dark .ant-menu-submenu-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:after,.ant-menu-dark .ant-menu-submenu-disabled>.ant-menu-submenu-title>.ant-menu-submenu-arrow:before{background:hsla(0,0%,100%,.35)!important}.ant-tooltip{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:absolute;z-index:1060;display:block;max-width:250px;visibility:visible}.ant-tooltip-hidden{display:none}.ant-tooltip-placement-top,.ant-tooltip-placement-topLeft,.ant-tooltip-placement-topRight{padding-bottom:8px}.ant-tooltip-placement-right,.ant-tooltip-placement-rightBottom,.ant-tooltip-placement-rightTop{padding-left:8px}.ant-tooltip-placement-bottom,.ant-tooltip-placement-bottomLeft,.ant-tooltip-placement-bottomRight{padding-top:8px}.ant-tooltip-placement-left,.ant-tooltip-placement-leftBottom,.ant-tooltip-placement-leftTop{padding-right:8px}.ant-tooltip-inner{min-width:30px;min-height:32px;padding:6px 8px;color:#fff;text-align:left;text-decoration:none;word-wrap:break-word;background-color:rgba(0,0,0,.75);border-radius:2px;box-shadow:0 2px 8px rgba(0,0,0,.15)}.ant-tooltip-arrow{position:absolute;width:0;height:0;border-color:transparent;border-style:solid}.ant-tooltip-placement-top .ant-tooltip-arrow,.ant-tooltip-placement-topLeft .ant-tooltip-arrow,.ant-tooltip-placement-topRight .ant-tooltip-arrow{bottom:3px;border-width:5px 5px 0;border-top-color:rgba(0,0,0,.75)}.ant-tooltip-placement-top .ant-tooltip-arrow{left:50%;margin-left:-5px}.ant-tooltip-placement-topLeft .ant-tooltip-arrow{left:16px}.ant-tooltip-placement-topRight .ant-tooltip-arrow{right:16px}.ant-tooltip-placement-right .ant-tooltip-arrow,.ant-tooltip-placement-rightBottom .ant-tooltip-arrow,.ant-tooltip-placement-rightTop .ant-tooltip-arrow{left:3px;border-width:5px 5px 5px 0;border-right-color:rgba(0,0,0,.75)}.ant-tooltip-placement-right .ant-tooltip-arrow{top:50%;margin-top:-5px}.ant-tooltip-placement-rightTop .ant-tooltip-arrow{top:8px}.ant-tooltip-placement-rightBottom .ant-tooltip-arrow{bottom:8px}.ant-tooltip-placement-left .ant-tooltip-arrow,.ant-tooltip-placement-leftBottom .ant-tooltip-arrow,.ant-tooltip-placement-leftTop .ant-tooltip-arrow{right:3px;border-width:5px 0 5px 5px;border-left-color:rgba(0,0,0,.75)}.ant-tooltip-placement-left .ant-tooltip-arrow{top:50%;margin-top:-5px}.ant-tooltip-placement-leftTop .ant-tooltip-arrow{top:8px}.ant-tooltip-placement-leftBottom .ant-tooltip-arrow{bottom:8px}.ant-tooltip-placement-bottom .ant-tooltip-arrow,.ant-tooltip-placement-bottomLeft .ant-tooltip-arrow,.ant-tooltip-placement-bottomRight .ant-tooltip-arrow{top:3px;border-width:0 5px 5px;border-bottom-color:rgba(0,0,0,.75)}.ant-tooltip-placement-bottom .ant-tooltip-arrow{left:50%;margin-left:-5px}.ant-tooltip-placement-bottomLeft .ant-tooltip-arrow{left:16px}.ant-tooltip-placement-bottomRight .ant-tooltip-arrow{right:16px}.ant-alert{box-sizing:border-box;margin:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;padding:8px 15px 8px 37px;border-radius:2px}.ant-alert.ant-alert-no-icon{padding:8px 15px}.ant-alert.ant-alert-closable{padding-right:30px}.ant-alert-icon{position:absolute;top:11.5px;left:16px}.ant-alert-description{display:none;font-size:14px;line-height:22px}.ant-alert-success{background-color:#f6ffed;border:1px solid #b7eb8f}.ant-alert-success .ant-alert-icon{color:#52c41a}.ant-alert-info{background-color:#e6f7ff;border:1px solid #91d5ff}.ant-alert-info .ant-alert-icon{color:#1890ff}.ant-alert-warning{background-color:#fffbe6;border:1px solid #ffe58f}.ant-alert-warning .ant-alert-icon{color:#faad14}.ant-alert-error{background-color:#fff1f0;border:1px solid #ffa39e}.ant-alert-error .ant-alert-icon{color:#f5222d}.ant-alert-close-icon{position:absolute;top:8px;right:16px;overflow:hidden;font-size:12px;line-height:22px;cursor:pointer}.ant-alert-close-icon .anticon-close{color:#c2cfd6;-webkit-transition:color .3s;transition:color .3s}.ant-alert-close-icon .anticon-close:hover{color:hsla(0,0%,100%,.75)}.ant-alert-close-text{position:absolute;right:16px}.ant-alert-with-description{position:relative;padding:15px 15px 15px 64px;color:#e3e6e8;line-height:1.5;border-radius:2px}.ant-alert-with-description.ant-alert-no-icon{padding:15px}.ant-alert-with-description .ant-alert-icon{position:absolute;top:16px;left:24px;font-size:24px}.ant-alert-with-description .ant-alert-close-icon{position:absolute;top:16px;right:16px;font-size:14px;cursor:pointer}.ant-alert-with-description .ant-alert-message{display:block;margin-bottom:4px;color:hsla(0,0%,100%,.85);font-size:16px}.ant-alert-with-description .ant-alert-description{display:block}.ant-alert.ant-alert-close{height:0!important;margin:0;padding-top:0;padding-bottom:0;-webkit-transform-origin:50% 0;transform-origin:50% 0;-webkit-transition:all .3s cubic-bezier(.78,.14,.15,.86);transition:all .3s cubic-bezier(.78,.14,.15,.86)}.ant-alert-slide-up-leave{-webkit-animation:antAlertSlideUpOut .3s cubic-bezier(.78,.14,.15,.86);animation:antAlertSlideUpOut .3s cubic-bezier(.78,.14,.15,.86);-webkit-animation-fill-mode:both;animation-fill-mode:both}.ant-alert-banner{margin-bottom:0;border:0;border-radius:0}@-webkit-keyframes antAlertSlideUpIn{0%{-webkit-transform:scaleY(0);transform:scaleY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@keyframes antAlertSlideUpIn{0%{-webkit-transform:scaleY(0);transform:scaleY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}to{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}}@-webkit-keyframes antAlertSlideUpOut{0%{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:scaleY(0);transform:scaleY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}@keyframes antAlertSlideUpOut{0%{-webkit-transform:scaleY(1);transform:scaleY(1);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:1}to{-webkit-transform:scaleY(0);transform:scaleY(0);-webkit-transform-origin:0 0;transform-origin:0 0;opacity:0}}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-nav-container{height:40px}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-ink-bar{visibility:hidden}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-tab{height:40px;margin:0 2px 0 0;padding:0 16px;line-height:38px;background:#5c6970;border:1px solid #454f54;border-radius:2px 2px 0 0;-webkit-transition:all .3s cubic-bezier(.645,.045,.355,1);transition:all .3s cubic-bezier(.645,.045,.355,1)}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-tab-active{height:40px;color:#0af;background:#2e3538;border-color:#454f54;border-bottom:1px solid #2e3538}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-tab-inactive{padding:0}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-nav-wrap{margin-bottom:0}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-tab .ant-tabs-close-x{width:16px;height:16px;height:14px;margin-right:-5px;margin-left:3px;overflow:hidden;color:#c2cfd6;font-size:12px;vertical-align:middle;-webkit-transition:all .3s;transition:all .3s}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-tab .ant-tabs-close-x:hover{color:hsla(0,0%,100%,.85)}.ant-tabs.ant-tabs-card .ant-tabs-card-content>.ant-tabs-tabpane,.ant-tabs.ant-tabs-editable-card .ant-tabs-card-content>.ant-tabs-tabpane{-webkit-transition:none!important;transition:none!important}.ant-tabs.ant-tabs-card .ant-tabs-card-content>.ant-tabs-tabpane-inactive,.ant-tabs.ant-tabs-editable-card .ant-tabs-card-content>.ant-tabs-tabpane-inactive{overflow:hidden}.ant-tabs.ant-tabs-card .ant-tabs-card-bar .ant-tabs-tab:hover .anticon-close{opacity:1}.ant-tabs-extra-content{line-height:40px}.ant-tabs-extra-content .ant-tabs-new-tab{position:relative;width:20px;height:20px;color:#e3e6e8;font-size:12px;line-height:20px;text-align:center;border:1px solid #454f54;border-radius:0;cursor:pointer;-webkit-transition:all .3s;transition:all .3s}.ant-tabs-extra-content .ant-tabs-new-tab:hover{color:#0af;border-color:#0af}.ant-tabs-extra-content .ant-tabs-new-tab svg{position:absolute;top:0;right:0;bottom:0;left:0;margin:auto}.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-nav-container,.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-nav-container{height:100%}.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-tab,.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-tab{margin-bottom:8px;border-bottom:1px solid #454f54}.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-tab-active,.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-tab-active{padding-bottom:4px}.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-tab:last-child,.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-tab:last-child{margin-bottom:8px}.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-new-tab,.ant-tabs-vertical.ant-tabs-card .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-new-tab{width:90%}.ant-tabs-vertical.ant-tabs-card.ant-tabs-left .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-nav-wrap{margin-right:0}.ant-tabs-vertical.ant-tabs-card.ant-tabs-left .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-tab{margin-right:1px;border-right:0;border-radius:2px 0 0 2px}.ant-tabs-vertical.ant-tabs-card.ant-tabs-left .ant-tabs-card-bar.ant-tabs-left-bar .ant-tabs-tab-active{margin-right:-1px;padding-right:18px}.ant-tabs-vertical.ant-tabs-card.ant-tabs-right .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-nav-wrap{margin-left:0}.ant-tabs-vertical.ant-tabs-card.ant-tabs-right .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-tab{margin-left:1px;border-left:0;border-radius:0 2px 2px 0}.ant-tabs-vertical.ant-tabs-card.ant-tabs-right .ant-tabs-card-bar.ant-tabs-right-bar .ant-tabs-tab-active{margin-left:-1px;padding-left:18px}.ant-tabs .ant-tabs-card-bar.ant-tabs-bottom-bar .ant-tabs-tab{height:auto;border-top:0;border-bottom:1px solid #454f54;border-radius:0 0 2px 2px}.ant-tabs .ant-tabs-card-bar.ant-tabs-bottom-bar .ant-tabs-tab-active{padding-top:1px;padding-bottom:0;color:#0af}.ant-tabs{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;overflow:hidden;zoom:1}.ant-tabs:after,.ant-tabs:before{display:table;content:""}.ant-tabs:after{clear:both}.ant-tabs-ink-bar{position:absolute;bottom:1px;left:0;z-index:1;box-sizing:border-box;height:2px;background-color:#0af;-webkit-transform-origin:0 0;transform-origin:0 0}.ant-tabs-bar{margin:0 0 16px;border-bottom:1px solid #454f54;outline:none}.ant-tabs-bar,.ant-tabs-nav-container{-webkit-transition:padding .3s cubic-bezier(.645,.045,.355,1);transition:padding .3s cubic-bezier(.645,.045,.355,1)}.ant-tabs-nav-container{position:relative;box-sizing:border-box;margin-bottom:-1px;overflow:hidden;font-size:14px;line-height:1.5;white-space:nowrap;zoom:1}.ant-tabs-nav-container:after,.ant-tabs-nav-container:before{display:table;content:""}.ant-tabs-nav-container:after{clear:both}.ant-tabs-nav-container-scrolling{padding-right:32px;padding-left:32px}.ant-tabs-bottom .ant-tabs-bottom-bar{margin-top:16px;margin-bottom:0;border-top:1px solid #454f54;border-bottom:none}.ant-tabs-bottom .ant-tabs-bottom-bar .ant-tabs-ink-bar{top:1px;bottom:auto}.ant-tabs-bottom .ant-tabs-bottom-bar .ant-tabs-nav-container{margin-top:-1px;margin-bottom:0}.ant-tabs-tab-next,.ant-tabs-tab-prev{position:absolute;z-index:2;width:0;height:100%;color:#c2cfd6;text-align:center;background-color:transparent;border:0;cursor:pointer;opacity:0;-webkit-transition:width .3s cubic-bezier(.645,.045,.355,1),opacity .3s cubic-bezier(.645,.045,.355,1),color .3s cubic-bezier(.645,.045,.355,1);transition:width .3s cubic-bezier(.645,.045,.355,1),opacity .3s cubic-bezier(.645,.045,.355,1),color .3s cubic-bezier(.645,.045,.355,1);-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;pointer-events:none}.ant-tabs-tab-next.ant-tabs-tab-arrow-show,.ant-tabs-tab-prev.ant-tabs-tab-arrow-show{width:32px;height:100%;opacity:1;pointer-events:auto}.ant-tabs-tab-next:hover,.ant-tabs-tab-prev:hover{color:#e3e6e8}.ant-tabs-tab-next-icon,.ant-tabs-tab-prev-icon{position:absolute;top:50%;left:50%;font-weight:700;font-style:normal;-webkit-font-feature-settings:normal;font-feature-settings:normal;font-variant:normal;line-height:inherit;text-align:center;text-transform:none;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%)}.ant-tabs-tab-next-icon-target,.ant-tabs-tab-prev-icon-target{display:block;display:inline-block;font-size:12px;font-size:10px\9;-webkit-transform:scale(.83333333) rotate(0deg);transform:scale(.83333333) rotate(0deg)}:root .ant-tabs-tab-next-icon-target,:root .ant-tabs-tab-prev-icon-target{font-size:12px}.ant-tabs-tab-btn-disabled{cursor:not-allowed}.ant-tabs-tab-btn-disabled,.ant-tabs-tab-btn-disabled:hover{color:hsla(0,0%,100%,.5)}.ant-tabs-tab-next{right:2px}.ant-tabs-tab-prev{left:0}:root .ant-tabs-tab-prev{-webkit-filter:none;filter:none}.ant-tabs-nav-wrap{margin-bottom:-1px;overflow:hidden}.ant-tabs-nav-scroll{overflow:hidden;white-space:nowrap}.ant-tabs-nav{position:relative;display:inline-block;box-sizing:border-box;margin:0;padding-left:0;list-style:none;-webkit-transition:-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1)}.ant-tabs-nav:after,.ant-tabs-nav:before{display:table;content:" "}.ant-tabs-nav:after{clear:both}.ant-tabs-nav .ant-tabs-tab{position:relative;display:inline-block;box-sizing:border-box;height:100%;margin:0 32px 0 0;padding:12px 16px;text-decoration:none;cursor:pointer;-webkit-transition:color .3s cubic-bezier(.645,.045,.355,1);transition:color .3s cubic-bezier(.645,.045,.355,1)}.ant-tabs-nav .ant-tabs-tab:last-child{margin-right:0}.ant-tabs-nav .ant-tabs-tab:hover{color:#29bfff}.ant-tabs-nav .ant-tabs-tab:active{color:#0089d9}.ant-tabs-nav .ant-tabs-tab .anticon{margin-right:8px}.ant-tabs-nav .ant-tabs-tab-disabled,.ant-tabs-nav .ant-tabs-tab-disabled:hover{color:hsla(0,0%,100%,.5);cursor:not-allowed}.ant-tabs-nav .ant-tabs-tab-active{color:#0af;font-weight:500}.ant-tabs .ant-tabs-large-bar .ant-tabs-nav-container{font-size:16px}.ant-tabs .ant-tabs-large-bar .ant-tabs-tab{padding:16px}.ant-tabs .ant-tabs-small-bar .ant-tabs-nav-container{font-size:14px}.ant-tabs .ant-tabs-small-bar .ant-tabs-tab{padding:8px 16px}.ant-tabs .ant-tabs-bottom-content,.ant-tabs .ant-tabs-top-content{width:100%}.ant-tabs .ant-tabs-bottom-content>.ant-tabs-tabpane,.ant-tabs .ant-tabs-top-content>.ant-tabs-tabpane{flex-shrink:0;width:100%;opacity:1;-webkit-transition:opacity .45s;transition:opacity .45s}.ant-tabs .ant-tabs-bottom-content>.ant-tabs-tabpane-inactive,.ant-tabs .ant-tabs-top-content>.ant-tabs-tabpane-inactive{height:0;padding:0!important;overflow:hidden;opacity:0;pointer-events:none}.ant-tabs .ant-tabs-bottom-content>.ant-tabs-tabpane-inactive input,.ant-tabs .ant-tabs-top-content>.ant-tabs-tabpane-inactive input{visibility:hidden}.ant-tabs .ant-tabs-bottom-content.ant-tabs-content-animated,.ant-tabs .ant-tabs-top-content.ant-tabs-content-animated{display:flex;flex-direction:row;-webkit-transition:margin-left .3s cubic-bezier(.645,.045,.355,1);transition:margin-left .3s cubic-bezier(.645,.045,.355,1);will-change:margin-left}.ant-tabs .ant-tabs-left-bar,.ant-tabs .ant-tabs-right-bar{height:100%;border-bottom:0}.ant-tabs .ant-tabs-left-bar-tab-next,.ant-tabs .ant-tabs-left-bar-tab-prev,.ant-tabs .ant-tabs-right-bar-tab-next,.ant-tabs .ant-tabs-right-bar-tab-prev{width:32px;height:0;-webkit-transition:height .3s cubic-bezier(.645,.045,.355,1),opacity .3s cubic-bezier(.645,.045,.355,1),color .3s cubic-bezier(.645,.045,.355,1);transition:height .3s cubic-bezier(.645,.045,.355,1),opacity .3s cubic-bezier(.645,.045,.355,1),color .3s cubic-bezier(.645,.045,.355,1)}.ant-tabs .ant-tabs-left-bar-tab-next.ant-tabs-tab-arrow-show,.ant-tabs .ant-tabs-left-bar-tab-prev.ant-tabs-tab-arrow-show,.ant-tabs .ant-tabs-right-bar-tab-next.ant-tabs-tab-arrow-show,.ant-tabs .ant-tabs-right-bar-tab-prev.ant-tabs-tab-arrow-show{width:100%;height:32px}.ant-tabs .ant-tabs-left-bar .ant-tabs-tab,.ant-tabs .ant-tabs-right-bar .ant-tabs-tab{display:block;float:none;margin:0 0 16px;padding:8px 24px}.ant-tabs .ant-tabs-left-bar .ant-tabs-tab:last-child,.ant-tabs .ant-tabs-right-bar .ant-tabs-tab:last-child{margin-bottom:0}.ant-tabs .ant-tabs-left-bar .ant-tabs-extra-content,.ant-tabs .ant-tabs-right-bar .ant-tabs-extra-content{text-align:center}.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-scroll,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-scroll{width:auto}.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-container,.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-wrap,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-container,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-wrap{height:100%}.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-container,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-container{margin-bottom:0}.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-container.ant-tabs-nav-container-scrolling,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-container.ant-tabs-nav-container-scrolling{padding:32px 0}.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-wrap,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-wrap{margin-bottom:0}.ant-tabs .ant-tabs-left-bar .ant-tabs-nav,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav{width:100%}.ant-tabs .ant-tabs-left-bar .ant-tabs-ink-bar,.ant-tabs .ant-tabs-right-bar .ant-tabs-ink-bar{top:0;bottom:auto;left:auto;width:2px;height:auto}.ant-tabs .ant-tabs-left-bar .ant-tabs-tab-next,.ant-tabs .ant-tabs-right-bar .ant-tabs-tab-next{bottom:0;width:100%;height:32px}.ant-tabs .ant-tabs-left-bar .ant-tabs-tab-prev,.ant-tabs .ant-tabs-right-bar .ant-tabs-tab-prev{top:0;width:100%;height:32px}.ant-tabs .ant-tabs-left-content,.ant-tabs .ant-tabs-right-content{width:auto;margin-top:0!important;overflow:hidden}.ant-tabs .ant-tabs-left-bar{float:left;margin-right:-1px;margin-bottom:0;border-right:1px solid #454f54}.ant-tabs .ant-tabs-left-bar .ant-tabs-tab{text-align:right}.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-container,.ant-tabs .ant-tabs-left-bar .ant-tabs-nav-wrap{margin-right:-1px}.ant-tabs .ant-tabs-left-bar .ant-tabs-ink-bar{right:1px}.ant-tabs .ant-tabs-left-content{padding-left:24px;border-left:1px solid #454f54}.ant-tabs .ant-tabs-right-bar{float:right;margin-bottom:0;margin-left:-1px;border-left:1px solid #454f54}.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-container,.ant-tabs .ant-tabs-right-bar .ant-tabs-nav-wrap{margin-left:-1px}.ant-tabs .ant-tabs-right-bar .ant-tabs-ink-bar{left:1px}.ant-tabs .ant-tabs-right-content{padding-right:24px;border-right:1px solid #454f54}.ant-tabs-bottom .ant-tabs-ink-bar-animated,.ant-tabs-top .ant-tabs-ink-bar-animated{-webkit-transition:width .3s cubic-bezier(.645,.045,.355,1),left .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:width .3s cubic-bezier(.645,.045,.355,1),left .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1),width .3s cubic-bezier(.645,.045,.355,1),left .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1),width .3s cubic-bezier(.645,.045,.355,1),left .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1)}.ant-tabs-left .ant-tabs-ink-bar-animated,.ant-tabs-right .ant-tabs-ink-bar-animated{-webkit-transition:height .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:height .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1),height .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1);transition:transform .3s cubic-bezier(.645,.045,.355,1),height .3s cubic-bezier(.645,.045,.355,1),top .3s cubic-bezier(.645,.045,.355,1),-webkit-transform .3s cubic-bezier(.645,.045,.355,1)}.ant-tabs-no-animation>.ant-tabs-content>.ant-tabs-content-animated,.no-flex>.ant-tabs-content>.ant-tabs-content-animated{margin-left:0!important;-webkit-transform:none!important;transform:none!important}.ant-tabs-no-animation>.ant-tabs-content>.ant-tabs-tabpane-inactive,.no-flex>.ant-tabs-content>.ant-tabs-tabpane-inactive{height:0;padding:0!important;overflow:hidden;opacity:0;pointer-events:none}.ant-tabs-no-animation>.ant-tabs-content>.ant-tabs-tabpane-inactive input,.no-flex>.ant-tabs-content>.ant-tabs-tabpane-inactive input{visibility:hidden}.ant-tabs-left-content>.ant-tabs-content-animated,.ant-tabs-right-content>.ant-tabs-content-animated{margin-left:0!important;-webkit-transform:none!important;transform:none!important}.ant-tabs-left-content>.ant-tabs-tabpane-inactive,.ant-tabs-right-content>.ant-tabs-tabpane-inactive{height:0;padding:0!important;overflow:hidden;opacity:0;pointer-events:none}.ant-tabs-left-content>.ant-tabs-tabpane-inactive input,.ant-tabs-right-content>.ant-tabs-tabpane-inactive input{visibility:hidden}.ant-input-number{box-sizing:border-box;font-variant:tabular-nums;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;width:100%;height:32px;color:#e3e6e8;font-size:14px;line-height:1.5;background-color:#22282a;background-image:none;-webkit-transition:all .3s;transition:all .3s;display:inline-block;width:90px;margin:0;padding:0;border:1px solid #5c6970;border-radius:2px}.ant-input-number::-moz-placeholder{color:#bfbfbf;opacity:1}.ant-input-number:-ms-input-placeholder{color:#bfbfbf}.ant-input-number::-webkit-input-placeholder{color:#bfbfbf}.ant-input-number:focus{border-color:#29bfff;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(0,170,255,.2)}.ant-input-number[disabled]{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);cursor:not-allowed;opacity:1}.ant-input-number[disabled]:hover{border-color:#757b7d;border-right-width:1px!important}textarea.ant-input-number{max-width:100%;height:auto;min-height:32px;vertical-align:bottom;-webkit-transition:all .3s,height 0s;transition:all .3s,height 0s}.ant-input-number-lg{height:40px;padding:6px 11px}.ant-input-number-sm{height:24px;padding:1px 7px}.ant-input-number-handler{position:relative;display:block;width:100%;height:50%;overflow:hidden;color:#c2cfd6;font-weight:700;line-height:0;text-align:center;-webkit-transition:all .1s linear;transition:all .1s linear}.ant-input-number-handler:active{background:#f4f4f4}.ant-input-number-handler:hover .ant-input-number-handler-down-inner,.ant-input-number-handler:hover .ant-input-number-handler-up-inner{color:#29bfff}.ant-input-number-handler-down-inner,.ant-input-number-handler-up-inner{display:inline-block;color:inherit;font-style:normal;line-height:0;text-align:center;text-transform:none;vertical-align:-.125em;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;position:absolute;right:4px;width:12px;height:12px;color:#c2cfd6;line-height:12px;-webkit-transition:all .1s linear;transition:all .1s linear;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.ant-input-number-handler-down-inner>*,.ant-input-number-handler-up-inner>*{line-height:1}.ant-input-number-handler-down-inner svg,.ant-input-number-handler-up-inner svg{display:inline-block}.ant-input-number-handler-down-inner:before,.ant-input-number-handler-up-inner:before{display:none}.ant-input-number-handler-down-inner .ant-input-number-handler-down-inner-icon,.ant-input-number-handler-down-inner .ant-input-number-handler-up-inner-icon,.ant-input-number-handler-up-inner .ant-input-number-handler-down-inner-icon,.ant-input-number-handler-up-inner .ant-input-number-handler-up-inner-icon{display:block}.ant-input-number-focused,.ant-input-number:hover{border-color:#29bfff;border-right-width:1px!important}.ant-input-number-focused{outline:0;box-shadow:0 0 0 2px rgba(0,170,255,.2)}.ant-input-number-disabled{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);cursor:not-allowed;opacity:1}.ant-input-number-disabled:hover{border-color:#757b7d;border-right-width:1px!important}.ant-input-number-disabled .ant-input-number-input{cursor:not-allowed}.ant-input-number-disabled .ant-input-number-handler-wrap{display:none}.ant-input-number-input{width:100%;height:30px;padding:0 11px;text-align:left;background-color:transparent;border:0;border-radius:2px;outline:0;-webkit-transition:all .3s linear;transition:all .3s linear;-moz-appearance:textfield}.ant-input-number-input::-moz-placeholder{color:#bfbfbf;opacity:1}.ant-input-number-input:-ms-input-placeholder{color:#bfbfbf}.ant-input-number-input::-webkit-input-placeholder{color:#bfbfbf}.ant-input-number-lg{padding:0;font-size:16px}.ant-input-number-lg input{height:38px}.ant-input-number-sm{padding:0}.ant-input-number-sm input{height:22px;padding:0 7px}.ant-input-number-handler-wrap{position:absolute;top:0;right:0;width:22px;height:100%;background:#2e3538;border-left:1px solid #5c6970;border-radius:0 2px 2px 0;opacity:0;-webkit-transition:opacity .24s linear .1s;transition:opacity .24s linear .1s}.ant-input-number-handler-wrap .ant-input-number-handler .ant-input-number-handler-down-inner,.ant-input-number-handler-wrap .ant-input-number-handler .ant-input-number-handler-up-inner{display:inline-block;font-size:12px;font-size:7px\9;-webkit-transform:scale(.58333333) rotate(0deg);transform:scale(.58333333) rotate(0deg);min-width:auto;margin-right:0}:root .ant-input-number-handler-wrap .ant-input-number-handler .ant-input-number-handler-down-inner,:root .ant-input-number-handler-wrap .ant-input-number-handler .ant-input-number-handler-up-inner{font-size:12px}.ant-input-number-handler-wrap:hover .ant-input-number-handler{height:40%}.ant-input-number:hover .ant-input-number-handler-wrap{opacity:1}.ant-input-number-handler-up{cursor:pointer}.ant-input-number-handler-up-inner{top:50%;margin-top:-5px;text-align:center}.ant-input-number-handler-up:hover{height:60%!important}.ant-input-number-handler-down{top:0;border-top:1px solid #5c6970;cursor:pointer}.ant-input-number-handler-down-inner{top:50%;margin-top:-6px;text-align:center}.ant-input-number-handler-down:hover{height:60%!important}.ant-input-number-handler-down-disabled,.ant-input-number-handler-up-disabled{cursor:not-allowed}.ant-input-number-handler-down-disabled:hover .ant-input-number-handler-down-inner,.ant-input-number-handler-up-disabled:hover .ant-input-number-handler-up-inner{color:hsla(0,0%,100%,.5)}.ant-select{box-sizing:border-box;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;display:inline-block;outline:0}.ant-select,.ant-select ol,.ant-select ul{margin:0;padding:0;list-style:none}.ant-select>ul>li>a{padding:0;background-color:#2e3538}.ant-select-arrow{display:inline-block;color:inherit;font-style:normal;line-height:0;text-align:center;text-transform:none;vertical-align:-.125em;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;position:absolute;top:50%;right:11px;margin-top:-6px;color:hsla(0,0%,100%,.5);font-size:12px;line-height:1;-webkit-transform-origin:50% 50%;transform-origin:50% 50%}.ant-select-arrow>*{line-height:1}.ant-select-arrow svg{display:inline-block}.ant-select-arrow:before{display:none}.ant-select-arrow .ant-select-arrow-icon{display:block}.ant-select-arrow .ant-select-arrow-icon svg{-webkit-transition:-webkit-transform .3s;transition:-webkit-transform .3s;transition:transform .3s;transition:transform .3s,-webkit-transform .3s}.ant-select-selection{display:block;box-sizing:border-box;background-color:#2e3538;border:1px solid #5c6970;border-top:1.02px solid #5c6970;border-radius:2px;outline:none;-webkit-transition:all .3s cubic-bezier(.645,.045,.355,1);transition:all .3s cubic-bezier(.645,.045,.355,1);-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.ant-select-selection:hover{border-color:#29bfff;border-right-width:1px!important}.ant-select-focused .ant-select-selection,.ant-select-selection:active,.ant-select-selection:focus{border-color:#29bfff;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(0,170,255,.2)}.ant-select-selection__clear{position:absolute;top:50%;right:11px;z-index:1;display:inline-block;width:12px;height:12px;margin-top:-6px;color:hsla(0,0%,100%,.5);font-size:12px;font-style:normal;line-height:12px;text-align:center;text-transform:none;background:#2e3538;cursor:pointer;opacity:0;-webkit-transition:color .3s ease,opacity .15s ease;transition:color .3s ease,opacity .15s ease;text-rendering:auto}.ant-select-selection__clear:before{display:block}.ant-select-selection__clear:hover{color:#c2cfd6}.ant-select-selection:hover .ant-select-selection__clear{opacity:1}.ant-select-selection-selected-value{float:left;max-width:100%;padding-right:20px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.ant-select-no-arrow .ant-select-selection-selected-value{padding-right:0}.ant-select-disabled{color:hsla(0,0%,100%,.5)}.ant-select-disabled .ant-select-selection{background:rgba(0,170,255,.2);cursor:not-allowed}.ant-select-disabled .ant-select-selection:active,.ant-select-disabled .ant-select-selection:focus,.ant-select-disabled .ant-select-selection:hover{border-color:#5c6970;box-shadow:none}.ant-select-disabled .ant-select-selection__clear{display:none;visibility:hidden;pointer-events:none}.ant-select-disabled .ant-select-selection--multiple .ant-select-selection__choice{padding-right:10px;color:hsla(0,0%,100%,.33);background:rgba(0,170,255,.2)}.ant-select-disabled .ant-select-selection--multiple .ant-select-selection__choice__remove{display:none}.ant-select-selection--single{position:relative;height:32px;cursor:pointer}.ant-select-selection__rendered{position:relative;display:block;margin-right:11px;margin-left:11px;line-height:30px}.ant-select-selection__rendered:after{display:inline-block;width:0;visibility:hidden;content:".";pointer-events:none}.ant-select-lg{font-size:16px}.ant-select-lg .ant-select-selection--single{height:40px}.ant-select-lg .ant-select-selection__rendered{line-height:38px}.ant-select-lg .ant-select-selection--multiple{min-height:40px}.ant-select-lg .ant-select-selection--multiple .ant-select-selection__rendered li{height:32px;line-height:32px}.ant-select-lg .ant-select-selection--multiple .ant-select-arrow,.ant-select-lg .ant-select-selection--multiple .ant-select-selection__clear{top:20px}.ant-select-sm .ant-select-selection--single{height:24px}.ant-select-sm .ant-select-selection__rendered{margin:0 7px;line-height:22px}.ant-select-sm .ant-select-selection--multiple{min-height:24px}.ant-select-sm .ant-select-selection--multiple .ant-select-selection__rendered li{height:16px;line-height:14px}.ant-select-sm .ant-select-selection--multiple .ant-select-arrow,.ant-select-sm .ant-select-selection--multiple .ant-select-selection__clear{top:12px}.ant-select-sm .ant-select-arrow,.ant-select-sm .ant-select-selection__clear{right:8px}.ant-select-disabled .ant-select-selection__choice__remove{color:hsla(0,0%,100%,.5);cursor:default}.ant-select-disabled .ant-select-selection__choice__remove:hover{color:hsla(0,0%,100%,.5)}.ant-select-search__field__wrap{position:relative;display:inline-block}.ant-select-search__field__placeholder,.ant-select-selection__placeholder{position:absolute;top:50%;right:9px;left:0;max-width:100%;height:20px;margin-top:-10px;overflow:hidden;color:#bfbfbf;line-height:20px;white-space:nowrap;text-align:left;text-overflow:ellipsis}.ant-select-search__field__placeholder{left:12px}.ant-select-search__field__mirror{position:absolute;top:0;left:0;white-space:pre;opacity:0;pointer-events:none}.ant-select-search--inline{position:absolute;width:100%;height:100%}.ant-select-search--inline .ant-select-search__field__wrap{width:100%;height:100%}.ant-select-search--inline .ant-select-search__field{width:100%;height:100%;font-size:100%;line-height:1;background:transparent;border-width:0;border-radius:2px;outline:0}.ant-select-search--inline>i{float:right}.ant-select-selection--multiple{min-height:32px;padding-bottom:3px;cursor:text;zoom:1}.ant-select-selection--multiple:after,.ant-select-selection--multiple:before{display:table;content:""}.ant-select-selection--multiple:after{clear:both}.ant-select-selection--multiple .ant-select-search--inline{position:static;float:left;width:auto;max-width:100%;padding:0}.ant-select-selection--multiple .ant-select-search--inline .ant-select-search__field{width:.75em;max-width:100%}.ant-select-selection--multiple .ant-select-selection__rendered{height:auto;margin-bottom:-3px;margin-left:5px}.ant-select-selection--multiple .ant-select-selection__placeholder{margin-left:6px}.ant-select-selection--multiple .ant-select-selection__rendered>ul>li,.ant-select-selection--multiple>ul>li{height:24px;margin-top:3px;line-height:22px}.ant-select-selection--multiple .ant-select-selection__choice{position:relative;float:left;max-width:99%;margin-right:4px;padding:0 20px 0 10px;overflow:hidden;color:#e3e6e8;background-color:#5c6970;border:1px solid #454f54;border-radius:0;cursor:default;-webkit-transition:padding .3s cubic-bezier(.645,.045,.355,1);transition:padding .3s cubic-bezier(.645,.045,.355,1)}.ant-select-selection--multiple .ant-select-selection__choice__disabled{padding:0 10px}.ant-select-selection--multiple .ant-select-selection__choice__content{display:inline-block;max-width:100%;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;-webkit-transition:margin .3s cubic-bezier(.645,.045,.355,1);transition:margin .3s cubic-bezier(.645,.045,.355,1)}.ant-select-selection--multiple .ant-select-selection__choice__remove{color:inherit;font-style:normal;line-height:0;text-align:center;text-transform:none;vertical-align:-.125em;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;position:absolute;right:4px;color:#c2cfd6;font-weight:700;line-height:inherit;cursor:pointer;-webkit-transition:all .3s;transition:all .3s;display:inline-block;font-size:12px;font-size:10px\9;-webkit-transform:scale(.83333333) rotate(0deg);transform:scale(.83333333) rotate(0deg)}.ant-select-selection--multiple .ant-select-selection__choice__remove>*{line-height:1}.ant-select-selection--multiple .ant-select-selection__choice__remove svg{display:inline-block}.ant-select-selection--multiple .ant-select-selection__choice__remove:before{display:none}.ant-select-selection--multiple .ant-select-selection__choice__remove .ant-select-selection--multiple .ant-select-selection__choice__remove-icon{display:block}:root .ant-select-selection--multiple .ant-select-selection__choice__remove{font-size:12px}.ant-select-selection--multiple .ant-select-selection__choice__remove:hover{color:hsla(0,0%,100%,.75)}.ant-select-selection--multiple .ant-select-arrow,.ant-select-selection--multiple .ant-select-selection__clear{top:16px}.ant-select-allow-clear .ant-select-selection--single .ant-select-selection-selected-value{padding-right:16px}.ant-select-allow-clear .ant-select-selection--multiple .ant-select-selection__rendered,.ant-select-show-arrow .ant-select-selection--multiple .ant-select-selection__rendered{margin-right:20px}.ant-select-open .ant-select-arrow-icon svg{-webkit-transform:rotate(180deg);transform:rotate(180deg)}.ant-select-open .ant-select-selection{border-color:#29bfff;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(0,170,255,.2)}.ant-select-combobox .ant-select-arrow{display:none}.ant-select-combobox .ant-select-search--inline{float:none;width:100%;height:100%}.ant-select-combobox .ant-select-search__field__wrap{width:100%;height:100%}.ant-select-combobox .ant-select-search__field{position:relative;z-index:1;width:100%;height:100%;box-shadow:none;-webkit-transition:all .3s cubic-bezier(.645,.045,.355,1),height 0s;transition:all .3s cubic-bezier(.645,.045,.355,1),height 0s}.ant-select-combobox.ant-select-allow-clear .ant-select-selection:hover .ant-select-selection__rendered,.ant-select-combobox.ant-select-show-arrow .ant-select-selection:hover .ant-select-selection__rendered{margin-right:20px}.ant-select-dropdown{margin:0;padding:0;color:#e3e6e8;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum",;font-feature-settings:"tnum","tnum",;position:absolute;top:-9999px;left:-9999px;z-index:1050;box-sizing:border-box;font-size:14px;font-variant:normal;background-color:#2e3538;border-radius:2px;outline:none;box-shadow:0 2px 8px rgba(0,0,0,.15)}.ant-select-dropdown.slide-up-appear.slide-up-appear-active.ant-select-dropdown-placement-bottomLeft,.ant-select-dropdown.slide-up-enter.slide-up-enter-active.ant-select-dropdown-placement-bottomLeft{-webkit-animation-name:antSlideUpIn;animation-name:antSlideUpIn}.ant-select-dropdown.slide-up-appear.slide-up-appear-active.ant-select-dropdown-placement-topLeft,.ant-select-dropdown.slide-up-enter.slide-up-enter-active.ant-select-dropdown-placement-topLeft{-webkit-animation-name:antSlideDownIn;animation-name:antSlideDownIn}.ant-select-dropdown.slide-up-leave.slide-up-leave-active.ant-select-dropdown-placement-bottomLeft{-webkit-animation-name:antSlideUpOut;animation-name:antSlideUpOut}.ant-select-dropdown.slide-up-leave.slide-up-leave-active.ant-select-dropdown-placement-topLeft{-webkit-animation-name:antSlideDownOut;animation-name:antSlideDownOut}.ant-select-dropdown-hidden{display:none}.ant-select-dropdown-menu{max-height:250px;margin-bottom:0;padding-left:0;overflow:auto;list-style:none;outline:none}.ant-select-dropdown-menu-item-group-list{margin:0;padding:0}.ant-select-dropdown-menu-item-group-list>.ant-select-dropdown-menu-item{padding-left:20px}.ant-select-dropdown-menu-item-group-title{height:32px;padding:0 12px;color:#c2cfd6;font-size:12px;line-height:32px}.ant-select-dropdown-menu-item-group-list .ant-select-dropdown-menu-item:first-child:not(:last-child),.ant-select-dropdown-menu-item-group:not(:last-child) .ant-select-dropdown-menu-item-group-list .ant-select-dropdown-menu-item:last-child{border-radius:0}.ant-select-dropdown-menu-item{position:relative;display:block;padding:5px 12px;overflow:hidden;color:#e3e6e8;font-weight:400;line-height:22px;white-space:nowrap;text-overflow:ellipsis;cursor:pointer;-webkit-transition:background .3s ease;transition:background .3s ease}.ant-select-dropdown-menu-item:hover:not(.ant-select-dropdown-menu-item-disabled){background-color:rgba(0,170,255,.1)}.ant-select-dropdown-menu-item:first-child{border-radius:2px 2px 0 0}.ant-select-dropdown-menu-item:last-child{border-radius:0 0 2px 2px}.ant-select-dropdown-menu-item-selected{color:#e3e6e8;font-weight:600;background-color:#5c6970}.ant-select-dropdown-menu-item-disabled,.ant-select-dropdown-menu-item-disabled:hover{color:hsla(0,0%,100%,.5);cursor:not-allowed}.ant-select-dropdown-menu-item-active:not(.ant-select-dropdown-menu-item-disabled){background-color:rgba(0,170,255,.2)}.ant-select-dropdown-menu-item-divider{height:1px;margin:1px 0;overflow:hidden;line-height:0;background-color:#454f54}.ant-select-dropdown.ant-select-dropdown--multiple .ant-select-dropdown-menu-item{padding-right:32px}.ant-select-dropdown.ant-select-dropdown--multiple .ant-select-dropdown-menu-item .ant-select-selected-icon{position:absolute;top:50%;right:12px;color:transparent;font-weight:700;font-size:12px;text-shadow:0 .1px 0,.1px 0 0,0 -.1px 0,-.1px 0;-webkit-transform:translateY(-50%);transform:translateY(-50%);-webkit-transition:all .2s;transition:all .2s}.ant-select-dropdown.ant-select-dropdown--multiple .ant-select-dropdown-menu-item:hover .ant-select-selected-icon{color:hsla(0,0%,100%,.87)}.ant-select-dropdown.ant-select-dropdown--multiple .ant-select-dropdown-menu-item-disabled .ant-select-selected-icon{display:none}.ant-select-dropdown.ant-select-dropdown--multiple .ant-select-dropdown-menu-item-selected .ant-select-selected-icon,.ant-select-dropdown.ant-select-dropdown--multiple .ant-select-dropdown-menu-item-selected:hover .ant-select-selected-icon{display:inline-block;color:#0af}.ant-select-dropdown--empty.ant-select-dropdown--multiple .ant-select-dropdown-menu-item{padding-right:12px}.ant-select-dropdown-container-open .ant-select-dropdown,.ant-select-dropdown-open .ant-select-dropdown{display:block}.ant-empty{margin:0 8px;font-size:14px;line-height:22px;text-align:center}.ant-empty-image{height:100px;margin-bottom:8px}.ant-empty-image img{height:100%}.ant-empty-description{margin:0}.ant-empty-footer{margin-top:16px}.ant-empty-normal{margin:32px 0;color:hsla(0,0%,100%,.5)}.ant-empty-normal .ant-empty-image{height:40px}.ant-empty-small{margin:8px 0;color:hsla(0,0%,100%,.5)}.ant-empty-small .ant-empty-image{height:35px}.ant-btn{line-height:1.499;position:relative;display:inline-block;font-weight:400;white-space:nowrap;text-align:center;background-image:none;box-shadow:0 2px 0 rgba(0,0,0,.015);cursor:pointer;-webkit-transition:all .3s cubic-bezier(.645,.045,.355,1);transition:all .3s cubic-bezier(.645,.045,.355,1);-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;touch-action:manipulation;height:32px;padding:0 15px;font-size:14px;border-radius:2px;color:#e3e6e8;background-color:#454f54;border:1px solid #5c6970}.ant-btn>.anticon{line-height:1}.ant-btn,.ant-btn:active,.ant-btn:focus{outline:0}.ant-btn:not([disabled]):hover{text-decoration:none}.ant-btn:not([disabled]):active{outline:0;box-shadow:none}.ant-btn.disabled,.ant-btn[disabled]{cursor:not-allowed}.ant-btn.disabled>*,.ant-btn[disabled]>*{pointer-events:none}.ant-btn-lg{height:40px;padding:0 15px;font-size:16px;border-radius:2px}.ant-btn-sm{height:24px;padding:0 7px;font-size:14px;border-radius:2px}.ant-btn>a:only-child{color:currentColor}.ant-btn>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn:focus,.ant-btn:hover{color:#29bfff;background-color:#454f54;border-color:#29bfff}.ant-btn:focus>a:only-child,.ant-btn:hover>a:only-child{color:currentColor}.ant-btn:focus>a:only-child:after,.ant-btn:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn.active,.ant-btn:active{color:#0089d9;background-color:#454f54;border-color:#0089d9}.ant-btn.active>a:only-child,.ant-btn:active>a:only-child{color:currentColor}.ant-btn.active>a:only-child:after,.ant-btn:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-disabled,.ant-btn-disabled.active,.ant-btn-disabled:active,.ant-btn-disabled:focus,.ant-btn-disabled:hover,.ant-btn.disabled,.ant-btn.disabled.active,.ant-btn.disabled:active,.ant-btn.disabled:focus,.ant-btn.disabled:hover,.ant-btn[disabled],.ant-btn[disabled].active,.ant-btn[disabled]:active,.ant-btn[disabled]:focus,.ant-btn[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-disabled.active>a:only-child,.ant-btn-disabled:active>a:only-child,.ant-btn-disabled:focus>a:only-child,.ant-btn-disabled:hover>a:only-child,.ant-btn-disabled>a:only-child,.ant-btn.disabled.active>a:only-child,.ant-btn.disabled:active>a:only-child,.ant-btn.disabled:focus>a:only-child,.ant-btn.disabled:hover>a:only-child,.ant-btn.disabled>a:only-child,.ant-btn[disabled].active>a:only-child,.ant-btn[disabled]:active>a:only-child,.ant-btn[disabled]:focus>a:only-child,.ant-btn[disabled]:hover>a:only-child,.ant-btn[disabled]>a:only-child{color:currentColor}.ant-btn-disabled.active>a:only-child:after,.ant-btn-disabled:active>a:only-child:after,.ant-btn-disabled:focus>a:only-child:after,.ant-btn-disabled:hover>a:only-child:after,.ant-btn-disabled>a:only-child:after,.ant-btn.disabled.active>a:only-child:after,.ant-btn.disabled:active>a:only-child:after,.ant-btn.disabled:focus>a:only-child:after,.ant-btn.disabled:hover>a:only-child:after,.ant-btn.disabled>a:only-child:after,.ant-btn[disabled].active>a:only-child:after,.ant-btn[disabled]:active>a:only-child:after,.ant-btn[disabled]:focus>a:only-child:after,.ant-btn[disabled]:hover>a:only-child:after,.ant-btn[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn.active,.ant-btn:active,.ant-btn:focus,.ant-btn:hover{text-decoration:none;background:#454f54}.ant-btn>i,.ant-btn>span{display:inline-block;pointer-events:none}.ant-btn-primary{color:#fff;background-color:#0af;border-color:#0af;text-shadow:0 -1px 0 rgba(0,0,0,.12);box-shadow:0 2px 0 rgba(0,0,0,.045)}.ant-btn-primary>a:only-child{color:currentColor}.ant-btn-primary>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-primary:focus,.ant-btn-primary:hover{color:#fff;background-color:#29bfff;border-color:#29bfff}.ant-btn-primary:focus>a:only-child,.ant-btn-primary:hover>a:only-child{color:currentColor}.ant-btn-primary:focus>a:only-child:after,.ant-btn-primary:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-primary.active,.ant-btn-primary:active{color:#fff;background-color:#0089d9;border-color:#0089d9}.ant-btn-primary.active>a:only-child,.ant-btn-primary:active>a:only-child{color:currentColor}.ant-btn-primary.active>a:only-child:after,.ant-btn-primary:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-primary-disabled,.ant-btn-primary-disabled.active,.ant-btn-primary-disabled:active,.ant-btn-primary-disabled:focus,.ant-btn-primary-disabled:hover,.ant-btn-primary.disabled,.ant-btn-primary.disabled.active,.ant-btn-primary.disabled:active,.ant-btn-primary.disabled:focus,.ant-btn-primary.disabled:hover,.ant-btn-primary[disabled],.ant-btn-primary[disabled].active,.ant-btn-primary[disabled]:active,.ant-btn-primary[disabled]:focus,.ant-btn-primary[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-primary-disabled.active>a:only-child,.ant-btn-primary-disabled:active>a:only-child,.ant-btn-primary-disabled:focus>a:only-child,.ant-btn-primary-disabled:hover>a:only-child,.ant-btn-primary-disabled>a:only-child,.ant-btn-primary.disabled.active>a:only-child,.ant-btn-primary.disabled:active>a:only-child,.ant-btn-primary.disabled:focus>a:only-child,.ant-btn-primary.disabled:hover>a:only-child,.ant-btn-primary.disabled>a:only-child,.ant-btn-primary[disabled].active>a:only-child,.ant-btn-primary[disabled]:active>a:only-child,.ant-btn-primary[disabled]:focus>a:only-child,.ant-btn-primary[disabled]:hover>a:only-child,.ant-btn-primary[disabled]>a:only-child{color:currentColor}.ant-btn-primary-disabled.active>a:only-child:after,.ant-btn-primary-disabled:active>a:only-child:after,.ant-btn-primary-disabled:focus>a:only-child:after,.ant-btn-primary-disabled:hover>a:only-child:after,.ant-btn-primary-disabled>a:only-child:after,.ant-btn-primary.disabled.active>a:only-child:after,.ant-btn-primary.disabled:active>a:only-child:after,.ant-btn-primary.disabled:focus>a:only-child:after,.ant-btn-primary.disabled:hover>a:only-child:after,.ant-btn-primary.disabled>a:only-child:after,.ant-btn-primary[disabled].active>a:only-child:after,.ant-btn-primary[disabled]:active>a:only-child:after,.ant-btn-primary[disabled]:focus>a:only-child:after,.ant-btn-primary[disabled]:hover>a:only-child:after,.ant-btn-primary[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-group .ant-btn-primary:not(:first-child):not(:last-child){border-right-color:#29bfff;border-left-color:#29bfff}.ant-btn-group .ant-btn-primary:not(:first-child):not(:last-child):disabled{border-color:#5c6970}.ant-btn-group .ant-btn-primary:first-child:not(:last-child){border-right-color:#29bfff}.ant-btn-group .ant-btn-primary:first-child:not(:last-child)[disabled]{border-right-color:#5c6970}.ant-btn-group .ant-btn-primary+.ant-btn-primary,.ant-btn-group .ant-btn-primary:last-child:not(:first-child){border-left-color:#29bfff}.ant-btn-group .ant-btn-primary+.ant-btn-primary[disabled],.ant-btn-group .ant-btn-primary:last-child:not(:first-child)[disabled]{border-left-color:#5c6970}.ant-btn-ghost{color:#e3e6e8;background-color:transparent;border-color:#5c6970}.ant-btn-ghost>a:only-child{color:currentColor}.ant-btn-ghost>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-ghost:focus,.ant-btn-ghost:hover{color:#29bfff;background-color:transparent;border-color:#29bfff}.ant-btn-ghost:focus>a:only-child,.ant-btn-ghost:hover>a:only-child{color:currentColor}.ant-btn-ghost:focus>a:only-child:after,.ant-btn-ghost:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-ghost.active,.ant-btn-ghost:active{color:#0089d9;background-color:transparent;border-color:#0089d9}.ant-btn-ghost.active>a:only-child,.ant-btn-ghost:active>a:only-child{color:currentColor}.ant-btn-ghost.active>a:only-child:after,.ant-btn-ghost:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-ghost-disabled,.ant-btn-ghost-disabled.active,.ant-btn-ghost-disabled:active,.ant-btn-ghost-disabled:focus,.ant-btn-ghost-disabled:hover,.ant-btn-ghost.disabled,.ant-btn-ghost.disabled.active,.ant-btn-ghost.disabled:active,.ant-btn-ghost.disabled:focus,.ant-btn-ghost.disabled:hover,.ant-btn-ghost[disabled],.ant-btn-ghost[disabled].active,.ant-btn-ghost[disabled]:active,.ant-btn-ghost[disabled]:focus,.ant-btn-ghost[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-ghost-disabled.active>a:only-child,.ant-btn-ghost-disabled:active>a:only-child,.ant-btn-ghost-disabled:focus>a:only-child,.ant-btn-ghost-disabled:hover>a:only-child,.ant-btn-ghost-disabled>a:only-child,.ant-btn-ghost.disabled.active>a:only-child,.ant-btn-ghost.disabled:active>a:only-child,.ant-btn-ghost.disabled:focus>a:only-child,.ant-btn-ghost.disabled:hover>a:only-child,.ant-btn-ghost.disabled>a:only-child,.ant-btn-ghost[disabled].active>a:only-child,.ant-btn-ghost[disabled]:active>a:only-child,.ant-btn-ghost[disabled]:focus>a:only-child,.ant-btn-ghost[disabled]:hover>a:only-child,.ant-btn-ghost[disabled]>a:only-child{color:currentColor}.ant-btn-ghost-disabled.active>a:only-child:after,.ant-btn-ghost-disabled:active>a:only-child:after,.ant-btn-ghost-disabled:focus>a:only-child:after,.ant-btn-ghost-disabled:hover>a:only-child:after,.ant-btn-ghost-disabled>a:only-child:after,.ant-btn-ghost.disabled.active>a:only-child:after,.ant-btn-ghost.disabled:active>a:only-child:after,.ant-btn-ghost.disabled:focus>a:only-child:after,.ant-btn-ghost.disabled:hover>a:only-child:after,.ant-btn-ghost.disabled>a:only-child:after,.ant-btn-ghost[disabled].active>a:only-child:after,.ant-btn-ghost[disabled]:active>a:only-child:after,.ant-btn-ghost[disabled]:focus>a:only-child:after,.ant-btn-ghost[disabled]:hover>a:only-child:after,.ant-btn-ghost[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-dashed{color:#e3e6e8;background-color:#454f54;border-color:#5c6970;border-style:dashed}.ant-btn-dashed>a:only-child{color:currentColor}.ant-btn-dashed>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-dashed:focus,.ant-btn-dashed:hover{color:#29bfff;background-color:#454f54;border-color:#29bfff}.ant-btn-dashed:focus>a:only-child,.ant-btn-dashed:hover>a:only-child{color:currentColor}.ant-btn-dashed:focus>a:only-child:after,.ant-btn-dashed:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-dashed.active,.ant-btn-dashed:active{color:#0089d9;background-color:#454f54;border-color:#0089d9}.ant-btn-dashed.active>a:only-child,.ant-btn-dashed:active>a:only-child{color:currentColor}.ant-btn-dashed.active>a:only-child:after,.ant-btn-dashed:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-dashed-disabled,.ant-btn-dashed-disabled.active,.ant-btn-dashed-disabled:active,.ant-btn-dashed-disabled:focus,.ant-btn-dashed-disabled:hover,.ant-btn-dashed.disabled,.ant-btn-dashed.disabled.active,.ant-btn-dashed.disabled:active,.ant-btn-dashed.disabled:focus,.ant-btn-dashed.disabled:hover,.ant-btn-dashed[disabled],.ant-btn-dashed[disabled].active,.ant-btn-dashed[disabled]:active,.ant-btn-dashed[disabled]:focus,.ant-btn-dashed[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-dashed-disabled.active>a:only-child,.ant-btn-dashed-disabled:active>a:only-child,.ant-btn-dashed-disabled:focus>a:only-child,.ant-btn-dashed-disabled:hover>a:only-child,.ant-btn-dashed-disabled>a:only-child,.ant-btn-dashed.disabled.active>a:only-child,.ant-btn-dashed.disabled:active>a:only-child,.ant-btn-dashed.disabled:focus>a:only-child,.ant-btn-dashed.disabled:hover>a:only-child,.ant-btn-dashed.disabled>a:only-child,.ant-btn-dashed[disabled].active>a:only-child,.ant-btn-dashed[disabled]:active>a:only-child,.ant-btn-dashed[disabled]:focus>a:only-child,.ant-btn-dashed[disabled]:hover>a:only-child,.ant-btn-dashed[disabled]>a:only-child{color:currentColor}.ant-btn-dashed-disabled.active>a:only-child:after,.ant-btn-dashed-disabled:active>a:only-child:after,.ant-btn-dashed-disabled:focus>a:only-child:after,.ant-btn-dashed-disabled:hover>a:only-child:after,.ant-btn-dashed-disabled>a:only-child:after,.ant-btn-dashed.disabled.active>a:only-child:after,.ant-btn-dashed.disabled:active>a:only-child:after,.ant-btn-dashed.disabled:focus>a:only-child:after,.ant-btn-dashed.disabled:hover>a:only-child:after,.ant-btn-dashed.disabled>a:only-child:after,.ant-btn-dashed[disabled].active>a:only-child:after,.ant-btn-dashed[disabled]:active>a:only-child:after,.ant-btn-dashed[disabled]:focus>a:only-child:after,.ant-btn-dashed[disabled]:hover>a:only-child:after,.ant-btn-dashed[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-danger{color:#f5222d;background-color:rgba(0,170,255,.2);border-color:#5c6970}.ant-btn-danger>a:only-child{color:currentColor}.ant-btn-danger>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-danger:hover{color:#fff;background-color:#ff4d4f;border-color:#ff4d4f}.ant-btn-danger:hover>a:only-child{color:currentColor}.ant-btn-danger:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-danger:focus{color:#ff4d4f;background-color:#2e3538;border-color:#ff4d4f}.ant-btn-danger:focus>a:only-child{color:currentColor}.ant-btn-danger:focus>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-danger.active,.ant-btn-danger:active{color:#fff;background-color:#cf1322;border-color:#cf1322}.ant-btn-danger.active>a:only-child,.ant-btn-danger:active>a:only-child{color:currentColor}.ant-btn-danger.active>a:only-child:after,.ant-btn-danger:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-danger-disabled,.ant-btn-danger-disabled.active,.ant-btn-danger-disabled:active,.ant-btn-danger-disabled:focus,.ant-btn-danger-disabled:hover,.ant-btn-danger.disabled,.ant-btn-danger.disabled.active,.ant-btn-danger.disabled:active,.ant-btn-danger.disabled:focus,.ant-btn-danger.disabled:hover,.ant-btn-danger[disabled],.ant-btn-danger[disabled].active,.ant-btn-danger[disabled]:active,.ant-btn-danger[disabled]:focus,.ant-btn-danger[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-danger-disabled.active>a:only-child,.ant-btn-danger-disabled:active>a:only-child,.ant-btn-danger-disabled:focus>a:only-child,.ant-btn-danger-disabled:hover>a:only-child,.ant-btn-danger-disabled>a:only-child,.ant-btn-danger.disabled.active>a:only-child,.ant-btn-danger.disabled:active>a:only-child,.ant-btn-danger.disabled:focus>a:only-child,.ant-btn-danger.disabled:hover>a:only-child,.ant-btn-danger.disabled>a:only-child,.ant-btn-danger[disabled].active>a:only-child,.ant-btn-danger[disabled]:active>a:only-child,.ant-btn-danger[disabled]:focus>a:only-child,.ant-btn-danger[disabled]:hover>a:only-child,.ant-btn-danger[disabled]>a:only-child{color:currentColor}.ant-btn-danger-disabled.active>a:only-child:after,.ant-btn-danger-disabled:active>a:only-child:after,.ant-btn-danger-disabled:focus>a:only-child:after,.ant-btn-danger-disabled:hover>a:only-child:after,.ant-btn-danger-disabled>a:only-child:after,.ant-btn-danger.disabled.active>a:only-child:after,.ant-btn-danger.disabled:active>a:only-child:after,.ant-btn-danger.disabled:focus>a:only-child:after,.ant-btn-danger.disabled:hover>a:only-child:after,.ant-btn-danger.disabled>a:only-child:after,.ant-btn-danger[disabled].active>a:only-child:after,.ant-btn-danger[disabled]:active>a:only-child:after,.ant-btn-danger[disabled]:focus>a:only-child:after,.ant-btn-danger[disabled]:hover>a:only-child:after,.ant-btn-danger[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-link{color:#0af;background-color:transparent;border-color:transparent;box-shadow:none}.ant-btn-link>a:only-child{color:currentColor}.ant-btn-link>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-link:focus,.ant-btn-link:hover{color:#29bfff;background-color:transparent;border-color:#29bfff}.ant-btn-link:focus>a:only-child,.ant-btn-link:hover>a:only-child{color:currentColor}.ant-btn-link:focus>a:only-child:after,.ant-btn-link:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-link.active,.ant-btn-link:active{color:#0089d9;background-color:transparent;border-color:#0089d9}.ant-btn-link.active>a:only-child,.ant-btn-link:active>a:only-child{color:currentColor}.ant-btn-link.active>a:only-child:after,.ant-btn-link:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-link-disabled,.ant-btn-link-disabled.active,.ant-btn-link-disabled:active,.ant-btn-link-disabled:focus,.ant-btn-link-disabled:hover,.ant-btn-link.disabled,.ant-btn-link.disabled.active,.ant-btn-link.disabled:active,.ant-btn-link.disabled:focus,.ant-btn-link.disabled:hover,.ant-btn-link[disabled],.ant-btn-link[disabled].active,.ant-btn-link[disabled]:active,.ant-btn-link[disabled]:focus,.ant-btn-link[disabled]:hover{background-color:rgba(0,170,255,.2);border-color:#5c6970}.ant-btn-link:active,.ant-btn-link:focus,.ant-btn-link:hover{border-color:transparent}.ant-btn-link-disabled,.ant-btn-link-disabled.active,.ant-btn-link-disabled:active,.ant-btn-link-disabled:focus,.ant-btn-link-disabled:hover,.ant-btn-link.disabled,.ant-btn-link.disabled.active,.ant-btn-link.disabled:active,.ant-btn-link.disabled:focus,.ant-btn-link.disabled:hover,.ant-btn-link[disabled],.ant-btn-link[disabled].active,.ant-btn-link[disabled]:active,.ant-btn-link[disabled]:focus,.ant-btn-link[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:transparent;border-color:transparent;text-shadow:none;box-shadow:none}.ant-btn-link-disabled.active>a:only-child,.ant-btn-link-disabled:active>a:only-child,.ant-btn-link-disabled:focus>a:only-child,.ant-btn-link-disabled:hover>a:only-child,.ant-btn-link-disabled>a:only-child,.ant-btn-link.disabled.active>a:only-child,.ant-btn-link.disabled:active>a:only-child,.ant-btn-link.disabled:focus>a:only-child,.ant-btn-link.disabled:hover>a:only-child,.ant-btn-link.disabled>a:only-child,.ant-btn-link[disabled].active>a:only-child,.ant-btn-link[disabled]:active>a:only-child,.ant-btn-link[disabled]:focus>a:only-child,.ant-btn-link[disabled]:hover>a:only-child,.ant-btn-link[disabled]>a:only-child{color:currentColor}.ant-btn-link-disabled.active>a:only-child:after,.ant-btn-link-disabled:active>a:only-child:after,.ant-btn-link-disabled:focus>a:only-child:after,.ant-btn-link-disabled:hover>a:only-child:after,.ant-btn-link-disabled>a:only-child:after,.ant-btn-link.disabled.active>a:only-child:after,.ant-btn-link.disabled:active>a:only-child:after,.ant-btn-link.disabled:focus>a:only-child:after,.ant-btn-link.disabled:hover>a:only-child:after,.ant-btn-link.disabled>a:only-child:after,.ant-btn-link[disabled].active>a:only-child:after,.ant-btn-link[disabled]:active>a:only-child:after,.ant-btn-link[disabled]:focus>a:only-child:after,.ant-btn-link[disabled]:hover>a:only-child:after,.ant-btn-link[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-round{height:32px;padding:0 16px;font-size:16px;border-radius:32px}.ant-btn-round.ant-btn-lg{height:40px;padding:0 20px;font-size:18px;border-radius:40px}.ant-btn-round.ant-btn-sm{height:24px;padding:0 12px;font-size:14px;border-radius:24px}.ant-btn-circle,.ant-btn-circle-outline{width:32px;height:32px;padding:0;font-size:16px;border-radius:50%}.ant-btn-circle-outline.ant-btn-lg,.ant-btn-circle.ant-btn-lg{width:40px;height:40px;padding:0;font-size:18px;border-radius:50%}.ant-btn-circle-outline.ant-btn-sm,.ant-btn-circle.ant-btn-sm{width:24px;height:24px;padding:0;font-size:14px;border-radius:50%}.ant-btn:before{position:absolute;top:-1px;right:-1px;bottom:-1px;left:-1px;z-index:1;display:none;background:#2e3538;border-radius:inherit;opacity:.35;-webkit-transition:opacity .2s;transition:opacity .2s;content:"";pointer-events:none}.ant-btn .anticon{-webkit-transition:margin-left .3s cubic-bezier(.645,.045,.355,1);transition:margin-left .3s cubic-bezier(.645,.045,.355,1)}.ant-btn .anticon.anticon-minus>svg,.ant-btn .anticon.anticon-plus>svg{shape-rendering:optimizeSpeed}.ant-btn.ant-btn-loading{position:relative;pointer-events:none}.ant-btn.ant-btn-loading:before{display:block}.ant-btn.ant-btn-loading:not(.ant-btn-circle):not(.ant-btn-circle-outline):not(.ant-btn-icon-only){padding-left:29px}.ant-btn.ant-btn-loading:not(.ant-btn-circle):not(.ant-btn-circle-outline):not(.ant-btn-icon-only) .anticon:not(:last-child){margin-left:-14px}.ant-btn-sm.ant-btn-loading:not(.ant-btn-circle):not(.ant-btn-circle-outline):not(.ant-btn-icon-only){padding-left:24px}.ant-btn-sm.ant-btn-loading:not(.ant-btn-circle):not(.ant-btn-circle-outline):not(.ant-btn-icon-only) .anticon{margin-left:-17px}.ant-btn-group{display:inline-block}.ant-btn-group,.ant-btn-group>.ant-btn,.ant-btn-group>span>.ant-btn{position:relative}.ant-btn-group>.ant-btn.active,.ant-btn-group>.ant-btn:active,.ant-btn-group>.ant-btn:focus,.ant-btn-group>.ant-btn:hover,.ant-btn-group>span>.ant-btn.active,.ant-btn-group>span>.ant-btn:active,.ant-btn-group>span>.ant-btn:focus,.ant-btn-group>span>.ant-btn:hover{z-index:2}.ant-btn-group>.ant-btn:disabled,.ant-btn-group>span>.ant-btn:disabled{z-index:0}.ant-btn-group-lg>.ant-btn,.ant-btn-group-lg>span>.ant-btn{height:40px;padding:0 15px;font-size:16px;border-radius:0;line-height:38px}.ant-btn-group-sm>.ant-btn,.ant-btn-group-sm>span>.ant-btn{height:24px;padding:0 7px;font-size:14px;border-radius:0;line-height:22px}.ant-btn-group-sm>.ant-btn>.anticon,.ant-btn-group-sm>span>.ant-btn>.anticon{font-size:14px}.ant-btn+.ant-btn-group,.ant-btn-group+.ant-btn,.ant-btn-group+.ant-btn-group,.ant-btn-group .ant-btn+.ant-btn,.ant-btn-group .ant-btn+span,.ant-btn-group>span+span,.ant-btn-group span+.ant-btn{margin-left:-1px}.ant-btn-group .ant-btn-primary+.ant-btn:not(.ant-btn-primary):not([disabled]){border-left-color:transparent}.ant-btn-group .ant-btn{border-radius:0}.ant-btn-group>.ant-btn:first-child,.ant-btn-group>span:first-child>.ant-btn{margin-left:0}.ant-btn-group>.ant-btn:only-child,.ant-btn-group>span:only-child>.ant-btn{border-radius:2px}.ant-btn-group>.ant-btn:first-child:not(:last-child),.ant-btn-group>span:first-child:not(:last-child)>.ant-btn{border-top-left-radius:2px;border-bottom-left-radius:2px}.ant-btn-group>.ant-btn:last-child:not(:first-child),.ant-btn-group>span:last-child:not(:first-child)>.ant-btn{border-top-right-radius:2px;border-bottom-right-radius:2px}.ant-btn-group-sm>.ant-btn:only-child,.ant-btn-group-sm>span:only-child>.ant-btn{border-radius:2px}.ant-btn-group-sm>.ant-btn:first-child:not(:last-child),.ant-btn-group-sm>span:first-child:not(:last-child)>.ant-btn{border-top-left-radius:2px;border-bottom-left-radius:2px}.ant-btn-group-sm>.ant-btn:last-child:not(:first-child),.ant-btn-group-sm>span:last-child:not(:first-child)>.ant-btn{border-top-right-radius:2px;border-bottom-right-radius:2px}.ant-btn-group>.ant-btn-group{float:left}.ant-btn-group>.ant-btn-group:not(:first-child):not(:last-child)>.ant-btn{border-radius:0}.ant-btn-group>.ant-btn-group:first-child:not(:last-child)>.ant-btn:last-child{padding-right:8px;border-top-right-radius:0;border-bottom-right-radius:0}.ant-btn-group>.ant-btn-group:last-child:not(:first-child)>.ant-btn:first-child{padding-left:8px;border-top-left-radius:0;border-bottom-left-radius:0}.ant-btn:not(.ant-btn-circle):not(.ant-btn-circle-outline).ant-btn-icon-only{padding-right:8px;padding-left:8px}.ant-btn:active>span,.ant-btn:focus>span{position:relative}.ant-btn>.anticon+span,.ant-btn>span+.anticon{margin-left:8px}.ant-btn-background-ghost{color:#2e3538;background:transparent!important;border-color:#2e3538}.ant-btn-background-ghost.ant-btn-primary{color:#0af;background-color:transparent;border-color:#0af;text-shadow:none}.ant-btn-background-ghost.ant-btn-primary>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-primary>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-primary:focus,.ant-btn-background-ghost.ant-btn-primary:hover{color:#29bfff;background-color:transparent;border-color:#29bfff}.ant-btn-background-ghost.ant-btn-primary:focus>a:only-child,.ant-btn-background-ghost.ant-btn-primary:hover>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-primary:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-primary.active,.ant-btn-background-ghost.ant-btn-primary:active{color:#0089d9;background-color:transparent;border-color:#0089d9}.ant-btn-background-ghost.ant-btn-primary.active>a:only-child,.ant-btn-background-ghost.ant-btn-primary:active>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-primary.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-primary-disabled,.ant-btn-background-ghost.ant-btn-primary-disabled.active,.ant-btn-background-ghost.ant-btn-primary-disabled:active,.ant-btn-background-ghost.ant-btn-primary-disabled:focus,.ant-btn-background-ghost.ant-btn-primary-disabled:hover,.ant-btn-background-ghost.ant-btn-primary.disabled,.ant-btn-background-ghost.ant-btn-primary.disabled.active,.ant-btn-background-ghost.ant-btn-primary.disabled:active,.ant-btn-background-ghost.ant-btn-primary.disabled:focus,.ant-btn-background-ghost.ant-btn-primary.disabled:hover,.ant-btn-background-ghost.ant-btn-primary[disabled],.ant-btn-background-ghost.ant-btn-primary[disabled].active,.ant-btn-background-ghost.ant-btn-primary[disabled]:active,.ant-btn-background-ghost.ant-btn-primary[disabled]:focus,.ant-btn-background-ghost.ant-btn-primary[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-background-ghost.ant-btn-primary-disabled.active>a:only-child,.ant-btn-background-ghost.ant-btn-primary-disabled:active>a:only-child,.ant-btn-background-ghost.ant-btn-primary-disabled:focus>a:only-child,.ant-btn-background-ghost.ant-btn-primary-disabled:hover>a:only-child,.ant-btn-background-ghost.ant-btn-primary-disabled>a:only-child,.ant-btn-background-ghost.ant-btn-primary.disabled.active>a:only-child,.ant-btn-background-ghost.ant-btn-primary.disabled:active>a:only-child,.ant-btn-background-ghost.ant-btn-primary.disabled:focus>a:only-child,.ant-btn-background-ghost.ant-btn-primary.disabled:hover>a:only-child,.ant-btn-background-ghost.ant-btn-primary.disabled>a:only-child,.ant-btn-background-ghost.ant-btn-primary[disabled].active>a:only-child,.ant-btn-background-ghost.ant-btn-primary[disabled]:active>a:only-child,.ant-btn-background-ghost.ant-btn-primary[disabled]:focus>a:only-child,.ant-btn-background-ghost.ant-btn-primary[disabled]:hover>a:only-child,.ant-btn-background-ghost.ant-btn-primary[disabled]>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-primary-disabled.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary-disabled:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary-disabled:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary-disabled:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary-disabled>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary.disabled.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary.disabled:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary.disabled:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary.disabled:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary.disabled>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary[disabled].active>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary[disabled]:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary[disabled]:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary[disabled]:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-primary[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-danger{color:#f5222d;background-color:transparent;border-color:#f5222d;text-shadow:none}.ant-btn-background-ghost.ant-btn-danger>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-danger>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-danger:focus,.ant-btn-background-ghost.ant-btn-danger:hover{color:#ff4d4f;background-color:transparent;border-color:#ff4d4f}.ant-btn-background-ghost.ant-btn-danger:focus>a:only-child,.ant-btn-background-ghost.ant-btn-danger:hover>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-danger:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-danger.active,.ant-btn-background-ghost.ant-btn-danger:active{color:#cf1322;background-color:transparent;border-color:#cf1322}.ant-btn-background-ghost.ant-btn-danger.active>a:only-child,.ant-btn-background-ghost.ant-btn-danger:active>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-danger.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-danger-disabled,.ant-btn-background-ghost.ant-btn-danger-disabled.active,.ant-btn-background-ghost.ant-btn-danger-disabled:active,.ant-btn-background-ghost.ant-btn-danger-disabled:focus,.ant-btn-background-ghost.ant-btn-danger-disabled:hover,.ant-btn-background-ghost.ant-btn-danger.disabled,.ant-btn-background-ghost.ant-btn-danger.disabled.active,.ant-btn-background-ghost.ant-btn-danger.disabled:active,.ant-btn-background-ghost.ant-btn-danger.disabled:focus,.ant-btn-background-ghost.ant-btn-danger.disabled:hover,.ant-btn-background-ghost.ant-btn-danger[disabled],.ant-btn-background-ghost.ant-btn-danger[disabled].active,.ant-btn-background-ghost.ant-btn-danger[disabled]:active,.ant-btn-background-ghost.ant-btn-danger[disabled]:focus,.ant-btn-background-ghost.ant-btn-danger[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-background-ghost.ant-btn-danger-disabled.active>a:only-child,.ant-btn-background-ghost.ant-btn-danger-disabled:active>a:only-child,.ant-btn-background-ghost.ant-btn-danger-disabled:focus>a:only-child,.ant-btn-background-ghost.ant-btn-danger-disabled:hover>a:only-child,.ant-btn-background-ghost.ant-btn-danger-disabled>a:only-child,.ant-btn-background-ghost.ant-btn-danger.disabled.active>a:only-child,.ant-btn-background-ghost.ant-btn-danger.disabled:active>a:only-child,.ant-btn-background-ghost.ant-btn-danger.disabled:focus>a:only-child,.ant-btn-background-ghost.ant-btn-danger.disabled:hover>a:only-child,.ant-btn-background-ghost.ant-btn-danger.disabled>a:only-child,.ant-btn-background-ghost.ant-btn-danger[disabled].active>a:only-child,.ant-btn-background-ghost.ant-btn-danger[disabled]:active>a:only-child,.ant-btn-background-ghost.ant-btn-danger[disabled]:focus>a:only-child,.ant-btn-background-ghost.ant-btn-danger[disabled]:hover>a:only-child,.ant-btn-background-ghost.ant-btn-danger[disabled]>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-danger-disabled.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger-disabled:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger-disabled:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger-disabled:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger-disabled>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger.disabled.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger.disabled:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger.disabled:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger.disabled:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger.disabled>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger[disabled].active>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger[disabled]:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger[disabled]:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger[disabled]:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-danger[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-link{color:#0af;background-color:transparent;border-color:transparent;text-shadow:none;color:#2e3538}.ant-btn-background-ghost.ant-btn-link>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-link>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-link:focus,.ant-btn-background-ghost.ant-btn-link:hover{color:#29bfff;background-color:transparent;border-color:transparent}.ant-btn-background-ghost.ant-btn-link:focus>a:only-child,.ant-btn-background-ghost.ant-btn-link:hover>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-link:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-link:hover>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-link.active,.ant-btn-background-ghost.ant-btn-link:active{color:#0089d9;background-color:transparent;border-color:transparent}.ant-btn-background-ghost.ant-btn-link.active>a:only-child,.ant-btn-background-ghost.ant-btn-link:active>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-link.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-link:active>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-background-ghost.ant-btn-link-disabled,.ant-btn-background-ghost.ant-btn-link-disabled.active,.ant-btn-background-ghost.ant-btn-link-disabled:active,.ant-btn-background-ghost.ant-btn-link-disabled:focus,.ant-btn-background-ghost.ant-btn-link-disabled:hover,.ant-btn-background-ghost.ant-btn-link.disabled,.ant-btn-background-ghost.ant-btn-link.disabled.active,.ant-btn-background-ghost.ant-btn-link.disabled:active,.ant-btn-background-ghost.ant-btn-link.disabled:focus,.ant-btn-background-ghost.ant-btn-link.disabled:hover,.ant-btn-background-ghost.ant-btn-link[disabled],.ant-btn-background-ghost.ant-btn-link[disabled].active,.ant-btn-background-ghost.ant-btn-link[disabled]:active,.ant-btn-background-ghost.ant-btn-link[disabled]:focus,.ant-btn-background-ghost.ant-btn-link[disabled]:hover{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);border-color:#5c6970;text-shadow:none;box-shadow:none}.ant-btn-background-ghost.ant-btn-link-disabled.active>a:only-child,.ant-btn-background-ghost.ant-btn-link-disabled:active>a:only-child,.ant-btn-background-ghost.ant-btn-link-disabled:focus>a:only-child,.ant-btn-background-ghost.ant-btn-link-disabled:hover>a:only-child,.ant-btn-background-ghost.ant-btn-link-disabled>a:only-child,.ant-btn-background-ghost.ant-btn-link.disabled.active>a:only-child,.ant-btn-background-ghost.ant-btn-link.disabled:active>a:only-child,.ant-btn-background-ghost.ant-btn-link.disabled:focus>a:only-child,.ant-btn-background-ghost.ant-btn-link.disabled:hover>a:only-child,.ant-btn-background-ghost.ant-btn-link.disabled>a:only-child,.ant-btn-background-ghost.ant-btn-link[disabled].active>a:only-child,.ant-btn-background-ghost.ant-btn-link[disabled]:active>a:only-child,.ant-btn-background-ghost.ant-btn-link[disabled]:focus>a:only-child,.ant-btn-background-ghost.ant-btn-link[disabled]:hover>a:only-child,.ant-btn-background-ghost.ant-btn-link[disabled]>a:only-child{color:currentColor}.ant-btn-background-ghost.ant-btn-link-disabled.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-link-disabled:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-link-disabled:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-link-disabled:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-link-disabled>a:only-child:after,.ant-btn-background-ghost.ant-btn-link.disabled.active>a:only-child:after,.ant-btn-background-ghost.ant-btn-link.disabled:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-link.disabled:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-link.disabled:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-link.disabled>a:only-child:after,.ant-btn-background-ghost.ant-btn-link[disabled].active>a:only-child:after,.ant-btn-background-ghost.ant-btn-link[disabled]:active>a:only-child:after,.ant-btn-background-ghost.ant-btn-link[disabled]:focus>a:only-child:after,.ant-btn-background-ghost.ant-btn-link[disabled]:hover>a:only-child:after,.ant-btn-background-ghost.ant-btn-link[disabled]>a:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;background:transparent;content:""}.ant-btn-two-chinese-chars:first-letter{letter-spacing:.34em}.ant-btn-two-chinese-chars>:not(.anticon){margin-right:-.34em;letter-spacing:.34em}.ant-btn-block{width:100%}.ant-btn:empty{vertical-align:top}a.ant-btn{line-height:30px}a.ant-btn-lg{line-height:38px}a.ant-btn-sm{line-height:22px}.ant-modal{box-sizing:border-box;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;top:100px;width:auto;margin:0 auto;padding:0 0 24px}.ant-modal-wrap{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1000;overflow:auto;outline:0;-webkit-overflow-scrolling:touch}.ant-modal-title{margin:0;color:hsla(0,0%,100%,.85);font-weight:500;font-size:16px;line-height:22px;word-wrap:break-word}.ant-modal-content{position:relative;background-color:#2e3538;background-clip:padding-box;border:0;border-radius:2px;box-shadow:0 4px 12px rgba(0,0,0,.15)}.ant-modal-close{position:absolute;top:0;right:0;z-index:10;padding:0;color:#c2cfd6;font-weight:700;line-height:1;text-decoration:none;background:transparent;border:0;outline:0;cursor:pointer;-webkit-transition:color .3s;transition:color .3s}.ant-modal-close-x{display:block;width:56px;height:56px;font-size:16px;font-style:normal;line-height:56px;text-align:center;text-transform:none;text-rendering:auto}.ant-modal-close:focus,.ant-modal-close:hover{color:hsla(0,0%,100%,.75);text-decoration:none}.ant-modal-header{padding:16px 24px;color:#e3e6e8;background:#2e3538;border-bottom:1px solid #454f54;border-radius:2px 2px 0 0}.ant-modal-body{padding:24px;font-size:14px;line-height:1.5;word-wrap:break-word}.ant-modal-footer{padding:10px 16px;text-align:right;background:transparent;border-top:1px solid #454f54;border-radius:0 0 2px 2px}.ant-modal-footer button+button{margin-bottom:0;margin-left:8px}.ant-modal.zoom-appear,.ant-modal.zoom-enter{-webkit-transform:none;transform:none;opacity:0;-webkit-animation-duration:.1s;animation-duration:.1s;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.ant-modal-mask{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1000;height:100%;background-color:rgba(0,0,0,.8);filter:alpha(opacity=50)}.ant-modal-mask-hidden{display:none}.ant-modal-open{overflow:hidden}.ant-modal-centered{text-align:center}.ant-modal-centered:before{display:inline-block;width:0;height:100%;vertical-align:middle;content:""}.ant-modal-centered .ant-modal{top:0;display:inline-block;text-align:left;vertical-align:middle}@media (max-width:767px){.ant-modal{max-width:calc(100vw - 16px);margin:8px auto}.ant-modal-centered .ant-modal{flex:1 1}}.ant-modal-confirm .ant-modal-close,.ant-modal-confirm .ant-modal-header{display:none}.ant-modal-confirm .ant-modal-body{padding:32px 32px 24px}.ant-modal-confirm-body-wrapper{zoom:1}.ant-modal-confirm-body-wrapper:after,.ant-modal-confirm-body-wrapper:before{display:table;content:""}.ant-modal-confirm-body-wrapper:after{clear:both}.ant-modal-confirm-body .ant-modal-confirm-title{display:block;overflow:hidden;color:hsla(0,0%,100%,.85);font-weight:500;font-size:16px;line-height:1.4}.ant-modal-confirm-body .ant-modal-confirm-content{margin-top:8px;color:#e3e6e8;font-size:14px}.ant-modal-confirm-body>.anticon{float:left;margin-right:16px;font-size:22px}.ant-modal-confirm-body>.anticon+.ant-modal-confirm-title+.ant-modal-confirm-content{margin-left:38px}.ant-modal-confirm .ant-modal-confirm-btns{float:right;margin-top:24px}.ant-modal-confirm .ant-modal-confirm-btns button+button{margin-bottom:0;margin-left:8px}.ant-modal-confirm-error .ant-modal-confirm-body>.anticon{color:#f5222d}.ant-modal-confirm-confirm .ant-modal-confirm-body>.anticon,.ant-modal-confirm-warning .ant-modal-confirm-body>.anticon{color:#faad14}.ant-modal-confirm-info .ant-modal-confirm-body>.anticon{color:#1890ff}.ant-modal-confirm-success .ant-modal-confirm-body>.anticon{color:#52c41a}.ant-form{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum"}.ant-form legend{display:block;width:100%;margin-bottom:20px;padding:0;color:#c2cfd6;font-size:16px;line-height:inherit;border:0;border-bottom:1px solid #5c6970}.ant-form label{font-size:14px}.ant-form input[type=search]{box-sizing:border-box}.ant-form input[type=checkbox],.ant-form input[type=radio]{line-height:normal}.ant-form input[type=file]{display:block}.ant-form input[type=range]{display:block;width:100%}.ant-form select[multiple],.ant-form select[size]{height:auto}.ant-form input[type=checkbox]:focus,.ant-form input[type=file]:focus,.ant-form input[type=radio]:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.ant-form output{display:block;padding-top:15px;color:#e3e6e8;font-size:14px;line-height:1.5}.ant-form-item-required:before{display:inline-block;margin-right:4px;color:#f5222d;font-size:14px;font-family:SimSun,sans-serif;line-height:1;content:"*"}.ant-form-hide-required-mark .ant-form-item-required:before{display:none}.ant-form-item-label>label{color:hsla(0,0%,100%,.85)}.ant-form-item-label>label:after{content:":";position:relative;top:-.5px;margin:0 8px 0 2px}.ant-form-item-label>label.ant-form-item-no-colon:after{content:" "}.ant-checkbox-inline.disabled,.ant-checkbox-vertical.disabled,.ant-checkbox.disabled label,.ant-radio-inline.disabled,.ant-radio-vertical.disabled,.ant-radio.disabled label,input[type=checkbox].disabled,input[type=checkbox][disabled],input[type=radio].disabled,input[type=radio][disabled]{cursor:not-allowed}.ant-form-item{box-sizing:border-box;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";margin:0 0 24px;vertical-align:top}.ant-form-item label{position:relative}.ant-form-item label>.anticon{font-size:14px;vertical-align:top}.ant-form-item-control{position:relative;line-height:40px;zoom:1}.ant-form-item-control:after,.ant-form-item-control:before{display:table;content:""}.ant-form-item-control:after{clear:both}.ant-form-item-children{position:relative}.ant-form-item-with-help{margin-bottom:5px}.ant-form-item-label{display:inline-block;overflow:hidden;line-height:39.9999px;white-space:nowrap;text-align:right;vertical-align:middle}.ant-form-item-label-left{text-align:left}.ant-form-item .ant-switch{margin:2px 0 4px}.ant-form-explain,.ant-form-extra{clear:both;min-height:22px;margin-top:-2px;color:#c2cfd6;font-size:14px;line-height:1.5;-webkit-transition:color .3s cubic-bezier(.215,.61,.355,1);transition:color .3s cubic-bezier(.215,.61,.355,1)}.ant-form-explain{margin-bottom:-1px}.ant-form-extra{padding-top:4px}.ant-form-text{display:inline-block;padding-right:8px}.ant-form-split{display:block;text-align:center}form .has-feedback .ant-input{padding-right:24px}form .has-feedback .ant-input-password-icon{margin-right:18px}form .has-feedback :not(.ant-input-group-addon)>.ant-select .ant-select-arrow,form .has-feedback :not(.ant-input-group-addon)>.ant-select .ant-select-selection__clear,form .has-feedback>.ant-select .ant-select-arrow,form .has-feedback>.ant-select .ant-select-selection__clear{right:28px}form .has-feedback :not(.ant-input-group-addon)>.ant-select .ant-select-selection-selected-value,form .has-feedback>.ant-select .ant-select-selection-selected-value{padding-right:42px}form .has-feedback .ant-cascader-picker-arrow{margin-right:17px}form .has-feedback .ant-calendar-picker-clear,form .has-feedback .ant-calendar-picker-icon,form .has-feedback .ant-cascader-picker-clear,form .has-feedback .ant-input-search:not(.ant-input-search-enter-button) .ant-input-suffix,form .has-feedback .ant-time-picker-clear,form .has-feedback .ant-time-picker-icon{right:28px}form .ant-mentions,form textarea.ant-input{height:auto;margin-bottom:4px}form .ant-upload{background:transparent}form input[type=checkbox],form input[type=radio]{width:14px;height:14px}form .ant-checkbox-inline,form .ant-radio-inline{display:inline-block;margin-left:8px;font-weight:400;vertical-align:middle;cursor:pointer}form .ant-checkbox-inline:first-child,form .ant-radio-inline:first-child{margin-left:0}form .ant-checkbox-vertical,form .ant-radio-vertical{display:block}form .ant-checkbox-vertical+.ant-checkbox-vertical,form .ant-radio-vertical+.ant-radio-vertical{margin-left:0}form .ant-input-number+.ant-form-text{margin-left:8px}form .ant-input-number-handler-wrap{z-index:2}form .ant-cascader-picker,form .ant-select{width:100%}form .ant-input-group .ant-cascader-picker,form .ant-input-group .ant-select{width:auto}form .ant-input-group-wrapper,form :not(.ant-input-group-wrapper)>.ant-input-group{position:relative;top:-1px;display:inline-block;vertical-align:middle}.ant-input-group-wrap .ant-select-selection{border-top-left-radius:0;border-bottom-left-radius:0}.ant-input-group-wrap .ant-select-selection:hover{border-color:#5c6970}.ant-input-group-wrap .ant-select-selection--single{height:40px;margin-left:-1px;background-color:hsla(0,0%,100%,.07)}.ant-input-group-wrap .ant-select-selection--single .ant-select-selection__rendered{padding-right:25px;padding-left:8px;line-height:30px}.ant-input-group-wrap .ant-select-open .ant-select-selection{border-color:#5c6970;box-shadow:none}.ant-col-24.ant-form-item-label,.ant-col-xl-24.ant-form-item-label,.ant-form-vertical .ant-form-item-label{display:block;margin:0;padding:0 0 8px;line-height:1.5;white-space:normal;text-align:left}.ant-col-24.ant-form-item-label label:after,.ant-col-xl-24.ant-form-item-label label:after,.ant-form-vertical .ant-form-item-label label:after{display:none}.ant-form-vertical .ant-form-item{padding-bottom:8px}.ant-form-vertical .ant-form-item-control{line-height:1.5}.ant-form-vertical .ant-form-explain{margin-top:2px;margin-bottom:-5px}.ant-form-vertical .ant-form-extra{margin-top:2px;margin-bottom:-4px}@media (max-width:575px){.ant-form-item-control-wrapper,.ant-form-item-label{display:block;width:100%}.ant-form-item-label{display:block;margin:0;padding:0 0 8px;line-height:1.5;white-space:normal;text-align:left}.ant-form-item-label label:after{display:none}.ant-col-xs-24.ant-form-item-label{display:block;margin:0;padding:0 0 8px;line-height:1.5;white-space:normal;text-align:left}.ant-col-xs-24.ant-form-item-label label:after{display:none}}@media (max-width:767px){.ant-col-sm-24.ant-form-item-label{display:block;margin:0;padding:0 0 8px;line-height:1.5;white-space:normal;text-align:left}.ant-col-sm-24.ant-form-item-label label:after{display:none}}@media (max-width:991px){.ant-col-md-24.ant-form-item-label{display:block;margin:0;padding:0 0 8px;line-height:1.5;white-space:normal;text-align:left}.ant-col-md-24.ant-form-item-label label:after{display:none}}@media (max-width:1199px){.ant-col-lg-24.ant-form-item-label{display:block;margin:0;padding:0 0 8px;line-height:1.5;white-space:normal;text-align:left}.ant-col-lg-24.ant-form-item-label label:after{display:none}}@media (max-width:1599px){.ant-col-xl-24.ant-form-item-label{display:block;margin:0;padding:0 0 8px;line-height:1.5;white-space:normal;text-align:left}.ant-col-xl-24.ant-form-item-label label:after{display:none}}.ant-form-inline .ant-form-item{display:inline-block;margin-right:16px;margin-bottom:0}.ant-form-inline .ant-form-item-with-help{margin-bottom:24px}.ant-form-inline .ant-form-item>.ant-form-item-control-wrapper,.ant-form-inline .ant-form-item>.ant-form-item-label{display:inline-block;vertical-align:top}.ant-form-inline .ant-form-text,.ant-form-inline .has-feedback{display:inline-block}.has-error.has-feedback .ant-form-item-children-icon,.has-success.has-feedback .ant-form-item-children-icon,.has-warning.has-feedback .ant-form-item-children-icon,.is-validating.has-feedback .ant-form-item-children-icon{position:absolute;top:50%;right:0;z-index:1;width:32px;height:20px;margin-top:-10px;font-size:14px;line-height:20px;text-align:center;visibility:visible;-webkit-animation:zoomIn .3s cubic-bezier(.12,.4,.29,1.46);animation:zoomIn .3s cubic-bezier(.12,.4,.29,1.46);pointer-events:none}.has-error.has-feedback .ant-form-item-children-icon svg,.has-success.has-feedback .ant-form-item-children-icon svg,.has-warning.has-feedback .ant-form-item-children-icon svg,.is-validating.has-feedback .ant-form-item-children-icon svg{position:absolute;top:0;right:0;bottom:0;left:0;margin:auto}.has-success.has-feedback .ant-form-item-children-icon{color:#52c41a;-webkit-animation-name:diffZoomIn1!important;animation-name:diffZoomIn1!important}.has-warning .ant-form-explain,.has-warning .ant-form-split{color:#faad14}.has-warning .ant-input,.has-warning .ant-input:hover{border-color:#faad14}.has-warning .ant-input:focus{border-color:#ffc53d;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(250,173,20,.2)}.has-warning .ant-input:not([disabled]):hover{border-color:#faad14}.has-warning .ant-calendar-picker-open .ant-calendar-picker-input{border-color:#ffc53d;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(250,173,20,.2)}.has-warning .ant-input-affix-wrapper .ant-input,.has-warning .ant-input-affix-wrapper .ant-input:hover{background-color:#22282a;border-color:#faad14}.has-warning .ant-input-affix-wrapper .ant-input:focus{border-color:#ffc53d;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(250,173,20,.2)}.has-warning .ant-input-affix-wrapper:hover .ant-input:not(.ant-input-disabled){border-color:#faad14}.has-warning .ant-input-prefix{color:#faad14}.has-warning .ant-input-group-addon{color:#faad14;background-color:#22282a;border-color:#faad14}.has-warning .has-feedback{color:#faad14}.has-warning.has-feedback .ant-form-item-children-icon{color:#faad14;-webkit-animation-name:diffZoomIn3!important;animation-name:diffZoomIn3!important}.has-warning .ant-select-selection,.has-warning .ant-select-selection:hover{border-color:#faad14}.has-warning .ant-select-focused .ant-select-selection,.has-warning .ant-select-open .ant-select-selection{border-color:#ffc53d;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(250,173,20,.2)}.has-warning .ant-calendar-picker-icon:after,.has-warning .ant-cascader-picker-arrow,.has-warning .ant-picker-icon:after,.has-warning .ant-select-arrow,.has-warning .ant-time-picker-icon:after{color:#faad14}.has-warning .ant-input-number,.has-warning .ant-time-picker-input{border-color:#faad14}.has-warning .ant-input-number-focused,.has-warning .ant-input-number:focus,.has-warning .ant-time-picker-input-focused,.has-warning .ant-time-picker-input:focus{border-color:#ffc53d;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(250,173,20,.2)}.has-warning .ant-input-number:not([disabled]):hover,.has-warning .ant-time-picker-input:not([disabled]):hover{border-color:#faad14}.has-warning .ant-cascader-picker:focus .ant-cascader-input{border-color:#ffc53d;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(250,173,20,.2)}.has-error .ant-form-explain,.has-error .ant-form-split{color:#f5222d}.has-error .ant-input,.has-error .ant-input:hover{border-color:#f5222d}.has-error .ant-input:focus{border-color:#ff4d4f;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(245,34,45,.2)}.has-error .ant-input:not([disabled]):hover{border-color:#f5222d}.has-error .ant-calendar-picker-open .ant-calendar-picker-input{border-color:#ff4d4f;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(245,34,45,.2)}.has-error .ant-input-affix-wrapper .ant-input,.has-error .ant-input-affix-wrapper .ant-input:hover{background-color:#22282a;border-color:#f5222d}.has-error .ant-input-affix-wrapper .ant-input:focus{border-color:#ff4d4f;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(245,34,45,.2)}.has-error .ant-input-affix-wrapper:hover .ant-input:not(.ant-input-disabled){border-color:#f5222d}.has-error .ant-input-prefix{color:#f5222d}.has-error .ant-input-group-addon{color:#f5222d;background-color:#22282a;border-color:#f5222d}.has-error .has-feedback{color:#f5222d}.has-error.has-feedback .ant-form-item-children-icon{color:#f5222d;-webkit-animation-name:diffZoomIn2!important;animation-name:diffZoomIn2!important}.has-error .ant-select-selection,.has-error .ant-select-selection:hover{border-color:#f5222d}.has-error .ant-select-focused .ant-select-selection,.has-error .ant-select-open .ant-select-selection{border-color:#ff4d4f;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(245,34,45,.2)}.has-error .ant-select.ant-select-auto-complete .ant-input:focus{border-color:#f5222d}.has-error .ant-input-group-addon .ant-select-selection{border-color:transparent;box-shadow:none}.has-error .ant-calendar-picker-icon:after,.has-error .ant-cascader-picker-arrow,.has-error .ant-picker-icon:after,.has-error .ant-select-arrow,.has-error .ant-time-picker-icon:after{color:#f5222d}.has-error .ant-input-number,.has-error .ant-time-picker-input{border-color:#f5222d}.has-error .ant-input-number-focused,.has-error .ant-input-number:focus,.has-error .ant-time-picker-input-focused,.has-error .ant-time-picker-input:focus{border-color:#ff4d4f;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(245,34,45,.2)}.has-error .ant-input-number:not([disabled]):hover,.has-error .ant-mention-wrapper .ant-mention-editor,.has-error .ant-mention-wrapper .ant-mention-editor:not([disabled]):hover,.has-error .ant-time-picker-input:not([disabled]):hover{border-color:#f5222d}.has-error .ant-cascader-picker:focus .ant-cascader-input,.has-error .ant-mention-wrapper.ant-mention-active:not([disabled]) .ant-mention-editor,.has-error .ant-mention-wrapper .ant-mention-editor:not([disabled]):focus{border-color:#ff4d4f;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(245,34,45,.2)}.has-error .ant-transfer-list{border-color:#f5222d}.has-error .ant-transfer-list-search:not([disabled]){border-color:#5c6970}.has-error .ant-transfer-list-search:not([disabled]):hover{border-color:#29bfff;border-right-width:1px!important}.has-error .ant-transfer-list-search:not([disabled]):focus{border-color:#29bfff;border-right-width:1px!important;outline:0;box-shadow:0 0 0 2px rgba(0,170,255,.2)}.is-validating.has-feedback .ant-form-item-children-icon{display:inline-block;color:#0af}.ant-advanced-search-form .ant-form-item{margin-bottom:24px}.ant-advanced-search-form .ant-form-item-with-help{margin-bottom:5px}.show-help-appear,.show-help-enter,.show-help-leave{-webkit-animation-duration:.3s;animation-duration:.3s;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-play-state:paused;animation-play-state:paused}.show-help-appear.show-help-appear-active,.show-help-enter.show-help-enter-active{-webkit-animation-name:antShowHelpIn;animation-name:antShowHelpIn;-webkit-animation-play-state:running;animation-play-state:running}.show-help-leave.show-help-leave-active{-webkit-animation-name:antShowHelpOut;animation-name:antShowHelpOut;-webkit-animation-play-state:running;animation-play-state:running;pointer-events:none}.show-help-appear,.show-help-enter{opacity:0}.show-help-appear,.show-help-enter,.show-help-leave{-webkit-animation-timing-function:cubic-bezier(.645,.045,.355,1);animation-timing-function:cubic-bezier(.645,.045,.355,1)}@-webkit-keyframes antShowHelpIn{0%{-webkit-transform:translateY(-5px);transform:translateY(-5px);opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);opacity:1}}@keyframes antShowHelpIn{0%{-webkit-transform:translateY(-5px);transform:translateY(-5px);opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);opacity:1}}@-webkit-keyframes antShowHelpOut{to{-webkit-transform:translateY(-5px);transform:translateY(-5px);opacity:0}}@keyframes antShowHelpOut{to{-webkit-transform:translateY(-5px);transform:translateY(-5px);opacity:0}}@-webkit-keyframes diffZoomIn1{0%{-webkit-transform:scale(0);transform:scale(0)}to{-webkit-transform:scale(1);transform:scale(1)}}@keyframes diffZoomIn1{0%{-webkit-transform:scale(0);transform:scale(0)}to{-webkit-transform:scale(1);transform:scale(1)}}@-webkit-keyframes diffZoomIn2{0%{-webkit-transform:scale(0);transform:scale(0)}to{-webkit-transform:scale(1);transform:scale(1)}}@keyframes diffZoomIn2{0%{-webkit-transform:scale(0);transform:scale(0)}to{-webkit-transform:scale(1);transform:scale(1)}}@-webkit-keyframes diffZoomIn3{0%{-webkit-transform:scale(0);transform:scale(0)}to{-webkit-transform:scale(1);transform:scale(1)}}@keyframes diffZoomIn3{0%{-webkit-transform:scale(0);transform:scale(0)}to{-webkit-transform:scale(1);transform:scale(1)}}.ant-row{position:relative;height:auto;margin-right:0;margin-left:0;zoom:1;display:block;box-sizing:border-box}.ant-row:after,.ant-row:before{display:table;content:""}.ant-row:after{clear:both}.ant-row-flex{display:flex;flex-flow:row wrap}.ant-row-flex:after,.ant-row-flex:before{display:flex}.ant-row-flex-start{justify-content:flex-start}.ant-row-flex-center{justify-content:center}.ant-row-flex-end{justify-content:flex-end}.ant-row-flex-space-between{justify-content:space-between}.ant-row-flex-space-around{justify-content:space-around}.ant-row-flex-top{align-items:flex-start}.ant-row-flex-middle{align-items:center}.ant-row-flex-bottom{align-items:flex-end}.ant-col{position:relative}.ant-col-1,.ant-col-2,.ant-col-3,.ant-col-4,.ant-col-5,.ant-col-6,.ant-col-7,.ant-col-8,.ant-col-9,.ant-col-10,.ant-col-11,.ant-col-12,.ant-col-13,.ant-col-14,.ant-col-15,.ant-col-16,.ant-col-17,.ant-col-18,.ant-col-19,.ant-col-20,.ant-col-21,.ant-col-22,.ant-col-23,.ant-col-24,.ant-col-lg-1,.ant-col-lg-2,.ant-col-lg-3,.ant-col-lg-4,.ant-col-lg-5,.ant-col-lg-6,.ant-col-lg-7,.ant-col-lg-8,.ant-col-lg-9,.ant-col-lg-10,.ant-col-lg-11,.ant-col-lg-12,.ant-col-lg-13,.ant-col-lg-14,.ant-col-lg-15,.ant-col-lg-16,.ant-col-lg-17,.ant-col-lg-18,.ant-col-lg-19,.ant-col-lg-20,.ant-col-lg-21,.ant-col-lg-22,.ant-col-lg-23,.ant-col-lg-24,.ant-col-md-1,.ant-col-md-2,.ant-col-md-3,.ant-col-md-4,.ant-col-md-5,.ant-col-md-6,.ant-col-md-7,.ant-col-md-8,.ant-col-md-9,.ant-col-md-10,.ant-col-md-11,.ant-col-md-12,.ant-col-md-13,.ant-col-md-14,.ant-col-md-15,.ant-col-md-16,.ant-col-md-17,.ant-col-md-18,.ant-col-md-19,.ant-col-md-20,.ant-col-md-21,.ant-col-md-22,.ant-col-md-23,.ant-col-md-24,.ant-col-sm-1,.ant-col-sm-2,.ant-col-sm-3,.ant-col-sm-4,.ant-col-sm-5,.ant-col-sm-6,.ant-col-sm-7,.ant-col-sm-8,.ant-col-sm-9,.ant-col-sm-10,.ant-col-sm-11,.ant-col-sm-12,.ant-col-sm-13,.ant-col-sm-14,.ant-col-sm-15,.ant-col-sm-16,.ant-col-sm-17,.ant-col-sm-18,.ant-col-sm-19,.ant-col-sm-20,.ant-col-sm-21,.ant-col-sm-22,.ant-col-sm-23,.ant-col-sm-24,.ant-col-xs-1,.ant-col-xs-2,.ant-col-xs-3,.ant-col-xs-4,.ant-col-xs-5,.ant-col-xs-6,.ant-col-xs-7,.ant-col-xs-8,.ant-col-xs-9,.ant-col-xs-10,.ant-col-xs-11,.ant-col-xs-12,.ant-col-xs-13,.ant-col-xs-14,.ant-col-xs-15,.ant-col-xs-16,.ant-col-xs-17,.ant-col-xs-18,.ant-col-xs-19,.ant-col-xs-20,.ant-col-xs-21,.ant-col-xs-22,.ant-col-xs-23,.ant-col-xs-24{position:relative;min-height:1px;padding-right:0;padding-left:0}.ant-col-1,.ant-col-2,.ant-col-3,.ant-col-4,.ant-col-5,.ant-col-6,.ant-col-7,.ant-col-8,.ant-col-9,.ant-col-10,.ant-col-11,.ant-col-12,.ant-col-13,.ant-col-14,.ant-col-15,.ant-col-16,.ant-col-17,.ant-col-18,.ant-col-19,.ant-col-20,.ant-col-21,.ant-col-22,.ant-col-23,.ant-col-24{flex:0 0 auto;float:left}.ant-col-24{display:block;box-sizing:border-box;width:100%}.ant-col-push-24{left:100%}.ant-col-pull-24{right:100%}.ant-col-offset-24{margin-left:100%}.ant-col-order-24{order:24}.ant-col-23{display:block;box-sizing:border-box;width:95.83333333%}.ant-col-push-23{left:95.83333333%}.ant-col-pull-23{right:95.83333333%}.ant-col-offset-23{margin-left:95.83333333%}.ant-col-order-23{order:23}.ant-col-22{display:block;box-sizing:border-box;width:91.66666667%}.ant-col-push-22{left:91.66666667%}.ant-col-pull-22{right:91.66666667%}.ant-col-offset-22{margin-left:91.66666667%}.ant-col-order-22{order:22}.ant-col-21{display:block;box-sizing:border-box;width:87.5%}.ant-col-push-21{left:87.5%}.ant-col-pull-21{right:87.5%}.ant-col-offset-21{margin-left:87.5%}.ant-col-order-21{order:21}.ant-col-20{display:block;box-sizing:border-box;width:83.33333333%}.ant-col-push-20{left:83.33333333%}.ant-col-pull-20{right:83.33333333%}.ant-col-offset-20{margin-left:83.33333333%}.ant-col-order-20{order:20}.ant-col-19{display:block;box-sizing:border-box;width:79.16666667%}.ant-col-push-19{left:79.16666667%}.ant-col-pull-19{right:79.16666667%}.ant-col-offset-19{margin-left:79.16666667%}.ant-col-order-19{order:19}.ant-col-18{display:block;box-sizing:border-box;width:75%}.ant-col-push-18{left:75%}.ant-col-pull-18{right:75%}.ant-col-offset-18{margin-left:75%}.ant-col-order-18{order:18}.ant-col-17{display:block;box-sizing:border-box;width:70.83333333%}.ant-col-push-17{left:70.83333333%}.ant-col-pull-17{right:70.83333333%}.ant-col-offset-17{margin-left:70.83333333%}.ant-col-order-17{order:17}.ant-col-16{display:block;box-sizing:border-box;width:66.66666667%}.ant-col-push-16{left:66.66666667%}.ant-col-pull-16{right:66.66666667%}.ant-col-offset-16{margin-left:66.66666667%}.ant-col-order-16{order:16}.ant-col-15{display:block;box-sizing:border-box;width:62.5%}.ant-col-push-15{left:62.5%}.ant-col-pull-15{right:62.5%}.ant-col-offset-15{margin-left:62.5%}.ant-col-order-15{order:15}.ant-col-14{display:block;box-sizing:border-box;width:58.33333333%}.ant-col-push-14{left:58.33333333%}.ant-col-pull-14{right:58.33333333%}.ant-col-offset-14{margin-left:58.33333333%}.ant-col-order-14{order:14}.ant-col-13{display:block;box-sizing:border-box;width:54.16666667%}.ant-col-push-13{left:54.16666667%}.ant-col-pull-13{right:54.16666667%}.ant-col-offset-13{margin-left:54.16666667%}.ant-col-order-13{order:13}.ant-col-12{display:block;box-sizing:border-box;width:50%}.ant-col-push-12{left:50%}.ant-col-pull-12{right:50%}.ant-col-offset-12{margin-left:50%}.ant-col-order-12{order:12}.ant-col-11{display:block;box-sizing:border-box;width:45.83333333%}.ant-col-push-11{left:45.83333333%}.ant-col-pull-11{right:45.83333333%}.ant-col-offset-11{margin-left:45.83333333%}.ant-col-order-11{order:11}.ant-col-10{display:block;box-sizing:border-box;width:41.66666667%}.ant-col-push-10{left:41.66666667%}.ant-col-pull-10{right:41.66666667%}.ant-col-offset-10{margin-left:41.66666667%}.ant-col-order-10{order:10}.ant-col-9{display:block;box-sizing:border-box;width:37.5%}.ant-col-push-9{left:37.5%}.ant-col-pull-9{right:37.5%}.ant-col-offset-9{margin-left:37.5%}.ant-col-order-9{order:9}.ant-col-8{display:block;box-sizing:border-box;width:33.33333333%}.ant-col-push-8{left:33.33333333%}.ant-col-pull-8{right:33.33333333%}.ant-col-offset-8{margin-left:33.33333333%}.ant-col-order-8{order:8}.ant-col-7{display:block;box-sizing:border-box;width:29.16666667%}.ant-col-push-7{left:29.16666667%}.ant-col-pull-7{right:29.16666667%}.ant-col-offset-7{margin-left:29.16666667%}.ant-col-order-7{order:7}.ant-col-6{display:block;box-sizing:border-box;width:25%}.ant-col-push-6{left:25%}.ant-col-pull-6{right:25%}.ant-col-offset-6{margin-left:25%}.ant-col-order-6{order:6}.ant-col-5{display:block;box-sizing:border-box;width:20.83333333%}.ant-col-push-5{left:20.83333333%}.ant-col-pull-5{right:20.83333333%}.ant-col-offset-5{margin-left:20.83333333%}.ant-col-order-5{order:5}.ant-col-4{display:block;box-sizing:border-box;width:16.66666667%}.ant-col-push-4{left:16.66666667%}.ant-col-pull-4{right:16.66666667%}.ant-col-offset-4{margin-left:16.66666667%}.ant-col-order-4{order:4}.ant-col-3{display:block;box-sizing:border-box;width:12.5%}.ant-col-push-3{left:12.5%}.ant-col-pull-3{right:12.5%}.ant-col-offset-3{margin-left:12.5%}.ant-col-order-3{order:3}.ant-col-2{display:block;box-sizing:border-box;width:8.33333333%}.ant-col-push-2{left:8.33333333%}.ant-col-pull-2{right:8.33333333%}.ant-col-offset-2{margin-left:8.33333333%}.ant-col-order-2{order:2}.ant-col-1{display:block;box-sizing:border-box;width:4.16666667%}.ant-col-push-1{left:4.16666667%}.ant-col-pull-1{right:4.16666667%}.ant-col-offset-1{margin-left:4.16666667%}.ant-col-order-1{order:1}.ant-col-0{display:none}.ant-col-offset-0{margin-left:0}.ant-col-order-0{order:0}.ant-col-xs-1,.ant-col-xs-2,.ant-col-xs-3,.ant-col-xs-4,.ant-col-xs-5,.ant-col-xs-6,.ant-col-xs-7,.ant-col-xs-8,.ant-col-xs-9,.ant-col-xs-10,.ant-col-xs-11,.ant-col-xs-12,.ant-col-xs-13,.ant-col-xs-14,.ant-col-xs-15,.ant-col-xs-16,.ant-col-xs-17,.ant-col-xs-18,.ant-col-xs-19,.ant-col-xs-20,.ant-col-xs-21,.ant-col-xs-22,.ant-col-xs-23,.ant-col-xs-24{flex:0 0 auto;float:left}.ant-col-xs-24{display:block;box-sizing:border-box;width:100%}.ant-col-xs-push-24{left:100%}.ant-col-xs-pull-24{right:100%}.ant-col-xs-offset-24{margin-left:100%}.ant-col-xs-order-24{order:24}.ant-col-xs-23{display:block;box-sizing:border-box;width:95.83333333%}.ant-col-xs-push-23{left:95.83333333%}.ant-col-xs-pull-23{right:95.83333333%}.ant-col-xs-offset-23{margin-left:95.83333333%}.ant-col-xs-order-23{order:23}.ant-col-xs-22{display:block;box-sizing:border-box;width:91.66666667%}.ant-col-xs-push-22{left:91.66666667%}.ant-col-xs-pull-22{right:91.66666667%}.ant-col-xs-offset-22{margin-left:91.66666667%}.ant-col-xs-order-22{order:22}.ant-col-xs-21{display:block;box-sizing:border-box;width:87.5%}.ant-col-xs-push-21{left:87.5%}.ant-col-xs-pull-21{right:87.5%}.ant-col-xs-offset-21{margin-left:87.5%}.ant-col-xs-order-21{order:21}.ant-col-xs-20{display:block;box-sizing:border-box;width:83.33333333%}.ant-col-xs-push-20{left:83.33333333%}.ant-col-xs-pull-20{right:83.33333333%}.ant-col-xs-offset-20{margin-left:83.33333333%}.ant-col-xs-order-20{order:20}.ant-col-xs-19{display:block;box-sizing:border-box;width:79.16666667%}.ant-col-xs-push-19{left:79.16666667%}.ant-col-xs-pull-19{right:79.16666667%}.ant-col-xs-offset-19{margin-left:79.16666667%}.ant-col-xs-order-19{order:19}.ant-col-xs-18{display:block;box-sizing:border-box;width:75%}.ant-col-xs-push-18{left:75%}.ant-col-xs-pull-18{right:75%}.ant-col-xs-offset-18{margin-left:75%}.ant-col-xs-order-18{order:18}.ant-col-xs-17{display:block;box-sizing:border-box;width:70.83333333%}.ant-col-xs-push-17{left:70.83333333%}.ant-col-xs-pull-17{right:70.83333333%}.ant-col-xs-offset-17{margin-left:70.83333333%}.ant-col-xs-order-17{order:17}.ant-col-xs-16{display:block;box-sizing:border-box;width:66.66666667%}.ant-col-xs-push-16{left:66.66666667%}.ant-col-xs-pull-16{right:66.66666667%}.ant-col-xs-offset-16{margin-left:66.66666667%}.ant-col-xs-order-16{order:16}.ant-col-xs-15{display:block;box-sizing:border-box;width:62.5%}.ant-col-xs-push-15{left:62.5%}.ant-col-xs-pull-15{right:62.5%}.ant-col-xs-offset-15{margin-left:62.5%}.ant-col-xs-order-15{order:15}.ant-col-xs-14{display:block;box-sizing:border-box;width:58.33333333%}.ant-col-xs-push-14{left:58.33333333%}.ant-col-xs-pull-14{right:58.33333333%}.ant-col-xs-offset-14{margin-left:58.33333333%}.ant-col-xs-order-14{order:14}.ant-col-xs-13{display:block;box-sizing:border-box;width:54.16666667%}.ant-col-xs-push-13{left:54.16666667%}.ant-col-xs-pull-13{right:54.16666667%}.ant-col-xs-offset-13{margin-left:54.16666667%}.ant-col-xs-order-13{order:13}.ant-col-xs-12{display:block;box-sizing:border-box;width:50%}.ant-col-xs-push-12{left:50%}.ant-col-xs-pull-12{right:50%}.ant-col-xs-offset-12{margin-left:50%}.ant-col-xs-order-12{order:12}.ant-col-xs-11{display:block;box-sizing:border-box;width:45.83333333%}.ant-col-xs-push-11{left:45.83333333%}.ant-col-xs-pull-11{right:45.83333333%}.ant-col-xs-offset-11{margin-left:45.83333333%}.ant-col-xs-order-11{order:11}.ant-col-xs-10{display:block;box-sizing:border-box;width:41.66666667%}.ant-col-xs-push-10{left:41.66666667%}.ant-col-xs-pull-10{right:41.66666667%}.ant-col-xs-offset-10{margin-left:41.66666667%}.ant-col-xs-order-10{order:10}.ant-col-xs-9{display:block;box-sizing:border-box;width:37.5%}.ant-col-xs-push-9{left:37.5%}.ant-col-xs-pull-9{right:37.5%}.ant-col-xs-offset-9{margin-left:37.5%}.ant-col-xs-order-9{order:9}.ant-col-xs-8{display:block;box-sizing:border-box;width:33.33333333%}.ant-col-xs-push-8{left:33.33333333%}.ant-col-xs-pull-8{right:33.33333333%}.ant-col-xs-offset-8{margin-left:33.33333333%}.ant-col-xs-order-8{order:8}.ant-col-xs-7{display:block;box-sizing:border-box;width:29.16666667%}.ant-col-xs-push-7{left:29.16666667%}.ant-col-xs-pull-7{right:29.16666667%}.ant-col-xs-offset-7{margin-left:29.16666667%}.ant-col-xs-order-7{order:7}.ant-col-xs-6{display:block;box-sizing:border-box;width:25%}.ant-col-xs-push-6{left:25%}.ant-col-xs-pull-6{right:25%}.ant-col-xs-offset-6{margin-left:25%}.ant-col-xs-order-6{order:6}.ant-col-xs-5{display:block;box-sizing:border-box;width:20.83333333%}.ant-col-xs-push-5{left:20.83333333%}.ant-col-xs-pull-5{right:20.83333333%}.ant-col-xs-offset-5{margin-left:20.83333333%}.ant-col-xs-order-5{order:5}.ant-col-xs-4{display:block;box-sizing:border-box;width:16.66666667%}.ant-col-xs-push-4{left:16.66666667%}.ant-col-xs-pull-4{right:16.66666667%}.ant-col-xs-offset-4{margin-left:16.66666667%}.ant-col-xs-order-4{order:4}.ant-col-xs-3{display:block;box-sizing:border-box;width:12.5%}.ant-col-xs-push-3{left:12.5%}.ant-col-xs-pull-3{right:12.5%}.ant-col-xs-offset-3{margin-left:12.5%}.ant-col-xs-order-3{order:3}.ant-col-xs-2{display:block;box-sizing:border-box;width:8.33333333%}.ant-col-xs-push-2{left:8.33333333%}.ant-col-xs-pull-2{right:8.33333333%}.ant-col-xs-offset-2{margin-left:8.33333333%}.ant-col-xs-order-2{order:2}.ant-col-xs-1{display:block;box-sizing:border-box;width:4.16666667%}.ant-col-xs-push-1{left:4.16666667%}.ant-col-xs-pull-1{right:4.16666667%}.ant-col-xs-offset-1{margin-left:4.16666667%}.ant-col-xs-order-1{order:1}.ant-col-xs-0{display:none}.ant-col-push-0{left:auto}.ant-col-pull-0{right:auto}.ant-col-xs-push-0{left:auto}.ant-col-xs-pull-0{right:auto}.ant-col-xs-offset-0{margin-left:0}.ant-col-xs-order-0{order:0}@media (min-width:576px){.ant-col-sm-1,.ant-col-sm-2,.ant-col-sm-3,.ant-col-sm-4,.ant-col-sm-5,.ant-col-sm-6,.ant-col-sm-7,.ant-col-sm-8,.ant-col-sm-9,.ant-col-sm-10,.ant-col-sm-11,.ant-col-sm-12,.ant-col-sm-13,.ant-col-sm-14,.ant-col-sm-15,.ant-col-sm-16,.ant-col-sm-17,.ant-col-sm-18,.ant-col-sm-19,.ant-col-sm-20,.ant-col-sm-21,.ant-col-sm-22,.ant-col-sm-23,.ant-col-sm-24{flex:0 0 auto;float:left}.ant-col-sm-24{display:block;box-sizing:border-box;width:100%}.ant-col-sm-push-24{left:100%}.ant-col-sm-pull-24{right:100%}.ant-col-sm-offset-24{margin-left:100%}.ant-col-sm-order-24{order:24}.ant-col-sm-23{display:block;box-sizing:border-box;width:95.83333333%}.ant-col-sm-push-23{left:95.83333333%}.ant-col-sm-pull-23{right:95.83333333%}.ant-col-sm-offset-23{margin-left:95.83333333%}.ant-col-sm-order-23{order:23}.ant-col-sm-22{display:block;box-sizing:border-box;width:91.66666667%}.ant-col-sm-push-22{left:91.66666667%}.ant-col-sm-pull-22{right:91.66666667%}.ant-col-sm-offset-22{margin-left:91.66666667%}.ant-col-sm-order-22{order:22}.ant-col-sm-21{display:block;box-sizing:border-box;width:87.5%}.ant-col-sm-push-21{left:87.5%}.ant-col-sm-pull-21{right:87.5%}.ant-col-sm-offset-21{margin-left:87.5%}.ant-col-sm-order-21{order:21}.ant-col-sm-20{display:block;box-sizing:border-box;width:83.33333333%}.ant-col-sm-push-20{left:83.33333333%}.ant-col-sm-pull-20{right:83.33333333%}.ant-col-sm-offset-20{margin-left:83.33333333%}.ant-col-sm-order-20{order:20}.ant-col-sm-19{display:block;box-sizing:border-box;width:79.16666667%}.ant-col-sm-push-19{left:79.16666667%}.ant-col-sm-pull-19{right:79.16666667%}.ant-col-sm-offset-19{margin-left:79.16666667%}.ant-col-sm-order-19{order:19}.ant-col-sm-18{display:block;box-sizing:border-box;width:75%}.ant-col-sm-push-18{left:75%}.ant-col-sm-pull-18{right:75%}.ant-col-sm-offset-18{margin-left:75%}.ant-col-sm-order-18{order:18}.ant-col-sm-17{display:block;box-sizing:border-box;width:70.83333333%}.ant-col-sm-push-17{left:70.83333333%}.ant-col-sm-pull-17{right:70.83333333%}.ant-col-sm-offset-17{margin-left:70.83333333%}.ant-col-sm-order-17{order:17}.ant-col-sm-16{display:block;box-sizing:border-box;width:66.66666667%}.ant-col-sm-push-16{left:66.66666667%}.ant-col-sm-pull-16{right:66.66666667%}.ant-col-sm-offset-16{margin-left:66.66666667%}.ant-col-sm-order-16{order:16}.ant-col-sm-15{display:block;box-sizing:border-box;width:62.5%}.ant-col-sm-push-15{left:62.5%}.ant-col-sm-pull-15{right:62.5%}.ant-col-sm-offset-15{margin-left:62.5%}.ant-col-sm-order-15{order:15}.ant-col-sm-14{display:block;box-sizing:border-box;width:58.33333333%}.ant-col-sm-push-14{left:58.33333333%}.ant-col-sm-pull-14{right:58.33333333%}.ant-col-sm-offset-14{margin-left:58.33333333%}.ant-col-sm-order-14{order:14}.ant-col-sm-13{display:block;box-sizing:border-box;width:54.16666667%}.ant-col-sm-push-13{left:54.16666667%}.ant-col-sm-pull-13{right:54.16666667%}.ant-col-sm-offset-13{margin-left:54.16666667%}.ant-col-sm-order-13{order:13}.ant-col-sm-12{display:block;box-sizing:border-box;width:50%}.ant-col-sm-push-12{left:50%}.ant-col-sm-pull-12{right:50%}.ant-col-sm-offset-12{margin-left:50%}.ant-col-sm-order-12{order:12}.ant-col-sm-11{display:block;box-sizing:border-box;width:45.83333333%}.ant-col-sm-push-11{left:45.83333333%}.ant-col-sm-pull-11{right:45.83333333%}.ant-col-sm-offset-11{margin-left:45.83333333%}.ant-col-sm-order-11{order:11}.ant-col-sm-10{display:block;box-sizing:border-box;width:41.66666667%}.ant-col-sm-push-10{left:41.66666667%}.ant-col-sm-pull-10{right:41.66666667%}.ant-col-sm-offset-10{margin-left:41.66666667%}.ant-col-sm-order-10{order:10}.ant-col-sm-9{display:block;box-sizing:border-box;width:37.5%}.ant-col-sm-push-9{left:37.5%}.ant-col-sm-pull-9{right:37.5%}.ant-col-sm-offset-9{margin-left:37.5%}.ant-col-sm-order-9{order:9}.ant-col-sm-8{display:block;box-sizing:border-box;width:33.33333333%}.ant-col-sm-push-8{left:33.33333333%}.ant-col-sm-pull-8{right:33.33333333%}.ant-col-sm-offset-8{margin-left:33.33333333%}.ant-col-sm-order-8{order:8}.ant-col-sm-7{display:block;box-sizing:border-box;width:29.16666667%}.ant-col-sm-push-7{left:29.16666667%}.ant-col-sm-pull-7{right:29.16666667%}.ant-col-sm-offset-7{margin-left:29.16666667%}.ant-col-sm-order-7{order:7}.ant-col-sm-6{display:block;box-sizing:border-box;width:25%}.ant-col-sm-push-6{left:25%}.ant-col-sm-pull-6{right:25%}.ant-col-sm-offset-6{margin-left:25%}.ant-col-sm-order-6{order:6}.ant-col-sm-5{display:block;box-sizing:border-box;width:20.83333333%}.ant-col-sm-push-5{left:20.83333333%}.ant-col-sm-pull-5{right:20.83333333%}.ant-col-sm-offset-5{margin-left:20.83333333%}.ant-col-sm-order-5{order:5}.ant-col-sm-4{display:block;box-sizing:border-box;width:16.66666667%}.ant-col-sm-push-4{left:16.66666667%}.ant-col-sm-pull-4{right:16.66666667%}.ant-col-sm-offset-4{margin-left:16.66666667%}.ant-col-sm-order-4{order:4}.ant-col-sm-3{display:block;box-sizing:border-box;width:12.5%}.ant-col-sm-push-3{left:12.5%}.ant-col-sm-pull-3{right:12.5%}.ant-col-sm-offset-3{margin-left:12.5%}.ant-col-sm-order-3{order:3}.ant-col-sm-2{display:block;box-sizing:border-box;width:8.33333333%}.ant-col-sm-push-2{left:8.33333333%}.ant-col-sm-pull-2{right:8.33333333%}.ant-col-sm-offset-2{margin-left:8.33333333%}.ant-col-sm-order-2{order:2}.ant-col-sm-1{display:block;box-sizing:border-box;width:4.16666667%}.ant-col-sm-push-1{left:4.16666667%}.ant-col-sm-pull-1{right:4.16666667%}.ant-col-sm-offset-1{margin-left:4.16666667%}.ant-col-sm-order-1{order:1}.ant-col-sm-0{display:none}.ant-col-push-0{left:auto}.ant-col-pull-0{right:auto}.ant-col-sm-push-0{left:auto}.ant-col-sm-pull-0{right:auto}.ant-col-sm-offset-0{margin-left:0}.ant-col-sm-order-0{order:0}}@media (min-width:768px){.ant-col-md-1,.ant-col-md-2,.ant-col-md-3,.ant-col-md-4,.ant-col-md-5,.ant-col-md-6,.ant-col-md-7,.ant-col-md-8,.ant-col-md-9,.ant-col-md-10,.ant-col-md-11,.ant-col-md-12,.ant-col-md-13,.ant-col-md-14,.ant-col-md-15,.ant-col-md-16,.ant-col-md-17,.ant-col-md-18,.ant-col-md-19,.ant-col-md-20,.ant-col-md-21,.ant-col-md-22,.ant-col-md-23,.ant-col-md-24{flex:0 0 auto;float:left}.ant-col-md-24{display:block;box-sizing:border-box;width:100%}.ant-col-md-push-24{left:100%}.ant-col-md-pull-24{right:100%}.ant-col-md-offset-24{margin-left:100%}.ant-col-md-order-24{order:24}.ant-col-md-23{display:block;box-sizing:border-box;width:95.83333333%}.ant-col-md-push-23{left:95.83333333%}.ant-col-md-pull-23{right:95.83333333%}.ant-col-md-offset-23{margin-left:95.83333333%}.ant-col-md-order-23{order:23}.ant-col-md-22{display:block;box-sizing:border-box;width:91.66666667%}.ant-col-md-push-22{left:91.66666667%}.ant-col-md-pull-22{right:91.66666667%}.ant-col-md-offset-22{margin-left:91.66666667%}.ant-col-md-order-22{order:22}.ant-col-md-21{display:block;box-sizing:border-box;width:87.5%}.ant-col-md-push-21{left:87.5%}.ant-col-md-pull-21{right:87.5%}.ant-col-md-offset-21{margin-left:87.5%}.ant-col-md-order-21{order:21}.ant-col-md-20{display:block;box-sizing:border-box;width:83.33333333%}.ant-col-md-push-20{left:83.33333333%}.ant-col-md-pull-20{right:83.33333333%}.ant-col-md-offset-20{margin-left:83.33333333%}.ant-col-md-order-20{order:20}.ant-col-md-19{display:block;box-sizing:border-box;width:79.16666667%}.ant-col-md-push-19{left:79.16666667%}.ant-col-md-pull-19{right:79.16666667%}.ant-col-md-offset-19{margin-left:79.16666667%}.ant-col-md-order-19{order:19}.ant-col-md-18{display:block;box-sizing:border-box;width:75%}.ant-col-md-push-18{left:75%}.ant-col-md-pull-18{right:75%}.ant-col-md-offset-18{margin-left:75%}.ant-col-md-order-18{order:18}.ant-col-md-17{display:block;box-sizing:border-box;width:70.83333333%}.ant-col-md-push-17{left:70.83333333%}.ant-col-md-pull-17{right:70.83333333%}.ant-col-md-offset-17{margin-left:70.83333333%}.ant-col-md-order-17{order:17}.ant-col-md-16{display:block;box-sizing:border-box;width:66.66666667%}.ant-col-md-push-16{left:66.66666667%}.ant-col-md-pull-16{right:66.66666667%}.ant-col-md-offset-16{margin-left:66.66666667%}.ant-col-md-order-16{order:16}.ant-col-md-15{display:block;box-sizing:border-box;width:62.5%}.ant-col-md-push-15{left:62.5%}.ant-col-md-pull-15{right:62.5%}.ant-col-md-offset-15{margin-left:62.5%}.ant-col-md-order-15{order:15}.ant-col-md-14{display:block;box-sizing:border-box;width:58.33333333%}.ant-col-md-push-14{left:58.33333333%}.ant-col-md-pull-14{right:58.33333333%}.ant-col-md-offset-14{margin-left:58.33333333%}.ant-col-md-order-14{order:14}.ant-col-md-13{display:block;box-sizing:border-box;width:54.16666667%}.ant-col-md-push-13{left:54.16666667%}.ant-col-md-pull-13{right:54.16666667%}.ant-col-md-offset-13{margin-left:54.16666667%}.ant-col-md-order-13{order:13}.ant-col-md-12{display:block;box-sizing:border-box;width:50%}.ant-col-md-push-12{left:50%}.ant-col-md-pull-12{right:50%}.ant-col-md-offset-12{margin-left:50%}.ant-col-md-order-12{order:12}.ant-col-md-11{display:block;box-sizing:border-box;width:45.83333333%}.ant-col-md-push-11{left:45.83333333%}.ant-col-md-pull-11{right:45.83333333%}.ant-col-md-offset-11{margin-left:45.83333333%}.ant-col-md-order-11{order:11}.ant-col-md-10{display:block;box-sizing:border-box;width:41.66666667%}.ant-col-md-push-10{left:41.66666667%}.ant-col-md-pull-10{right:41.66666667%}.ant-col-md-offset-10{margin-left:41.66666667%}.ant-col-md-order-10{order:10}.ant-col-md-9{display:block;box-sizing:border-box;width:37.5%}.ant-col-md-push-9{left:37.5%}.ant-col-md-pull-9{right:37.5%}.ant-col-md-offset-9{margin-left:37.5%}.ant-col-md-order-9{order:9}.ant-col-md-8{display:block;box-sizing:border-box;width:33.33333333%}.ant-col-md-push-8{left:33.33333333%}.ant-col-md-pull-8{right:33.33333333%}.ant-col-md-offset-8{margin-left:33.33333333%}.ant-col-md-order-8{order:8}.ant-col-md-7{display:block;box-sizing:border-box;width:29.16666667%}.ant-col-md-push-7{left:29.16666667%}.ant-col-md-pull-7{right:29.16666667%}.ant-col-md-offset-7{margin-left:29.16666667%}.ant-col-md-order-7{order:7}.ant-col-md-6{display:block;box-sizing:border-box;width:25%}.ant-col-md-push-6{left:25%}.ant-col-md-pull-6{right:25%}.ant-col-md-offset-6{margin-left:25%}.ant-col-md-order-6{order:6}.ant-col-md-5{display:block;box-sizing:border-box;width:20.83333333%}.ant-col-md-push-5{left:20.83333333%}.ant-col-md-pull-5{right:20.83333333%}.ant-col-md-offset-5{margin-left:20.83333333%}.ant-col-md-order-5{order:5}.ant-col-md-4{display:block;box-sizing:border-box;width:16.66666667%}.ant-col-md-push-4{left:16.66666667%}.ant-col-md-pull-4{right:16.66666667%}.ant-col-md-offset-4{margin-left:16.66666667%}.ant-col-md-order-4{order:4}.ant-col-md-3{display:block;box-sizing:border-box;width:12.5%}.ant-col-md-push-3{left:12.5%}.ant-col-md-pull-3{right:12.5%}.ant-col-md-offset-3{margin-left:12.5%}.ant-col-md-order-3{order:3}.ant-col-md-2{display:block;box-sizing:border-box;width:8.33333333%}.ant-col-md-push-2{left:8.33333333%}.ant-col-md-pull-2{right:8.33333333%}.ant-col-md-offset-2{margin-left:8.33333333%}.ant-col-md-order-2{order:2}.ant-col-md-1{display:block;box-sizing:border-box;width:4.16666667%}.ant-col-md-push-1{left:4.16666667%}.ant-col-md-pull-1{right:4.16666667%}.ant-col-md-offset-1{margin-left:4.16666667%}.ant-col-md-order-1{order:1}.ant-col-md-0{display:none}.ant-col-push-0{left:auto}.ant-col-pull-0{right:auto}.ant-col-md-push-0{left:auto}.ant-col-md-pull-0{right:auto}.ant-col-md-offset-0{margin-left:0}.ant-col-md-order-0{order:0}}@media (min-width:992px){.ant-col-lg-1,.ant-col-lg-2,.ant-col-lg-3,.ant-col-lg-4,.ant-col-lg-5,.ant-col-lg-6,.ant-col-lg-7,.ant-col-lg-8,.ant-col-lg-9,.ant-col-lg-10,.ant-col-lg-11,.ant-col-lg-12,.ant-col-lg-13,.ant-col-lg-14,.ant-col-lg-15,.ant-col-lg-16,.ant-col-lg-17,.ant-col-lg-18,.ant-col-lg-19,.ant-col-lg-20,.ant-col-lg-21,.ant-col-lg-22,.ant-col-lg-23,.ant-col-lg-24{flex:0 0 auto;float:left}.ant-col-lg-24{display:block;box-sizing:border-box;width:100%}.ant-col-lg-push-24{left:100%}.ant-col-lg-pull-24{right:100%}.ant-col-lg-offset-24{margin-left:100%}.ant-col-lg-order-24{order:24}.ant-col-lg-23{display:block;box-sizing:border-box;width:95.83333333%}.ant-col-lg-push-23{left:95.83333333%}.ant-col-lg-pull-23{right:95.83333333%}.ant-col-lg-offset-23{margin-left:95.83333333%}.ant-col-lg-order-23{order:23}.ant-col-lg-22{display:block;box-sizing:border-box;width:91.66666667%}.ant-col-lg-push-22{left:91.66666667%}.ant-col-lg-pull-22{right:91.66666667%}.ant-col-lg-offset-22{margin-left:91.66666667%}.ant-col-lg-order-22{order:22}.ant-col-lg-21{display:block;box-sizing:border-box;width:87.5%}.ant-col-lg-push-21{left:87.5%}.ant-col-lg-pull-21{right:87.5%}.ant-col-lg-offset-21{margin-left:87.5%}.ant-col-lg-order-21{order:21}.ant-col-lg-20{display:block;box-sizing:border-box;width:83.33333333%}.ant-col-lg-push-20{left:83.33333333%}.ant-col-lg-pull-20{right:83.33333333%}.ant-col-lg-offset-20{margin-left:83.33333333%}.ant-col-lg-order-20{order:20}.ant-col-lg-19{display:block;box-sizing:border-box;width:79.16666667%}.ant-col-lg-push-19{left:79.16666667%}.ant-col-lg-pull-19{right:79.16666667%}.ant-col-lg-offset-19{margin-left:79.16666667%}.ant-col-lg-order-19{order:19}.ant-col-lg-18{display:block;box-sizing:border-box;width:75%}.ant-col-lg-push-18{left:75%}.ant-col-lg-pull-18{right:75%}.ant-col-lg-offset-18{margin-left:75%}.ant-col-lg-order-18{order:18}.ant-col-lg-17{display:block;box-sizing:border-box;width:70.83333333%}.ant-col-lg-push-17{left:70.83333333%}.ant-col-lg-pull-17{right:70.83333333%}.ant-col-lg-offset-17{margin-left:70.83333333%}.ant-col-lg-order-17{order:17}.ant-col-lg-16{display:block;box-sizing:border-box;width:66.66666667%}.ant-col-lg-push-16{left:66.66666667%}.ant-col-lg-pull-16{right:66.66666667%}.ant-col-lg-offset-16{margin-left:66.66666667%}.ant-col-lg-order-16{order:16}.ant-col-lg-15{display:block;box-sizing:border-box;width:62.5%}.ant-col-lg-push-15{left:62.5%}.ant-col-lg-pull-15{right:62.5%}.ant-col-lg-offset-15{margin-left:62.5%}.ant-col-lg-order-15{order:15}.ant-col-lg-14{display:block;box-sizing:border-box;width:58.33333333%}.ant-col-lg-push-14{left:58.33333333%}.ant-col-lg-pull-14{right:58.33333333%}.ant-col-lg-offset-14{margin-left:58.33333333%}.ant-col-lg-order-14{order:14}.ant-col-lg-13{display:block;box-sizing:border-box;width:54.16666667%}.ant-col-lg-push-13{left:54.16666667%}.ant-col-lg-pull-13{right:54.16666667%}.ant-col-lg-offset-13{margin-left:54.16666667%}.ant-col-lg-order-13{order:13}.ant-col-lg-12{display:block;box-sizing:border-box;width:50%}.ant-col-lg-push-12{left:50%}.ant-col-lg-pull-12{right:50%}.ant-col-lg-offset-12{margin-left:50%}.ant-col-lg-order-12{order:12}.ant-col-lg-11{display:block;box-sizing:border-box;width:45.83333333%}.ant-col-lg-push-11{left:45.83333333%}.ant-col-lg-pull-11{right:45.83333333%}.ant-col-lg-offset-11{margin-left:45.83333333%}.ant-col-lg-order-11{order:11}.ant-col-lg-10{display:block;box-sizing:border-box;width:41.66666667%}.ant-col-lg-push-10{left:41.66666667%}.ant-col-lg-pull-10{right:41.66666667%}.ant-col-lg-offset-10{margin-left:41.66666667%}.ant-col-lg-order-10{order:10}.ant-col-lg-9{display:block;box-sizing:border-box;width:37.5%}.ant-col-lg-push-9{left:37.5%}.ant-col-lg-pull-9{right:37.5%}.ant-col-lg-offset-9{margin-left:37.5%}.ant-col-lg-order-9{order:9}.ant-col-lg-8{display:block;box-sizing:border-box;width:33.33333333%}.ant-col-lg-push-8{left:33.33333333%}.ant-col-lg-pull-8{right:33.33333333%}.ant-col-lg-offset-8{margin-left:33.33333333%}.ant-col-lg-order-8{order:8}.ant-col-lg-7{display:block;box-sizing:border-box;width:29.16666667%}.ant-col-lg-push-7{left:29.16666667%}.ant-col-lg-pull-7{right:29.16666667%}.ant-col-lg-offset-7{margin-left:29.16666667%}.ant-col-lg-order-7{order:7}.ant-col-lg-6{display:block;box-sizing:border-box;width:25%}.ant-col-lg-push-6{left:25%}.ant-col-lg-pull-6{right:25%}.ant-col-lg-offset-6{margin-left:25%}.ant-col-lg-order-6{order:6}.ant-col-lg-5{display:block;box-sizing:border-box;width:20.83333333%}.ant-col-lg-push-5{left:20.83333333%}.ant-col-lg-pull-5{right:20.83333333%}.ant-col-lg-offset-5{margin-left:20.83333333%}.ant-col-lg-order-5{order:5}.ant-col-lg-4{display:block;box-sizing:border-box;width:16.66666667%}.ant-col-lg-push-4{left:16.66666667%}.ant-col-lg-pull-4{right:16.66666667%}.ant-col-lg-offset-4{margin-left:16.66666667%}.ant-col-lg-order-4{order:4}.ant-col-lg-3{display:block;box-sizing:border-box;width:12.5%}.ant-col-lg-push-3{left:12.5%}.ant-col-lg-pull-3{right:12.5%}.ant-col-lg-offset-3{margin-left:12.5%}.ant-col-lg-order-3{order:3}.ant-col-lg-2{display:block;box-sizing:border-box;width:8.33333333%}.ant-col-lg-push-2{left:8.33333333%}.ant-col-lg-pull-2{right:8.33333333%}.ant-col-lg-offset-2{margin-left:8.33333333%}.ant-col-lg-order-2{order:2}.ant-col-lg-1{display:block;box-sizing:border-box;width:4.16666667%}.ant-col-lg-push-1{left:4.16666667%}.ant-col-lg-pull-1{right:4.16666667%}.ant-col-lg-offset-1{margin-left:4.16666667%}.ant-col-lg-order-1{order:1}.ant-col-lg-0{display:none}.ant-col-push-0{left:auto}.ant-col-pull-0{right:auto}.ant-col-lg-push-0{left:auto}.ant-col-lg-pull-0{right:auto}.ant-col-lg-offset-0{margin-left:0}.ant-col-lg-order-0{order:0}}@media (min-width:1200px){.ant-col-xl-1,.ant-col-xl-2,.ant-col-xl-3,.ant-col-xl-4,.ant-col-xl-5,.ant-col-xl-6,.ant-col-xl-7,.ant-col-xl-8,.ant-col-xl-9,.ant-col-xl-10,.ant-col-xl-11,.ant-col-xl-12,.ant-col-xl-13,.ant-col-xl-14,.ant-col-xl-15,.ant-col-xl-16,.ant-col-xl-17,.ant-col-xl-18,.ant-col-xl-19,.ant-col-xl-20,.ant-col-xl-21,.ant-col-xl-22,.ant-col-xl-23,.ant-col-xl-24{flex:0 0 auto;float:left}.ant-col-xl-24{display:block;box-sizing:border-box;width:100%}.ant-col-xl-push-24{left:100%}.ant-col-xl-pull-24{right:100%}.ant-col-xl-offset-24{margin-left:100%}.ant-col-xl-order-24{order:24}.ant-col-xl-23{display:block;box-sizing:border-box;width:95.83333333%}.ant-col-xl-push-23{left:95.83333333%}.ant-col-xl-pull-23{right:95.83333333%}.ant-col-xl-offset-23{margin-left:95.83333333%}.ant-col-xl-order-23{order:23}.ant-col-xl-22{display:block;box-sizing:border-box;width:91.66666667%}.ant-col-xl-push-22{left:91.66666667%}.ant-col-xl-pull-22{right:91.66666667%}.ant-col-xl-offset-22{margin-left:91.66666667%}.ant-col-xl-order-22{order:22}.ant-col-xl-21{display:block;box-sizing:border-box;width:87.5%}.ant-col-xl-push-21{left:87.5%}.ant-col-xl-pull-21{right:87.5%}.ant-col-xl-offset-21{margin-left:87.5%}.ant-col-xl-order-21{order:21}.ant-col-xl-20{display:block;box-sizing:border-box;width:83.33333333%}.ant-col-xl-push-20{left:83.33333333%}.ant-col-xl-pull-20{right:83.33333333%}.ant-col-xl-offset-20{margin-left:83.33333333%}.ant-col-xl-order-20{order:20}.ant-col-xl-19{display:block;box-sizing:border-box;width:79.16666667%}.ant-col-xl-push-19{left:79.16666667%}.ant-col-xl-pull-19{right:79.16666667%}.ant-col-xl-offset-19{margin-left:79.16666667%}.ant-col-xl-order-19{order:19}.ant-col-xl-18{display:block;box-sizing:border-box;width:75%}.ant-col-xl-push-18{left:75%}.ant-col-xl-pull-18{right:75%}.ant-col-xl-offset-18{margin-left:75%}.ant-col-xl-order-18{order:18}.ant-col-xl-17{display:block;box-sizing:border-box;width:70.83333333%}.ant-col-xl-push-17{left:70.83333333%}.ant-col-xl-pull-17{right:70.83333333%}.ant-col-xl-offset-17{margin-left:70.83333333%}.ant-col-xl-order-17{order:17}.ant-col-xl-16{display:block;box-sizing:border-box;width:66.66666667%}.ant-col-xl-push-16{left:66.66666667%}.ant-col-xl-pull-16{right:66.66666667%}.ant-col-xl-offset-16{margin-left:66.66666667%}.ant-col-xl-order-16{order:16}.ant-col-xl-15{display:block;box-sizing:border-box;width:62.5%}.ant-col-xl-push-15{left:62.5%}.ant-col-xl-pull-15{right:62.5%}.ant-col-xl-offset-15{margin-left:62.5%}.ant-col-xl-order-15{order:15}.ant-col-xl-14{display:block;box-sizing:border-box;width:58.33333333%}.ant-col-xl-push-14{left:58.33333333%}.ant-col-xl-pull-14{right:58.33333333%}.ant-col-xl-offset-14{margin-left:58.33333333%}.ant-col-xl-order-14{order:14}.ant-col-xl-13{display:block;box-sizing:border-box;width:54.16666667%}.ant-col-xl-push-13{left:54.16666667%}.ant-col-xl-pull-13{right:54.16666667%}.ant-col-xl-offset-13{margin-left:54.16666667%}.ant-col-xl-order-13{order:13}.ant-col-xl-12{display:block;box-sizing:border-box;width:50%}.ant-col-xl-push-12{left:50%}.ant-col-xl-pull-12{right:50%}.ant-col-xl-offset-12{margin-left:50%}.ant-col-xl-order-12{order:12}.ant-col-xl-11{display:block;box-sizing:border-box;width:45.83333333%}.ant-col-xl-push-11{left:45.83333333%}.ant-col-xl-pull-11{right:45.83333333%}.ant-col-xl-offset-11{margin-left:45.83333333%}.ant-col-xl-order-11{order:11}.ant-col-xl-10{display:block;box-sizing:border-box;width:41.66666667%}.ant-col-xl-push-10{left:41.66666667%}.ant-col-xl-pull-10{right:41.66666667%}.ant-col-xl-offset-10{margin-left:41.66666667%}.ant-col-xl-order-10{order:10}.ant-col-xl-9{display:block;box-sizing:border-box;width:37.5%}.ant-col-xl-push-9{left:37.5%}.ant-col-xl-pull-9{right:37.5%}.ant-col-xl-offset-9{margin-left:37.5%}.ant-col-xl-order-9{order:9}.ant-col-xl-8{display:block;box-sizing:border-box;width:33.33333333%}.ant-col-xl-push-8{left:33.33333333%}.ant-col-xl-pull-8{right:33.33333333%}.ant-col-xl-offset-8{margin-left:33.33333333%}.ant-col-xl-order-8{order:8}.ant-col-xl-7{display:block;box-sizing:border-box;width:29.16666667%}.ant-col-xl-push-7{left:29.16666667%}.ant-col-xl-pull-7{right:29.16666667%}.ant-col-xl-offset-7{margin-left:29.16666667%}.ant-col-xl-order-7{order:7}.ant-col-xl-6{display:block;box-sizing:border-box;width:25%}.ant-col-xl-push-6{left:25%}.ant-col-xl-pull-6{right:25%}.ant-col-xl-offset-6{margin-left:25%}.ant-col-xl-order-6{order:6}.ant-col-xl-5{display:block;box-sizing:border-box;width:20.83333333%}.ant-col-xl-push-5{left:20.83333333%}.ant-col-xl-pull-5{right:20.83333333%}.ant-col-xl-offset-5{margin-left:20.83333333%}.ant-col-xl-order-5{order:5}.ant-col-xl-4{display:block;box-sizing:border-box;width:16.66666667%}.ant-col-xl-push-4{left:16.66666667%}.ant-col-xl-pull-4{right:16.66666667%}.ant-col-xl-offset-4{margin-left:16.66666667%}.ant-col-xl-order-4{order:4}.ant-col-xl-3{display:block;box-sizing:border-box;width:12.5%}.ant-col-xl-push-3{left:12.5%}.ant-col-xl-pull-3{right:12.5%}.ant-col-xl-offset-3{margin-left:12.5%}.ant-col-xl-order-3{order:3}.ant-col-xl-2{display:block;box-sizing:border-box;width:8.33333333%}.ant-col-xl-push-2{left:8.33333333%}.ant-col-xl-pull-2{right:8.33333333%}.ant-col-xl-offset-2{margin-left:8.33333333%}.ant-col-xl-order-2{order:2}.ant-col-xl-1{display:block;box-sizing:border-box;width:4.16666667%}.ant-col-xl-push-1{left:4.16666667%}.ant-col-xl-pull-1{right:4.16666667%}.ant-col-xl-offset-1{margin-left:4.16666667%}.ant-col-xl-order-1{order:1}.ant-col-xl-0{display:none}.ant-col-push-0{left:auto}.ant-col-pull-0{right:auto}.ant-col-xl-push-0{left:auto}.ant-col-xl-pull-0{right:auto}.ant-col-xl-offset-0{margin-left:0}.ant-col-xl-order-0{order:0}}@media (min-width:1600px){.ant-col-xxl-1,.ant-col-xxl-2,.ant-col-xxl-3,.ant-col-xxl-4,.ant-col-xxl-5,.ant-col-xxl-6,.ant-col-xxl-7,.ant-col-xxl-8,.ant-col-xxl-9,.ant-col-xxl-10,.ant-col-xxl-11,.ant-col-xxl-12,.ant-col-xxl-13,.ant-col-xxl-14,.ant-col-xxl-15,.ant-col-xxl-16,.ant-col-xxl-17,.ant-col-xxl-18,.ant-col-xxl-19,.ant-col-xxl-20,.ant-col-xxl-21,.ant-col-xxl-22,.ant-col-xxl-23,.ant-col-xxl-24{flex:0 0 auto;float:left}.ant-col-xxl-24{display:block;box-sizing:border-box;width:100%}.ant-col-xxl-push-24{left:100%}.ant-col-xxl-pull-24{right:100%}.ant-col-xxl-offset-24{margin-left:100%}.ant-col-xxl-order-24{order:24}.ant-col-xxl-23{display:block;box-sizing:border-box;width:95.83333333%}.ant-col-xxl-push-23{left:95.83333333%}.ant-col-xxl-pull-23{right:95.83333333%}.ant-col-xxl-offset-23{margin-left:95.83333333%}.ant-col-xxl-order-23{order:23}.ant-col-xxl-22{display:block;box-sizing:border-box;width:91.66666667%}.ant-col-xxl-push-22{left:91.66666667%}.ant-col-xxl-pull-22{right:91.66666667%}.ant-col-xxl-offset-22{margin-left:91.66666667%}.ant-col-xxl-order-22{order:22}.ant-col-xxl-21{display:block;box-sizing:border-box;width:87.5%}.ant-col-xxl-push-21{left:87.5%}.ant-col-xxl-pull-21{right:87.5%}.ant-col-xxl-offset-21{margin-left:87.5%}.ant-col-xxl-order-21{order:21}.ant-col-xxl-20{display:block;box-sizing:border-box;width:83.33333333%}.ant-col-xxl-push-20{left:83.33333333%}.ant-col-xxl-pull-20{right:83.33333333%}.ant-col-xxl-offset-20{margin-left:83.33333333%}.ant-col-xxl-order-20{order:20}.ant-col-xxl-19{display:block;box-sizing:border-box;width:79.16666667%}.ant-col-xxl-push-19{left:79.16666667%}.ant-col-xxl-pull-19{right:79.16666667%}.ant-col-xxl-offset-19{margin-left:79.16666667%}.ant-col-xxl-order-19{order:19}.ant-col-xxl-18{display:block;box-sizing:border-box;width:75%}.ant-col-xxl-push-18{left:75%}.ant-col-xxl-pull-18{right:75%}.ant-col-xxl-offset-18{margin-left:75%}.ant-col-xxl-order-18{order:18}.ant-col-xxl-17{display:block;box-sizing:border-box;width:70.83333333%}.ant-col-xxl-push-17{left:70.83333333%}.ant-col-xxl-pull-17{right:70.83333333%}.ant-col-xxl-offset-17{margin-left:70.83333333%}.ant-col-xxl-order-17{order:17}.ant-col-xxl-16{display:block;box-sizing:border-box;width:66.66666667%}.ant-col-xxl-push-16{left:66.66666667%}.ant-col-xxl-pull-16{right:66.66666667%}.ant-col-xxl-offset-16{margin-left:66.66666667%}.ant-col-xxl-order-16{order:16}.ant-col-xxl-15{display:block;box-sizing:border-box;width:62.5%}.ant-col-xxl-push-15{left:62.5%}.ant-col-xxl-pull-15{right:62.5%}.ant-col-xxl-offset-15{margin-left:62.5%}.ant-col-xxl-order-15{order:15}.ant-col-xxl-14{display:block;box-sizing:border-box;width:58.33333333%}.ant-col-xxl-push-14{left:58.33333333%}.ant-col-xxl-pull-14{right:58.33333333%}.ant-col-xxl-offset-14{margin-left:58.33333333%}.ant-col-xxl-order-14{order:14}.ant-col-xxl-13{display:block;box-sizing:border-box;width:54.16666667%}.ant-col-xxl-push-13{left:54.16666667%}.ant-col-xxl-pull-13{right:54.16666667%}.ant-col-xxl-offset-13{margin-left:54.16666667%}.ant-col-xxl-order-13{order:13}.ant-col-xxl-12{display:block;box-sizing:border-box;width:50%}.ant-col-xxl-push-12{left:50%}.ant-col-xxl-pull-12{right:50%}.ant-col-xxl-offset-12{margin-left:50%}.ant-col-xxl-order-12{order:12}.ant-col-xxl-11{display:block;box-sizing:border-box;width:45.83333333%}.ant-col-xxl-push-11{left:45.83333333%}.ant-col-xxl-pull-11{right:45.83333333%}.ant-col-xxl-offset-11{margin-left:45.83333333%}.ant-col-xxl-order-11{order:11}.ant-col-xxl-10{display:block;box-sizing:border-box;width:41.66666667%}.ant-col-xxl-push-10{left:41.66666667%}.ant-col-xxl-pull-10{right:41.66666667%}.ant-col-xxl-offset-10{margin-left:41.66666667%}.ant-col-xxl-order-10{order:10}.ant-col-xxl-9{display:block;box-sizing:border-box;width:37.5%}.ant-col-xxl-push-9{left:37.5%}.ant-col-xxl-pull-9{right:37.5%}.ant-col-xxl-offset-9{margin-left:37.5%}.ant-col-xxl-order-9{order:9}.ant-col-xxl-8{display:block;box-sizing:border-box;width:33.33333333%}.ant-col-xxl-push-8{left:33.33333333%}.ant-col-xxl-pull-8{right:33.33333333%}.ant-col-xxl-offset-8{margin-left:33.33333333%}.ant-col-xxl-order-8{order:8}.ant-col-xxl-7{display:block;box-sizing:border-box;width:29.16666667%}.ant-col-xxl-push-7{left:29.16666667%}.ant-col-xxl-pull-7{right:29.16666667%}.ant-col-xxl-offset-7{margin-left:29.16666667%}.ant-col-xxl-order-7{order:7}.ant-col-xxl-6{display:block;box-sizing:border-box;width:25%}.ant-col-xxl-push-6{left:25%}.ant-col-xxl-pull-6{right:25%}.ant-col-xxl-offset-6{margin-left:25%}.ant-col-xxl-order-6{order:6}.ant-col-xxl-5{display:block;box-sizing:border-box;width:20.83333333%}.ant-col-xxl-push-5{left:20.83333333%}.ant-col-xxl-pull-5{right:20.83333333%}.ant-col-xxl-offset-5{margin-left:20.83333333%}.ant-col-xxl-order-5{order:5}.ant-col-xxl-4{display:block;box-sizing:border-box;width:16.66666667%}.ant-col-xxl-push-4{left:16.66666667%}.ant-col-xxl-pull-4{right:16.66666667%}.ant-col-xxl-offset-4{margin-left:16.66666667%}.ant-col-xxl-order-4{order:4}.ant-col-xxl-3{display:block;box-sizing:border-box;width:12.5%}.ant-col-xxl-push-3{left:12.5%}.ant-col-xxl-pull-3{right:12.5%}.ant-col-xxl-offset-3{margin-left:12.5%}.ant-col-xxl-order-3{order:3}.ant-col-xxl-2{display:block;box-sizing:border-box;width:8.33333333%}.ant-col-xxl-push-2{left:8.33333333%}.ant-col-xxl-pull-2{right:8.33333333%}.ant-col-xxl-offset-2{margin-left:8.33333333%}.ant-col-xxl-order-2{order:2}.ant-col-xxl-1{display:block;box-sizing:border-box;width:4.16666667%}.ant-col-xxl-push-1{left:4.16666667%}.ant-col-xxl-pull-1{right:4.16666667%}.ant-col-xxl-offset-1{margin-left:4.16666667%}.ant-col-xxl-order-1{order:1}.ant-col-xxl-0{display:none}.ant-col-push-0{left:auto}.ant-col-pull-0{right:auto}.ant-col-xxl-push-0{left:auto}.ant-col-xxl-pull-0{right:auto}.ant-col-xxl-offset-0{margin-left:0}.ant-col-xxl-order-0{order:0}}.ant-input{box-sizing:border-box;margin:0;font-variant:tabular-nums;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;display:inline-block;width:100%;height:32px;padding:4px 11px;color:#e3e6e8;font-size:14px;line-height:1.5;background-color:#22282a;background-image:none;border:1px solid #5c6970;border-radius:2px;-webkit-transition:all .3s;transition:all .3s}.ant-input::-moz-placeholder{color:#bfbfbf;opacity:1}.ant-input:-ms-input-placeholder{color:#bfbfbf}.ant-input::-webkit-input-placeholder{color:#bfbfbf}.ant-input:focus,.ant-input:hover{border-color:#29bfff;border-right-width:1px!important}.ant-input:focus{outline:0;box-shadow:0 0 0 2px rgba(0,170,255,.2)}.ant-input-disabled{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);cursor:not-allowed;opacity:1}.ant-input-disabled:hover{border-color:#757b7d;border-right-width:1px!important}.ant-input[disabled]{color:hsla(0,0%,100%,.5);background-color:rgba(0,170,255,.2);cursor:not-allowed;opacity:1}.ant-input[disabled]:hover{border-color:#757b7d;border-right-width:1px!important}textarea.ant-input{max-width:100%;height:auto;min-height:32px;vertical-align:bottom;-webkit-transition:all .3s,height 0s;transition:all .3s,height 0s}.ant-input-lg{height:40px;padding:6px 11px;font-size:16px}.ant-input-sm{height:24px;padding:1px 7px}.ant-input-group{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;display:table;width:100%;border-collapse:separate;border-spacing:0}.ant-input-group[class*=col-]{float:none;padding-right:0;padding-left:0}.ant-input-group>[class*=col-]{padding-right:8px}.ant-input-group>[class*=col-]:last-child{padding-right:0}.ant-input-group-addon,.ant-input-group-wrap,.ant-input-group>.ant-input{display:table-cell}.ant-input-group-addon:not(:first-child):not(:last-child),.ant-input-group-wrap:not(:first-child):not(:last-child),.ant-input-group>.ant-input:not(:first-child):not(:last-child){border-radius:0}.ant-input-group-addon,.ant-input-group-wrap{width:1px;white-space:nowrap;vertical-align:middle}.ant-input-group-wrap>*{display:block!important}.ant-input-group .ant-input{float:left;width:100%;margin-bottom:0;text-align:inherit}.ant-input-group .ant-input:focus,.ant-input-group .ant-input:hover{z-index:1;border-right-width:1px}.ant-input-group-addon{position:relative;padding:0 11px;color:#e3e6e8;font-weight:400;font-size:14px;line-height:1;text-align:center;background-color:#5c6970;border:1px solid #5c6970;border-radius:2px;-webkit-transition:all .3s;transition:all .3s}.ant-input-group-addon .ant-select{margin:-5px -11px}.ant-input-group-addon .ant-select .ant-select-selection{margin:-1px;background-color:inherit;border:1px solid transparent;box-shadow:none}.ant-input-group-addon .ant-select-focused .ant-select-selection,.ant-input-group-addon .ant-select-open .ant-select-selection{color:#0af}.ant-input-group-addon>i:only-child:after{position:absolute;top:0;right:0;bottom:0;left:0;content:""}.ant-input-group-addon:first-child,.ant-input-group-addon:first-child .ant-select .ant-select-selection,.ant-input-group>.ant-input:first-child,.ant-input-group>.ant-input:first-child .ant-select .ant-select-selection{border-top-right-radius:0;border-bottom-right-radius:0}.ant-input-group>.ant-input-affix-wrapper:not(:first-child) .ant-input{border-top-left-radius:0;border-bottom-left-radius:0}.ant-input-group>.ant-input-affix-wrapper:not(:last-child) .ant-input{border-top-right-radius:0;border-bottom-right-radius:0}.ant-input-group-addon:first-child{border-right:0}.ant-input-group-addon:last-child{border-left:0}.ant-input-group-addon:last-child,.ant-input-group-addon:last-child .ant-select .ant-select-selection,.ant-input-group>.ant-input:last-child,.ant-input-group>.ant-input:last-child .ant-select .ant-select-selection{border-top-left-radius:0;border-bottom-left-radius:0}.ant-input-group-lg .ant-input,.ant-input-group-lg>.ant-input-group-addon{height:40px;padding:6px 11px;font-size:16px}.ant-input-group-sm .ant-input,.ant-input-group-sm>.ant-input-group-addon{height:24px;padding:1px 7px}.ant-input-group-lg .ant-select-selection--single{height:40px}.ant-input-group-sm .ant-select-selection--single{height:24px}.ant-input-group .ant-input-affix-wrapper{display:table-cell;float:left;width:100%}.ant-input-group.ant-input-group-compact{display:block;zoom:1}.ant-input-group.ant-input-group-compact:after,.ant-input-group.ant-input-group-compact:before{display:table;content:""}.ant-input-group.ant-input-group-compact:after{clear:both}.ant-input-group.ant-input-group-compact-addon:not(:first-child):not(:last-child),.ant-input-group.ant-input-group-compact-wrap:not(:first-child):not(:last-child),.ant-input-group.ant-input-group-compact>.ant-input:not(:first-child):not(:last-child){border-right-width:1px}.ant-input-group.ant-input-group-compact-addon:not(:first-child):not(:last-child):focus,.ant-input-group.ant-input-group-compact-addon:not(:first-child):not(:last-child):hover,.ant-input-group.ant-input-group-compact-wrap:not(:first-child):not(:last-child):focus,.ant-input-group.ant-input-group-compact-wrap:not(:first-child):not(:last-child):hover,.ant-input-group.ant-input-group-compact>.ant-input:not(:first-child):not(:last-child):focus,.ant-input-group.ant-input-group-compact>.ant-input:not(:first-child):not(:last-child):hover{z-index:1}.ant-input-group.ant-input-group-compact>*{display:inline-block;float:none;vertical-align:top;border-radius:0}.ant-input-group.ant-input-group-compact>:not(:last-child){margin-right:-1px;border-right-width:1px}.ant-input-group.ant-input-group-compact .ant-input{float:none}.ant-input-group.ant-input-group-compact>.ant-calendar-picker .ant-input,.ant-input-group.ant-input-group-compact>.ant-cascader-picker .ant-input,.ant-input-group.ant-input-group-compact>.ant-mention-wrapper .ant-mention-editor,.ant-input-group.ant-input-group-compact>.ant-select-auto-complete .ant-input,.ant-input-group.ant-input-group-compact>.ant-select>.ant-select-selection,.ant-input-group.ant-input-group-compact>.ant-time-picker .ant-time-picker-input{border-right-width:1px;border-radius:0}.ant-input-group.ant-input-group-compact>.ant-calendar-picker .ant-input:focus,.ant-input-group.ant-input-group-compact>.ant-calendar-picker .ant-input:hover,.ant-input-group.ant-input-group-compact>.ant-cascader-picker .ant-input:focus,.ant-input-group.ant-input-group-compact>.ant-cascader-picker .ant-input:hover,.ant-input-group.ant-input-group-compact>.ant-mention-wrapper .ant-mention-editor:focus,.ant-input-group.ant-input-group-compact>.ant-mention-wrapper .ant-mention-editor:hover,.ant-input-group.ant-input-group-compact>.ant-select-auto-complete .ant-input:focus,.ant-input-group.ant-input-group-compact>.ant-select-auto-complete .ant-input:hover,.ant-input-group.ant-input-group-compact>.ant-select>.ant-select-selection:focus,.ant-input-group.ant-input-group-compact>.ant-select>.ant-select-selection:hover,.ant-input-group.ant-input-group-compact>.ant-time-picker .ant-time-picker-input:focus,.ant-input-group.ant-input-group-compact>.ant-time-picker .ant-time-picker-input:hover{z-index:1}.ant-input-group.ant-input-group-compact>.ant-calendar-picker:first-child .ant-input,.ant-input-group.ant-input-group-compact>.ant-cascader-picker:first-child .ant-input,.ant-input-group.ant-input-group-compact>.ant-mention-wrapper:first-child .ant-mention-editor,.ant-input-group.ant-input-group-compact>.ant-select-auto-complete:first-child .ant-input,.ant-input-group.ant-input-group-compact>.ant-select:first-child>.ant-select-selection,.ant-input-group.ant-input-group-compact>.ant-time-picker:first-child .ant-time-picker-input,.ant-input-group.ant-input-group-compact>:first-child{border-top-left-radius:2px;border-bottom-left-radius:2px}.ant-input-group.ant-input-group-compact>.ant-calendar-picker:last-child .ant-input,.ant-input-group.ant-input-group-compact>.ant-cascader-picker-focused:last-child .ant-input,.ant-input-group.ant-input-group-compact>.ant-cascader-picker:last-child .ant-input,.ant-input-group.ant-input-group-compact>.ant-mention-wrapper:last-child .ant-mention-editor,.ant-input-group.ant-input-group-compact>.ant-select-auto-complete:last-child .ant-input,.ant-input-group.ant-input-group-compact>.ant-select:last-child>.ant-select-selection,.ant-input-group.ant-input-group-compact>.ant-time-picker:last-child .ant-time-picker-input,.ant-input-group.ant-input-group-compact>:last-child{border-right-width:1px;border-top-right-radius:2px;border-bottom-right-radius:2px}.ant-input-group.ant-input-group-compact>.ant-select-auto-complete .ant-input{vertical-align:top}.ant-input-group-wrapper{display:inline-block;width:100%;text-align:start;vertical-align:top}.ant-input-affix-wrapper{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";position:relative;display:inline-block;width:100%;text-align:start}.ant-input-affix-wrapper:hover .ant-input:not(.ant-input-disabled){border-color:#29bfff;border-right-width:1px!important}.ant-input-affix-wrapper .ant-input{position:relative;text-align:inherit}.ant-input-affix-wrapper .ant-input-prefix,.ant-input-affix-wrapper .ant-input-suffix{position:absolute;top:50%;z-index:2;color:#e3e6e8;line-height:0;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.ant-input-affix-wrapper .ant-input-prefix :not(.anticon),.ant-input-affix-wrapper .ant-input-suffix :not(.anticon){line-height:1.5}.ant-input-affix-wrapper .ant-input-prefix{left:12px}.ant-input-affix-wrapper .ant-input-suffix{right:12px}.ant-input-affix-wrapper .ant-input:not(:first-child){padding-left:30px}.ant-input-affix-wrapper .ant-input:not(:last-child){padding-right:30px}.ant-input-affix-wrapper .ant-input{min-height:100%}.ant-input-password-icon{color:#c2cfd6;cursor:pointer;-webkit-transition:all .3s;transition:all .3s}.ant-input-password-icon:hover{color:#333}.ant-input-clear-icon{color:hsla(0,0%,100%,.5);font-size:12px;vertical-align:top;cursor:pointer;-webkit-transition:color .3s;transition:color .3s}.ant-input-clear-icon:hover{color:#c2cfd6}.ant-input-clear-icon:active{color:#e3e6e8}.ant-input-clear-icon+i{margin-left:6px}.ant-input-search-icon{color:#c2cfd6;cursor:pointer;-webkit-transition:all .3s;transition:all .3s}.ant-input-search-icon:hover{color:hsla(0,0%,100%,.8)}.ant-input-search-enter-button input{border-right:0}.ant-input-search-enter-button+.ant-input-group-addon,.ant-input-search-enter-button input+.ant-input-group-addon{padding:0;border:0}.ant-input-search-enter-button+.ant-input-group-addon .ant-input-search-button,.ant-input-search-enter-button input+.ant-input-group-addon .ant-input-search-button{width:100%;border-top-left-radius:0;border-bottom-left-radius:0}.ant-upload{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";outline:0}.ant-upload p{margin:0}.ant-upload-btn{display:block;width:100%;outline:none}.ant-upload input[type=file]{cursor:pointer}.ant-upload.ant-upload-select{display:inline-block}.ant-upload.ant-upload-disabled{cursor:not-allowed}.ant-upload.ant-upload-select-picture-card{display:table;width:104px;height:104px;margin-right:8px;margin-bottom:8px;text-align:center;vertical-align:top;background-color:#5c6970;border:1px dashed #5c6970;border-radius:2px;cursor:pointer;-webkit-transition:border-color .3s ease;transition:border-color .3s ease}.ant-upload.ant-upload-select-picture-card>.ant-upload{display:table-cell;width:100%;height:100%;padding:8px;text-align:center;vertical-align:middle}.ant-upload.ant-upload-select-picture-card:hover{border-color:#0af}.ant-upload.ant-upload-drag{position:relative;width:100%;height:100%;text-align:center;background:#5c6970;border:1px dashed #5c6970;border-radius:2px;cursor:pointer;-webkit-transition:border-color .3s;transition:border-color .3s}.ant-upload.ant-upload-drag .ant-upload{padding:16px 0}.ant-upload.ant-upload-drag.ant-upload-drag-hover:not(.ant-upload-disabled){border-color:#0089d9}.ant-upload.ant-upload-drag.ant-upload-disabled{cursor:not-allowed}.ant-upload.ant-upload-drag .ant-upload-btn{display:table;height:100%}.ant-upload.ant-upload-drag .ant-upload-drag-container{display:table-cell;vertical-align:middle}.ant-upload.ant-upload-drag:not(.ant-upload-disabled):hover{border-color:#29bfff}.ant-upload.ant-upload-drag p.ant-upload-drag-icon{margin-bottom:20px}.ant-upload.ant-upload-drag p.ant-upload-drag-icon .anticon{color:#29bfff;font-size:48px}.ant-upload.ant-upload-drag p.ant-upload-text{margin:0 0 4px;color:hsla(0,0%,100%,.85);font-size:16px}.ant-upload.ant-upload-drag p.ant-upload-hint{color:#c2cfd6;font-size:14px}.ant-upload.ant-upload-drag .anticon-plus{color:hsla(0,0%,100%,.5);font-size:30px;-webkit-transition:all .3s;transition:all .3s}.ant-upload.ant-upload-drag .anticon-plus:hover,.ant-upload.ant-upload-drag:hover .anticon-plus{color:#c2cfd6}.ant-upload-list{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";zoom:1}.ant-upload-list:after,.ant-upload-list:before{display:table;content:""}.ant-upload-list:after{clear:both}.ant-upload-list-item{position:relative;height:22px;margin-top:8px;font-size:14px}.ant-upload-list-item-name{display:inline-block;width:100%;padding-left:22px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.ant-upload-list-item-info{height:100%;padding:0 12px 0 4px;-webkit-transition:background-color .3s;transition:background-color .3s}.ant-upload-list-item-info>span{display:block}.ant-upload-list-item-info .anticon-loading,.ant-upload-list-item-info .anticon-paper-clip{position:absolute;top:5px;color:#c2cfd6;font-size:14px}.ant-upload-list-item .anticon-close{display:inline-block;font-size:12px;font-size:10px\9;-webkit-transform:scale(.83333333) rotate(0deg);transform:scale(.83333333) rotate(0deg);position:absolute;top:6px;right:4px;color:#c2cfd6;line-height:0;cursor:pointer;opacity:0;-webkit-transition:all .3s;transition:all .3s}:root .ant-upload-list-item .anticon-close{font-size:12px}.ant-upload-list-item .anticon-close:hover{color:#e3e6e8}.ant-upload-list-item:hover .ant-upload-list-item-info{background-color:rgba(0,170,255,.1)}.ant-upload-list-item:hover .anticon-close{opacity:1}.ant-upload-list-item-error,.ant-upload-list-item-error .ant-upload-list-item-name,.ant-upload-list-item-error .anticon-paper-clip{color:#f5222d}.ant-upload-list-item-error .anticon-close{color:#f5222d!important;opacity:1}.ant-upload-list-item-progress{position:absolute;bottom:-12px;width:100%;padding-left:26px;font-size:14px;line-height:0}.ant-upload-list-picture-card .ant-upload-list-item,.ant-upload-list-picture .ant-upload-list-item{position:relative;height:66px;padding:8px;border:1px solid #5c6970;border-radius:2px}.ant-upload-list-picture-card .ant-upload-list-item:hover,.ant-upload-list-picture .ant-upload-list-item:hover{background:transparent}.ant-upload-list-picture-card .ant-upload-list-item-error,.ant-upload-list-picture .ant-upload-list-item-error{border-color:#f5222d}.ant-upload-list-picture-card .ant-upload-list-item-info,.ant-upload-list-picture .ant-upload-list-item-info{padding:0}.ant-upload-list-picture-card .ant-upload-list-item:hover .ant-upload-list-item-info,.ant-upload-list-picture .ant-upload-list-item:hover .ant-upload-list-item-info{background:transparent}.ant-upload-list-picture-card .ant-upload-list-item-uploading,.ant-upload-list-picture .ant-upload-list-item-uploading{border-style:dashed}.ant-upload-list-picture-card .ant-upload-list-item-thumbnail,.ant-upload-list-picture .ant-upload-list-item-thumbnail{position:absolute;top:8px;left:8px;width:48px;height:48px;font-size:26px;line-height:54px;text-align:center;opacity:.8}.ant-upload-list-picture-card .ant-upload-list-item-icon,.ant-upload-list-picture .ant-upload-list-item-icon{position:absolute;top:50%;left:50%;font-size:26px;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%)}.ant-upload-list-picture-card .ant-upload-list-item-thumbnail img,.ant-upload-list-picture .ant-upload-list-item-thumbnail img{display:block;width:48px;height:48px;overflow:hidden}.ant-upload-list-picture-card .ant-upload-list-item-name,.ant-upload-list-picture .ant-upload-list-item-name{display:inline-block;box-sizing:border-box;max-width:100%;margin:0 0 0 8px;padding-right:8px;padding-left:48px;overflow:hidden;line-height:44px;white-space:nowrap;text-overflow:ellipsis;-webkit-transition:all .3s;transition:all .3s}.ant-upload-list-picture-card .ant-upload-list-item-uploading .ant-upload-list-item-name,.ant-upload-list-picture .ant-upload-list-item-uploading .ant-upload-list-item-name{line-height:28px}.ant-upload-list-picture-card .ant-upload-list-item-progress,.ant-upload-list-picture .ant-upload-list-item-progress{bottom:14px;width:calc(100% - 24px);margin-top:0;padding-left:56px}.ant-upload-list-picture-card .anticon-close,.ant-upload-list-picture .anticon-close{position:absolute;top:8px;right:8px;line-height:1;opacity:1}.ant-upload-list-picture-card{float:left}.ant-upload-list-picture-card.ant-upload-list:after{display:none}.ant-upload-list-picture-card .ant-upload-list-item{float:left;width:104px;height:104px;margin:0 8px 8px 0}.ant-upload-list-picture-card .ant-upload-list-item-info{position:relative;height:100%;overflow:hidden}.ant-upload-list-picture-card .ant-upload-list-item-info:before{position:absolute;z-index:1;width:100%;height:100%;background-color:hsla(0,0%,100%,.5);opacity:0;-webkit-transition:all .3s;transition:all .3s;content:" "}.ant-upload-list-picture-card .ant-upload-list-item:hover .ant-upload-list-item-info:before{opacity:1}.ant-upload-list-picture-card .ant-upload-list-item-actions{position:absolute;top:50%;left:50%;z-index:10;white-space:nowrap;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%);opacity:0;-webkit-transition:all .3s;transition:all .3s}.ant-upload-list-picture-card .ant-upload-list-item-actions .anticon-delete,.ant-upload-list-picture-card .ant-upload-list-item-actions .anticon-eye-o{z-index:10;width:16px;margin:0 4px;color:hsla(0,0%,100%,.85);font-size:16px;cursor:pointer;-webkit-transition:all .3s;transition:all .3s}.ant-upload-list-picture-card .ant-upload-list-item-actions .anticon-delete:hover,.ant-upload-list-picture-card .ant-upload-list-item-actions .anticon-eye-o:hover{color:#000}.ant-upload-list-picture-card .ant-upload-list-item-actions:hover,.ant-upload-list-picture-card .ant-upload-list-item-info:hover+.ant-upload-list-item-actions{opacity:1}.ant-upload-list-picture-card .ant-upload-list-item-thumbnail,.ant-upload-list-picture-card .ant-upload-list-item-thumbnail img{position:static;display:block;width:100%;height:100%}.ant-upload-list-picture-card .ant-upload-list-item-name{display:none;margin:8px 0 0;padding:0;line-height:1.5;text-align:center}.ant-upload-list-picture-card .anticon-picture+.ant-upload-list-item-name{display:block}.ant-upload-list-picture-card .ant-upload-list-item-uploading.ant-upload-list-item{background-color:#5c6970}.ant-upload-list-picture-card .ant-upload-list-item-uploading .ant-upload-list-item-info{height:auto}.ant-upload-list-picture-card .ant-upload-list-item-uploading .ant-upload-list-item-info .anticon-delete,.ant-upload-list-picture-card .ant-upload-list-item-uploading .ant-upload-list-item-info .anticon-eye-o,.ant-upload-list-picture-card .ant-upload-list-item-uploading .ant-upload-list-item-info:before{display:none}.ant-upload-list-picture-card .ant-upload-list-item-uploading-text{margin-top:18px;color:#c2cfd6}.ant-upload-list-picture-card .ant-upload-list-item-progress{bottom:32px;padding-left:0}.ant-upload-list .ant-upload-success-icon{color:#52c41a;font-weight:700}.ant-upload-list .ant-upload-animate-enter,.ant-upload-list .ant-upload-animate-inline-enter,.ant-upload-list .ant-upload-animate-inline-leave,.ant-upload-list .ant-upload-animate-leave{-webkit-animation-duration:.3s;animation-duration:.3s;-webkit-animation-fill-mode:cubic-bezier(.78,.14,.15,.86);animation-fill-mode:cubic-bezier(.78,.14,.15,.86)}.ant-upload-list .ant-upload-animate-enter{-webkit-animation-name:uploadAnimateIn;animation-name:uploadAnimateIn}.ant-upload-list .ant-upload-animate-leave{-webkit-animation-name:uploadAnimateOut;animation-name:uploadAnimateOut}.ant-upload-list .ant-upload-animate-inline-enter{-webkit-animation-name:uploadAnimateInlineIn;animation-name:uploadAnimateInlineIn}.ant-upload-list .ant-upload-animate-inline-leave{-webkit-animation-name:uploadAnimateInlineOut;animation-name:uploadAnimateInlineOut}@-webkit-keyframes uploadAnimateIn{0%{height:0;margin:0;padding:0;opacity:0}}@keyframes uploadAnimateIn{0%{height:0;margin:0;padding:0;opacity:0}}@-webkit-keyframes uploadAnimateOut{to{height:0;margin:0;padding:0;opacity:0}}@keyframes uploadAnimateOut{to{height:0;margin:0;padding:0;opacity:0}}@-webkit-keyframes uploadAnimateInlineIn{0%{width:0;height:0;margin:0;padding:0;opacity:0}}@keyframes uploadAnimateInlineIn{0%{width:0;height:0;margin:0;padding:0;opacity:0}}@-webkit-keyframes uploadAnimateInlineOut{to{width:0;height:0;margin:0;padding:0;opacity:0}}@keyframes uploadAnimateInlineOut{to{width:0;height:0;margin:0;padding:0;opacity:0}}.ant-progress{box-sizing:border-box;margin:0;padding:0;color:#e3e6e8;font-size:14px;font-variant:tabular-nums;line-height:1.5;list-style:none;-webkit-font-feature-settings:"tnum","tnum";font-feature-settings:"tnum","tnum";display:inline-block}.ant-progress-line{position:relative;width:100%;font-size:14px}.ant-progress-small.ant-progress-line,.ant-progress-small.ant-progress-line .ant-progress-text .anticon{font-size:12px}.ant-progress-outer{display:inline-block;width:100%;margin-right:0;padding-right:0}.ant-progress-show-info .ant-progress-outer{margin-right:calc(-2em - 8px);padding-right:calc(2em + 8px)}.ant-progress-inner{position:relative;display:inline-block;width:100%;vertical-align:middle;background-color:rgba(0,170,255,.2);border-radius:100px}.ant-progress-circle-trail{stroke:rgba(0,170,255,.2)}.ant-progress-circle-path{-webkit-animation:ant-progress-appear .3s;animation:ant-progress-appear .3s;stroke:#1890ff}.ant-progress-bg,.ant-progress-success-bg{position:relative;background-color:#1890ff;-webkit-transition:all .4s cubic-bezier(.08,.82,.17,1) 0s;transition:all .4s cubic-bezier(.08,.82,.17,1) 0s}.ant-progress-success-bg{position:absolute;top:0;left:0;background-color:#52c41a}.ant-progress-text{display:inline-block;width:2em;margin-left:8px;color:#c2cfd6;font-size:1em;line-height:1;white-space:nowrap;text-align:left;vertical-align:middle;word-break:normal}.ant-progress-text .anticon{font-size:14px}.ant-progress-status-active .ant-progress-bg:before{position:absolute;top:0;right:0;bottom:0;left:0;background:#2e3538;border-radius:10px;opacity:0;-webkit-animation:ant-progress-active 2.4s cubic-bezier(.23,1,.32,1) infinite;animation:ant-progress-active 2.4s cubic-bezier(.23,1,.32,1) infinite;content:""}.ant-progress-status-exception .ant-progress-bg{background-color:#f5222d}.ant-progress-status-exception .ant-progress-text{color:#f5222d}.ant-progress-status-exception .ant-progress-circle-path{stroke:#f5222d}.ant-progress-status-success .ant-progress-bg{background-color:#52c41a}.ant-progress-status-success .ant-progress-text{color:#52c41a}.ant-progress-status-success .ant-progress-circle-path{stroke:#52c41a}.ant-progress-circle .ant-progress-inner{position:relative;line-height:1;background-color:transparent}.ant-progress-circle .ant-progress-text{position:absolute;top:50%;left:50%;width:100%;margin:0;padding:0;color:#e3e6e8;line-height:1;white-space:normal;text-align:center;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%)}.ant-progress-circle .ant-progress-text .anticon{font-size:1.16666667em}.ant-progress-circle.ant-progress-status-exception .ant-progress-text{color:#f5222d}.ant-progress-circle.ant-progress-status-success .ant-progress-text{color:#52c41a}@-webkit-keyframes ant-progress-active{0%{width:0;opacity:.1}20%{width:0;opacity:.5}to{width:100%;opacity:0}}@keyframes ant-progress-active{0%{width:0;opacity:.1}20%{width:0;opacity:.5}to{width:100%;opacity:0}}.ReactVirtualized__Table__headerRow{font-weight:700;text-transform:uppercase}.ReactVirtualized__Table__headerRow,.ReactVirtualized__Table__row{display:flex;flex-direction:row;align-items:center}.ReactVirtualized__Table__headerTruncatedText{display:inline-block;max-width:100%;white-space:nowrap;text-overflow:ellipsis;overflow:hidden}.ReactVirtualized__Table__headerColumn,.ReactVirtualized__Table__rowColumn{margin-right:10px;min-width:0}.ReactVirtualized__Table__rowColumn{text-overflow:ellipsis;white-space:nowrap}.ReactVirtualized__Table__headerColumn:first-of-type,.ReactVirtualized__Table__rowColumn:first-of-type{margin-left:10px}.ReactVirtualized__Table__sortableHeaderColumn{cursor:pointer}.ReactVirtualized__Table__sortableHeaderIconContainer{display:flex;align-items:center}.ReactVirtualized__Table__sortableHeaderIcon{flex:0 0 24px;height:1em;width:1em;fill:currentColor} +/*# sourceMappingURL=2.5fcf4e3c.chunk.css.map */ \ No newline at end of file diff --git a/static/css/2.5fcf4e3c.chunk.css.map b/static/css/2.5fcf4e3c.chunk.css.map new file mode 100644 index 00000000..b655f2e9 --- /dev/null +++ b/static/css/2.5fcf4e3c.chunk.css.map @@ -0,0 +1 @@ +{"version":3,"sources":["C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/color/bezierEasing.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/mixins/size.less","index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/base.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/mixins/clearfix.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/mixins/iconfont.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/iconfont.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/mixins/motion.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/motion/fade.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/motion/move.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/motion/other.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/motion/slide.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/motion/swing.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/motion/zoom.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/core/motion.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/mixins/reset.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/menu/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/menu/style/dark.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/color/tinyColor.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/tooltip/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/alert/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/tabs/style/card-style.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/tabs/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/input/style/mixin.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/input-number/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/style/mixins/compatibility.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/select/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/empty/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/button/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/button/style/mixin.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/modal/style/modal.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/modal/style/confirm.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/form/style/mixin.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/form/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/grid/style/mixin.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/grid/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/input/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/input/style/search-input.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/upload/style/index.less","C:/Users/Daan/code/phantasmal-world/node_modules/antd/es/progress/style/index.less","source/styles.css","styles.css"],"names":[],"mappings":"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B6HN,CgB9WC,yHIQC,QAAA,CACA,mBAAA,CACA,eAAA,CACA,SAAA,CACA,mBpB0WF,CgBtXC,qIIcG,iBpB4WJ,CgB1XC,uHIyPK,YAAA,CACA,kBAAA,CACA,iEAAA,CAAA,yDAAA,CACA,uBpBqIN,CgBjYC,2DImQG,WAAA,CACA,epBkIJ,CoBjII,0JAEE,UAAA,CACA,QAAA,CACA,gJAAA,CAAA,wIpBqIN,CgB9YC,0PI6QK,UAAA,CACA,WpBuIN,CgBrZC,uFIkRK,aAAA,CACA,UAAA,CACA,eAAA,CACA,gBpBuIN,CoBrIM,6GACE,epBwIR,CgBhaC,2GI6RK,iBpBuIN,CgBpaC,qGIiSK,UpBuIN,CgBxaC,4MIsSK,WpBwIN,CgB9aC,2GI0SK,epBwIN,CgBlbC,6KI6SO,cpByIR,CgBtbC,iGIkTK,epBwIN,CgB1bC,uFIsTK,UpBwIN,CgB9bC,+FI0TK,KAAA,CACA,WAAA,CACA,SAAA,CACA,SAAA,CACA,WpBwIN,CgBtcC,iGIkUK,QAAA,CACA,UAAA,CACA,WpBwIN,CgB5cC,iGIwUK,KAAA,CACA,UAAA,CACA,WpBwIN,CgBldC,mEIiVG,UAAA,CACA,sBAAA,CACA,epBqIJ,CgBxdC,6BIwVG,UAAA,CACA,iBAAA,CACA,eAAA,CACA,8BpBmIJ,CgB9dC,2CI6VK,gBpBoIN,CgBjeC,qGImWK,iBpBoIN,CgBveC,+CIsWK,SpBoIN,CgB1eC,iCI0WG,iBAAA,CACA,6BpBmIJ,CgB9eC,8BIgXG,WAAA,CACA,eAAA,CACA,gBAAA,CACA,6BpBiIJ,CgBpfC,uGIwXK,gBpBkIN,CgB1fC,gDI2XK,QpBkIN,CgB7fC,kCI+XG,kBAAA,CACA,8BpBiIJ,CgBjgBC,qFIsYC,wJAAA,CAAA,gJAAA,CAAA,wIAAA,CAAA,6LpB+HF,CgBrgBC,qFI2YC,wJAAA,CAAA,gJAAA,CAAA,wIAAA,CAAA,6LpB8HF,CgBzgBC,0HIiZG,uBAAA,CACA,gCAAA,CAAA,wBpB4HJ,CgB9gBC,0HIQC,QAAA,CACA,mBAAA,CACA,eAAA,CACA,SAAA,CACA,mBpB0gBF,CgBthBC,sIIcG,iBpB4gBJ,CgB1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A,UAAA,CACA,WAAA,CACA,SAAA,CACA,SAAA,CACA,YAAA,CACA,kBAAA,CACA,qBAAA,CACA,WAAA,CACA,8BAAA,CAAA,sBAAA,CACA,UAAA,CACA,mB1Bw0BJ,CFz6BC,kB4BqGG,iEAAA,CAAA,yD1Bu0BJ,CF56BC,uE4B4GO,6B1Bo0BR,C0B/zBE,yBACE,iBAAA,CACA,mB1Bi0BJ,C0B9zBE,gCACE,a1Bg0BJ,C0B7zBE,mGACE,iB1B+zBJ,C0Bh0BE,6HAGI,iB1Bg0BN,C0B5zBE,sGACE,iB1B8zBJ,C0B/zBE,+GAGI,iB1B+zBN,C0B3zBE,eCJA,oB3Bm0BF,C0B/zBE,oECLA,iB3Bw0BF,C2Bn0BI,wQAIE,S3By0BN,C2Bv0BI,uEACE,S3B00BN,CFz9BC,2D6BGC,WAAA,CACA,cAAA,CACA,cAAA,CACA,eAAA,CAgJE,gB3B20BJ,CFj+BC,2D6BGC,WAAA,CACA,aAAA,CACA,cAAA,CACA,eAAA,CAqJE,gB3B80BJ,CFz+BC,6E6B6JK,c3Bg1BN,C0Br2BE,kMCmJE,gB3B2tBJ,C0B92BE,+ECsJE,6B3B2tBJ,C0Bj3BE,wBCyJE,e3B2tBJ,C0Bp3BE,6EC6JE,a3B2tBJ,C0Bx3BE,2ECmKE,iB3B2tBJ,C0B93BE,+GCuKE,0BAAA,CACA,6B3B2tBJ,C0Bn4BE,+GC4KE,2BAAA,CACA,8B3B2tBJ,C2BztBE,iFAKI,iB3B0tBN,C2B/tBE,qHASI,0BAAA,CACA,6B3B0tBN,C2BpuBE,qHAcI,2BAAA,CACA,8B3B0tBN,C2BvtBE,8BACE,U3BytBJ,CFniCC,0E6B6UG,e3BytBJ,C2BvtBE,+EAEI,iBAAA,CACA,yBAAA,CACA,4B3BwtBN,CF3iCC,gF6BuVG,gBAAA,CACA,wBAAA,CACA,2B3ButBJ,C0Bp6BE,6EACE,iBAAA,CACA,gB1Bs6BJ,C0Bl6BE,yCAEE,iB1Bo6BJ,CFxjCC,8C4B0JG,e1Bk6BJ,C0B/5BE,0BACE,aAAA,CACA,gCAAA,CACA,oB1Bi6BJ,C0B95BE,0CCnDA,UAAA,CACA,4BAAA,CACA,iBAAA,CAxBA,gB3B6+BF,C0Bp6BE,uDC7CE,kB3Bo9BJ,C2Bn9BI,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wCrBwcF,CF/dC,gFgC0CK,oB9BwbN,C+BpBA,+BD/ZI,a9BsbJ,C+BvBA,oCD3ZI,aAAA,CACA,wBAAA,CACA,oB9BqbJ,C+B5BA,2BDrZI,a9BobJ,CF7eC,uDiCkdG,aAAA,CACA,4CAAA,CAAA,oC/B8BJ,C+BvBM,4EACE,oB/B4BR,CFvfC,2GuBoBC,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,wCrBueF,C+BhDA,iMA4BI,a/B2BJ,C+BvDA,mEAkCI,oB/ByBJ,C+BxBI,kKV7dF,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,wCrB2fF,C+B7BI,+GACE,oB/BgCN,C+BxEA,4DV1bE,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,wCrBqgBF,C+B7BA,wDD1fI,a9B2hBJ,C8BvhBI,kDAEE,oB9ByhBN,C8BthBI,4BTMF,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,uCrBmhBF,C8BxhBI,4CACE,oB9B0hBN,C+B9CA,gEV3eE,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,uCrB4hBF,C8BrhBM,oGAEE,wBAAA,CACA,oB9BuhBR,C8BphBM,qDThBJ,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,uCrBuiBF,CF9jBC,8EgC0CK,oB9BuhBN,C+BlEA,6BDhdI,a9BqhBJ,C+BrEA,kCD5cI,aAAA,CACA,wBAAA,CACA,oB9BohBJ,C+B1EA,yBDtcI,a9BmhBJ,CF5kBC,qDiCmgBG,aAAA,CACA,4CAAA,CAAA,oC/B4EJ,C+BrEM,wEACE,oB/B0ER,CFtlBC,uGuBoBC,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,uCrBskBF,C+B9FA,iEAwBM,oB/ByEN,C+BpEI,wDACE,wBAAA,CACA,e/BsEN,C+BrGA,uLAyCI,a/BmEJ,C+B5GA,+DA+CI,oB/BiEJ,C+BhEI,0JV3hBF,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,uCrBimBF,C+B/DM,yOAEE,oB/BqER,C+BjIA,2NV3eE,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,uCrBsnBF,C+BnEI,8BACE,oB/BqEN,C+BnEM,qDACE,oB/BqER,C+BnEQ,2DVpjBN,oBAAA,CACA,gCrB0nBF,C+BnEQ,2DVhkBN,oBAAA,CACA,gCAAA,CACA,SAAA,CACA,uCrBsoBF,CF7pBC,yDiC8lBG,oBAAA,CACA,U/BkEJ,CFjqBC,yCiCqmBG,kB/B+DJ,C+B7DI,mDACE,iB/B+DN,CFvqBC,oDOQC,8BAAA,CAAA,sBAAA,CACA,gCAAA,CAAA,wBAAA,CAaE,mCAAA,CAAA,2BL4pBJ,CFlrBC,kFO0BG,oCAAA,CAAA,4BAAA,CACA,oCAAA,CAAA,4BL4pBJ,CFvrBC,wCO8BG,qCAAA,CAAA,6BAAA,CACA,oCAAA,CAAA,4BAAA,CACA,mBL4pBJ,CF5rBC,mCiCinBG,S/BgFJ,CFjsBC,oDiCknBG,gEAAA,CAAA,wD/BkFJ,C+BzEA,iCACE,GACE,kCAAA,CAAA,0BAAA,CACA,S/B2EF,C+BzEA,GACE,+BAAA,CAAA,uBAAA,CACA,S/B2EF,CACF,C+BnFA,yBACE,GACE,kCAAA,CAAA,0BAAA,CACA,S/B2EF,C+BzEA,GACE,+BAAA,CAAA,uBAAA,CACA,S/B2EF,CACF,C+BxEA,kCACE,GACE,kCAAA,CAAA,0BAAA,CACA,S/B0EF,CACF,C+B9EA,0BACE,GACE,kCAAA,CAAA,0BAAA,CACA,S/B0EF,CACF,C+BrEA,+BACE,GACE,0BAAA,CAAA,kB/BuEF,C+BrEA,GACE,0BAAA,CAAA,kB/BuEF,CACF,C+B7EA,uBACE,GACE,0BAAA,CAAA,kB/BuEF,C+BrEA,GACE,0BAAA,CAAA,kB/BuEF,CACF,C+BpEA,+BACE,GACE,0BAAA,CAAA,kB/BsEF,C+BpEA,GACE,0BAAA,CAAA,kB/BsEF,CACF,C+B5EA,uBACE,GACE,0BAAA,CAAA,kB/BsEF,C+BpEA,GACE,0BAAA,CAAA,kB/BsEF,CACF,C+BnEA,+BACE,GACE,0BAAA,CAAA,kB/BqEF,C+BnEA,GACE,0BAAA,CAAA,kB/BqEF,CACF,C+B3EA,uBACE,GACE,0BAAA,CAAA,kB/BqEF,C+BnEA,GACE,0BAAA,CAAA,kB/BqEF,CACF,CgB5uBC,SgBKC,iBAAA,CACA,WAAA,CACA,cAAA,CACA,aAAA,C9BLA,MAAA,C+BKA,aAAA,CACA,qBjCGF,CERE,+BAEE,aAAA,CACA,UFUJ,CERE,eACE,UFUJ,CgBpBC,ciBaC,YAAA,CACA,kBjCUF,CiCRE,yCAEE,YjCUJ,CgB5BC,oBiBwBC,0BjCOF,CgB/BC,qBiB6BC,sBjCKF,CgBlCC,kBiBkCC,wBjCGF,CgBrCC,4BiBuCC,6BjCCF,CgBxCC,2BiB4CC,4BjCDF,CgB3CC,kBiBiDC,sBjCHF,CgB9CC,qBiBsDC,kBjCLF,CgBjDC,qBiB2DC,oBjCPF,CgBpDC,SiB+DC,iBjCRF,CgBvDC,mpDgBuBK,iBAAA,CAEA,cAAA,CACA,eAAA,CACA,chCyJN,CgBpLC,uRgB+CK,aAAA,CACA,UhC+JN,CgB/MC,YgBwDG,aAAA,CACA,qBAAA,CACA,UhC0JJ,CgBpNC,iBgB6DG,ShC0JJ,CgBvNC,iBgBgEG,UhC0JJ,CgB1NC,mBgBmEG,gBhC0JJ,CgB7NC,kBgBsEG,QhC0JJ,CgBhOC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhC2KJ,CgBrOC,iBgB6DG,iBhC2KJ,CgBxOC,iBgBgEG,kBhC2KJ,CgB3OC,mBgBmEG,wBhC2KJ,CgB9OC,kBgBsEG,QhC2KJ,CgBjPC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhC4LJ,CgBtPC,iBgB6DG,iBhC4LJ,CgBzPC,iBgBgEG,kBhC4LJ,CgB5PC,mBgBmEG,wBhC4LJ,CgB/PC,kBgBsEG,QhC4LJ,CgBlQC,YgBwDG,aAAA,CACA,qBAAA,CACA,WhC6MJ,CgBvQC,iBgB6DG,UhC6MJ,CgB1QC,iBgBgEG,WhC6MJ,CgB7QC,mBgBmEG,iBhC6MJ,CgBhRC,kBgBsEG,QhC6MJ,CgBnRC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhC8NJ,CgBxRC,iBgB6DG,iBhC8NJ,CgB3RC,iBgBgEG,kBhC8NJ,CgB9RC,mBgBmEG,wBhC8NJ,CgBjSC,kBgBsEG,QhC8NJ,CgBpSC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhC+OJ,CgBzSC,iBgB6DG,iBhC+OJ,CgB5SC,iBgBgEG,kBhC+OJ,CgB/SC,mBgBmEG,wBhC+OJ,CgBlTC,kBgBsEG,QhC+OJ,CgBrTC,YgBwDG,aAAA,CACA,qBAAA,CACA,ShCgQJ,CgB1TC,iBgB6DG,QhCgQJ,CgB7TC,iBgBgEG,ShCgQJ,CgBhUC,mBgBmEG,ehCgQJ,CgBnUC,kBgBsEG,QhCgQJ,CgBtUC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhCiRJ,CgB3UC,iBgB6DG,iBhCiRJ,CgB9UC,iBgBgEG,kBhCiRJ,CgBjVC,mBgBmEG,wBhCiRJ,CgBpVC,kBgBsEG,QhCiRJ,CgBvVC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhCkSJ,CgB5VC,iBgB6DG,iBhCkSJ,CgB/VC,iBgBgEG,kBhCkSJ,CgBlWC,mBgBmEG,wBhCkSJ,CgBrWC,kBgBsEG,QhCkSJ,CgBxWC,YgBwDG,aAAA,CACA,qBAAA,CACA,WhCmTJ,CgB7WC,iBgB6DG,UhCmTJ,CgBhXC,iBgBgEG,WhCmTJ,CgBnXC,mBgBmEG,iBhCmTJ,CgBtXC,kBgBsEG,QhCmTJ,CgBzXC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhCoUJ,CgB9XC,iBgB6DG,iBhCoUJ,CgBjYC,iBgBgEG,kBhCoUJ,CgBpYC,mBgBmEG,wBhCoUJ,CgBvYC,kBgBsEG,QhCoUJ,CgB1YC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhCqVJ,CgB/YC,iBgB6DG,iBhCqVJ,CgBlZC,iBgBgEG,kBhCqVJ,CgBrZC,mBgBmEG,wBhCqVJ,CgBxZC,kBgBsEG,QhCqVJ,CgB3ZC,YgBwDG,aAAA,CACA,qBAAA,CACA,ShCsWJ,CgBhaC,iBgB6DG,QhCsWJ,CgBnaC,iBgBgEG,ShCsWJ,CgBtaC,mBgBmEG,ehCsWJ,CgBzaC,kBgBsEG,QhCsWJ,CgB5aC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhCuXJ,CgBjbC,iBgB6DG,iBhCuXJ,CgBpbC,iBgBgEG,kBhCuXJ,CgBvbC,mBgBmEG,wBhCuXJ,CgB1bC,kBgBsEG,QhCuXJ,CgB7bC,YgBwDG,aAAA,CACA,qBAAA,CACA,kBhCwYJ,CgBlcC,iBgB6DG,iBhCwYJ,CgBrcC,iBgBgEG,kBhCwYJ,CgBxcC,mBgBmEG,wBhCwYJ,CgB3cC,kBgBsEG,QhCwYJ,CgB9cC,WgBwDG,aAAA,CACA,qBAAA,CACA,WhCyZJ,CgBndC,gBgB6DG,UhCyZJ,CgBtdC,gBgBgEG,WhCyZJ,CgBzdC,kBgBmEG,iBhCyZJ,CgB5dC,iBgBsEG,OhCyZJ,CgB/dC,WgBwDG,aAAA,CACA,qBAAA,CACA,kBhC0aJ,CgBpeC,gBgB6DG,iBhC0aJ,CgBveC,gBgBgEG,kBhC0aJ,CgB1eC,kBgBmEG,wBhC0aJ,CgB7eC,iBgBsEG,OhC0aJ,CgBhfC,WgBwDG,aAAA,CACA,qBAAA,CACA,kBhC2bJ,CgBrfC,gBgB6DG,iBhC2bJ,CgBxfC,gBgBgEG,kBhC2bJ,CgB3fC,kBgBmEG,wBhC2bJ,CgB9fC,iBgBsEG,OhC2bJ,CgBjgBC,WgBwDG,aAAA,CACA,qBAAA,CACA,ShC4cJ,CgBtgBC,gBgB6DG,QhC4cJ,CgBzgBC,gBgBgEG,ShC4cJ,CgB5gBC,kBgBmEG,ehC4cJ,CgB/gBC,iBgBsEG,OhC4cJ,CgBlhBC,WgBwDG,aAAA,CACA,qBAAA,CACA,kBhC6dJ,CgBvhBC,gBgB6DG,iBhC6dJ,CgB1hBC,gBgBgEG,kBhC6dJ,CgB7hBC,kBgBmEG,wBhC6dJ,CgBhiBC,iBgBsEG,OhC6dJ,CgBniBC,WgBwDG,aAAA,CACA,qBAAA,CACA,kBhC8eJ,CgBxiBC,gBgB6DG,iBhC8eJ,CgB3iBC,gBgBgEG,kBhC8eJ,CgB9iBC,kBgBmEG,wBhC8eJ,CgBjjBC,iBgBsEG,OhC8eJ,CgBpjBC,WgBwDG,aAAA,CACA,qBAAA,CACA,WhC+fJ,CgBzjBC,gBgB6DG,UhC+fJ,CgB5jBC,gBgBgEG,WhC+fJ,CgB/jBC,kBgBmEG,iBhC+fJ,CgBlkBC,iBgBsEG,OhC+fJ,CgBrkBC,WgBwDG,aAAA,CACA,qBAAA,CACA,iBhCghBJ,CgB1kBC,gBgB6DG,gBhCghBJ,CgB7kBC,gBgBgEG,iBhCghBJ,CgBhlBC,kBgBmEG,uBhCghBJ,CgBnlBC,iBgBsEG,OhCghBJ,CgBtlBC,WgBwDG,aAAA,CACA,qBAAA,CACA,iBhCiiBJ,CgB3lBC,gBgB6DG,gBhCiiBJ,CgB9lBC,gBgBgEG,iBhCiiBJ,CgBjmBC,kBgBmEG,uBhCiiBJ,CgBpmBC,iBgBsEG,OhCiiBJ,CgBvmBC,WgB6EG,YhC6hBJ,CgB1mBC,kBgB4FG,ahC6hBJ,CgBznBC,iBgB+FG,OhC6hBJ,CgB5nBC,+VgB+CK,aAAA,CACA,UhCumBN,CgBvpBC,egBwDG,aAAA,CACA,qBAAA,CACA,UhCkmBJ,CgB5pBC,oBgB6DG,ShCkmBJ,CgB/pBC,oBgBgEG,UhCkmBJ,CgBlqBC,sBgBmEG,gBhCkmBJ,CgBrqBC,qBgBsEG,QhCkmBJ,CgBxqBC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCmnBJ,CgB7qBC,oBgB6DG,iBhCmnBJ,CgBhrBC,oBgBgEG,kBhCmnBJ,CgBnrBC,sBgBmEG,wBhCmnBJ,CgBtrBC,qBgBsEG,QhCmnBJ,CgBzrBC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCooBJ,CgB9rBC,oBgB6DG,iBhCooBJ,CgBjsBC,oBgBgEG,kBhCooBJ,CgBpsBC,sBgBmEG,wBhCooBJ,CgBvsBC,qBgBsEG,QhCooBJ,CgB1sBC,egBwDG,aAAA,CACA,qBAAA,CACA,WhCqpBJ,CgB/sBC,oBgB6DG,UhCqpBJ,CgBltBC,oBgBgEG,WhCqpBJ,CgBrtBC,sBgBmEG,iBhCqpBJ,CgBxtBC,qBgBsEG,QhCqpBJ,CgB3tBC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCsqBJ,CgBhuBC,oBgB6DG,iBhCsqBJ,CgBnuBC,oBgBgEG,kBhCsqBJ,CgBtuBC,sBgBmEG,wBhCsqBJ,CgBzuBC,qBgBsEG,QhCsqBJ,CgB5uBC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCurBJ,CgBjvBC,oBgB6DG,iBhCurBJ,CgBpvBC,oBgBgEG,kBhCurBJ,CgBvvBC,sBgBmEG,wBhCurBJ,CgB1vBC,qBgBsEG,QhCurBJ,CgB7vBC,egBwDG,aAAA,CACA,qBAAA,CACA,ShCwsBJ,CgBlwBC,oBgB6DG,QhCwsBJ,CgBrwBC,oBgBgEG,ShCwsBJ,CgBxwBC,sBgBmEG,ehCwsBJ,CgB3wBC,qBgBsEG,QhCwsBJ,CgB9wBC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCytBJ,CgBnxBC,oBgB6DG,iBhCytBJ,CgBtxBC,oBgBgEG,kBhCytBJ,CgBzxBC,sBgBmEG,wBhCytBJ,CgB5xBC,qBgBsEG,QhCytBJ,CgB/xBC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC0uBJ,CgBpyBC,oBgB6DG,iBhC0uBJ,CgBvyBC,oBgBgEG,kBhC0uBJ,CgB1yBC,sBgBmEG,wBhC0uBJ,CgB7yBC,qBgBsEG,QhC0uBJ,CgBhzBC,egBwDG,aAAA,CACA,qBAAA,CACA,WhC2vBJ,CgBrzBC,oBgB6DG,UhC2vBJ,CgBxzBC,oBgBgEG,WhC2vBJ,CgB3zBC,sBgBmEG,iBhC2vBJ,CgB9zBC,qBgBsEG,QhC2vBJ,CgBj0BC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC4wBJ,CgBt0BC,oBgB6DG,iBhC4wBJ,CgBz0BC,oBgBgEG,kBhC4wBJ,CgB50BC,sBgBmEG,wBhC4wBJ,CgB/0BC,qBgBsEG,QhC4wBJ,CgBl1BC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC6xBJ,CgBv1BC,oBgB6DG,iBhC6xBJ,CgB11BC,oBgBgEG,kBhC6xBJ,CgB71BC,sBgBmEG,wBhC6xBJ,CgBh2BC,qBgBsEG,QhC6xBJ,CgBn2BC,egBwDG,aAAA,CACA,qBAAA,CACA,ShC8yBJ,CgBx2BC,oBgB6DG,QhC8yBJ,CgB32BC,oBgBgEG,ShC8yBJ,CgB92BC,sBgBmEG,ehC8yBJ,CgBj3BC,qBgBsEG,QhC8yBJ,CgBp3BC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC+zBJ,CgBz3BC,oBgB6DG,iBhC+zBJ,CgB53BC,oBgBgEG,kBhC+zBJ,CgB/3BC,sBgBmEG,wBhC+zBJ,CgBl4BC,qBgBsEG,QhC+zBJ,CgBr4BC,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCg1BJ,CgB14BC,oBgB6DG,iBhCg1BJ,CgB74BC,oBgBgEG,kBhCg1BJ,CgBh5BC,sBgBmEG,wBhCg1BJ,CgBn5BC,qBgBsEG,QhCg1BJ,CgBt5BC,cgBwDG,aAAA,CACA,qBAAA,CACA,WhCi2BJ,CgB35BC,mBgB6DG,UhCi2BJ,CgB95BC,mBgBgEG,WhCi2BJ,CgBj6BC,qBgBmEG,iBhCi2BJ,CgBp6BC,oBgBsEG,OhCi2BJ,CgBv6BC,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCk3BJ,CgB56BC,mBgB6DG,iBhCk3BJ,CgB/6BC,mBgBgEG,kBhCk3BJ,CgBl7BC,qBgBmEG,wBhCk3BJ,CgBr7BC,oBgBsEG,OhCk3BJ,CgBx7BC,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCm4BJ,CgB77BC,mBgB6DG,iBhCm4BJ,CgBh8BC,mBgBgEG,kBhCm4BJ,CgBn8BC,qBgBmEG,wBhCm4BJ,CgBt8BC,oBgBsEG,OhCm4BJ,CgBz8BC,cgBwDG,aAAA,CACA,qBAAA,CACA,ShCo5BJ,CgB98BC,mBgB6DG,QhCo5BJ,CgBj9BC,mBgBgEG,ShCo5BJ,CgBp9BC,qBgBmEG,ehCo5BJ,CgBv9BC,oBgBsEG,OhCo5BJ,CgB19BC,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCq6BJ,CgB/9BC,mBgB6DG,iBhCq6BJ,CgBl+BC,mBgBgEG,kBhCq6BJ,CgBr+BC,qBgBmEG,wBhCq6BJ,CgBx+BC,oBgBsEG,OhCq6BJ,CgB3+BC,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCs7BJ,CgBh/BC,mBgB6DG,iBhCs7BJ,CgBn/BC,mBgBgEG,kBhCs7BJ,CgBt/BC,qBgBmEG,wBhCs7BJ,CgBz/BC,oBgBsEG,OhCs7BJ,CgB5/BC,cgBwDG,aAAA,CACA,qBAAA,CACA,WhCu8BJ,CgBjgCC,mBgB6DG,UhCu8BJ,CgBpgCC,mBgBgEG,WhCu8BJ,CgBvgCC,qBgBmEG,iBhCu8BJ,CgB1gCC,oBgBsEG,OhCu8BJ,CgB7gCC,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhCw9BJ,CgBlhCC,mBgB6DG,gBhCw9BJ,CgBrhCC,mBgBgEG,iBhCw9BJ,CgBxhCC,qBgBmEG,uBhCw9BJ,CgB3hCC,oBgBsEG,OhCw9BJ,CgB9hCC,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhCy+BJ,CgBniCC,mBgB6DG,gBhCy+BJ,CgBtiCC,mBgBgEG,iBhCy+BJ,CgBziCC,qBgBmEG,uBhCy+BJ,CgB5iCC,oBgBsEG,OhCy+BJ,CgB/iCC,cgB6EG,YhCq+BJ,CgBljCC,gBgBgFG,ShCq+BJ,CgBrjCC,gBgBmFG,UhCq+BJ,CgBxjCC,mBgBsFG,ShCq+BJ,CgB3jCC,mBgByFG,UhCq+BJ,CgB9jCC,qBgB4FG,ahCq+BJ,CgBjkCC,oBgB+FG,OhCq+BJ,CiCn/BA,yBjBjFC,+VgB+CK,aAAA,CACA,UhCgjCJ,CgBhmCD,egBwDG,aAAA,CACA,qBAAA,CACA,UhC2iCF,CgBrmCD,oBgB6DG,ShC2iCF,CgBxmCD,oBgBgEG,UhC2iCF,CgB3mCD,sBgBmEG,gBhC2iCF,CgB9mCD,qBgBsEG,QhC2iCF,CgBjnCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC4jCF,CgBtnCD,oBgB6DG,iBhC4jCF,CgBznCD,oBgBgEG,kBhC4jCF,CgB5nCD,sBgBmEG,wBhC4jCF,CgB/nCD,qBgBsEG,QhC4jCF,CgBloCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC6kCF,CgBvoCD,oBgB6DG,iBhC6kCF,CgB1oCD,oBgBgEG,kBhC6kCF,CgB7oCD,sBgBmEG,wBhC6kCF,CgBhpCD,qBgBsEG,QhC6kCF,CgBnpCD,egBwDG,aAAA,CACA,qBAAA,CACA,WhC8lCF,CgBxpCD,oBgB6DG,UhC8lCF,CgB3pCD,oBgBgEG,WhC8lCF,CgB9pCD,sBgBmEG,iBhC8lCF,CgBjqCD,qBgBsEG,QhC8lCF,CgBpqCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC+mCF,CgBzqCD,oBgB6DG,iBhC+mCF,CgB5qCD,oBgBgEG,kBhC+mCF,CgB/qCD,sBgBmEG,wBhC+mCF,CgBlrCD,qBgBsEG,QhC+mCF,CgBrrCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCgoCF,CgB1rCD,oBgB6DG,iBhCgoCF,CgB7rCD,oBgBgEG,kBhCgoCF,CgBhsCD,sBgBmEG,wBhCgoCF,CgBnsCD,qBgBsEG,QhCgoCF,CgBtsCD,egBwDG,aAAA,CACA,qBAAA,CACA,ShCipCF,CgB3sCD,oBgB6DG,QhCipCF,CgB9sCD,oBgBgEG,ShCipCF,CgBjtCD,sBgBmEG,ehCipCF,CgBptCD,qBgBsEG,QhCipCF,CgBvtCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCkqCF,CgB5tCD,oBgB6DG,iBhCkqCF,CgB/tCD,oBgBgEG,kBhCkqCF,CgBluCD,sBgBmEG,wBhCkqCF,CgBruCD,qBgBsEG,QhCkqCF,CgBxuCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCmrCF,CgB7uCD,oBgB6DG,iBhCmrCF,CgBhvCD,oBgBgEG,kBhCmrCF,CgBnvCD,sBgBmEG,wBhCmrCF,CgBtvCD,qBgBsEG,QhCmrCF,CgBzvCD,egBwDG,aAAA,CACA,qBAAA,CACA,WhCosCF,CgB9vCD,oBgB6DG,UhCosCF,CgBjwCD,oBgBgEG,WhCosCF,CgBpwCD,sBgBmEG,iBhCosCF,CgBvwCD,qBgBsEG,QhCosCF,CgB1wCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCqtCF,CgB/wCD,oBgB6DG,iBhCqtCF,CgBlxCD,oBgBgEG,kBhCqtCF,CgBrxCD,sBgBmEG,wBhCqtCF,CgBxxCD,qBgBsEG,QhCqtCF,CgB3xCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCsuCF,CgBhyCD,oBgB6DG,iBhCsuCF,CgBnyCD,oBgBgEG,kBhCsuCF,CgBtyCD,sBgBmEG,wBhCsuCF,CgBzyCD,qBgBsEG,QhCsuCF,CgB5yCD,egBwDG,aAAA,CACA,qBAAA,CACA,ShCuvCF,CgBjzCD,oBgB6DG,QhCuvCF,CgBpzCD,oBgBgEG,ShCuvCF,CgBvzCD,sBgBmEG,ehCuvCF,CgB1zCD,qBgBsEG,QhCuvCF,CgB7zCD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCwwCF,CgBl0CD,oBgB6DG,iBhCwwCF,CgBr0CD,oBgBgEG,kBhCwwCF,CgBx0CD,sBgBmEG,wBhCwwCF,CgB30CD,qBgBsEG,QhCwwCF,CgB90CD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCyxCF,CgBn1CD,oBgB6DG,iBhCyxCF,CgBt1CD,oBgBgEG,kBhCyxCF,CgBz1CD,sBgBmEG,wBhCyxCF,CgB51CD,qBgBsEG,QhCyxCF,CgB/1CD,cgBwDG,aAAA,CACA,qBAAA,CACA,WhC0yCF,CgBp2CD,mBgB6DG,UhC0yCF,CgBv2CD,mBgBgEG,WhC0yCF,CgB12CD,qBgBmEG,iBhC0yCF,CgB72CD,oBgBsEG,OhC0yCF,CgBh3CD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC2zCF,CgBr3CD,mBgB6DG,iBhC2zCF,CgBx3CD,mBgBgEG,kBhC2zCF,CgB33CD,qBgBmEG,wBhC2zCF,CgB93CD,oBgBsEG,OhC2zCF,CgBj4CD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC40CF,CgBt4CD,mBgB6DG,iBhC40CF,CgBz4CD,mBgBgEG,kBhC40CF,CgB54CD,qBgBmEG,wBhC40CF,CgB/4CD,oBgBsEG,OhC40CF,CgBl5CD,cgBwDG,aAAA,CACA,qBAAA,CACA,ShC61CF,CgBv5CD,mBgB6DG,QhC61CF,CgB15CD,mBgBgEG,ShC61CF,CgB75CD,qBgBmEG,ehC61CF,CgBh6CD,oBgBsEG,OhC61CF,CgBn6CD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC82CF,CgBx6CD,mBgB6DG,iBhC82CF,CgB36CD,mBgBgEG,kBhC82CF,CgB96CD,qBgBmEG,wBhC82CF,CgBj7CD,oBgBsEG,OhC82CF,CgBp7CD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC+3CF,CgBz7CD,mBgB6DG,iBhC+3CF,CgB57CD,mBgBgEG,kBhC+3CF,CgB/7CD,qBgBmEG,wBhC+3CF,CgBl8CD,oBgBsEG,OhC+3CF,CgBr8CD,cgBwDG,aAAA,CACA,qBAAA,CACA,WhCg5CF,CgB18CD,mBgB6DG,UhCg5CF,CgB78CD,mBgBgEG,WhCg5CF,CgBh9CD,qBgBmEG,iBhCg5CF,CgBn9CD,oBgBsEG,OhCg5CF,CgBt9CD,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhCi6CF,CgB39CD,mBgB6DG,gBhCi6CF,CgB99CD,mBgBgEG,iBhCi6CF,CgBj+CD,qBgBmEG,uBhCi6CF,CgBp+CD,oBgBsEG,OhCi6CF,CgBv+CD,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhCk7CF,CgB5+CD,mBgB6DG,gBhCk7CF,CgB/+CD,mBgBgEG,iBhCk7CF,CgBl/CD,qBgBmEG,uBhCk7CF,CgBr/CD,oBgBsEG,OhCk7CF,CgBx/CD,cgB6EG,YhC86CF,CgB3/CD,gBgBgFG,ShC86CF,CgB9/CD,gBgBmFG,UhC86CF,CgBjgDD,mBgBsFG,ShC86CF,CgBpgDD,mBgByFG,UhC86CF,CgBvgDD,qBgB4FG,ahC86CF,CgB1gDD,oBgB+FG,OhC86CF,CACF,CiCr7CA,yBjBzFC,+VgB+CK,aAAA,CACA,UhC0/CJ,CgB1iDD,egBwDG,aAAA,CACA,qBAAA,CACA,UhCq/CF,CgB/iDD,oBgB6DG,ShCq/CF,CgBljDD,oBgBgEG,UhCq/CF,CgBrjDD,sBgBmEG,gBhCq/CF,CgBxjDD,qBgBsEG,QhCq/CF,CgB3jDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCsgDF,CgBhkDD,oBgB6DG,iBhCsgDF,CgBnkDD,oBgBgEG,kBhCsgDF,CgBtkDD,sBgBmEG,wBhCsgDF,CgBzkDD,qBgBsEG,QhCsgDF,CgB5kDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCuhDF,CgBjlDD,oBgB6DG,iBhCuhDF,CgBplDD,oBgBgEG,kBhCuhDF,CgBvlDD,sBgBmEG,wBhCuhDF,CgB1lDD,qBgBsEG,QhCuhDF,CgB7lDD,egBwDG,aAAA,CACA,qBAAA,CACA,WhCwiDF,CgBlmDD,oBgB6DG,UhCwiDF,CgBrmDD,oBgBgEG,WhCwiDF,CgBxmDD,sBgBmEG,iBhCwiDF,CgB3mDD,qBgBsEG,QhCwiDF,CgB9mDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCyjDF,CgBnnDD,oBgB6DG,iBhCyjDF,CgBtnDD,oBgBgEG,kBhCyjDF,CgBznDD,sBgBmEG,wBhCyjDF,CgB5nDD,qBgBsEG,QhCyjDF,CgB/nDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC0kDF,CgBpoDD,oBgB6DG,iBhC0kDF,CgBvoDD,oBgBgEG,kBhC0kDF,CgB1oDD,sBgBmEG,wBhC0kDF,CgB7oDD,qBgBsEG,QhC0kDF,CgBhpDD,egBwDG,aAAA,CACA,qBAAA,CACA,ShC2lDF,CgBrpDD,oBgB6DG,QhC2lDF,CgBxpDD,oBgBgEG,ShC2lDF,CgB3pDD,sBgBmEG,ehC2lDF,CgB9pDD,qBgBsEG,QhC2lDF,CgBjqDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC4mDF,CgBtqDD,oBgB6DG,iBhC4mDF,CgBzqDD,oBgBgEG,kBhC4mDF,CgB5qDD,sBgBmEG,wBhC4mDF,CgB/qDD,qBgBsEG,QhC4mDF,CgBlrDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC6nDF,CgBvrDD,oBgB6DG,iBhC6nDF,CgB1rDD,oBgBgEG,kBhC6nDF,CgB7rDD,sBgBmEG,wBhC6nDF,CgBhsDD,qBgBsEG,QhC6nDF,CgBnsDD,egBwDG,aAAA,CACA,qBAAA,CACA,WhC8oDF,CgBxsDD,oBgB6DG,UhC8oDF,CgB3sDD,oBgBgEG,WhC8oDF,CgB9sDD,sBgBmEG,iBhC8oDF,CgBjtDD,qBgBsEG,QhC8oDF,CgBptDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC+pDF,CgBztDD,oBgB6DG,iBhC+pDF,CgB5tDD,oBgBgEG,kBhC+pDF,CgB/tDD,sBgBmEG,wBhC+pDF,CgBluDD,qBgBsEG,QhC+pDF,CgBruDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCgrDF,CgB1uDD,oBgB6DG,iBhCgrDF,CgB7uDD,oBgBgEG,kBhCgrDF,CgBhvDD,sBgBmEG,wBhCgrDF,CgBnvDD,qBgBsEG,QhCgrDF,CgBtvDD,egBwDG,aAAA,CACA,qBAAA,CACA,ShCisDF,CgB3vDD,oBgB6DG,QhCisDF,CgB9vDD,oBgBgEG,ShCisDF,CgBjwDD,sBgBmEG,ehCisDF,CgBpwDD,qBgBsEG,QhCisDF,CgBvwDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCktDF,CgB5wDD,oBgB6DG,iBhCktDF,CgB/wDD,oBgBgEG,kBhCktDF,CgBlxDD,sBgBmEG,wBhCktDF,CgBrxDD,qBgBsEG,QhCktDF,CgBxxDD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCmuDF,CgB7xDD,oBgB6DG,iBhCmuDF,CgBhyDD,oBgBgEG,kBhCmuDF,CgBnyDD,sBgBmEG,wBhCmuDF,CgBtyDD,qBgBsEG,QhCmuDF,CgBzyDD,cgBwDG,aAAA,CACA,qBAAA,CACA,WhCovDF,CgB9yDD,mBgB6DG,UhCovDF,CgBjzDD,mBgBgEG,WhCovDF,CgBpzDD,qBgBmEG,iBhCovDF,CgBvzDD,oBgBsEG,OhCovDF,CgB1zDD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCqwDF,CgB/zDD,mBgB6DG,iBhCqwDF,CgBl0DD,mBgBgEG,kBhCqwDF,CgBr0DD,qBgBmEG,wBhCqwDF,CgBx0DD,oBgBsEG,OhCqwDF,CgB30DD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCsxDF,CgBh1DD,mBgB6DG,iBhCsxDF,CgBn1DD,mBgBgEG,kBhCsxDF,CgBt1DD,qBgBmEG,wBhCsxDF,CgBz1DD,oBgBsEG,OhCsxDF,CgB51DD,cgBwDG,aAAA,CACA,qBAAA,CACA,ShCuyDF,CgBj2DD,mBgB6DG,QhCuyDF,CgBp2DD,mBgBgEG,ShCuyDF,CgBv2DD,qBgBmEG,ehCuyDF,CgB12DD,oBgBsEG,OhCuyDF,CgB72DD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCwzDF,CgBl3DD,mBgB6DG,iBhCwzDF,CgBr3DD,mBgBgEG,kBhCwzDF,CgBx3DD,qBgBmEG,wBhCwzDF,CgB33DD,oBgBsEG,OhCwzDF,CgB93DD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCy0DF,CgBn4DD,mBgB6DG,iBhCy0DF,CgBt4DD,mBgBgEG,kBhCy0DF,CgBz4DD,qBgBmEG,wBhCy0DF,CgB54DD,oBgBsEG,OhCy0DF,CgB/4DD,cgBwDG,aAAA,CACA,qBAAA,CACA,WhC01DF,CgBp5DD,mBgB6DG,UhC01DF,CgBv5DD,mBgBgEG,WhC01DF,CgB15DD,qBgBmEG,iBhC01DF,CgB75DD,oBgBsEG,OhC01DF,CgBh6DD,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhC22DF,CgBr6DD,mBgB6DG,gBhC22DF,CgBx6DD,mBgBgEG,iBhC22DF,CgB36DD,qBgBmEG,uBhC22DF,CgB96DD,oBgBsEG,OhC22DF,CgBj7DD,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhC43DF,CgBt7DD,mBgB6DG,gBhC43DF,CgBz7DD,mBgBgEG,iBhC43DF,CgB57DD,qBgBmEG,uBhC43DF,CgB/7DD,oBgBsEG,OhC43DF,CgBl8DD,cgB6EG,YhCw3DF,CgBr8DD,gBgBgFG,ShCw3DF,CgBx8DD,gBgBmFG,UhCw3DF,CgB38DD,mBgBsFG,ShCw3DF,CgB98DD,mBgByFG,UhCw3DF,CgBj9DD,qBgB4FG,ahCw3DF,CgBp9DD,oBgB+FG,OhCw3DF,CACF,CiCv3DA,yBjBjGC,+VgB+CK,aAAA,CACA,UhCo8DJ,CgBp/DD,egBwDG,aAAA,CACA,qBAAA,CACA,UhC+7DF,CgBz/DD,oBgB6DG,ShC+7DF,CgB5/DD,oBgBgEG,UhC+7DF,CgB//DD,sBgBmEG,gBhC+7DF,CgBlgED,qBgBsEG,QhC+7DF,CgBrgED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCg9DF,CgB1gED,oBgB6DG,iBhCg9DF,CgB7gED,oBgBgEG,kBhCg9DF,CgBhhED,sBgBmEG,wBhCg9DF,CgBnhED,qBgBsEG,QhCg9DF,CgBthED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCi+DF,CgB3hED,oBgB6DG,iBhCi+DF,CgB9hED,oBgBgEG,kBhCi+DF,CgBjiED,sBgBmEG,wBhCi+DF,CgBpiED,qBgBsEG,QhCi+DF,CgBviED,egBwDG,aAAA,CACA,qBAAA,CACA,WhCk/DF,CgB5iED,oBgB6DG,UhCk/DF,CgB/iED,oBgBgEG,WhCk/DF,CgBljED,sBgBmEG,iBhCk/DF,CgBrjED,qBgBsEG,QhCk/DF,CgBxjED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCmgEF,CgB7jED,oBgB6DG,iBhCmgEF,CgBhkED,oBgBgEG,kBhCmgEF,CgBnkED,sBgBmEG,wBhCmgEF,CgBtkED,qBgBsEG,QhCmgEF,CgBzkED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCohEF,CgB9kED,oBgB6DG,iBhCohEF,CgBjlED,oBgBgEG,kBhCohEF,CgBplED,sBgBmEG,wBhCohEF,CgBvlED,qBgBsEG,QhCohEF,CgB1lED,egBwDG,aAAA,CACA,qBAAA,CACA,ShCqiEF,CgB/lED,oBgB6DG,QhCqiEF,CgBlmED,oBgBgEG,ShCqiEF,CgBrmED,sBgBmEG,ehCqiEF,CgBxmED,qBgBsEG,QhCqiEF,CgB3mED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCsjEF,CgBhnED,oBgB6DG,iBhCsjEF,CgBnnED,oBgBgEG,kBhCsjEF,CgBtnED,sBgBmEG,wBhCsjEF,CgBznED,qBgBsEG,QhCsjEF,CgB5nED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCukEF,CgBjoED,oBgB6DG,iBhCukEF,CgBpoED,oBgBgEG,kBhCukEF,CgBvoED,sBgBmEG,wBhCukEF,CgB1oED,qBgBsEG,QhCukEF,CgB7oED,egBwDG,aAAA,CACA,qBAAA,CACA,WhCwlEF,CgBlpED,oBgB6DG,UhCwlEF,CgBrpED,oBgBgEG,WhCwlEF,CgBxpED,sBgBmEG,iBhCwlEF,CgB3pED,qBgBsEG,QhCwlEF,CgB9pED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCymEF,CgBnqED,oBgB6DG,iBhCymEF,CgBtqED,oBgBgEG,kBhCymEF,CgBzqED,sBgBmEG,wBhCymEF,CgB5qED,qBgBsEG,QhCymEF,CgB/qED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC0nEF,CgBprED,oBgB6DG,iBhC0nEF,CgBvrED,oBgBgEG,kBhC0nEF,CgB1rED,sBgBmEG,wBhC0nEF,CgB7rED,qBgBsEG,QhC0nEF,CgBhsED,egBwDG,aAAA,CACA,qBAAA,CACA,ShC2oEF,CgBrsED,oBgB6DG,QhC2oEF,CgBxsED,oBgBgEG,ShC2oEF,CgB3sED,sBgBmEG,ehC2oEF,CgB9sED,qBgBsEG,QhC2oEF,CgBjtED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC4pEF,CgBttED,oBgB6DG,iBhC4pEF,CgBztED,oBgBgEG,kBhC4pEF,CgB5tED,sBgBmEG,wBhC4pEF,CgB/tED,qBgBsEG,QhC4pEF,CgBluED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC6qEF,CgBvuED,oBgB6DG,iBhC6qEF,CgB1uED,oBgBgEG,kBhC6qEF,CgB7uED,sBgBmEG,wBhC6qEF,CgBhvED,qBgBsEG,QhC6qEF,CgBnvED,cgBwDG,aAAA,CACA,qBAAA,CACA,WhC8rEF,CgBxvED,mBgB6DG,UhC8rEF,CgB3vED,mBgBgEG,WhC8rEF,CgB9vED,qBgBmEG,iBhC8rEF,CgBjwED,oBgBsEG,OhC8rEF,CgBpwED,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC+sEF,CgBzwED,mBgB6DG,iBhC+sEF,CgB5wED,mBgBgEG,kBhC+sEF,CgB/wED,qBgBmEG,wBhC+sEF,CgBlxED,oBgBsEG,OhC+sEF,CgBrxED,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCguEF,CgB1xED,mBgB6DG,iBhCguEF,CgB7xED,mBgBgEG,kBhCguEF,CgBhyED,qBgBmEG,wBhCguEF,CgBnyED,oBgBsEG,OhCguEF,CgBtyED,cgBwDG,aAAA,CACA,qBAAA,CACA,ShCivEF,CgB3yED,mBgB6DG,QhCivEF,CgB9yED,mBgBgEG,ShCivEF,CgBjzED,qBgBmEG,ehCivEF,CgBpzED,oBgBsEG,OhCivEF,CgBvzED,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCkwEF,CgB5zED,mBgB6DG,iBhCkwEF,CgB/zED,mBgBgEG,kBhCkwEF,CgBl0ED,qBgBmEG,wBhCkwEF,CgBr0ED,oBgBsEG,OhCkwEF,CgBx0ED,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCmxEF,CgB70ED,mBgB6DG,iBhCmxEF,CgBh1ED,mBgBgEG,kBhCmxEF,CgBn1ED,qBgBmEG,wBhCmxEF,CgBt1ED,oBgBsEG,OhCmxEF,CgBz1ED,cgBwDG,aAAA,CACA,qBAAA,CACA,WhCoyEF,CgB91ED,mBgB6DG,UhCoyEF,CgBj2ED,mBgBgEG,WhCoyEF,CgBp2ED,qBgBmEG,iBhCoyEF,CgBv2ED,oBgBsEG,OhCoyEF,CgB12ED,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhCqzEF,CgB/2ED,mBgB6DG,gBhCqzEF,CgBl3ED,mBgBgEG,iBhCqzEF,CgBr3ED,qBgBmEG,uBhCqzEF,CgBx3ED,oBgBsEG,OhCqzEF,CgB33ED,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhCs0EF,CgBh4ED,mBgB6DG,gBhCs0EF,CgBn4ED,mBgBgEG,iBhCs0EF,CgBt4ED,qBgBmEG,uBhCs0EF,CgBz4ED,oBgBsEG,OhCs0EF,CgB54ED,cgB6EG,YhCk0EF,CgB/4ED,gBgBgFG,ShCk0EF,CgBl5ED,gBgBmFG,UhCk0EF,CgBr5ED,mBgBsFG,ShCk0EF,CgBx5ED,mBgByFG,UhCk0EF,CgB35ED,qBgB4FG,ahCk0EF,CgB95ED,oBgB+FG,OhCk0EF,CACF,CiCzzEA,0BjBzGC,+VgB+CK,aAAA,CACA,UhC84EJ,CgB97ED,egBwDG,aAAA,CACA,qBAAA,CACA,UhCy4EF,CgBn8ED,oBgB6DG,ShCy4EF,CgBt8ED,oBgBgEG,UhCy4EF,CgBz8ED,sBgBmEG,gBhCy4EF,CgB58ED,qBgBsEG,QhCy4EF,CgB/8ED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC05EF,CgBp9ED,oBgB6DG,iBhC05EF,CgBv9ED,oBgBgEG,kBhC05EF,CgB19ED,sBgBmEG,wBhC05EF,CgB79ED,qBgBsEG,QhC05EF,CgBh+ED,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC26EF,CgBr+ED,oBgB6DG,iBhC26EF,CgBx+ED,oBgBgEG,kBhC26EF,CgB3+ED,sBgBmEG,wBhC26EF,CgB9+ED,qBgBsEG,QhC26EF,CgBj/ED,egBwDG,aAAA,CACA,qBAAA,CACA,WhC47EF,CgBt/ED,oBgB6DG,UhC47EF,CgBz/ED,oBgBgEG,WhC47EF,CgB5/ED,sBgBmEG,iBhC47EF,CgB//ED,qBgBsEG,QhC47EF,CgBlgFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC68EF,CgBvgFD,oBgB6DG,iBhC68EF,CgB1gFD,oBgBgEG,kBhC68EF,CgB7gFD,sBgBmEG,wBhC68EF,CgBhhFD,qBgBsEG,QhC68EF,CgBnhFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhC89EF,CgBxhFD,oBgB6DG,iBhC89EF,CgB3hFD,oBgBgEG,kBhC89EF,CgB9hFD,sBgBmEG,wBhC89EF,CgBjiFD,qBgBsEG,QhC89EF,CgBpiFD,egBwDG,aAAA,CACA,qBAAA,CACA,ShC++EF,CgBziFD,oBgB6DG,QhC++EF,CgB5iFD,oBgBgEG,ShC++EF,CgB/iFD,sBgBmEG,ehC++EF,CgBljFD,qBgBsEG,QhC++EF,CgBrjFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCggFF,CgB1jFD,oBgB6DG,iBhCggFF,CgB7jFD,oBgBgEG,kBhCggFF,CgBhkFD,sBgBmEG,wBhCggFF,CgBnkFD,qBgBsEG,QhCggFF,CgBtkFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCihFF,CgB3kFD,oBgB6DG,iBhCihFF,CgB9kFD,oBgBgEG,kBhCihFF,CgBjlFD,sBgBmEG,wBhCihFF,CgBplFD,qBgBsEG,QhCihFF,CgBvlFD,egBwDG,aAAA,CACA,qBAAA,CACA,WhCkiFF,CgB5lFD,oBgB6DG,UhCkiFF,CgB/lFD,oBgBgEG,WhCkiFF,CgBlmFD,sBgBmEG,iBhCkiFF,CgBrmFD,qBgBsEG,QhCkiFF,CgBxmFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCmjFF,CgB7mFD,oBgB6DG,iBhCmjFF,CgBhnFD,oBgBgEG,kBhCmjFF,CgBnnFD,sBgBmEG,wBhCmjFF,CgBtnFD,qBgBsEG,QhCmjFF,CgBznFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCokFF,CgB9nFD,oBgB6DG,iBhCokFF,CgBjoFD,oBgBgEG,kBhCokFF,CgBpoFD,sBgBmEG,wBhCokFF,CgBvoFD,qBgBsEG,QhCokFF,CgB1oFD,egBwDG,aAAA,CACA,qBAAA,CACA,ShCqlFF,CgB/oFD,oBgB6DG,QhCqlFF,CgBlpFD,oBgBgEG,ShCqlFF,CgBrpFD,sBgBmEG,ehCqlFF,CgBxpFD,qBgBsEG,QhCqlFF,CgB3pFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCsmFF,CgBhqFD,oBgB6DG,iBhCsmFF,CgBnqFD,oBgBgEG,kBhCsmFF,CgBtqFD,sBgBmEG,wBhCsmFF,CgBzqFD,qBgBsEG,QhCsmFF,CgB5qFD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCunFF,CgBjrFD,oBgB6DG,iBhCunFF,CgBprFD,oBgBgEG,kBhCunFF,CgBvrFD,sBgBmEG,wBhCunFF,CgB1rFD,qBgBsEG,QhCunFF,CgB7rFD,cgBwDG,aAAA,CACA,qBAAA,CACA,WhCwoFF,CgBlsFD,mBgB6DG,UhCwoFF,CgBrsFD,mBgBgEG,WhCwoFF,CgBxsFD,qBgBmEG,iBhCwoFF,CgB3sFD,oBgBsEG,OhCwoFF,CgB9sFD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhCypFF,CgBntFD,mBgB6DG,iBhCypFF,CgBttFD,mBgBgEG,kBhCypFF,CgBztFD,qBgBmEG,wBhCypFF,CgB5tFD,oBgBsEG,OhCypFF,CgB/tFD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC0qFF,CgBpuFD,mBgB6DG,iBhC0qFF,CgBvuFD,mBgBgEG,kBhC0qFF,CgB1uFD,qBgBmEG,wBhC0qFF,CgB7uFD,oBgBsEG,OhC0qFF,CgBhvFD,cgBwDG,aAAA,CACA,qBAAA,CACA,ShC2rFF,CgBrvFD,mBgB6DG,QhC2rFF,CgBxvFD,mBgBgEG,ShC2rFF,CgB3vFD,qBgBmEG,ehC2rFF,CgB9vFD,oBgBsEG,OhC2rFF,CgBjwFD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC4sFF,CgBtwFD,mBgB6DG,iBhC4sFF,CgBzwFD,mBgBgEG,kBhC4sFF,CgB5wFD,qBgBmEG,wBhC4sFF,CgB/wFD,oBgBsEG,OhC4sFF,CgBlxFD,cgBwDG,aAAA,CACA,qBAAA,CACA,kBhC6tFF,CgBvxFD,mBgB6DG,iBhC6tFF,CgB1xFD,mBgBgEG,kBhC6tFF,CgB7xFD,qBgBmEG,wBhC6tFF,CgBhyFD,oBgBsEG,OhC6tFF,CgBnyFD,cgBwDG,aAAA,CACA,qBAAA,CACA,WhC8uFF,CgBxyFD,mBgB6DG,UhC8uFF,CgB3yFD,mBgBgEG,WhC8uFF,CgB9yFD,qBgBmEG,iBhC8uFF,CgBjzFD,oBgBsEG,OhC8uFF,CgBpzFD,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhC+vFF,CgBzzFD,mBgB6DG,gBhC+vFF,CgB5zFD,mBgBgEG,iBhC+vFF,CgB/zFD,qBgBmEG,uBhC+vFF,CgBl0FD,oBgBsEG,OhC+vFF,CgBr0FD,cgBwDG,aAAA,CACA,qBAAA,CACA,iBhCgxFF,CgB10FD,mBgB6DG,gBhCgxFF,CgB70FD,mBgBgEG,iBhCgxFF,CgBh1FD,qBgBmEG,uBhCgxFF,CgBn1FD,oBgBsEG,OhCgxFF,CgBt1FD,cgB6EG,YhC4wFF,CgBz1FD,gBgBgFG,ShC4wFF,CgB51FD,gBgBmFG,UhC4wFF,CgB/1FD,mBgBsFG,ShC4wFF,CgBl2FD,mBgByFG,UhC4wFF,CgBr2FD,qBgB4FG,ahC4wFF,CgBx2FD,oBgB+FG,OhC4wFF,CACF,CiC3vFA,0BjBjHC,uXgB+CK,aAAA,CACA,UhCw1FJ,CgBx4FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,UhCm1FF,CgB74FD,qBgB6DG,ShCm1FF,CgBh5FD,qBgBgEG,UhCm1FF,CgBn5FD,uBgBmEG,gBhCm1FF,CgBt5FD,sBgBsEG,QhCm1FF,CgBz5FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhCo2FF,CgB95FD,qBgB6DG,iBhCo2FF,CgBj6FD,qBgBgEG,kBhCo2FF,CgBp6FD,uBgBmEG,wBhCo2FF,CgBv6FD,sBgBsEG,QhCo2FF,CgB16FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhCq3FF,CgB/6FD,qBgB6DG,iBhCq3FF,CgBl7FD,qBgBgEG,kBhCq3FF,CgBr7FD,uBgBmEG,wBhCq3FF,CgBx7FD,sBgBsEG,QhCq3FF,CgB37FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,WhCs4FF,CgBh8FD,qBgB6DG,UhCs4FF,CgBn8FD,qBgBgEG,WhCs4FF,CgBt8FD,uBgBmEG,iBhCs4FF,CgBz8FD,sBgBsEG,QhCs4FF,CgB58FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhCu5FF,CgBj9FD,qBgB6DG,iBhCu5FF,CgBp9FD,qBgBgEG,kBhCu5FF,CgBv9FD,uBgBmEG,wBhCu5FF,CgB19FD,sBgBsEG,QhCu5FF,CgB79FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhCw6FF,CgBl+FD,qBgB6DG,iBhCw6FF,CgBr+FD,qBgBgEG,kBhCw6FF,CgBx+FD,uBgBmEG,wBhCw6FF,CgB3+FD,sBgBsEG,QhCw6FF,CgB9+FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,ShCy7FF,CgBn/FD,qBgB6DG,QhCy7FF,CgBt/FD,qBgBgEG,ShCy7FF,CgBz/FD,uBgBmEG,ehCy7FF,CgB5/FD,sBgBsEG,QhCy7FF,CgB//FD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhC08FF,CgBpgGD,qBgB6DG,iBhC08FF,CgBvgGD,qBgBgEG,kBhC08FF,CgB1gGD,uBgBmEG,wBhC08FF,CgB7gGD,sBgBsEG,QhC08FF,CgBhhGD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhC29FF,CgBrhGD,qBgB6DG,iBhC29FF,CgBxhGD,qBgBgEG,kBhC29FF,CgB3hGD,uBgBmEG,wBhC29FF,CgB9hGD,sBgBsEG,QhC29FF,CgBjiGD,gBgBwDG,aAAA,CACA,qBAAA,CACA,WhC4+FF,CgBtiGD,qBgB6DG,UhC4+FF,CgBziGD,qBgBgEG,WhC4+FF,CgB5iGD,uBgBmEG,iBhC4+FF,CgB/iGD,sBgBsEG,QhC4+FF,CgBljGD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhC6/FF,CgBvjGD,qBgB6DG,iBhC6/FF,CgB1jGD,qBgBgEG,kBhC6/FF,CgB7jGD,uBgBmEG,wBhC6/FF,CgBhkGD,sBgBsEG,QhC6/FF,CgBnkGD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhC8gGF,CgBxkGD,qBgB6DG,iBhC8gGF,CgB3kGD,qBgBgEG,kBhC8gGF,CgB9kGD,uBgBmEG,wBhC8gGF,CgBjlGD,sBgBsEG,QhC8gGF,CgBplGD,gBgBwDG,aAAA,CACA,qBAAA,CACA,ShC+hGF,CgBzlGD,qBgB6DG,QhC+hGF,CgB5lGD,qBgBgEG,ShC+hGF,CgB/lGD,uBgBmEG,ehC+hGF,CgBlmGD,sBgBsEG,QhC+hGF,CgBrmGD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhCgjGF,CgB1mGD,qBgB6DG,iBhCgjGF,CgB7mGD,qBgBgEG,kBhCgjGF,CgBhnGD,uBgBmEG,wBhCgjGF,CgBnnGD,sBgBsEG,QhCgjGF,CgBtnGD,gBgBwDG,aAAA,CACA,qBAAA,CACA,kBhCikGF,CgB3nGD,qBgB6DG,iBhCikGF,CgB9nGD,qBgBgEG,kBhCikGF,CgBjoGD,uBgBmEG,wBhCikGF,CgBpoGD,sBgBsEG,QhCikGF,CgBvoGD,egBwDG,aAAA,CACA,qBAAA,CACA,WhCklGF,CgB5oGD,oBgB6DG,UhCklGF,CgB/oGD,oBgBgEG,WhCklGF,CgBlpGD,sBgBmEG,iBhCklGF,CgBrpGD,qBgBsEG,OhCklGF,CgBxpGD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCmmGF,CgB7pGD,oBgB6DG,iBhCmmGF,CgBhqGD,oBgBgEG,kBhCmmGF,CgBnqGD,sBgBmEG,wBhCmmGF,CgBtqGD,qBgBsEG,OhCmmGF,CgBzqGD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhConGF,CgB9qGD,oBgB6DG,iBhConGF,CgBjrGD,oBgBgEG,kBhConGF,CgBprGD,sBgBmEG,wBhConGF,CgBvrGD,qBgBsEG,OhConGF,CgB1rGD,egBwDG,aAAA,CACA,qBAAA,CACA,ShCqoGF,CgB/rGD,oBgB6DG,QhCqoGF,CgBlsGD,oBgBgEG,ShCqoGF,CgBrsGD,sBgBmEG,ehCqoGF,CgBxsGD,qBgBsEG,OhCqoGF,CgB3sGD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCspGF,CgBhtGD,oBgB6DG,iBhCspGF,CgBntGD,oBgBgEG,kBhCspGF,CgBttGD,sBgBmEG,wBhCspGF,CgBztGD,qBgBsEG,OhCspGF,CgB5tGD,egBwDG,aAAA,CACA,qBAAA,CACA,kBhCuqGF,CgBjuGD,oBgB6DG,iBhCuqGF,CgBpuGD,oBgBgEG,kBhCuqGF,CgBvuGD,sBgBmEG,wBhCuqGF,CgB1uGD,qBgBsEG,OhCuqGF,CgB7uGD,egBwDG,aAAA,CACA,qBAAA,CACA,WhCwrGF,CgBlvGD,oBgB6DG,UhCwrGF,CgBrvGD,oBgBgEG,WhCwrGF,CgBxvGD,sBgBmEG,iBhCwrGF,CgB3vGD,qBgBsEG,OhCwrGF,CgB9vGD,egBwDG,aAAA,CACA,qBAAA,CACA,iBhCysGF,CgBnwGD,oBgB6DG,gBhCysGF,CgBtwGD,oBgBgEG,iBhCysGF,CgBzwGD,sBgBmEG,uB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","file":"2.5fcf4e3c.chunk.css","sourcesContent":["/* stylelint-disable */\n.bezierEasingMixin() {\n@functions: ~`(function() {\n var NEWTON_ITERATIONS = 4;\n var NEWTON_MIN_SLOPE = 0.001;\n var SUBDIVISION_PRECISION = 0.0000001;\n var SUBDIVISION_MAX_ITERATIONS = 10;\n\n var kSplineTableSize = 11;\n var kSampleStepSize = 1.0 / (kSplineTableSize - 1.0);\n\n var float32ArraySupported = typeof Float32Array === 'function';\n\n function A (aA1, aA2) { return 1.0 - 3.0 * aA2 + 3.0 * aA1; }\n function B (aA1, aA2) { return 3.0 * aA2 - 6.0 * aA1; }\n function C (aA1) { return 3.0 * aA1; }\n\n // Returns x(t) given t, x1, and x2, or y(t) given t, y1, and y2.\n function calcBezier (aT, aA1, aA2) { return ((A(aA1, aA2) * aT + B(aA1, aA2)) * aT + C(aA1)) * aT; }\n\n // Returns dx/dt given t, x1, and x2, or dy/dt given t, y1, and y2.\n function getSlope (aT, aA1, aA2) { return 3.0 * A(aA1, aA2) * aT * aT + 2.0 * B(aA1, aA2) * aT + C(aA1); }\n\n function binarySubdivide (aX, aA, aB, mX1, mX2) {\n var currentX, currentT, i = 0;\n do {\n currentT = aA + (aB - aA) / 2.0;\n currentX = calcBezier(currentT, mX1, mX2) - aX;\n if (currentX > 0.0) {\n aB = currentT;\n } else {\n aA = currentT;\n }\n } while (Math.abs(currentX) > SUBDIVISION_PRECISION && ++i < SUBDIVISION_MAX_ITERATIONS);\n return currentT;\n }\n\n function newtonRaphsonIterate (aX, aGuessT, mX1, mX2) {\n for (var i = 0; i < NEWTON_ITERATIONS; ++i) {\n var currentSlope = getSlope(aGuessT, mX1, mX2);\n if (currentSlope === 0.0) {\n return aGuessT;\n }\n var currentX = calcBezier(aGuessT, mX1, mX2) - aX;\n aGuessT -= currentX / currentSlope;\n }\n return aGuessT;\n }\n\n var BezierEasing = function (mX1, mY1, mX2, mY2) {\n if (!(0 <= mX1 && mX1 <= 1 && 0 <= mX2 && mX2 <= 1)) {\n throw new Error('bezier x values must be in [0, 1] range');\n }\n\n // Precompute samples table\n var sampleValues = float32ArraySupported ? new Float32Array(kSplineTableSize) : new Array(kSplineTableSize);\n if (mX1 !== mY1 || mX2 !== mY2) {\n for (var i = 0; i < kSplineTableSize; ++i) {\n sampleValues[i] = calcBezier(i * kSampleStepSize, mX1, mX2);\n }\n }\n\n function getTForX (aX) {\n var intervalStart = 0.0;\n var currentSample = 1;\n var lastSample = kSplineTableSize - 1;\n\n for (; currentSample !== lastSample && sampleValues[currentSample] <= aX; ++currentSample) {\n intervalStart += kSampleStepSize;\n }\n --currentSample;\n\n // Interpolate to provide an initial guess for t\n var dist = (aX - sampleValues[currentSample]) / (sampleValues[currentSample + 1] - sampleValues[currentSample]);\n var guessForT = intervalStart + dist * kSampleStepSize;\n\n var initialSlope = getSlope(guessForT, mX1, mX2);\n if (initialSlope >= NEWTON_MIN_SLOPE) {\n return newtonRaphsonIterate(aX, guessForT, mX1, mX2);\n } else if (initialSlope === 0.0) {\n return guessForT;\n } else {\n return binarySubdivide(aX, intervalStart, intervalStart + kSampleStepSize, mX1, mX2);\n }\n }\n\n return function BezierEasing (x) {\n if (mX1 === mY1 && mX2 === mY2) {\n return x; // linear\n }\n // Because JavaScript number are imprecise, we should guarantee the extremes are right.\n if (x === 0) {\n return 0;\n }\n if (x === 1) {\n return 1;\n }\n return calcBezier(getTForX(x), mY1, mY2);\n };\n };\n\n this.colorEasing = BezierEasing(0.26, 0.09, 0.37, 0.18);\n // less 3 requires a return\n return '';\n})()`;\n}\n// It is hacky way to make this function will be compiled preferentially by less\n// resolve error: `ReferenceError: colorPalette is not defined`\n// https://github.com/ant-design/ant-motion/issues/44\n.bezierEasingMixin();\n","// Sizing shortcuts\n\n.size(@width; @height) {\n width: @width;\n height: @height;\n}\n\n.square(@size) {\n .size(@size; @size);\n}\n","/* stylelint-disable at-rule-empty-line-before,at-rule-name-space-after,at-rule-no-unknown */\n/* stylelint-disable no-duplicate-selectors */\n/* stylelint-disable */\n/* stylelint-disable declaration-bang-space-before,no-duplicate-selectors,string-no-newline */\n.ant-progress {\n box-sizing: border-box;\n margin: 0;\n padding: 0;\n color: hsl(200, 10%, 90%);\n font-size: 14px;\n font-variant: tabular-nums;\n line-height: 1.5;\n list-style: none;\n font-feature-settings: 'tnum';\n display: inline-block;\n}\n.ant-progress-line {\n position: relative;\n width: 100%;\n font-size: 14px;\n}\n.ant-progress-small.ant-progress-line,\n.ant-progress-small.ant-progress-line .ant-progress-text .anticon {\n font-size: 12px;\n}\n.ant-progress-outer {\n display: inline-block;\n width: 100%;\n margin-right: 0;\n padding-right: 0;\n}\n.ant-progress-show-info .ant-progress-outer {\n margin-right: calc(-2em - 8px);\n padding-right: calc(2em + 8px);\n}\n.ant-progress-inner {\n position: relative;\n display: inline-block;\n width: 100%;\n vertical-align: middle;\n background-color: hsla(200, 100%, 50%, 0.2);\n border-radius: 100px;\n}\n.ant-progress-circle-trail {\n stroke: hsla(200, 100%, 50%, 0.2);\n}\n.ant-progress-circle-path {\n animation: ant-progress-appear 0.3s;\n stroke: #1890ff;\n}\n.ant-progress-success-bg,\n.ant-progress-bg {\n position: relative;\n background-color: #1890ff;\n transition: all 0.4s cubic-bezier(0.08, 0.82, 0.17, 1) 0s;\n}\n.ant-progress-success-bg {\n position: absolute;\n top: 0;\n left: 0;\n background-color: #52c41a;\n}\n.ant-progress-text {\n display: inline-block;\n width: 2em;\n margin-left: 8px;\n color: hsl(200, 20%, 80%);\n font-size: 1em;\n line-height: 1;\n white-space: nowrap;\n text-align: left;\n vertical-align: middle;\n word-break: normal;\n}\n.ant-progress-text .anticon {\n font-size: 14px;\n}\n.ant-progress-status-active .ant-progress-bg::before {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n background: hsl(200, 10%, 20%);\n border-radius: 10px;\n opacity: 0;\n animation: ant-progress-active 2.4s cubic-bezier(0.23, 1, 0.32, 1) infinite;\n content: '';\n}\n.ant-progress-status-exception .ant-progress-bg {\n background-color: #f5222d;\n}\n.ant-progress-status-exception .ant-progress-text {\n color: #f5222d;\n}\n.ant-progress-status-exception .ant-progress-circle-path {\n stroke: #f5222d;\n}\n.ant-progress-status-success .ant-progress-bg {\n background-color: #52c41a;\n}\n.ant-progress-status-success .ant-progress-text {\n color: #52c41a;\n}\n.ant-progress-status-success .ant-progress-circle-path {\n stroke: #52c41a;\n}\n.ant-progress-circle .ant-progress-inner {\n position: relative;\n line-height: 1;\n background-color: transparent;\n}\n.ant-progress-circle .ant-progress-text {\n position: absolute;\n top: 50%;\n left: 50%;\n width: 100%;\n margin: 0;\n padding: 0;\n color: hsl(200, 10%, 90%);\n line-height: 1;\n white-space: normal;\n text-align: center;\n transform: translate(-50%, -50%);\n}\n.ant-progress-circle .ant-progress-text .anticon {\n font-size: 1.16666667em;\n}\n.ant-progress-circle.ant-progress-status-exception .ant-progress-text {\n color: #f5222d;\n}\n.ant-progress-circle.ant-progress-status-success .ant-progress-text {\n color: #52c41a;\n}\n@keyframes ant-progress-active {\n 0% {\n width: 0;\n opacity: 0.1;\n }\n 20% {\n width: 0;\n opacity: 0.5;\n }\n 100% {\n width: 100%;\n opacity: 0;\n }\n}\n","/* stylelint-disable at-rule-no-unknown */\n\n// Reboot\n//\n// Normalization of HTML elements, manually forked from Normalize.css to remove\n// styles targeting irrelevant browsers while applying new styles.\n//\n// Normalize is licensed MIT. https://github.com/necolas/normalize.css\n\n// HTML & Body reset\n@{html-selector},\nbody {\n .square(100%);\n}\n\n// remove the clear button of a text input control in IE10+\ninput::-ms-clear,\ninput::-ms-reveal {\n display: none;\n}\n\n// Document\n//\n// 1. Change from `box-sizing: content-box` so that `width` is not affected by `padding` or `border`.\n// 2. Change the default font family in all browsers.\n// 3. Correct the line height in all browsers.\n// 4. Prevent adjustments of font size after orientation changes in IE on Windows Phone and in iOS.\n// 5. Setting @viewport causes scrollbars to overlap content in IE11 and Edge, so\n// we force a non-overlapping, non-auto-hiding scrollbar to counteract.\n// 6. Change the default tap highlight to be completely transparent in iOS.\n\n*,\n*::before,\n*::after {\n box-sizing: border-box; // 1\n}\n\n@{html-selector} {\n font-family: sans-serif; // 2\n line-height: 1.15; // 3\n -webkit-text-size-adjust: 100%; // 4\n -ms-text-size-adjust: 100%; // 4\n -ms-overflow-style: scrollbar; // 5\n -webkit-tap-highlight-color: fade(@black, 0%); // 6\n}\n\n// IE10+ doesn't honor `` in some cases.\n@-ms-viewport {\n width: device-width;\n}\n\n// Shim for \"new\" HTML5 structural elements to display correctly (IE10, older browsers)\narticle,\naside,\ndialog,\nfigcaption,\nfigure,\nfooter,\nheader,\nhgroup,\nmain,\nnav,\nsection {\n display: block;\n}\n\n// Body\n//\n// 1. remove the margin in all browsers.\n// 2. As a best practice, apply a default `body-background`.\n\nbody {\n margin: 0; // 1\n color: @text-color;\n font-size: @font-size-base;\n font-family: @font-family;\n font-variant: @font-variant-base;\n line-height: @line-height-base;\n background-color: @body-background; // 2\n font-feature-settings: @font-feature-settings-base;\n}\n\n// Suppress the focus outline on elements that cannot be accessed via keyboard.\n// This prevents an unwanted focus outline from appearing around elements that\n// might still respond to pointer events.\n//\n// Credit: https://github.com/suitcss/base\n[tabindex='-1']:focus {\n outline: none !important;\n}\n\n// Content grouping\n//\n// 1. Add the correct box sizing in Firefox.\n// 2. Show the overflow in Edge and IE.\n\nhr {\n box-sizing: content-box; // 1\n height: 0; // 1\n overflow: visible; // 2\n}\n\n//\n// Typography\n//\n\n// remove top margins from headings\n//\n// By default, `

`-`

` all receive top and bottom margins. We nuke the top\n// margin for easier control within type scales as it avoids margin collapsing.\nh1,\nh2,\nh3,\nh4,\nh5,\nh6 {\n margin-top: 0;\n margin-bottom: 0.5em;\n color: @heading-color;\n font-weight: 500;\n}\n\n// Reset margins on paragraphs\n//\n// Similarly, the top margin on `

`s get reset. However, we also reset the\n// bottom margin to use `em` units instead of `em`.\np {\n margin-top: 0;\n margin-bottom: 1em;\n}\n\n// Abbreviations\n//\n// 1. remove the bottom border in Firefox 39-.\n// 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari.\n// 3. Add explicit cursor to indicate changed behavior.\n// 4. Duplicate behavior to the data-* attribute for our tooltip plugin\n\nabbr[title],\nabbr[data-original-title] {\n // 4\n text-decoration: underline; // 2\n text-decoration: underline dotted; // 2\n border-bottom: 0; // 1\n cursor: help; // 3\n}\n\naddress {\n margin-bottom: 1em;\n font-style: normal;\n line-height: inherit;\n}\n\ninput[type='text'],\ninput[type='password'],\ninput[type='number'],\ntextarea {\n -webkit-appearance: none;\n}\n\nol,\nul,\ndl {\n margin-top: 0;\n margin-bottom: 1em;\n}\n\nol ol,\nul ul,\nol ul,\nul ol {\n margin-bottom: 0;\n}\n\ndt {\n font-weight: 500;\n}\n\ndd {\n margin-bottom: 0.5em;\n margin-left: 0; // Undo browser default\n}\n\nblockquote {\n margin: 0 0 1em;\n}\n\ndfn {\n font-style: italic; // Add the correct font style in Android 4.3-\n}\n\nb,\nstrong {\n font-weight: bolder; // Add the correct font weight in Chrome, Edge, and Safari\n}\n\nsmall {\n font-size: 80%; // Add the correct font size in all browsers\n}\n\n//\n// Prevent `sub` and `sup` elements from affecting the line height in\n// all browsers.\n//\n\nsub,\nsup {\n position: relative;\n font-size: 75%;\n line-height: 0;\n vertical-align: baseline;\n}\n\nsub {\n bottom: -0.25em;\n}\nsup {\n top: -0.5em;\n}\n\n//\n// Links\n//\n\na {\n color: @link-color;\n text-decoration: @link-decoration;\n background-color: transparent; // remove the gray background on active links in IE 10.\n outline: none;\n cursor: pointer;\n transition: color 0.3s;\n -webkit-text-decoration-skip: objects; // remove gaps in links underline in iOS 8+ and Safari 8+.\n\n &:hover {\n color: @link-hover-color;\n }\n\n &:active {\n color: @link-active-color;\n }\n\n &:active,\n &:hover {\n text-decoration: @link-hover-decoration;\n outline: 0;\n }\n\n &[disabled] {\n color: @disabled-color;\n cursor: not-allowed;\n pointer-events: none;\n }\n}\n\n//\n// Code\n//\n\npre,\ncode,\nkbd,\nsamp {\n font-size: 1em; // Correct the odd `em` font sizing in all browsers.\n font-family: @code-family;\n}\n\npre {\n // remove browser default top margin\n margin-top: 0;\n // Reset browser default of `1em` to use `em`s\n margin-bottom: 1em;\n // Don't allow content to break outside\n overflow: auto;\n}\n\n//\n// Figures\n//\nfigure {\n // Apply a consistent margin strategy (matches our type styles).\n margin: 0 0 1em;\n}\n\n//\n// Images and content\n//\n\nimg {\n vertical-align: middle;\n border-style: none; // remove the border on images inside links in IE 10-.\n}\n\nsvg:not(:root) {\n overflow: hidden; // Hide the overflow in IE\n}\n\n// Avoid 300ms click delay on touch devices that support the `touch-action` CSS property.\n//\n// In particular, unlike most other browsers, IE11+Edge on Windows 10 on touch devices and IE Mobile 10-11\n// DON'T remove the click delay when `` is present.\n// However, they DO support emoving the click delay via `touch-action: manipulation`.\n// See:\n// * https://getbootstrap.com/docs/4.0/content/reboot/#click-delay-optimization-for-touch\n// * http://caniuse.com/#feat=css-touch-action\n// * https://patrickhlauke.github.io/touch/tests/results/#suppressing-300ms-delay\n\na,\narea,\nbutton,\n[role='button'],\ninput:not([type='range']),\nlabel,\nselect,\nsummary,\ntextarea {\n touch-action: manipulation;\n}\n\n//\n// Tables\n//\n\ntable {\n border-collapse: collapse; // Prevent double borders\n}\n\ncaption {\n padding-top: 0.75em;\n padding-bottom: 0.3em;\n color: @text-color-secondary;\n text-align: left;\n caption-side: bottom;\n}\n\nth {\n // Matches default `` alignment by inheriting from the ``, or the\n // closest parent with a set `text-align`.\n text-align: inherit;\n}\n\n//\n// Forms\n//\n\ninput,\nbutton,\nselect,\noptgroup,\ntextarea {\n margin: 0; // remove the margin in Firefox and Safari\n color: inherit;\n font-size: inherit;\n font-family: inherit;\n line-height: inherit;\n}\n\nbutton,\ninput {\n overflow: visible; // Show the overflow in Edge\n}\n\nbutton,\nselect {\n text-transform: none; // remove the inheritance of text transform in Firefox\n}\n\n// 1. Prevent a WebKit bug where (2) destroys native `audio` and `video`\n// controls in Android 4.\n// 2. Correct the inability to style clickable types in iOS and Safari.\nbutton,\n@{html-selector} [type=\"button\"], /* 1 */\n[type=\"reset\"],\n[type=\"submit\"] {\n -webkit-appearance: button; // 2\n}\n\n// remove inner border and padding from Firefox, but don't restore the outline like Normalize.\nbutton::-moz-focus-inner,\n[type='button']::-moz-focus-inner,\n[type='reset']::-moz-focus-inner,\n[type='submit']::-moz-focus-inner {\n padding: 0;\n border-style: none;\n}\n\ninput[type='radio'],\ninput[type='checkbox'] {\n box-sizing: border-box; // 1. Add the correct box sizing in IE 10-\n padding: 0; // 2. remove the padding in IE 10-\n}\n\ninput[type='date'],\ninput[type='time'],\ninput[type='datetime-local'],\ninput[type='month'] {\n // remove the default appearance of temporal inputs to avoid a Mobile Safari\n // bug where setting a custom line-height prevents text from being vertically\n // centered within the input.\n // See https://bugs.webkit.org/show_bug.cgi?id=139848\n // and https://github.com/twbs/bootstrap/issues/11266\n -webkit-appearance: listbox;\n}\n\ntextarea {\n overflow: auto; // remove the default vertical scrollbar in IE.\n // Textareas should really only resize vertically so they don't break their (horizontal) containers.\n resize: vertical;\n}\n\nfieldset {\n // Browsers set a default `min-width: min-content;` on fieldsets,\n // unlike e.g. `

`s, which have `min-width: 0;` by default.\n // So we reset that to ensure fieldsets behave more like a standard block element.\n // See https://github.com/twbs/bootstrap/issues/12359\n // and https://html.spec.whatwg.org/multipage/#the-fieldset-and-legend-elements\n min-width: 0;\n margin: 0;\n // Reset the default outline behavior of fieldsets so they don't affect page layout.\n padding: 0;\n border: 0;\n}\n\n// 1. Correct the text wrapping in Edge and IE.\n// 2. Correct the color inheritance from `fieldset` elements in IE.\nlegend {\n display: block;\n width: 100%;\n max-width: 100%; // 1\n margin-bottom: 0.5em;\n padding: 0;\n color: inherit; // 2\n font-size: 1.5em;\n line-height: inherit;\n white-space: normal; // 1\n}\n\nprogress {\n vertical-align: baseline; // Add the correct vertical alignment in Chrome, Firefox, and Opera.\n}\n\n// Correct the cursor style of incement and decement buttons in Chrome.\n[type='number']::-webkit-inner-spin-button,\n[type='number']::-webkit-outer-spin-button {\n height: auto;\n}\n\n[type='search'] {\n // This overrides the extra rounded corners on search inputs in iOS so that our\n // `.form-control` class can properly style them. Note that this cannot simply\n // be added to `.form-control` as it's not specific enough. For details, see\n // https://github.com/twbs/bootstrap/issues/11586.\n outline-offset: -2px; // 2. Correct the outline style in Safari.\n -webkit-appearance: none;\n}\n\n//\n// remove the inner padding and cancel buttons in Chrome and Safari on macOS.\n//\n\n[type='search']::-webkit-search-cancel-button,\n[type='search']::-webkit-search-decoration {\n -webkit-appearance: none;\n}\n\n//\n// 1. Correct the inability to style clickable types in iOS and Safari.\n// 2. Change font properties to `inherit` in Safari.\n//\n\n::-webkit-file-upload-button {\n font: inherit; // 2\n -webkit-appearance: button; // 1\n}\n\n//\n// Correct element displays\n//\n\noutput {\n display: inline-block;\n}\n\nsummary {\n display: list-item; // Add the correct display in all browsers\n}\n\ntemplate {\n display: none; // Add the correct display in IE\n}\n\n// Always hide an element with the `hidden` HTML attribute (from PureCSS).\n// Needed for proper display in IE 10-.\n[hidden] {\n display: none !important;\n}\n\nmark {\n padding: 0.2em;\n background-color: @yellow-1;\n}\n\n::selection {\n color: @text-color-inverse;\n background: @text-selection-bg;\n}\n\n// Utility classes\n.clearfix {\n .clearfix();\n}\n","// mixins for clearfix\n// ------------------------\n.clearfix() {\n zoom: 1;\n &::before,\n &::after {\n display: table;\n content: '';\n }\n &::after {\n clear: both;\n }\n}\n",".iconfont-mixin() {\n display: inline-block;\n color: @icon-color;\n font-style: normal;\n line-height: 0;\n text-align: center;\n text-transform: none;\n vertical-align: -0.125em; // for SVG icon, see https://blog.prototypr.io/align-svg-icons-to-text-and-say-goodbye-to-font-icons-d44b3d7b26b4\n text-rendering: optimizeLegibility;\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n\n > * {\n line-height: 1;\n }\n\n svg {\n display: inline-block;\n }\n\n &::before {\n display: none; // dont display old icon.\n }\n\n & &-icon {\n display: block;\n }\n}\n\n// for iconfont font size\n// fix chrome 12px bug, support ie\n.iconfont-size-under-12px(@size, @rotate: 0deg) {\n display: inline-block;\n @font-scale: unit(@size / 12px);\n\n font-size: 12px;\n // IE9\n font-size: ~'@{size} \\9';\n transform: scale(@font-scale) rotate(@rotate);\n :root & {\n font-size: @font-size-sm; // reset IE9 and above\n }\n}\n","@import '../themes/index';\n@import '../mixins/iconfont';\n\n.@{iconfont-css-prefix} {\n .iconfont-mixin();\n\n &[tabindex] {\n cursor: pointer;\n }\n}\n\n.@{iconfont-css-prefix}-spin::before {\n display: inline-block;\n animation: loadingCircle 1s infinite linear;\n}\n.@{iconfont-css-prefix}-spin {\n display: inline-block;\n animation: loadingCircle 1s infinite linear;\n}\n","@import '../themes/index';\n\n.motion-common(@duration: @animation-duration-base) {\n animation-duration: @duration;\n animation-fill-mode: both;\n}\n\n.motion-common-leave(@duration: @animation-duration-base) {\n animation-duration: @duration;\n animation-fill-mode: both;\n}\n\n.make-motion(@className, @keyframeName, @duration: @animation-duration-base) {\n .@{className}-enter,\n .@{className}-appear {\n .motion-common(@duration);\n\n animation-play-state: paused;\n }\n .@{className}-leave {\n .motion-common-leave(@duration);\n\n animation-play-state: paused;\n }\n .@{className}-enter.@{className}-enter-active,\n .@{className}-appear.@{className}-appear-active {\n animation-name: ~'@{keyframeName}In';\n animation-play-state: running;\n }\n .@{className}-leave.@{className}-leave-active {\n animation-name: ~'@{keyframeName}Out';\n animation-play-state: running;\n pointer-events: none;\n }\n}\n",".fade-motion(@className, @keyframeName) {\n .make-motion(@className, @keyframeName);\n .@{className}-enter,\n .@{className}-appear {\n opacity: 0;\n animation-timing-function: linear;\n }\n .@{className}-leave {\n animation-timing-function: linear;\n }\n}\n\n.fade-motion(fade, antFade);\n\n@keyframes antFadeIn {\n 0% {\n opacity: 0;\n }\n 100% {\n opacity: 1;\n }\n}\n\n@keyframes antFadeOut {\n 0% {\n opacity: 1;\n }\n 100% {\n opacity: 0;\n }\n}\n",".move-motion(@className, @keyframeName) {\n .make-motion(@className, @keyframeName);\n .@{className}-enter,\n .@{className}-appear {\n opacity: 0;\n animation-timing-function: @ease-out-circ;\n }\n .@{className}-leave {\n animation-timing-function: @ease-in-circ;\n }\n}\n\n.move-motion(move-up, antMoveUp);\n.move-motion(move-down, antMoveDown);\n.move-motion(move-left, antMoveLeft);\n.move-motion(move-right, antMoveRight);\n\n@keyframes antMoveDownIn {\n 0% {\n transform: translateY(100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n 100% {\n transform: translateY(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n}\n\n@keyframes antMoveDownOut {\n 0% {\n transform: translateY(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n 100% {\n transform: translateY(100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n}\n\n@keyframes antMoveLeftIn {\n 0% {\n transform: translateX(-100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n 100% {\n transform: translateX(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n}\n\n@keyframes antMoveLeftOut {\n 0% {\n transform: translateX(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n 100% {\n transform: translateX(-100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n}\n\n@keyframes antMoveRightIn {\n 0% {\n transform: translateX(100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n 100% {\n transform: translateX(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n}\n\n@keyframes antMoveRightOut {\n 0% {\n transform: translateX(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n 100% {\n transform: translateX(100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n}\n\n@keyframes antMoveUpIn {\n 0% {\n transform: translateY(-100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n 100% {\n transform: translateY(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n}\n\n@keyframes antMoveUpOut {\n 0% {\n transform: translateY(0%);\n transform-origin: 0 0;\n opacity: 1;\n }\n 100% {\n transform: translateY(-100%);\n transform-origin: 0 0;\n opacity: 0;\n }\n}\n","@keyframes loadingCircle {\n 100% {\n transform: rotate(360deg);\n }\n}\n\n[ant-click-animating='true'],\n[ant-click-animating-without-extra-node='true'] {\n position: relative;\n}\n\nhtml {\n --antd-wave-shadow-color: @primary-color;\n}\n\n[ant-click-animating-without-extra-node='true']::after,\n.ant-click-animating-node {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n display: block;\n border-radius: inherit;\n box-shadow: 0 0 0 0 @primary-color;\n box-shadow: 0 0 0 0 var(--antd-wave-shadow-color);\n opacity: 0.2;\n animation: fadeEffect 2s @ease-out-circ, waveEffect 0.4s @ease-out-circ;\n animation-fill-mode: forwards;\n content: '';\n pointer-events: none;\n}\n\n@keyframes waveEffect {\n 100% {\n box-shadow: 0 0 0 @primary-color;\n box-shadow: 0 0 0 @wave-animation-width var(--antd-wave-shadow-color);\n }\n}\n\n@keyframes fadeEffect {\n 100% {\n opacity: 0;\n }\n}\n",".slide-motion(@className, @keyframeName) {\n .make-motion(@className, @keyframeName);\n .@{className}-enter,\n .@{className}-appear {\n opacity: 0;\n animation-timing-function: @ease-out-quint;\n }\n .@{className}-leave {\n animation-timing-function: @ease-in-quint;\n }\n}\n\n.slide-motion(slide-up, antSlideUp);\n.slide-motion(slide-down, antSlideDown);\n.slide-motion(slide-left, antSlideLeft);\n.slide-motion(slide-right, antSlideRight);\n\n@keyframes antSlideUpIn {\n 0% {\n transform: scaleY(0.8);\n transform-origin: 0% 0%;\n opacity: 0;\n }\n 100% {\n transform: scaleY(1);\n transform-origin: 0% 0%;\n opacity: 1;\n }\n}\n\n@keyframes antSlideUpOut {\n 0% {\n transform: scaleY(1);\n transform-origin: 0% 0%;\n opacity: 1;\n }\n 100% {\n transform: scaleY(0.8);\n transform-origin: 0% 0%;\n opacity: 0;\n }\n}\n\n@keyframes antSlideDownIn {\n 0% {\n transform: scaleY(0.8);\n transform-origin: 100% 100%;\n opacity: 0;\n }\n 100% {\n transform: scaleY(1);\n transform-origin: 100% 100%;\n opacity: 1;\n }\n}\n\n@keyframes antSlideDownOut {\n 0% {\n transform: scaleY(1);\n transform-origin: 100% 100%;\n opacity: 1;\n }\n 100% {\n transform: scaleY(0.8);\n transform-origin: 100% 100%;\n opacity: 0;\n }\n}\n\n@keyframes antSlideLeftIn {\n 0% {\n transform: scaleX(0.8);\n transform-origin: 0% 0%;\n opacity: 0;\n }\n 100% {\n transform: scaleX(1);\n transform-origin: 0% 0%;\n opacity: 1;\n }\n}\n\n@keyframes antSlideLeftOut {\n 0% {\n transform: scaleX(1);\n transform-origin: 0% 0%;\n opacity: 1;\n }\n 100% {\n transform: scaleX(0.8);\n transform-origin: 0% 0%;\n opacity: 0;\n }\n}\n\n@keyframes antSlideRightIn {\n 0% {\n transform: scaleX(0.8);\n transform-origin: 100% 0%;\n opacity: 0;\n }\n 100% {\n transform: scaleX(1);\n transform-origin: 100% 0%;\n opacity: 1;\n }\n}\n\n@keyframes antSlideRightOut {\n 0% {\n transform: scaleX(1);\n transform-origin: 100% 0%;\n opacity: 1;\n }\n 100% {\n transform: scaleX(0.8);\n transform-origin: 100% 0%;\n opacity: 0;\n }\n}\n",".swing-motion(@className, @keyframeName) {\n .@{className}-enter,\n .@{className}-appear {\n .motion-common();\n\n animation-play-state: paused;\n }\n .@{className}-enter.@{className}-enter-active,\n .@{className}-appear.@{className}-appear-active {\n animation-name: ~'@{keyframeName}In';\n animation-play-state: running;\n }\n}\n\n.swing-motion(swing, antSwing);\n\n@keyframes antSwingIn {\n 0%,\n 100% {\n transform: translateX(0);\n }\n 20% {\n transform: translateX(-10px);\n }\n 40% {\n transform: translateX(10px);\n }\n 60% {\n transform: translateX(-5px);\n }\n 80% {\n transform: translateX(5px);\n }\n}\n",".zoom-motion(@className, @keyframeName, @duration: @animation-duration-base) {\n .make-motion(@className, @keyframeName, @duration);\n .@{className}-enter,\n .@{className}-appear {\n transform: scale(0); // need this by yiminghe\n opacity: 0;\n animation-timing-function: @ease-out-circ;\n }\n .@{className}-leave {\n animation-timing-function: @ease-in-out-circ;\n }\n}\n\n// For Modal, Select choosen item\n.zoom-motion(zoom, antZoom);\n// For Popover, Popconfirm, Dropdown\n.zoom-motion(zoom-big, antZoomBig);\n// For Tooltip\n.zoom-motion(zoom-big-fast, antZoomBig, @animation-duration-fast);\n\n.zoom-motion(zoom-up, antZoomUp);\n.zoom-motion(zoom-down, antZoomDown);\n.zoom-motion(zoom-left, antZoomLeft);\n.zoom-motion(zoom-right, antZoomRight);\n\n@keyframes antZoomIn {\n 0% {\n transform: scale(0.2);\n opacity: 0;\n }\n 100% {\n transform: scale(1);\n opacity: 1;\n }\n}\n\n@keyframes antZoomOut {\n 0% {\n transform: scale(1);\n }\n 100% {\n transform: scale(0.2);\n opacity: 0;\n }\n}\n\n@keyframes antZoomBigIn {\n 0% {\n transform: scale(0.8);\n opacity: 0;\n }\n 100% {\n transform: scale(1);\n opacity: 1;\n }\n}\n\n@keyframes antZoomBigOut {\n 0% {\n transform: scale(1);\n }\n 100% {\n transform: scale(0.8);\n opacity: 0;\n }\n}\n\n@keyframes antZoomUpIn {\n 0% {\n transform: scale(0.8);\n transform-origin: 50% 0%;\n opacity: 0;\n }\n 100% {\n transform: scale(1);\n transform-origin: 50% 0%;\n }\n}\n\n@keyframes antZoomUpOut {\n 0% {\n transform: scale(1);\n transform-origin: 50% 0%;\n }\n 100% {\n transform: scale(0.8);\n transform-origin: 50% 0%;\n opacity: 0;\n }\n}\n\n@keyframes antZoomLeftIn {\n 0% {\n transform: scale(0.8);\n transform-origin: 0% 50%;\n opacity: 0;\n }\n 100% {\n transform: scale(1);\n transform-origin: 0% 50%;\n }\n}\n\n@keyframes antZoomLeftOut {\n 0% {\n transform: scale(1);\n transform-origin: 0% 50%;\n }\n 100% {\n transform: scale(0.8);\n transform-origin: 0% 50%;\n opacity: 0;\n }\n}\n\n@keyframes antZoomRightIn {\n 0% {\n transform: scale(0.8);\n transform-origin: 100% 50%;\n opacity: 0;\n }\n 100% {\n transform: scale(1);\n transform-origin: 100% 50%;\n }\n}\n\n@keyframes antZoomRightOut {\n 0% {\n transform: scale(1);\n transform-origin: 100% 50%;\n }\n 100% {\n transform: scale(0.8);\n transform-origin: 100% 50%;\n opacity: 0;\n }\n}\n\n@keyframes antZoomDownIn {\n 0% {\n transform: scale(0.8);\n transform-origin: 50% 100%;\n opacity: 0;\n }\n 100% {\n transform: scale(1);\n transform-origin: 50% 100%;\n }\n}\n\n@keyframes antZoomDownOut {\n 0% {\n transform: scale(1);\n transform-origin: 50% 100%;\n }\n 100% {\n transform: scale(0.8);\n transform-origin: 50% 100%;\n opacity: 0;\n }\n}\n","@import '../mixins/motion';\n@import 'motion/fade';\n@import 'motion/move';\n@import 'motion/other';\n@import 'motion/slide';\n@import 'motion/swing';\n@import 'motion/zoom';\n\n// For common/openAnimation\n.ant-motion-collapse-legacy {\n overflow: hidden;\n &-active {\n transition: height 0.15s @ease-in-out, opacity 0.15s @ease-in-out !important;\n }\n}\n\n.ant-motion-collapse {\n overflow: hidden;\n transition: height 0.15s @ease-in-out, opacity 0.15s @ease-in-out !important;\n}\n","@import '../themes/index';\n\n.reset-component() {\n box-sizing: border-box;\n margin: 0;\n padding: 0;\n color: @text-color;\n font-size: @font-size-base;\n font-variant: @font-variant-base;\n line-height: @line-height-base;\n list-style: none;\n font-feature-settings: @font-feature-settings-base;\n}\n","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n\n@menu-prefix-cls: ~'@{ant-prefix}-menu';\n\n// default theme\n.@{menu-prefix-cls} {\n .reset-component;\n\n margin-bottom: 0;\n padding-left: 0; // Override default ul/ol\n color: @menu-item-color;\n line-height: 0; // Fix display inline-block gap\n list-style: none;\n background: @menu-bg;\n outline: none;\n box-shadow: @box-shadow-base;\n transition: background 0.3s, width 0.2s;\n .clearfix;\n\n ul,\n ol {\n margin: 0;\n padding: 0;\n list-style: none;\n }\n\n &-hidden {\n display: none;\n }\n\n &-item-group-title {\n padding: 8px 16px;\n color: @menu-item-group-title-color;\n font-size: @font-size-base;\n line-height: @line-height-base;\n transition: all 0.3s;\n }\n\n &-submenu,\n &-submenu-inline {\n transition: border-color 0.3s @ease-in-out, background 0.3s @ease-in-out,\n padding 0.15s @ease-in-out;\n }\n\n &-item:active,\n &-submenu-title:active {\n background: @menu-item-active-bg;\n }\n\n &-submenu &-sub {\n cursor: initial;\n transition: background 0.3s @ease-in-out, padding 0.3s @ease-in-out;\n }\n\n &-item > a {\n display: block;\n color: @menu-item-color;\n &:hover {\n color: @menu-highlight-color;\n }\n &::before {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n background-color: transparent;\n content: '';\n }\n }\n\n &-item-divider {\n height: 1px;\n overflow: hidden;\n line-height: 0;\n background-color: @border-color-split;\n }\n\n &-item:hover,\n &-item-active,\n &:not(&-inline) &-submenu-open,\n &-submenu-active,\n &-submenu-title:hover {\n color: @menu-highlight-color;\n }\n\n &-horizontal &-item,\n &-horizontal &-submenu {\n margin-top: -1px;\n }\n\n &-horizontal > &-item:hover,\n &-horizontal > &-item-active,\n &-horizontal > &-submenu &-submenu-title:hover {\n background-color: transparent;\n }\n\n &-item-selected {\n color: @menu-highlight-color;\n > a,\n > a:hover {\n color: @menu-highlight-color;\n }\n }\n\n &:not(&-horizontal) &-item-selected {\n background-color: @menu-item-active-bg;\n }\n\n &-inline,\n &-vertical,\n &-vertical-left {\n border-right: @border-width-base @border-style-base @border-color-split;\n }\n &-vertical-right {\n border-left: @border-width-base @border-style-base @border-color-split;\n }\n\n &-vertical&-sub,\n &-vertical-left&-sub,\n &-vertical-right&-sub {\n min-width: 160px;\n padding: 0;\n border-right: 0;\n transform-origin: 0 0;\n\n .@{menu-prefix-cls}-item {\n left: 0;\n margin-left: 0;\n border-right: 0;\n &::after {\n border-right: 0;\n }\n }\n > .@{menu-prefix-cls}-item,\n > .@{menu-prefix-cls}-submenu {\n transform-origin: 0 0;\n }\n }\n\n &-horizontal&-sub {\n min-width: 114px; // in case of submenu width is too big: https://codesandbox.io/s/qvpwm6mk66\n }\n\n &-item,\n &-submenu-title {\n position: relative;\n display: block;\n margin: 0;\n padding: 0 20px;\n white-space: nowrap;\n cursor: pointer;\n transition: color 0.3s @ease-in-out, border-color 0.3s @ease-in-out,\n background 0.3s @ease-in-out, padding 0.15s @ease-in-out;\n .@{iconfont-css-prefix} {\n min-width: 14px;\n margin-right: 10px;\n font-size: @font-size-base;\n transition: font-size 0.15s @ease-out, margin 0.3s @ease-in-out;\n + span {\n opacity: 1;\n transition: opacity 0.3s @ease-in-out, width 0.3s @ease-in-out;\n }\n }\n }\n\n & > &-item-divider {\n height: 1px;\n margin: 1px 0;\n padding: 0;\n overflow: hidden;\n line-height: 0;\n background-color: @border-color-split;\n }\n\n &-submenu {\n &-popup {\n position: absolute;\n z-index: @zindex-dropdown;\n background: @menu-popup-bg;\n border-radius: @border-radius-base;\n\n .submenu-title-wrapper {\n padding-right: 20px;\n }\n\n &::before {\n position: absolute;\n top: -7px;\n right: 0;\n bottom: 0;\n left: 0;\n opacity: 0.0001;\n content: ' ';\n }\n }\n\n > .@{menu-prefix-cls} {\n background-color: @menu-bg;\n border-radius: @border-radius-base;\n &-submenu-title::after {\n transition: transform 0.3s @ease-in-out;\n }\n }\n\n &-vertical,\n &-vertical-left,\n &-vertical-right,\n &-inline {\n > .@{menu-prefix-cls}-submenu-title .@{menu-prefix-cls}-submenu-arrow {\n position: absolute;\n top: 50%;\n right: 16px;\n width: 10px;\n transition: transform 0.3s @ease-in-out;\n &::before,\n &::after {\n position: absolute;\n width: 6px;\n height: 1.5px;\n // background + background-image to makes before & after cross have same color.\n // Since `linear-gradient` not work on IE9, we should hack it.\n // ref: https://github.com/ant-design/ant-design/issues/15910\n background: @menu-bg;\n background: ~'@{menu-item-color} \\9';\n background-image: linear-gradient(to right, @menu-item-color, @menu-item-color);\n background-image: ~'none \\9';\n border-radius: 2px;\n transition: background 0.3s @ease-in-out, transform 0.3s @ease-in-out,\n top 0.3s @ease-in-out;\n content: '';\n }\n &::before {\n transform: rotate(45deg) translateY(-2px);\n }\n &::after {\n transform: rotate(-45deg) translateY(2px);\n }\n }\n > .@{menu-prefix-cls}-submenu-title:hover .@{menu-prefix-cls}-submenu-arrow {\n &::after,\n &::before {\n background: linear-gradient(to right, @menu-highlight-color, @menu-highlight-color);\n }\n }\n }\n\n &-inline > .@{menu-prefix-cls}-submenu-title .@{menu-prefix-cls}-submenu-arrow {\n &::before {\n transform: rotate(-45deg) translateX(2px);\n }\n &::after {\n transform: rotate(45deg) translateX(-2px);\n }\n }\n\n &-open {\n &.@{menu-prefix-cls}-submenu-inline\n > .@{menu-prefix-cls}-submenu-title\n .@{menu-prefix-cls}-submenu-arrow {\n transform: translateY(-2px);\n &::after {\n transform: rotate(-45deg) translateX(-2px);\n }\n &::before {\n transform: rotate(45deg) translateX(2px);\n }\n }\n }\n }\n\n &-vertical &-submenu-selected,\n &-vertical-left &-submenu-selected,\n &-vertical-right &-submenu-selected {\n color: @menu-highlight-color;\n > a {\n color: @menu-highlight-color;\n }\n }\n\n &-horizontal {\n line-height: 46px;\n white-space: nowrap;\n border: 0;\n border-bottom: @border-width-base @border-style-base @border-color-split;\n box-shadow: none;\n\n > .@{menu-prefix-cls}-item,\n > .@{menu-prefix-cls}-submenu {\n position: relative;\n top: 1px;\n display: inline-block;\n vertical-align: bottom;\n border-bottom: 2px solid transparent;\n\n &:hover,\n &-active,\n &-open,\n &-selected {\n color: @menu-highlight-color;\n border-bottom: 2px solid @menu-highlight-color;\n }\n }\n\n > .@{menu-prefix-cls}-item {\n > a {\n display: block;\n color: @menu-item-color;\n &:hover {\n color: @menu-highlight-color;\n }\n &::before {\n bottom: -2px;\n }\n }\n &-selected > a {\n color: @menu-highlight-color;\n }\n }\n\n &::after {\n display: block;\n clear: both;\n height: 0;\n content: '\\20';\n }\n }\n\n &-vertical,\n &-vertical-left,\n &-vertical-right,\n &-inline {\n .@{menu-prefix-cls}-item {\n position: relative;\n &::after {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n border-right: @menu-item-active-border-width solid @menu-highlight-color;\n transform: scaleY(0.0001);\n opacity: 0;\n transition: transform 0.15s @ease-out, opacity 0.15s @ease-out;\n content: '';\n }\n }\n\n .@{menu-prefix-cls}-item,\n .@{menu-prefix-cls}-submenu-title {\n height: @menu-item-height;\n margin-top: 4px;\n margin-bottom: 4px;\n padding: 0 16px;\n overflow: hidden;\n font-size: @font-size-base;\n line-height: @menu-item-height;\n text-overflow: ellipsis;\n }\n\n // disable margin collapsed\n .@{menu-prefix-cls}-submenu {\n padding-bottom: 0.01px;\n }\n\n .@{menu-prefix-cls}-item:not(:last-child) {\n margin-bottom: 8px;\n }\n\n > .@{menu-prefix-cls}-item,\n > .@{menu-prefix-cls}-submenu > .@{menu-prefix-cls}-submenu-title {\n height: @menu-inline-toplevel-item-height;\n line-height: @menu-inline-toplevel-item-height;\n }\n }\n\n &-inline {\n width: 100%;\n .@{menu-prefix-cls}-selected,\n .@{menu-prefix-cls}-item-selected {\n &::after {\n transform: scaleY(1);\n opacity: 1;\n transition: transform 0.15s @ease-in-out, opacity 0.15s @ease-in-out;\n }\n }\n\n .@{menu-prefix-cls}-item,\n .@{menu-prefix-cls}-submenu-title {\n width: ~'calc(100% + 1px)';\n }\n\n .@{menu-prefix-cls}-submenu-title {\n padding-right: 34px;\n }\n }\n\n &-inline-collapsed {\n width: @menu-collapsed-width;\n > .@{menu-prefix-cls}-item,\n > .@{menu-prefix-cls}-item-group\n > .@{menu-prefix-cls}-item-group-list\n > .@{menu-prefix-cls}-item,\n > .@{menu-prefix-cls}-item-group\n > .@{menu-prefix-cls}-item-group-list\n > .@{menu-prefix-cls}-submenu\n > .@{menu-prefix-cls}-submenu-title,\n > .@{menu-prefix-cls}-submenu > .@{menu-prefix-cls}-submenu-title {\n left: 0;\n padding: 0 (@menu-collapsed-width - 16px) / 2 !important;\n text-overflow: clip;\n .@{menu-prefix-cls}-submenu-arrow {\n display: none;\n }\n .@{iconfont-css-prefix} {\n margin: 0;\n font-size: 16px;\n line-height: @menu-item-height;\n + span {\n display: inline-block;\n max-width: 0;\n opacity: 0;\n }\n }\n }\n &-tooltip {\n pointer-events: none;\n .@{iconfont-css-prefix} {\n display: none;\n }\n a {\n color: @text-color-dark;\n }\n }\n\n .@{menu-prefix-cls}-item-group-title {\n padding-right: 4px;\n padding-left: 4px;\n overflow: hidden;\n white-space: nowrap;\n text-overflow: ellipsis;\n }\n }\n\n &-item-group-list {\n margin: 0;\n padding: 0;\n .@{menu-prefix-cls}-item,\n .@{menu-prefix-cls}-submenu-title {\n padding: 0 16px 0 28px;\n }\n }\n\n &-root&-vertical,\n &-root&-vertical-left,\n &-root&-vertical-right,\n &-root&-inline {\n box-shadow: none;\n }\n\n &-sub&-inline {\n padding: 0;\n border: 0;\n border-radius: 0;\n box-shadow: none;\n & > .@{menu-prefix-cls}-item,\n & > .@{menu-prefix-cls}-submenu > .@{menu-prefix-cls}-submenu-title {\n height: @menu-item-height;\n line-height: @menu-item-height;\n list-style-position: inside;\n list-style-type: disc;\n }\n\n & .@{menu-prefix-cls}-item-group-title {\n padding-left: 32px;\n }\n }\n\n // Disabled state sets text to gray and nukes hover/tab effects\n &-item-disabled,\n &-submenu-disabled {\n color: @disabled-color !important;\n background: none;\n border-color: transparent !important;\n cursor: not-allowed;\n > a {\n color: @disabled-color !important;\n pointer-events: none;\n }\n > .@{menu-prefix-cls}-submenu-title {\n color: @disabled-color !important;\n cursor: not-allowed;\n > .@{menu-prefix-cls}-submenu-arrow {\n &::before,\n &::after {\n background: @disabled-color !important;\n }\n }\n }\n }\n}\n\n@import './dark';\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;",".@{menu-prefix-cls} {\n // dark theme\n &-dark,\n &-dark &-sub {\n color: @menu-dark-color;\n background: @menu-dark-bg;\n .@{menu-prefix-cls}-submenu-title .@{menu-prefix-cls}-submenu-arrow {\n opacity: 0.45;\n transition: all 0.3s;\n &::after,\n &::before {\n background: @menu-dark-arrow-color;\n }\n }\n }\n\n &-dark&-submenu-popup {\n background: transparent;\n }\n\n &-dark &-inline&-sub {\n background: @menu-dark-submenu-bg;\n box-shadow: 0 2px 8px fade(@black, 45%) inset;\n }\n\n &-dark&-horizontal {\n border-bottom: 0;\n }\n\n &-dark&-horizontal > &-item,\n &-dark&-horizontal > &-submenu {\n top: 0;\n margin-top: 0;\n border-color: @menu-dark-bg;\n border-bottom: 0;\n }\n\n &-dark&-horizontal > &-item > a::before {\n bottom: 0;\n }\n\n &-dark &-item,\n &-dark &-item-group-title,\n &-dark &-item > a {\n color: @menu-dark-color;\n }\n\n &-dark&-inline,\n &-dark&-vertical,\n &-dark&-vertical-left,\n &-dark&-vertical-right {\n border-right: 0;\n }\n\n &-dark&-inline &-item,\n &-dark&-vertical &-item,\n &-dark&-vertical-left &-item,\n &-dark&-vertical-right &-item {\n left: 0;\n margin-left: 0;\n border-right: 0;\n &::after {\n border-right: 0;\n }\n }\n\n &-dark&-inline &-item,\n &-dark&-inline &-submenu-title {\n width: 100%;\n }\n\n &-dark &-item:hover,\n &-dark &-item-active,\n &-dark &-submenu-active,\n &-dark &-submenu-open,\n &-dark &-submenu-selected,\n &-dark &-submenu-title:hover {\n color: @menu-dark-highlight-color;\n background-color: transparent;\n > a {\n color: @menu-dark-highlight-color;\n }\n > .@{menu-prefix-cls}-submenu-title,\n > .@{menu-prefix-cls}-submenu-title:hover {\n > .@{menu-prefix-cls}-submenu-arrow {\n opacity: 1;\n &::after,\n &::before {\n background: @menu-dark-highlight-color;\n }\n }\n }\n }\n\n &-dark &-item-selected {\n color: @menu-dark-highlight-color;\n border-right: 0;\n &::after {\n border-right: 0;\n }\n > a,\n > a:hover {\n color: @menu-dark-highlight-color;\n }\n }\n\n &&-dark &-item-selected,\n &-submenu-popup&-dark &-item-selected {\n background-color: @menu-dark-item-active-bg;\n }\n\n // Disabled state sets text to dark gray and nukes hover/tab effects\n &-dark &-item-disabled,\n &-dark &-submenu-disabled {\n &,\n > a {\n color: @disabled-color-dark !important;\n opacity: 0.8;\n }\n > .@{menu-prefix-cls}-submenu-title {\n color: @disabled-color-dark !important;\n > .@{menu-prefix-cls}-submenu-arrow {\n &::before,\n &::after {\n background: @disabled-color-dark !important;\n }\n }\n }\n }\n}\n","/* stylelint-disable declaration-bang-space-before,no-duplicate-selectors,string-no-newline */\n.tinyColorMixin() {\n@functions: ~`(function() {\n// TinyColor v1.4.1\n// https://github.com/bgrins/TinyColor\n// 2016-07-07, Brian Grinstead, MIT License\nvar trimLeft = /^\\s+/,\n trimRight = /\\s+$/,\n tinyCounter = 0,\n mathRound = Math.round,\n mathMin = Math.min,\n mathMax = Math.max,\n mathRandom = Math.random;\n\nfunction tinycolor (color, opts) {\n\n color = (color) ? color : '';\n opts = opts || { };\n\n // If input is already a tinycolor, return itself\n if (color instanceof tinycolor) {\n return color;\n }\n // If we are called as a function, call using new instead\n if (!(this instanceof tinycolor)) {\n return new tinycolor(color, opts);\n }\n\n var rgb = inputToRGB(color);\n this._originalInput = color,\n this._r = rgb.r,\n this._g = rgb.g,\n this._b = rgb.b,\n this._a = rgb.a,\n this._roundA = mathRound(100*this._a) / 100,\n this._format = opts.format || rgb.format;\n this._gradientType = opts.gradientType;\n\n // Don't let the range of [0,255] come back in [0,1].\n // Potentially lose a little bit of precision here, but will fix issues where\n // .5 gets interpreted as half of the total, instead of half of 1\n // If it was supposed to be 128, this was already taken care of by inputToRgb\n if (this._r < 1) { this._r = mathRound(this._r); }\n if (this._g < 1) { this._g = mathRound(this._g); }\n if (this._b < 1) { this._b = mathRound(this._b); }\n\n this._ok = rgb.ok;\n this._tc_id = tinyCounter++;\n}\n\ntinycolor.prototype = {\n isDark: function() {\n return this.getBrightness() < 128;\n },\n isLight: function() {\n return !this.isDark();\n },\n isValid: function() {\n return this._ok;\n },\n getOriginalInput: function() {\n return this._originalInput;\n },\n getFormat: function() {\n return this._format;\n },\n getAlpha: function() {\n return this._a;\n },\n getBrightness: function() {\n //http://www.w3.org/TR/AERT#color-contrast\n var rgb = this.toRgb();\n return (rgb.r * 299 + rgb.g * 587 + rgb.b * 114) / 1000;\n },\n getLuminance: function() {\n //http://www.w3.org/TR/2008/REC-WCAG20-20081211/#relativeluminancedef\n var rgb = this.toRgb();\n var RsRGB, GsRGB, BsRGB, R, G, B;\n RsRGB = rgb.r/255;\n GsRGB = rgb.g/255;\n BsRGB = rgb.b/255;\n\n if (RsRGB <= 0.03928) {R = RsRGB / 12.92;} else {R = Math.pow(((RsRGB + 0.055) / 1.055), 2.4);}\n if (GsRGB <= 0.03928) {G = GsRGB / 12.92;} else {G = Math.pow(((GsRGB + 0.055) / 1.055), 2.4);}\n if (BsRGB <= 0.03928) {B = BsRGB / 12.92;} else {B = Math.pow(((BsRGB + 0.055) / 1.055), 2.4);}\n return (0.2126 * R) + (0.7152 * G) + (0.0722 * B);\n },\n setAlpha: function(value) {\n this._a = boundAlpha(value);\n this._roundA = mathRound(100*this._a) / 100;\n return this;\n },\n toHsv: function() {\n var hsv = rgbToHsv(this._r, this._g, this._b);\n return { h: hsv.h * 360, s: hsv.s, v: hsv.v, a: this._a };\n },\n toHsvString: function() {\n var hsv = rgbToHsv(this._r, this._g, this._b);\n var h = mathRound(hsv.h * 360), s = mathRound(hsv.s * 100), v = mathRound(hsv.v * 100);\n return (this._a == 1) ?\n \"hsv(\" + h + \", \" + s + \"%, \" + v + \"%)\" :\n \"hsva(\" + h + \", \" + s + \"%, \" + v + \"%, \"+ this._roundA + \")\";\n },\n toHsl: function() {\n var hsl = rgbToHsl(this._r, this._g, this._b);\n return { h: hsl.h * 360, s: hsl.s, l: hsl.l, a: this._a };\n },\n toHslString: function() {\n var hsl = rgbToHsl(this._r, this._g, this._b);\n var h = mathRound(hsl.h * 360), s = mathRound(hsl.s * 100), l = mathRound(hsl.l * 100);\n return (this._a == 1) ?\n \"hsl(\" + h + \", \" + s + \"%, \" + l + \"%)\" :\n \"hsla(\" + h + \", \" + s + \"%, \" + l + \"%, \"+ this._roundA + \")\";\n },\n toHex: function(allow3Char) {\n return rgbToHex(this._r, this._g, this._b, allow3Char);\n },\n toHexString: function(allow3Char) {\n return '#' + this.toHex(allow3Char);\n },\n toHex8: function(allow4Char) {\n return rgbaToHex(this._r, this._g, this._b, this._a, allow4Char);\n },\n toHex8String: function(allow4Char) {\n return '#' + this.toHex8(allow4Char);\n },\n toRgb: function() {\n return { r: mathRound(this._r), g: mathRound(this._g), b: mathRound(this._b), a: this._a };\n },\n toRgbString: function() {\n return (this._a == 1) ?\n \"rgb(\" + mathRound(this._r) + \", \" + mathRound(this._g) + \", \" + mathRound(this._b) + \")\" :\n \"rgba(\" + mathRound(this._r) + \", \" + mathRound(this._g) + \", \" + mathRound(this._b) + \", \" + this._roundA + \")\";\n },\n toPercentageRgb: function() {\n return { r: mathRound(bound01(this._r, 255) * 100) + \"%\", g: mathRound(bound01(this._g, 255) * 100) + \"%\", b: mathRound(bound01(this._b, 255) * 100) + \"%\", a: this._a };\n },\n toPercentageRgbString: function() {\n return (this._a == 1) ?\n \"rgb(\" + mathRound(bound01(this._r, 255) * 100) + \"%, \" + mathRound(bound01(this._g, 255) * 100) + \"%, \" + mathRound(bound01(this._b, 255) * 100) + \"%)\" :\n \"rgba(\" + mathRound(bound01(this._r, 255) * 100) + \"%, \" + mathRound(bound01(this._g, 255) * 100) + \"%, \" + mathRound(bound01(this._b, 255) * 100) + \"%, \" + this._roundA + \")\";\n },\n toName: function() {\n if (this._a === 0) {\n return \"transparent\";\n }\n\n if (this._a < 1) {\n return false;\n }\n\n return hexNames[rgbToHex(this._r, this._g, this._b, true)] || false;\n },\n toFilter: function(secondColor) {\n var hex8String = '#' + rgbaToArgbHex(this._r, this._g, this._b, this._a);\n var secondHex8String = hex8String;\n var gradientType = this._gradientType ? \"GradientType = 1, \" : \"\";\n\n if (secondColor) {\n var s = tinycolor(secondColor);\n secondHex8String = '#' + rgbaToArgbHex(s._r, s._g, s._b, s._a);\n }\n\n return \"progid:DXImageTransform.Microsoft.gradient(\"+gradientType+\"startColorstr=\"+hex8String+\",endColorstr=\"+secondHex8String+\")\";\n },\n toString: function(format) {\n var formatSet = !!format;\n format = format || this._format;\n\n var formattedString = false;\n var hasAlpha = this._a < 1 && this._a >= 0;\n var needsAlphaFormat = !formatSet && hasAlpha && (format === \"hex\" || format === \"hex6\" || format === \"hex3\" || format === \"hex4\" || format === \"hex8\" || format === \"name\");\n\n if (needsAlphaFormat) {\n // Special case for \"transparent\", all other non-alpha formats\n // will return rgba when there is transparency.\n if (format === \"name\" && this._a === 0) {\n return this.toName();\n }\n return this.toRgbString();\n }\n if (format === \"rgb\") {\n formattedString = this.toRgbString();\n }\n if (format === \"prgb\") {\n formattedString = this.toPercentageRgbString();\n }\n if (format === \"hex\" || format === \"hex6\") {\n formattedString = this.toHexString();\n }\n if (format === \"hex3\") {\n formattedString = this.toHexString(true);\n }\n if (format === \"hex4\") {\n formattedString = this.toHex8String(true);\n }\n if (format === \"hex8\") {\n formattedString = this.toHex8String();\n }\n if (format === \"name\") {\n formattedString = this.toName();\n }\n if (format === \"hsl\") {\n formattedString = this.toHslString();\n }\n if (format === \"hsv\") {\n formattedString = this.toHsvString();\n }\n\n return formattedString || this.toHexString();\n },\n clone: function() {\n return tinycolor(this.toString());\n },\n\n _applyModification: function(fn, args) {\n var color = fn.apply(null, [this].concat([].slice.call(args)));\n this._r = color._r;\n this._g = color._g;\n this._b = color._b;\n this.setAlpha(color._a);\n return this;\n },\n lighten: function() {\n return this._applyModification(lighten, arguments);\n },\n brighten: function() {\n return this._applyModification(brighten, arguments);\n },\n darken: function() {\n return this._applyModification(darken, arguments);\n },\n desaturate: function() {\n return this._applyModification(desaturate, arguments);\n },\n saturate: function() {\n return this._applyModification(saturate, arguments);\n },\n greyscale: function() {\n return this._applyModification(greyscale, arguments);\n },\n spin: function() {\n return this._applyModification(spin, arguments);\n },\n\n _applyCombination: function(fn, args) {\n return fn.apply(null, [this].concat([].slice.call(args)));\n },\n analogous: function() {\n return this._applyCombination(analogous, arguments);\n },\n complement: function() {\n return this._applyCombination(complement, arguments);\n },\n monochromatic: function() {\n return this._applyCombination(monochromatic, arguments);\n },\n splitcomplement: function() {\n return this._applyCombination(splitcomplement, arguments);\n },\n triad: function() {\n return this._applyCombination(triad, arguments);\n },\n tetrad: function() {\n return this._applyCombination(tetrad, arguments);\n }\n};\n\n// If input is an object, force 1 into \"1.0\" to handle ratios properly\n// String input requires \"1.0\" as input, so 1 will be treated as 1\ntinycolor.fromRatio = function(color, opts) {\n if (typeof color == \"object\") {\n var newColor = {};\n for (var i in color) {\n if (color.hasOwnProperty(i)) {\n if (i === \"a\") {\n newColor[i] = color[i];\n }\n else {\n newColor[i] = convertToPercentage(color[i]);\n }\n }\n }\n color = newColor;\n }\n\n return tinycolor(color, opts);\n};\n\n// Given a string or object, convert that input to RGB\n// Possible string inputs:\n//\n// \"red\"\n// \"#f00\" or \"f00\"\n// \"#ff0000\" or \"ff0000\"\n// \"#ff000000\" or \"ff000000\"\n// \"rgb 255 0 0\" or \"rgb (255, 0, 0)\"\n// \"rgb 1.0 0 0\" or \"rgb (1, 0, 0)\"\n// \"rgba (255, 0, 0, 1)\" or \"rgba 255, 0, 0, 1\"\n// \"rgba (1.0, 0, 0, 1)\" or \"rgba 1.0, 0, 0, 1\"\n// \"hsl(0, 100%, 50%)\" or \"hsl 0 100% 50%\"\n// \"hsla(0, 100%, 50%, 1)\" or \"hsla 0 100% 50%, 1\"\n// \"hsv(0, 100%, 100%)\" or \"hsv 0 100% 100%\"\n//\nfunction inputToRGB(color) {\n\n var rgb = { r: 0, g: 0, b: 0 };\n var a = 1;\n var s = null;\n var v = null;\n var l = null;\n var ok = false;\n var format = false;\n\n if (typeof color == \"string\") {\n color = stringInputToObject(color);\n }\n\n if (typeof color == \"object\") {\n if (isValidCSSUnit(color.r) && isValidCSSUnit(color.g) && isValidCSSUnit(color.b)) {\n rgb = rgbToRgb(color.r, color.g, color.b);\n ok = true;\n format = String(color.r).substr(-1) === \"%\" ? \"prgb\" : \"rgb\";\n }\n else if (isValidCSSUnit(color.h) && isValidCSSUnit(color.s) && isValidCSSUnit(color.v)) {\n s = convertToPercentage(color.s);\n v = convertToPercentage(color.v);\n rgb = hsvToRgb(color.h, s, v);\n ok = true;\n format = \"hsv\";\n }\n else if (isValidCSSUnit(color.h) && isValidCSSUnit(color.s) && isValidCSSUnit(color.l)) {\n s = convertToPercentage(color.s);\n l = convertToPercentage(color.l);\n rgb = hslToRgb(color.h, s, l);\n ok = true;\n format = \"hsl\";\n }\n\n if (color.hasOwnProperty(\"a\")) {\n a = color.a;\n }\n }\n\n a = boundAlpha(a);\n\n return {\n ok: ok,\n format: color.format || format,\n r: mathMin(255, mathMax(rgb.r, 0)),\n g: mathMin(255, mathMax(rgb.g, 0)),\n b: mathMin(255, mathMax(rgb.b, 0)),\n a: a\n };\n}\n\n// Conversion Functions\n// --------------------\n\n// rgbToHsl, rgbToHsv, hslToRgb, hsvToRgb modified from:\n// \n\n// rgbToRgb\n// Handle bounds / percentage checking to conform to CSS color spec\n// \n// *Assumes:* r, g, b in [0, 255] or [0, 1]\n// *Returns:* { r, g, b } in [0, 255]\nfunction rgbToRgb(r, g, b){\n return {\n r: bound01(r, 255) * 255,\n g: bound01(g, 255) * 255,\n b: bound01(b, 255) * 255\n };\n}\n\n// rgbToHsl\n// Converts an RGB color value to HSL.\n// *Assumes:* r, g, and b are contained in [0, 255] or [0, 1]\n// *Returns:* { h, s, l } in [0,1]\nfunction rgbToHsl(r, g, b) {\n\n r = bound01(r, 255);\n g = bound01(g, 255);\n b = bound01(b, 255);\n\n var max = mathMax(r, g, b), min = mathMin(r, g, b);\n var h, s, l = (max + min) / 2;\n\n if(max == min) {\n h = s = 0; // achromatic\n }\n else {\n var d = max - min;\n s = l > 0.5 ? d / (2 - max - min) : d / (max + min);\n switch(max) {\n case r: h = (g - b) / d + (g < b ? 6 : 0); break;\n case g: h = (b - r) / d + 2; break;\n case b: h = (r - g) / d + 4; break;\n }\n\n h /= 6;\n }\n\n return { h: h, s: s, l: l };\n}\n\n// hslToRgb\n// Converts an HSL color value to RGB.\n// *Assumes:* h is contained in [0, 1] or [0, 360] and s and l are contained [0, 1] or [0, 100]\n// *Returns:* { r, g, b } in the set [0, 255]\nfunction hslToRgb(h, s, l) {\n var r, g, b;\n\n h = bound01(h, 360);\n s = bound01(s, 100);\n l = bound01(l, 100);\n\n function hue2rgb(p, q, t) {\n if(t < 0) t += 1;\n if(t > 1) t -= 1;\n if(t < 1/6) return p + (q - p) * 6 * t;\n if(t < 1/2) return q;\n if(t < 2/3) return p + (q - p) * (2/3 - t) * 6;\n return p;\n }\n\n if(s === 0) {\n r = g = b = l; // achromatic\n }\n else {\n var q = l < 0.5 ? l * (1 + s) : l + s - l * s;\n var p = 2 * l - q;\n r = hue2rgb(p, q, h + 1/3);\n g = hue2rgb(p, q, h);\n b = hue2rgb(p, q, h - 1/3);\n }\n\n return { r: r * 255, g: g * 255, b: b * 255 };\n}\n\n// rgbToHsv\n// Converts an RGB color value to HSV\n// *Assumes:* r, g, and b are contained in the set [0, 255] or [0, 1]\n// *Returns:* { h, s, v } in [0,1]\nfunction rgbToHsv(r, g, b) {\n\n r = bound01(r, 255);\n g = bound01(g, 255);\n b = bound01(b, 255);\n\n var max = mathMax(r, g, b), min = mathMin(r, g, b);\n var h, s, v = max;\n\n var d = max - min;\n s = max === 0 ? 0 : d / max;\n\n if(max == min) {\n h = 0; // achromatic\n }\n else {\n switch(max) {\n case r: h = (g - b) / d + (g < b ? 6 : 0); break;\n case g: h = (b - r) / d + 2; break;\n case b: h = (r - g) / d + 4; break;\n }\n h /= 6;\n }\n return { h: h, s: s, v: v };\n}\n\n// hsvToRgb\n// Converts an HSV color value to RGB.\n// *Assumes:* h is contained in [0, 1] or [0, 360] and s and v are contained in [0, 1] or [0, 100]\n// *Returns:* { r, g, b } in the set [0, 255]\n function hsvToRgb(h, s, v) {\n\n h = bound01(h, 360) * 6;\n s = bound01(s, 100);\n v = bound01(v, 100);\n\n var i = Math.floor(h),\n f = h - i,\n p = v * (1 - s),\n q = v * (1 - f * s),\n t = v * (1 - (1 - f) * s),\n mod = i % 6,\n r = [v, q, p, p, t, v][mod],\n g = [t, v, v, q, p, p][mod],\n b = [p, p, t, v, v, q][mod];\n\n return { r: r * 255, g: g * 255, b: b * 255 };\n}\n\n// rgbToHex\n// Converts an RGB color to hex\n// Assumes r, g, and b are contained in the set [0, 255]\n// Returns a 3 or 6 character hex\nfunction rgbToHex(r, g, b, allow3Char) {\n\n var hex = [\n pad2(mathRound(r).toString(16)),\n pad2(mathRound(g).toString(16)),\n pad2(mathRound(b).toString(16))\n ];\n\n // Return a 3 character hex if possible\n if (allow3Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1)) {\n return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0);\n }\n\n return hex.join(\"\");\n}\n\n// rgbaToHex\n// Converts an RGBA color plus alpha transparency to hex\n// Assumes r, g, b are contained in the set [0, 255] and\n// a in [0, 1]. Returns a 4 or 8 character rgba hex\nfunction rgbaToHex(r, g, b, a, allow4Char) {\n\n var hex = [\n pad2(mathRound(r).toString(16)),\n pad2(mathRound(g).toString(16)),\n pad2(mathRound(b).toString(16)),\n pad2(convertDecimalToHex(a))\n ];\n\n // Return a 4 character hex if possible\n if (allow4Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1) && hex[3].charAt(0) == hex[3].charAt(1)) {\n return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0) + hex[3].charAt(0);\n }\n\n return hex.join(\"\");\n}\n\n// rgbaToArgbHex\n// Converts an RGBA color to an ARGB Hex8 string\n// Rarely used, but required for \"toFilter()\"\nfunction rgbaToArgbHex(r, g, b, a) {\n\n var hex = [\n pad2(convertDecimalToHex(a)),\n pad2(mathRound(r).toString(16)),\n pad2(mathRound(g).toString(16)),\n pad2(mathRound(b).toString(16))\n ];\n\n return hex.join(\"\");\n}\n\n// equals\n// Can be called with any tinycolor input\ntinycolor.equals = function (color1, color2) {\n if (!color1 || !color2) { return false; }\n return tinycolor(color1).toRgbString() == tinycolor(color2).toRgbString();\n};\n\ntinycolor.random = function() {\n return tinycolor.fromRatio({\n r: mathRandom(),\n g: mathRandom(),\n b: mathRandom()\n });\n};\n\n// Modification Functions\n// ----------------------\n// Thanks to less.js for some of the basics here\n// \n\nfunction desaturate(color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.s -= amount / 100;\n hsl.s = clamp01(hsl.s);\n return tinycolor(hsl);\n}\n\nfunction saturate(color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.s += amount / 100;\n hsl.s = clamp01(hsl.s);\n return tinycolor(hsl);\n}\n\nfunction greyscale(color) {\n return tinycolor(color).desaturate(100);\n}\n\nfunction lighten (color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.l += amount / 100;\n hsl.l = clamp01(hsl.l);\n return tinycolor(hsl);\n}\n\nfunction brighten(color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var rgb = tinycolor(color).toRgb();\n rgb.r = mathMax(0, mathMin(255, rgb.r - mathRound(255 * - (amount / 100))));\n rgb.g = mathMax(0, mathMin(255, rgb.g - mathRound(255 * - (amount / 100))));\n rgb.b = mathMax(0, mathMin(255, rgb.b - mathRound(255 * - (amount / 100))));\n return tinycolor(rgb);\n}\n\nfunction darken (color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.l -= amount / 100;\n hsl.l = clamp01(hsl.l);\n return tinycolor(hsl);\n}\n\n// Spin takes a positive or negative amount within [-360, 360] indicating the change of hue.\n// Values outside of this range will be wrapped into this range.\nfunction spin(color, amount) {\n var hsl = tinycolor(color).toHsl();\n var hue = (hsl.h + amount) % 360;\n hsl.h = hue < 0 ? 360 + hue : hue;\n return tinycolor(hsl);\n}\n\n// Combination Functions\n// ---------------------\n// Thanks to jQuery xColor for some of the ideas behind these\n// \n\nfunction complement(color) {\n var hsl = tinycolor(color).toHsl();\n hsl.h = (hsl.h + 180) % 360;\n return tinycolor(hsl);\n}\n\nfunction triad(color) {\n var hsl = tinycolor(color).toHsl();\n var h = hsl.h;\n return [\n tinycolor(color),\n tinycolor({ h: (h + 120) % 360, s: hsl.s, l: hsl.l }),\n tinycolor({ h: (h + 240) % 360, s: hsl.s, l: hsl.l })\n ];\n}\n\nfunction tetrad(color) {\n var hsl = tinycolor(color).toHsl();\n var h = hsl.h;\n return [\n tinycolor(color),\n tinycolor({ h: (h + 90) % 360, s: hsl.s, l: hsl.l }),\n tinycolor({ h: (h + 180) % 360, s: hsl.s, l: hsl.l }),\n tinycolor({ h: (h + 270) % 360, s: hsl.s, l: hsl.l })\n ];\n}\n\nfunction splitcomplement(color) {\n var hsl = tinycolor(color).toHsl();\n var h = hsl.h;\n return [\n tinycolor(color),\n tinycolor({ h: (h + 72) % 360, s: hsl.s, l: hsl.l}),\n tinycolor({ h: (h + 216) % 360, s: hsl.s, l: hsl.l})\n ];\n}\n\nfunction analogous(color, results, slices) {\n results = results || 6;\n slices = slices || 30;\n\n var hsl = tinycolor(color).toHsl();\n var part = 360 / slices;\n var ret = [tinycolor(color)];\n\n for (hsl.h = ((hsl.h - (part * results >> 1)) + 720) % 360; --results; ) {\n hsl.h = (hsl.h + part) % 360;\n ret.push(tinycolor(hsl));\n }\n return ret;\n}\n\nfunction monochromatic(color, results) {\n results = results || 6;\n var hsv = tinycolor(color).toHsv();\n var h = hsv.h, s = hsv.s, v = hsv.v;\n var ret = [];\n var modification = 1 / results;\n\n while (results--) {\n ret.push(tinycolor({ h: h, s: s, v: v}));\n v = (v + modification) % 1;\n }\n\n return ret;\n}\n\n// Utility Functions\n// ---------------------\n\ntinycolor.mix = function(color1, color2, amount) {\n amount = (amount === 0) ? 0 : (amount || 50);\n\n var rgb1 = tinycolor(color1).toRgb();\n var rgb2 = tinycolor(color2).toRgb();\n\n var p = amount / 100;\n\n var rgba = {\n r: ((rgb2.r - rgb1.r) * p) + rgb1.r,\n g: ((rgb2.g - rgb1.g) * p) + rgb1.g,\n b: ((rgb2.b - rgb1.b) * p) + rgb1.b,\n a: ((rgb2.a - rgb1.a) * p) + rgb1.a\n };\n\n return tinycolor(rgba);\n};\n\n// Readability Functions\n// ---------------------\n// false\n// tinycolor.isReadable(\"#000\", \"#111\",{level:\"AA\",size:\"large\"}) => false\ntinycolor.isReadable = function(color1, color2, wcag2) {\n var readability = tinycolor.readability(color1, color2);\n var wcag2Parms, out;\n\n out = false;\n\n wcag2Parms = validateWCAG2Parms(wcag2);\n switch (wcag2Parms.level + wcag2Parms.size) {\n case \"AAsmall\":\n case \"AAAlarge\":\n out = readability >= 4.5;\n break;\n case \"AAlarge\":\n out = readability >= 3;\n break;\n case \"AAAsmall\":\n out = readability >= 7;\n break;\n }\n return out;\n\n};\n\n// mostReadable\n// Given a base color and a list of possible foreground or background\n// colors for that base, returns the most readable color.\n// Optionally returns Black or White if the most readable color is unreadable.\n// *Example*\n// tinycolor.mostReadable(tinycolor.mostReadable(\"#123\", [\"#124\", \"#125\"],{includeFallbackColors:false}).toHexString(); // \"#112255\"\n// tinycolor.mostReadable(tinycolor.mostReadable(\"#123\", [\"#124\", \"#125\"],{includeFallbackColors:true}).toHexString(); // \"#ffffff\"\n// tinycolor.mostReadable(\"#a8015a\", [\"#faf3f3\"],{includeFallbackColors:true,level:\"AAA\",size:\"large\"}).toHexString(); // \"#faf3f3\"\n// tinycolor.mostReadable(\"#a8015a\", [\"#faf3f3\"],{includeFallbackColors:true,level:\"AAA\",size:\"small\"}).toHexString(); // \"#ffffff\"\ntinycolor.mostReadable = function(baseColor, colorList, args) {\n var bestColor = null;\n var bestScore = 0;\n var readability;\n var includeFallbackColors, level, size ;\n args = args || {};\n includeFallbackColors = args.includeFallbackColors ;\n level = args.level;\n size = args.size;\n\n for (var i= 0; i < colorList.length ; i++) {\n readability = tinycolor.readability(baseColor, colorList[i]);\n if (readability > bestScore) {\n bestScore = readability;\n bestColor = tinycolor(colorList[i]);\n }\n }\n\n if (tinycolor.isReadable(baseColor, bestColor, {\"level\":level,\"size\":size}) || !includeFallbackColors) {\n return bestColor;\n }\n else {\n args.includeFallbackColors=false;\n return tinycolor.mostReadable(baseColor,[\"#fff\", \"#000\"],args);\n }\n};\n\n// Big List of Colors\n// ------------------\n// \nvar names = tinycolor.names = {\n aliceblue: \"f0f8ff\",\n antiquewhite: \"faebd7\",\n aqua: \"0ff\",\n aquamarine: \"7fffd4\",\n azure: \"f0ffff\",\n beige: \"f5f5dc\",\n bisque: \"ffe4c4\",\n black: \"000\",\n blanchedalmond: \"ffebcd\",\n blue: \"00f\",\n blueviolet: \"8a2be2\",\n brown: \"a52a2a\",\n burlywood: \"deb887\",\n burntsienna: \"ea7e5d\",\n cadetblue: \"5f9ea0\",\n chartreuse: \"7fff00\",\n chocolate: \"d2691e\",\n coral: \"ff7f50\",\n cornflowerblue: \"6495ed\",\n cornsilk: \"fff8dc\",\n crimson: \"dc143c\",\n cyan: \"0ff\",\n darkblue: \"00008b\",\n darkcyan: \"008b8b\",\n darkgoldenrod: \"b8860b\",\n darkgray: \"a9a9a9\",\n darkgreen: \"006400\",\n darkgrey: \"a9a9a9\",\n darkkhaki: \"bdb76b\",\n darkmagenta: \"8b008b\",\n darkolivegreen: \"556b2f\",\n darkorange: \"ff8c00\",\n darkorchid: \"9932cc\",\n darkred: \"8b0000\",\n darksalmon: \"e9967a\",\n darkseagreen: \"8fbc8f\",\n darkslateblue: \"483d8b\",\n darkslategray: \"2f4f4f\",\n darkslategrey: \"2f4f4f\",\n darkturquoise: \"00ced1\",\n darkviolet: \"9400d3\",\n deeppink: \"ff1493\",\n deepskyblue: \"00bfff\",\n dimgray: \"696969\",\n dimgrey: \"696969\",\n dodgerblue: \"1e90ff\",\n firebrick: \"b22222\",\n floralwhite: \"fffaf0\",\n forestgreen: \"228b22\",\n fuchsia: \"f0f\",\n gainsboro: \"dcdcdc\",\n ghostwhite: \"f8f8ff\",\n gold: \"ffd700\",\n goldenrod: \"daa520\",\n gray: \"808080\",\n green: \"008000\",\n greenyellow: \"adff2f\",\n grey: \"808080\",\n honeydew: \"f0fff0\",\n hotpink: \"ff69b4\",\n indianred: \"cd5c5c\",\n indigo: \"4b0082\",\n ivory: \"fffff0\",\n khaki: \"f0e68c\",\n lavender: \"e6e6fa\",\n lavenderblush: \"fff0f5\",\n lawngreen: \"7cfc00\",\n lemonchiffon: \"fffacd\",\n lightblue: \"add8e6\",\n lightcoral: \"f08080\",\n lightcyan: \"e0ffff\",\n lightgoldenrodyellow: \"fafad2\",\n lightgray: \"d3d3d3\",\n lightgreen: \"90ee90\",\n lightgrey: \"d3d3d3\",\n lightpink: \"ffb6c1\",\n lightsalmon: \"ffa07a\",\n lightseagreen: \"20b2aa\",\n lightskyblue: \"87cefa\",\n lightslategray: \"789\",\n lightslategrey: \"789\",\n lightsteelblue: \"b0c4de\",\n lightyellow: \"ffffe0\",\n lime: \"0f0\",\n limegreen: \"32cd32\",\n linen: \"faf0e6\",\n magenta: \"f0f\",\n maroon: \"800000\",\n mediumaquamarine: \"66cdaa\",\n mediumblue: \"0000cd\",\n mediumorchid: \"ba55d3\",\n mediumpurple: \"9370db\",\n mediumseagreen: \"3cb371\",\n mediumslateblue: \"7b68ee\",\n mediumspringgreen: \"00fa9a\",\n mediumturquoise: \"48d1cc\",\n mediumvioletred: \"c71585\",\n midnightblue: \"191970\",\n mintcream: \"f5fffa\",\n mistyrose: \"ffe4e1\",\n moccasin: \"ffe4b5\",\n navajowhite: \"ffdead\",\n navy: \"000080\",\n oldlace: \"fdf5e6\",\n olive: \"808000\",\n olivedrab: \"6b8e23\",\n orange: \"ffa500\",\n orangered: \"ff4500\",\n orchid: \"da70d6\",\n palegoldenrod: \"eee8aa\",\n palegreen: \"98fb98\",\n paleturquoise: \"afeeee\",\n palevioletred: \"db7093\",\n papayawhip: \"ffefd5\",\n peachpuff: \"ffdab9\",\n peru: \"cd853f\",\n pink: \"ffc0cb\",\n plum: \"dda0dd\",\n powderblue: \"b0e0e6\",\n purple: \"800080\",\n rebeccapurple: \"663399\",\n red: \"f00\",\n rosybrown: \"bc8f8f\",\n royalblue: \"4169e1\",\n saddlebrown: \"8b4513\",\n salmon: \"fa8072\",\n sandybrown: \"f4a460\",\n seagreen: \"2e8b57\",\n seashell: \"fff5ee\",\n sienna: \"a0522d\",\n silver: \"c0c0c0\",\n skyblue: \"87ceeb\",\n slateblue: \"6a5acd\",\n slategray: \"708090\",\n slategrey: \"708090\",\n snow: \"fffafa\",\n springgreen: \"00ff7f\",\n steelblue: \"4682b4\",\n tan: \"d2b48c\",\n teal: \"008080\",\n thistle: \"d8bfd8\",\n tomato: \"ff6347\",\n turquoise: \"40e0d0\",\n violet: \"ee82ee\",\n wheat: \"f5deb3\",\n white: \"fff\",\n whitesmoke: \"f5f5f5\",\n yellow: \"ff0\",\n yellowgreen: \"9acd32\"\n};\n\n// Make it easy to access colors via hexNames[hex]\nvar hexNames = tinycolor.hexNames = flip(names);\n\n// Utilities\n// ---------\n\n// { 'name1': 'val1' } becomes { 'val1': 'name1' }\nfunction flip(o) {\n var flipped = { };\n for (var i in o) {\n if (o.hasOwnProperty(i)) {\n flipped[o[i]] = i;\n }\n }\n return flipped;\n}\n\n// Return a valid alpha value [0,1] with all invalid values being set to 1\nfunction boundAlpha(a) {\n a = parseFloat(a);\n\n if (isNaN(a) || a < 0 || a > 1) {\n a = 1;\n }\n\n return a;\n}\n\n// Take input from [0, n] and return it as [0, 1]\nfunction bound01(n, max) {\n if (isOnePointZero(n)) { n = \"100%\"; }\n\n var processPercent = isPercentage(n);\n n = mathMin(max, mathMax(0, parseFloat(n)));\n\n // Automatically convert percentage into number\n if (processPercent) {\n n = parseInt(n * max, 10) / 100;\n }\n\n // Handle floating point rounding errors\n if ((Math.abs(n - max) < 0.000001)) {\n return 1;\n }\n\n // Convert into [0, 1] range if it isn't already\n return (n % max) / parseFloat(max);\n}\n\n// Force a number between 0 and 1\nfunction clamp01(val) {\n return mathMin(1, mathMax(0, val));\n}\n\n// Parse a base-16 hex value into a base-10 integer\nfunction parseIntFromHex(val) {\n return parseInt(val, 16);\n}\n\n// Need to handle 1.0 as 100%, since once it is a number, there is no difference between it and 1\n// \nfunction isOnePointZero(n) {\n return typeof n == \"string\" && n.indexOf('.') != -1 && parseFloat(n) === 1;\n}\n\n// Check to see if string passed in is a percentage\nfunction isPercentage(n) {\n return typeof n === \"string\" && n.indexOf('%') != -1;\n}\n\n// Force a hex value to have 2 characters\nfunction pad2(c) {\n return c.length == 1 ? '0' + c : '' + c;\n}\n\n// Replace a decimal with it's percentage value\nfunction convertToPercentage(n) {\n if (n <= 1) {\n n = (n * 100) + \"%\";\n }\n\n return n;\n}\n\n// Converts a decimal to a hex value\nfunction convertDecimalToHex(d) {\n return Math.round(parseFloat(d) * 255).toString(16);\n}\n// Converts a hex value to a decimal\nfunction convertHexToDecimal(h) {\n return (parseIntFromHex(h) / 255);\n}\n\nvar matchers = (function() {\n\n // \n var CSS_INTEGER = \"[-\\\\+]?\\\\d+%?\";\n\n // \n var CSS_NUMBER = \"[-\\\\+]?\\\\d*\\\\.\\\\d+%?\";\n\n // Allow positive/negative integer/number. Don't capture the either/or, just the entire outcome.\n var CSS_UNIT = \"(?:\" + CSS_NUMBER + \")|(?:\" + CSS_INTEGER + \")\";\n\n // Actual matching.\n // Parentheses and commas are optional, but not required.\n // Whitespace can take the place of commas or opening paren\n var PERMISSIVE_MATCH3 = \"[\\\\s|\\\\(]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")\\\\s*\\\\)?\";\n var PERMISSIVE_MATCH4 = \"[\\\\s|\\\\(]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")\\\\s*\\\\)?\";\n\n return {\n CSS_UNIT: new RegExp(CSS_UNIT),\n rgb: new RegExp(\"rgb\" + PERMISSIVE_MATCH3),\n rgba: new RegExp(\"rgba\" + PERMISSIVE_MATCH4),\n hsl: new RegExp(\"hsl\" + PERMISSIVE_MATCH3),\n hsla: new RegExp(\"hsla\" + PERMISSIVE_MATCH4),\n hsv: new RegExp(\"hsv\" + PERMISSIVE_MATCH3),\n hsva: new RegExp(\"hsva\" + PERMISSIVE_MATCH4),\n hex3: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,\n hex6: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/,\n hex4: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,\n hex8: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/\n };\n})();\n\n// isValidCSSUnit\n// Take in a single string / number and check to see if it looks like a CSS unit\n// (see matchers above for definition).\nfunction isValidCSSUnit(color) {\n return !!matchers.CSS_UNIT.exec(color);\n}\n\n// stringInputToObject\n// Permissive string parsing. Take in a number of formats, and output an object\n// based on detected format. Returns { r, g, b } or { h, s, l } or { h, s, v}\nfunction stringInputToObject(color) {\n\n color = color.replace(trimLeft, '').replace(trimRight, '').toLowerCase();\n var named = false;\n if (names[color]) {\n color = names[color];\n named = true;\n }\n else if (color == 'transparent') {\n return { r: 0, g: 0, b: 0, a: 0, format: \"name\" };\n }\n\n // Try to match string input using regular expressions.\n // Keep most of the number bounding out of this function - don't worry about [0,1] or [0,100] or [0,360]\n // Just return an object and let the conversion functions handle that.\n // This way the result will be the same whether the tinycolor is initialized with string or object.\n var match;\n if ((match = matchers.rgb.exec(color))) {\n return { r: match[1], g: match[2], b: match[3] };\n }\n if ((match = matchers.rgba.exec(color))) {\n return { r: match[1], g: match[2], b: match[3], a: match[4] };\n }\n if ((match = matchers.hsl.exec(color))) {\n return { h: match[1], s: match[2], l: match[3] };\n }\n if ((match = matchers.hsla.exec(color))) {\n return { h: match[1], s: match[2], l: match[3], a: match[4] };\n }\n if ((match = matchers.hsv.exec(color))) {\n return { h: match[1], s: match[2], v: match[3] };\n }\n if ((match = matchers.hsva.exec(color))) {\n return { h: match[1], s: match[2], v: match[3], a: match[4] };\n }\n if ((match = matchers.hex8.exec(color))) {\n return {\n r: parseIntFromHex(match[1]),\n g: parseIntFromHex(match[2]),\n b: parseIntFromHex(match[3]),\n a: convertHexToDecimal(match[4]),\n format: named ? \"name\" : \"hex8\"\n };\n }\n if ((match = matchers.hex6.exec(color))) {\n return {\n r: parseIntFromHex(match[1]),\n g: parseIntFromHex(match[2]),\n b: parseIntFromHex(match[3]),\n format: named ? \"name\" : \"hex\"\n };\n }\n if ((match = matchers.hex4.exec(color))) {\n return {\n r: parseIntFromHex(match[1] + '' + match[1]),\n g: parseIntFromHex(match[2] + '' + match[2]),\n b: parseIntFromHex(match[3] + '' + match[3]),\n a: convertHexToDecimal(match[4] + '' + match[4]),\n format: named ? \"name\" : \"hex8\"\n };\n }\n if ((match = matchers.hex3.exec(color))) {\n return {\n r: parseIntFromHex(match[1] + '' + match[1]),\n g: parseIntFromHex(match[2] + '' + match[2]),\n b: parseIntFromHex(match[3] + '' + match[3]),\n format: named ? \"name\" : \"hex\"\n };\n }\n\n return false;\n}\n\nfunction validateWCAG2Parms(parms) {\n // return valid WCAG2 parms for isReadable.\n // If input parms are invalid, return {\"level\":\"AA\", \"size\":\"small\"}\n var level, size;\n parms = parms || {\"level\":\"AA\", \"size\":\"small\"};\n level = (parms.level || \"AA\").toUpperCase();\n size = (parms.size || \"small\").toLowerCase();\n if (level !== \"AA\" && level !== \"AAA\") {\n level = \"AA\";\n }\n if (size !== \"small\" && size !== \"large\") {\n size = \"small\";\n }\n return {\"level\":level, \"size\":size};\n}\n\nthis.tinycolor = tinycolor;\n\n})()`;\n}\n// It is hacky way to make this function will be compiled preferentially by less\n// resolve error: `ReferenceError: colorPalette is not defined`\n// https://github.com/ant-design/ant-motion/issues/44\n.tinyColorMixin();\n","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n\n@tooltip-prefix-cls: ~'@{ant-prefix}-tooltip';\n\n// Base class\n.@{tooltip-prefix-cls} {\n .reset-component;\n\n position: absolute;\n z-index: @zindex-tooltip;\n display: block;\n max-width: @tooltip-max-width;\n visibility: visible;\n\n &-hidden {\n display: none;\n }\n\n &-placement-top,\n &-placement-topLeft,\n &-placement-topRight {\n padding-bottom: @tooltip-distance;\n }\n &-placement-right,\n &-placement-rightTop,\n &-placement-rightBottom {\n padding-left: @tooltip-distance;\n }\n &-placement-bottom,\n &-placement-bottomLeft,\n &-placement-bottomRight {\n padding-top: @tooltip-distance;\n }\n &-placement-left,\n &-placement-leftTop,\n &-placement-leftBottom {\n padding-right: @tooltip-distance;\n }\n\n // Wrapper for the tooltip content\n &-inner {\n min-width: 30px;\n min-height: 32px;\n padding: 6px 8px;\n color: @tooltip-color;\n text-align: left;\n text-decoration: none;\n word-wrap: break-word;\n background-color: @tooltip-bg;\n border-radius: @border-radius-base;\n box-shadow: @box-shadow-base;\n }\n\n // Arrows\n &-arrow {\n position: absolute;\n width: 0;\n height: 0;\n border-color: transparent;\n border-style: solid;\n }\n\n &-placement-top &-arrow,\n &-placement-topLeft &-arrow,\n &-placement-topRight &-arrow {\n bottom: @tooltip-distance - @tooltip-arrow-width;\n border-width: @tooltip-arrow-width @tooltip-arrow-width 0;\n border-top-color: @tooltip-arrow-color;\n }\n\n &-placement-top &-arrow {\n left: 50%;\n margin-left: -@tooltip-arrow-width;\n }\n\n &-placement-topLeft &-arrow {\n left: 16px;\n }\n\n &-placement-topRight &-arrow {\n right: 16px;\n }\n\n &-placement-right &-arrow,\n &-placement-rightTop &-arrow,\n &-placement-rightBottom &-arrow {\n left: @tooltip-distance - @tooltip-arrow-width;\n border-width: @tooltip-arrow-width @tooltip-arrow-width @tooltip-arrow-width 0;\n border-right-color: @tooltip-arrow-color;\n }\n\n &-placement-right &-arrow {\n top: 50%;\n margin-top: -@tooltip-arrow-width;\n }\n\n &-placement-rightTop &-arrow {\n top: 8px;\n }\n\n &-placement-rightBottom &-arrow {\n bottom: 8px;\n }\n\n &-placement-left &-arrow,\n &-placement-leftTop &-arrow,\n &-placement-leftBottom &-arrow {\n right: @tooltip-distance - @tooltip-arrow-width;\n border-width: @tooltip-arrow-width 0 @tooltip-arrow-width @tooltip-arrow-width;\n border-left-color: @tooltip-arrow-color;\n }\n\n &-placement-left &-arrow {\n top: 50%;\n margin-top: -@tooltip-arrow-width;\n }\n\n &-placement-leftTop &-arrow {\n top: 8px;\n }\n\n &-placement-leftBottom &-arrow {\n bottom: 8px;\n }\n\n &-placement-bottom &-arrow,\n &-placement-bottomLeft &-arrow,\n &-placement-bottomRight &-arrow {\n top: @tooltip-distance - @tooltip-arrow-width;\n border-width: 0 @tooltip-arrow-width @tooltip-arrow-width;\n border-bottom-color: @tooltip-arrow-color;\n }\n\n &-placement-bottom &-arrow {\n left: 50%;\n margin-left: -@tooltip-arrow-width;\n }\n\n &-placement-bottomLeft &-arrow {\n left: 16px;\n }\n\n &-placement-bottomRight &-arrow {\n right: 16px;\n }\n}\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n\n@alert-prefix-cls: ~'@{ant-prefix}-alert';\n\n@alert-message-color: @heading-color;\n@alert-text-color: @text-color;\n@alert-close-color: @text-color-secondary;\n@alert-close-hover-color: @icon-color-hover;\n\n.@{alert-prefix-cls} {\n .reset-component;\n\n position: relative;\n padding: 8px 15px 8px 37px;\n border-radius: @border-radius-base;\n\n &&-no-icon {\n padding: 8px 15px;\n }\n\n &&-closable {\n padding-right: 30px;\n }\n\n &-icon {\n position: absolute;\n top: 8px + @font-size-base * @line-height-base / 2 - @font-size-base / 2;\n left: 16px;\n }\n\n &-description {\n display: none;\n font-size: @font-size-base;\n line-height: 22px;\n }\n\n &-success {\n background-color: @alert-success-bg-color;\n border: @border-width-base @border-style-base @alert-success-border-color;\n .@{alert-prefix-cls}-icon {\n color: @alert-success-icon-color;\n }\n }\n\n &-info {\n background-color: @alert-info-bg-color;\n border: @border-width-base @border-style-base @alert-info-border-color;\n .@{alert-prefix-cls}-icon {\n color: @alert-info-icon-color;\n }\n }\n\n &-warning {\n background-color: @alert-warning-bg-color;\n border: @border-width-base @border-style-base @alert-warning-border-color;\n .@{alert-prefix-cls}-icon {\n color: @alert-warning-icon-color;\n }\n }\n\n &-error {\n background-color: @alert-error-bg-color;\n border: @border-width-base @border-style-base @alert-error-border-color;\n .@{alert-prefix-cls}-icon {\n color: @alert-error-icon-color;\n }\n }\n\n &-close-icon {\n position: absolute;\n top: 8px;\n right: 16px;\n overflow: hidden;\n font-size: @font-size-sm;\n line-height: 22px;\n cursor: pointer;\n\n .@{iconfont-css-prefix}-close {\n color: @alert-close-color;\n transition: color 0.3s;\n &:hover {\n color: @alert-close-hover-color;\n }\n }\n }\n\n &-close-text {\n position: absolute;\n right: 16px;\n }\n\n &-with-description {\n position: relative;\n padding: 15px 15px 15px 64px;\n color: @alert-text-color;\n line-height: @line-height-base;\n border-radius: @border-radius-base;\n }\n\n &-with-description&-no-icon {\n padding: 15px;\n }\n\n &-with-description &-icon {\n position: absolute;\n top: 16px;\n left: 24px;\n font-size: 24px;\n }\n\n &-with-description &-close-icon {\n position: absolute;\n top: 16px;\n right: 16px;\n font-size: @font-size-base;\n cursor: pointer;\n }\n\n &-with-description &-message {\n display: block;\n margin-bottom: 4px;\n color: @alert-message-color;\n font-size: @font-size-lg;\n }\n\n &-with-description &-description {\n display: block;\n }\n\n &&-close {\n height: 0 !important;\n margin: 0;\n padding-top: 0;\n padding-bottom: 0;\n transform-origin: 50% 0;\n transition: all 0.3s @ease-in-out-circ;\n }\n\n &-slide-up-leave {\n animation: antAlertSlideUpOut 0.3s @ease-in-out-circ;\n animation-fill-mode: both;\n }\n\n &-banner {\n margin-bottom: 0;\n border: 0;\n border-radius: 0;\n }\n}\n\n@keyframes antAlertSlideUpIn {\n 0% {\n transform: scaleY(0);\n transform-origin: 0% 0%;\n opacity: 0;\n }\n 100% {\n transform: scaleY(1);\n transform-origin: 0% 0%;\n opacity: 1;\n }\n}\n\n@keyframes antAlertSlideUpOut {\n 0% {\n transform: scaleY(1);\n transform-origin: 0% 0%;\n opacity: 1;\n }\n 100% {\n transform: scaleY(0);\n transform-origin: 0% 0%;\n opacity: 0;\n }\n}\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n\n@tab-prefix-cls: ~'@{ant-prefix}-tabs';\n\n// card style\n.@{tab-prefix-cls} {\n &&-card &-card-bar &-nav-container {\n height: @tabs-card-height;\n }\n &&-card &-card-bar &-ink-bar {\n visibility: hidden;\n }\n &&-card &-card-bar &-tab {\n height: @tabs-card-height;\n margin: 0;\n margin-right: 2px;\n padding: 0 16px;\n line-height: @tabs-card-height - 2px;\n background: @tabs-card-head-background;\n border: @border-width-base @border-style-base @border-color-split;\n border-radius: @border-radius-base @border-radius-base 0 0;\n transition: all 0.3s @ease-in-out;\n }\n &&-card &-card-bar &-tab-active {\n height: @tabs-card-height;\n color: @tabs-card-active-color;\n background: @component-background;\n border-color: @border-color-split;\n border-bottom: @border-width-base solid @component-background;\n }\n &&-card &-card-bar &-tab-inactive {\n padding: 0;\n }\n &&-card &-card-bar &-nav-wrap {\n margin-bottom: 0;\n }\n &&-card &-card-bar &-tab &-close-x {\n width: 16px;\n height: 16px;\n height: @font-size-base;\n margin-right: -5px;\n margin-left: 3px;\n overflow: hidden;\n color: @text-color-secondary;\n font-size: @font-size-sm;\n vertical-align: middle;\n transition: all 0.3s;\n &:hover {\n color: @heading-color;\n }\n }\n\n &&-card &-card-content > &-tabpane,\n &&-editable-card &-card-content > &-tabpane {\n transition: none !important;\n &-inactive {\n overflow: hidden;\n }\n }\n\n &&-card &-card-bar &-tab:hover .@{iconfont-css-prefix}-close {\n opacity: 1;\n }\n\n &-extra-content {\n line-height: @tabs-card-height;\n\n .@{tab-prefix-cls}-new-tab {\n position: relative;\n width: 20px;\n height: 20px;\n color: @text-color;\n font-size: 12px;\n line-height: 20px;\n text-align: center;\n border: @border-width-base @border-style-base @border-color-split;\n border-radius: @border-radius-sm;\n cursor: pointer;\n transition: all 0.3s;\n &:hover {\n color: @tabs-card-active-color;\n border-color: @tabs-card-active-color;\n }\n svg {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n margin: auto;\n }\n }\n }\n\n // https://github.com/ant-design/ant-design/issues/4669\n &-vertical&-card &-card-bar&-left-bar,\n &-vertical&-card &-card-bar&-right-bar {\n .@{tab-prefix-cls}-nav-container {\n height: 100%;\n }\n .@{tab-prefix-cls}-tab {\n margin-bottom: 8px;\n border-bottom: @border-width-base @border-style-base @border-color-split;\n &-active {\n padding-bottom: 4px;\n }\n &:last-child {\n margin-bottom: 8px;\n }\n }\n .@{tab-prefix-cls}-new-tab {\n width: 90%;\n }\n }\n\n &-vertical&-card&-left &-card-bar&-left-bar {\n .@{tab-prefix-cls}-nav-wrap {\n margin-right: 0;\n }\n .@{tab-prefix-cls}-tab {\n margin-right: 1px;\n border-right: 0;\n border-radius: @border-radius-base 0 0 @border-radius-base;\n &-active {\n margin-right: -1px;\n padding-right: 18px;\n }\n }\n }\n\n &-vertical&-card&-right &-card-bar&-right-bar {\n .@{tab-prefix-cls}-nav-wrap {\n margin-left: 0;\n }\n .@{tab-prefix-cls}-tab {\n margin-left: 1px;\n border-left: 0;\n border-radius: 0 @border-radius-base @border-radius-base 0;\n &-active {\n margin-left: -1px;\n padding-left: 18px;\n }\n }\n }\n\n // https://github.com/ant-design/ant-design/issues/9104\n & &-card-bar&-bottom-bar &-tab {\n height: auto;\n border-top: 0;\n border-bottom: @border-width-base @border-style-base @border-color-split;\n border-radius: 0 0 @border-radius-base @border-radius-base;\n }\n\n & &-card-bar&-bottom-bar &-tab-active {\n padding-top: 1px;\n padding-bottom: 0;\n color: @primary-color;\n }\n}\n","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n@import './card-style';\n\n@tab-prefix-cls: ~'@{ant-prefix}-tabs';\n\n// Hidden content\n.tabs-hidden-content() {\n height: 0;\n padding: 0 !important;\n overflow: hidden;\n opacity: 0;\n pointer-events: none;\n input {\n visibility: hidden;\n }\n}\n\n.@{tab-prefix-cls} {\n .reset-component;\n\n position: relative;\n overflow: hidden;\n .clearfix;\n\n &-ink-bar {\n position: absolute;\n bottom: 1px;\n left: 0;\n z-index: 1;\n box-sizing: border-box;\n height: 2px;\n background-color: @tabs-ink-bar-color;\n transform-origin: 0 0;\n }\n\n &-bar {\n margin: @tabs-bar-margin;\n border-bottom: @border-width-base @border-style-base @border-color-split;\n outline: none;\n transition: padding 0.3s @ease-in-out;\n }\n\n &-nav-container {\n position: relative;\n box-sizing: border-box;\n margin-bottom: -1px;\n overflow: hidden;\n font-size: @tabs-title-font-size;\n line-height: @line-height-base;\n white-space: nowrap;\n transition: padding 0.3s @ease-in-out;\n .clearfix;\n\n &-scrolling {\n padding-right: @tabs-scrolling-size;\n padding-left: @tabs-scrolling-size;\n }\n }\n\n // https://github.com/ant-design/ant-design/issues/9104\n &-bottom &-bottom-bar {\n margin-top: 16px;\n margin-bottom: 0;\n border-top: @border-width-base @border-style-base @border-color-split;\n border-bottom: none;\n }\n\n &-bottom &-bottom-bar &-ink-bar {\n top: 1px;\n bottom: auto;\n }\n\n &-bottom &-bottom-bar &-nav-container {\n margin-top: -1px;\n margin-bottom: 0;\n }\n\n &-tab-prev,\n &-tab-next {\n position: absolute;\n z-index: 2;\n width: 0;\n height: 100%;\n color: @text-color-secondary;\n text-align: center;\n background-color: transparent;\n border: 0;\n cursor: pointer;\n opacity: 0;\n transition: width 0.3s @ease-in-out, opacity 0.3s @ease-in-out, color 0.3s @ease-in-out;\n user-select: none;\n pointer-events: none;\n\n &.@{tab-prefix-cls}-tab-arrow-show {\n width: @tabs-scrolling-size;\n height: 100%;\n opacity: 1;\n pointer-events: auto;\n }\n\n &:hover {\n color: @text-color;\n }\n\n &-icon {\n position: absolute;\n top: 50%;\n left: 50%;\n font-weight: bold;\n font-style: normal;\n font-variant: normal;\n line-height: inherit;\n text-align: center;\n text-transform: none;\n transform: translate(-50%, -50%);\n\n &-target {\n display: block;\n .iconfont-size-under-12px(10px);\n }\n }\n }\n\n &-tab-btn-disabled {\n cursor: not-allowed;\n &,\n &:hover {\n color: @disabled-color;\n }\n }\n\n &-tab-next {\n right: 2px;\n }\n\n &-tab-prev {\n left: 0;\n :root & {\n filter: none;\n }\n }\n\n &-nav-wrap {\n margin-bottom: -1px;\n overflow: hidden;\n }\n\n &-nav-scroll {\n overflow: hidden;\n white-space: nowrap;\n }\n\n &-nav {\n position: relative;\n display: inline-block;\n box-sizing: border-box;\n margin: 0;\n padding-left: 0;\n list-style: none;\n transition: transform 0.3s @ease-in-out;\n\n &::before,\n &::after {\n display: table;\n content: ' ';\n }\n\n &::after {\n clear: both;\n }\n\n .@{tab-prefix-cls}-tab {\n position: relative;\n display: inline-block;\n box-sizing: border-box;\n height: 100%;\n margin: @tabs-horizontal-margin;\n padding: @tabs-horizontal-padding;\n text-decoration: none;\n cursor: pointer;\n transition: color 0.3s @ease-in-out;\n\n &:last-child {\n margin-right: 0;\n }\n\n &:hover {\n color: @tabs-hover-color;\n }\n\n &:active {\n color: @tabs-active-color;\n }\n\n .@{iconfont-css-prefix} {\n margin-right: 8px;\n }\n\n &-disabled {\n &,\n &:hover {\n color: @disabled-color;\n cursor: not-allowed;\n }\n }\n\n &-active {\n color: @tabs-highlight-color;\n font-weight: 500;\n }\n }\n }\n\n .@{tab-prefix-cls}-large-bar {\n .@{tab-prefix-cls}-nav-container {\n font-size: @tabs-title-font-size-lg;\n }\n .@{tab-prefix-cls}-tab {\n padding: @tabs-horizontal-padding-lg;\n }\n }\n\n .@{tab-prefix-cls}-small-bar {\n .@{tab-prefix-cls}-nav-container {\n font-size: @tabs-title-font-size-sm;\n }\n .@{tab-prefix-cls}-tab {\n padding: @tabs-horizontal-padding-sm;\n }\n }\n\n // Horizontal Content\n .@{tab-prefix-cls}-top-content,\n .@{tab-prefix-cls}-bottom-content {\n width: 100%;\n\n > .@{tab-prefix-cls}-tabpane {\n flex-shrink: 0;\n width: 100%;\n opacity: 1;\n transition: opacity 0.45s;\n }\n\n > .@{tab-prefix-cls}-tabpane-inactive {\n .tabs-hidden-content();\n }\n\n &.@{tab-prefix-cls}-content-animated {\n display: flex;\n flex-direction: row;\n transition: margin-left 0.3s @ease-in-out;\n will-change: margin-left;\n }\n }\n\n // Vertical Bar\n .@{tab-prefix-cls}-left-bar,\n .@{tab-prefix-cls}-right-bar {\n height: 100%;\n border-bottom: 0;\n &-tab-prev,\n &-tab-next {\n width: @tabs-scrolling-size;\n height: 0;\n transition: height 0.3s @ease-in-out, opacity 0.3s @ease-in-out, color 0.3s @ease-in-out;\n }\n &-tab-prev.@{tab-prefix-cls}-tab-arrow-show,\n &-tab-next.@{tab-prefix-cls}-tab-arrow-show {\n width: 100%;\n height: @tabs-scrolling-size;\n }\n\n .@{tab-prefix-cls}-tab {\n display: block;\n float: none;\n margin: @tabs-vertical-margin;\n padding: @tabs-vertical-padding;\n\n &:last-child {\n margin-bottom: 0;\n }\n }\n\n .@{tab-prefix-cls}-extra-content {\n text-align: center;\n }\n\n .@{tab-prefix-cls}-nav-scroll {\n width: auto;\n }\n\n .@{tab-prefix-cls}-nav-container,\n .@{tab-prefix-cls}-nav-wrap {\n height: 100%;\n }\n\n .@{tab-prefix-cls}-nav-container {\n margin-bottom: 0;\n\n &.@{tab-prefix-cls}-nav-container-scrolling {\n padding: @tabs-scrolling-size 0;\n }\n }\n\n .@{tab-prefix-cls}-nav-wrap {\n margin-bottom: 0;\n }\n\n .@{tab-prefix-cls}-nav {\n width: 100%;\n }\n\n .@{tab-prefix-cls}-ink-bar {\n top: 0;\n bottom: auto;\n left: auto;\n width: 2px;\n height: auto;\n }\n\n .@{tab-prefix-cls}-tab-next {\n bottom: 0;\n width: 100%;\n height: @tabs-scrolling-size;\n }\n\n .@{tab-prefix-cls}-tab-prev {\n top: 0;\n width: 100%;\n height: @tabs-scrolling-size;\n }\n }\n\n // Vertical Content\n .@{tab-prefix-cls}-left-content,\n .@{tab-prefix-cls}-right-content {\n width: auto;\n margin-top: 0 !important;\n overflow: hidden;\n }\n\n // Vertical - Left\n .@{tab-prefix-cls}-left-bar {\n float: left;\n margin-right: -1px;\n margin-bottom: 0;\n border-right: @border-width-base @border-style-base @border-color-split;\n .@{tab-prefix-cls}-tab {\n text-align: right;\n }\n .@{tab-prefix-cls}-nav-container {\n margin-right: -1px;\n }\n .@{tab-prefix-cls}-nav-wrap {\n margin-right: -1px;\n }\n .@{tab-prefix-cls}-ink-bar {\n right: 1px;\n }\n }\n .@{tab-prefix-cls}-left-content {\n padding-left: 24px;\n border-left: @border-width-base @border-style-base @border-color-split;\n }\n\n // Vertical - Right\n .@{tab-prefix-cls}-right-bar {\n float: right;\n margin-bottom: 0;\n margin-left: -1px;\n border-left: @border-width-base @border-style-base @border-color-split;\n .@{tab-prefix-cls}-nav-container {\n margin-left: -1px;\n }\n .@{tab-prefix-cls}-nav-wrap {\n margin-left: -1px;\n }\n .@{tab-prefix-cls}-ink-bar {\n left: 1px;\n }\n }\n .@{tab-prefix-cls}-right-content {\n padding-right: 24px;\n border-right: @border-width-base @border-style-base @border-color-split;\n }\n}\n\n.@{tab-prefix-cls}-top .@{tab-prefix-cls}-ink-bar-animated,\n.@{tab-prefix-cls}-bottom .@{tab-prefix-cls}-ink-bar-animated {\n transition: transform 0.3s @ease-in-out, width 0.3s @ease-in-out, left 0.3s @ease-in-out;\n}\n\n.@{tab-prefix-cls}-left .@{tab-prefix-cls}-ink-bar-animated,\n.@{tab-prefix-cls}-right .@{tab-prefix-cls}-ink-bar-animated {\n transition: transform 0.3s @ease-in-out, height 0.3s @ease-in-out, top 0.3s @ease-in-out;\n}\n\n// No animation\n.tabs-no-animation() {\n > .@{tab-prefix-cls}-content-animated {\n margin-left: 0 !important;\n transform: none !important;\n }\n > .@{tab-prefix-cls}-tabpane-inactive {\n .tabs-hidden-content();\n }\n}\n\n.no-flex,\n.@{tab-prefix-cls}-no-animation {\n > .@{tab-prefix-cls}-content {\n .tabs-no-animation();\n }\n}\n\n.@{tab-prefix-cls}-left-content,\n.@{tab-prefix-cls}-right-content {\n .tabs-no-animation();\n}\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n\n@input-affix-width: 19px;\n\n// size mixins for input\n.input-lg() {\n height: @input-height-lg;\n padding: @input-padding-vertical-lg @input-padding-horizontal-lg;\n font-size: @font-size-lg;\n}\n\n.input-sm() {\n height: @input-height-sm;\n padding: @input-padding-vertical-sm @input-padding-horizontal-sm;\n}\n\n// input status\n// == when focus or actived\n.active(@color: @outline-color) {\n border-color: ~`colorPalette('@{color}', 5) `;\n border-right-width: @border-width-base !important;\n outline: 0;\n box-shadow: @input-outline-offset @outline-blur-size @outline-width fade(@color, 20%);\n}\n\n// == when hoverd\n.hover(@color: @input-hover-border-color) {\n border-color: ~`colorPalette('@{color}', 5) `;\n border-right-width: @border-width-base !important;\n}\n\n.disabled() {\n color: @disabled-color;\n background-color: @input-disabled-bg;\n cursor: not-allowed;\n opacity: 1;\n &:hover {\n .hover(@input-border-color);\n }\n}\n\n// Basic style for input\n.input() {\n position: relative;\n display: inline-block;\n width: 100%;\n height: @input-height-base;\n padding: @input-padding-vertical-base @input-padding-horizontal-base;\n color: @input-color;\n font-size: @font-size-base;\n line-height: @line-height-base;\n background-color: @input-bg;\n background-image: none;\n border: @border-width-base @border-style-base @input-border-color;\n border-radius: @border-radius-base;\n transition: all 0.3s;\n .placeholder(); // Reset placeholder\n\n &:hover {\n .hover();\n }\n\n &:focus {\n .active();\n }\n\n &-disabled {\n .disabled();\n }\n\n &[disabled] {\n .disabled();\n }\n\n // Reset height for `textarea`s\n textarea& {\n max-width: 100%; // prevent textearea resize from coming out of its container\n height: auto;\n min-height: @input-height-base;\n vertical-align: bottom;\n transition: all 0.3s, height 0s;\n }\n\n // Size\n &-lg {\n .input-lg();\n }\n\n &-sm {\n .input-sm();\n }\n}\n\n// label input\n.input-group(@inputClass) {\n position: relative;\n display: table;\n width: 100%;\n border-collapse: separate;\n border-spacing: 0;\n\n // Undo padding and float of grid classes\n &[class*='col-'] {\n float: none;\n padding-right: 0;\n padding-left: 0;\n }\n\n > [class*='col-'] {\n padding-right: 8px;\n &:last-child {\n padding-right: 0;\n }\n }\n\n &-addon,\n &-wrap,\n > .@{inputClass} {\n display: table-cell;\n\n &:not(:first-child):not(:last-child) {\n border-radius: 0;\n }\n }\n\n &-addon,\n &-wrap {\n width: 1px; // To make addon/wrap as small as possible\n white-space: nowrap;\n vertical-align: middle;\n }\n\n &-wrap > * {\n display: block !important;\n }\n\n .@{inputClass} {\n float: left;\n width: 100%;\n margin-bottom: 0;\n text-align: inherit;\n &:focus {\n z-index: 1; // Fix https://gw.alipayobjects.com/zos/rmsportal/DHNpoqfMXSfrSnlZvhsJ.png\n border-right-width: 1px;\n }\n &:hover {\n z-index: 1;\n border-right-width: 1px;\n }\n }\n\n &-addon {\n position: relative;\n padding: 0 @input-padding-horizontal-base;\n color: @input-color;\n font-weight: normal;\n font-size: @font-size-base;\n line-height: 1;\n text-align: center;\n background-color: @input-addon-bg;\n border: @border-width-base @border-style-base @input-border-color;\n border-radius: @border-radius-base;\n transition: all 0.3s;\n\n // Reset Select's style in addon\n .@{ant-prefix}-select {\n margin: -(@input-padding-vertical-base + 1px) (-@input-padding-horizontal-base);\n\n .@{ant-prefix}-select-selection {\n margin: -1px;\n background-color: inherit;\n border: @border-width-base @border-style-base transparent;\n box-shadow: none;\n }\n\n &-open,\n &-focused {\n .@{ant-prefix}-select-selection {\n color: @primary-color;\n }\n }\n }\n\n // Expand addon icon click area\n // https://github.com/ant-design/ant-design/issues/3714\n > i:only-child::after {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n content: '';\n }\n }\n\n // Reset rounded corners\n > .@{inputClass}:first-child,\n &-addon:first-child {\n border-top-right-radius: 0;\n border-bottom-right-radius: 0;\n\n // Reset Select's style in addon\n .@{ant-prefix}-select .@{ant-prefix}-select-selection {\n border-top-right-radius: 0;\n border-bottom-right-radius: 0;\n }\n }\n\n > .@{inputClass}-affix-wrapper {\n &:not(:first-child) .@{inputClass} {\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n }\n\n &:not(:last-child) .@{inputClass} {\n border-top-right-radius: 0;\n border-bottom-right-radius: 0;\n }\n }\n\n &-addon:first-child {\n border-right: 0;\n }\n &-addon:last-child {\n border-left: 0;\n }\n\n > .@{inputClass}:last-child,\n &-addon:last-child {\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n\n // Reset Select's style in addon\n .@{ant-prefix}-select .@{ant-prefix}-select-selection {\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n }\n }\n\n // Sizing options\n &-lg .@{inputClass},\n &-lg > &-addon {\n .input-lg();\n }\n\n &-sm .@{inputClass},\n &-sm > &-addon {\n .input-sm();\n }\n\n // Fix https://github.com/ant-design/ant-design/issues/5754\n &-lg .@{ant-prefix}-select-selection--single {\n height: @input-height-lg;\n }\n\n &-sm .@{ant-prefix}-select-selection--single {\n height: @input-height-sm;\n }\n\n .@{inputClass}-affix-wrapper {\n display: table-cell;\n float: left;\n width: 100%;\n }\n\n &&-compact {\n display: block;\n .clearfix;\n\n &-addon,\n &-wrap,\n > .@{inputClass} {\n &:not(:first-child):not(:last-child) {\n border-right-width: @border-width-base;\n &:hover {\n z-index: 1;\n }\n &:focus {\n z-index: 1;\n }\n }\n }\n\n & > * {\n display: inline-block;\n float: none;\n vertical-align: top; // https://github.com/ant-design/ant-design-pro/issues/139\n border-radius: 0;\n }\n\n & > *:not(:last-child) {\n margin-right: -@border-width-base;\n border-right-width: @border-width-base;\n }\n\n // Undo float for .ant-input-group .ant-input\n .@{inputClass} {\n float: none;\n }\n\n // reset border for Select, DatePicker, AutoComplete, Cascader, Mention, TimePicker\n & > .@{ant-prefix}-select > .@{ant-prefix}-select-selection,\n & > .@{ant-prefix}-calendar-picker .@{ant-prefix}-input,\n & > .@{ant-prefix}-select-auto-complete .@{ant-prefix}-input,\n & > .@{ant-prefix}-cascader-picker .@{ant-prefix}-input,\n & > .@{ant-prefix}-mention-wrapper .@{ant-prefix}-mention-editor,\n & > .@{ant-prefix}-time-picker .@{ant-prefix}-time-picker-input {\n border-right-width: @border-width-base;\n border-radius: 0;\n &:hover {\n z-index: 1;\n }\n &:focus {\n z-index: 1;\n }\n }\n\n & > *:first-child,\n & > .@{ant-prefix}-select:first-child > .@{ant-prefix}-select-selection,\n & > .@{ant-prefix}-calendar-picker:first-child .@{ant-prefix}-input,\n & > .@{ant-prefix}-select-auto-complete:first-child .@{ant-prefix}-input,\n & > .@{ant-prefix}-cascader-picker:first-child .@{ant-prefix}-input,\n & > .@{ant-prefix}-mention-wrapper:first-child .@{ant-prefix}-mention-editor,\n & > .@{ant-prefix}-time-picker:first-child .@{ant-prefix}-time-picker-input {\n border-top-left-radius: @border-radius-base;\n border-bottom-left-radius: @border-radius-base;\n }\n\n & > *:last-child,\n & > .@{ant-prefix}-select:last-child > .@{ant-prefix}-select-selection,\n & > .@{ant-prefix}-calendar-picker:last-child .@{ant-prefix}-input,\n & > .@{ant-prefix}-select-auto-complete:last-child .@{ant-prefix}-input,\n & > .@{ant-prefix}-cascader-picker:last-child .@{ant-prefix}-input,\n & > .@{ant-prefix}-cascader-picker-focused:last-child .@{ant-prefix}-input,\n & > .@{ant-prefix}-mention-wrapper:last-child .@{ant-prefix}-mention-editor,\n & > .@{ant-prefix}-time-picker:last-child .@{ant-prefix}-time-picker-input {\n border-right-width: @border-width-base;\n border-top-right-radius: @border-radius-base;\n border-bottom-right-radius: @border-radius-base;\n }\n\n // https://github.com/ant-design/ant-design/issues/12493\n & > .@{ant-prefix}-select-auto-complete .@{ant-prefix}-input {\n vertical-align: top;\n }\n }\n}\n\n.input-affix-wrapper(@inputClass) {\n position: relative;\n display: inline-block;\n width: 100%;\n text-align: start;\n\n &:hover .@{inputClass}:not(.@{inputClass}-disabled) {\n .hover();\n }\n\n .@{inputClass} {\n position: relative;\n text-align: inherit;\n }\n\n .@{inputClass}-prefix,\n .@{inputClass}-suffix {\n position: absolute;\n top: 50%;\n z-index: 2;\n color: @input-color;\n line-height: 0;\n transform: translateY(-50%);\n :not(.anticon) {\n line-height: @line-height-base;\n }\n }\n\n .@{inputClass}-prefix {\n left: @input-padding-horizontal-base + 1px;\n }\n\n .@{inputClass}-suffix {\n right: @input-padding-horizontal-base + 1px;\n }\n\n .@{inputClass}:not(:first-child) {\n padding-left: @input-padding-horizontal-base + @input-affix-width;\n }\n\n .@{inputClass}:not(:last-child) {\n padding-right: @input-padding-horizontal-base + @input-affix-width;\n }\n}\n","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n@import '../../input/style/mixin';\n\n@input-number-prefix-cls: ~'@{ant-prefix}-input-number';\n\n.@{input-number-prefix-cls} {\n .reset-component;\n .input;\n\n display: inline-block;\n width: 90px;\n margin: 0;\n padding: 0;\n border: @border-width-base @border-style-base @border-color-base;\n border-radius: @border-radius-base;\n\n &-handler {\n position: relative;\n display: block;\n width: 100%;\n height: 50%;\n overflow: hidden;\n color: @text-color-secondary;\n font-weight: bold;\n line-height: 0;\n text-align: center;\n transition: all 0.1s linear;\n &:active {\n background: @input-number-handler-active-bg;\n }\n &:hover &-up-inner,\n &:hover &-down-inner {\n color: @primary-5;\n }\n }\n\n &-handler-up-inner,\n &-handler-down-inner {\n .iconfont-mixin();\n\n position: absolute;\n right: 4px;\n width: 12px;\n height: 12px;\n color: @text-color-secondary;\n line-height: 12px;\n transition: all 0.1s linear;\n user-select: none;\n }\n\n &:hover {\n .hover();\n }\n\n &-focused {\n .active();\n }\n\n &-disabled {\n .disabled();\n .@{input-number-prefix-cls}-input {\n cursor: not-allowed;\n }\n .@{input-number-prefix-cls}-handler-wrap {\n display: none;\n }\n }\n\n &-input {\n width: 100%;\n height: @input-height-base - 2px;\n padding: 0 @control-padding-horizontal - 1px;\n text-align: left;\n background-color: transparent;\n border: 0;\n border-radius: @border-radius-base;\n outline: 0;\n transition: all 0.3s linear;\n -moz-appearance: textfield;\n .placeholder();\n }\n\n &-lg {\n padding: 0;\n font-size: @font-size-lg;\n\n input {\n height: @input-height-lg - 2px;\n }\n }\n\n &-sm {\n padding: 0;\n\n input {\n height: @input-height-sm - 2px;\n padding: 0 @control-padding-horizontal-sm - 1px;\n }\n }\n\n &-handler-wrap {\n position: absolute;\n top: 0;\n right: 0;\n width: 22px;\n height: 100%;\n background: @component-background;\n border-left: @border-width-base @border-style-base @border-color-base;\n border-radius: 0 @border-radius-base @border-radius-base 0;\n opacity: 0;\n transition: opacity 0.24s linear 0.1s;\n\n // Fix input number inside Menu makes icon too large\n // We arise the selector priority by nest selector here\n // https://github.com/ant-design/ant-design/issues/14367\n .@{input-number-prefix-cls}-handler {\n .@{input-number-prefix-cls}-handler-up-inner,\n .@{input-number-prefix-cls}-handler-down-inner {\n .iconfont-size-under-12px(7px);\n\n min-width: auto;\n margin-right: 0;\n }\n }\n }\n\n &-handler-wrap:hover &-handler {\n height: 40%;\n }\n\n &:hover &-handler-wrap {\n opacity: 1;\n }\n\n &-handler-up {\n cursor: pointer;\n &-inner {\n top: 50%;\n margin-top: -5px;\n text-align: center;\n }\n &:hover {\n height: 60% !important;\n }\n }\n\n &-handler-down {\n top: 0;\n border-top: @border-width-base @border-style-base @border-color-base;\n cursor: pointer;\n &-inner {\n top: 50%;\n margin-top: -6px;\n text-align: center;\n }\n &:hover {\n height: 60% !important;\n }\n }\n\n &-handler-up-disabled,\n &-handler-down-disabled {\n cursor: not-allowed;\n }\n\n &-handler-up-disabled:hover &-handler-up-inner,\n &-handler-down-disabled:hover &-handler-down-inner {\n color: @disabled-color;\n }\n}\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;","// Compatibility for browsers.\n\n// Placeholder text\n.placeholder(@color: @input-placeholder-color) {\n // Firefox\n &::-moz-placeholder {\n color: @color;\n opacity: 1; // Override Firefox's unusual default opacity; see https://github.com/twbs/bootstrap/pull/11526\n }\n // Internet Explorer 10+\n &:-ms-input-placeholder {\n color: @color;\n }\n // Safari and Chrome\n &::-webkit-input-placeholder {\n color: @color;\n }\n}\n","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n@import '../../input/style/mixin';\n\n@select-prefix-cls: ~'@{ant-prefix}-select';\n\n.selection__clear() {\n position: absolute;\n top: 50%;\n right: @control-padding-horizontal - 1px;\n z-index: 1;\n display: inline-block;\n width: 12px;\n height: 12px;\n margin-top: -6px;\n color: @disabled-color;\n font-size: @font-size-sm;\n font-style: normal;\n line-height: 12px;\n text-align: center;\n text-transform: none;\n background: @component-background;\n cursor: pointer;\n opacity: 0;\n transition: color 0.3s ease, opacity 0.15s ease;\n text-rendering: auto;\n &::before {\n display: block;\n }\n &:hover {\n color: @text-color-secondary;\n }\n}\n\n.@{select-prefix-cls} {\n .reset-component;\n\n position: relative;\n display: inline-block;\n outline: 0;\n\n ul,\n ol {\n margin: 0;\n padding: 0;\n list-style: none;\n }\n\n > ul > li > a {\n padding: 0;\n background-color: @component-background;\n }\n\n // arrow\n &-arrow {\n .iconfont-mixin();\n\n position: absolute;\n top: 50%;\n right: @control-padding-horizontal - 1px;\n margin-top: -@font-size-sm / 2;\n color: @disabled-color;\n font-size: @font-size-sm;\n line-height: 1;\n transform-origin: 50% 50%;\n\n & &-icon svg {\n transition: transform 0.3s;\n }\n }\n\n &-selection {\n display: block;\n box-sizing: border-box;\n background-color: @component-background;\n border: @border-width-base @border-style-base @select-border-color;\n // strange align fix for chrome but works\n // https://gw.alipayobjects.com/zos/rmsportal/VFTfKXJuogBAXcvfAUWJ.gif\n border-top-width: @border-width-base + 0.02px;\n border-radius: @border-radius-base;\n outline: none;\n transition: all 0.3s @ease-in-out;\n user-select: none;\n\n &:hover {\n .hover;\n }\n\n .@{select-prefix-cls}-focused &,\n &:focus,\n &:active {\n .active;\n }\n\n &__clear {\n .selection__clear();\n }\n\n &:hover &__clear {\n opacity: 1;\n }\n\n &-selected-value {\n float: left;\n max-width: 100%;\n padding-right: 20px;\n overflow: hidden;\n white-space: nowrap;\n text-overflow: ellipsis;\n }\n }\n\n &-no-arrow &-selection-selected-value {\n padding-right: 0;\n }\n\n &-disabled {\n color: @disabled-color;\n }\n\n &-disabled &-selection {\n background: @input-disabled-bg;\n cursor: not-allowed;\n &:hover,\n &:focus,\n &:active {\n border-color: @select-border-color;\n box-shadow: none;\n }\n\n &__clear {\n display: none;\n visibility: hidden;\n pointer-events: none;\n }\n }\n\n &-disabled &-selection--multiple &-selection__choice {\n padding-right: 10px;\n color: fade(@black, 33%);\n background: @background-color-base;\n &__remove {\n display: none;\n }\n }\n\n &-selection--single {\n position: relative;\n height: @input-height-base;\n cursor: pointer;\n }\n\n &-selection__rendered {\n position: relative;\n display: block;\n margin-right: @control-padding-horizontal - 1px;\n margin-left: @control-padding-horizontal - 1px;\n line-height: @input-height-base - 2px;\n // https://github.com/ant-design/ant-design/issues/3481#issuecomment-254721026\n &::after {\n display: inline-block;\n width: 0;\n visibility: hidden;\n content: '.';\n pointer-events: none;\n }\n }\n\n &-lg {\n font-size: @font-size-lg;\n .@{select-prefix-cls}-selection--single {\n height: @input-height-lg;\n }\n .@{select-prefix-cls}-selection__rendered {\n line-height: @input-height-lg - 2px;\n }\n .@{select-prefix-cls}-selection--multiple {\n min-height: @input-height-lg;\n .@{select-prefix-cls}-selection__rendered {\n li {\n height: @input-height-lg - 8px;\n line-height: @input-height-lg - 8px;\n }\n }\n .@{select-prefix-cls}-selection__clear,\n .@{select-prefix-cls}-arrow {\n top: @input-height-lg / 2;\n }\n }\n }\n\n &-sm {\n .@{select-prefix-cls}-selection--single {\n height: @input-height-sm;\n }\n .@{select-prefix-cls}-selection__rendered {\n margin: 0 @control-padding-horizontal-sm - 1px;\n line-height: @input-height-sm - 2px;\n }\n .@{select-prefix-cls}-selection--multiple {\n min-height: @input-height-sm;\n .@{select-prefix-cls}-selection__rendered {\n li {\n height: @input-height-sm - 8px;\n line-height: @input-height-sm - 10px;\n }\n }\n .@{select-prefix-cls}-selection__clear,\n .@{select-prefix-cls}-arrow {\n top: @input-height-sm / 2;\n }\n }\n .@{select-prefix-cls}-selection__clear,\n .@{select-prefix-cls}-arrow {\n right: @control-padding-horizontal-sm;\n }\n }\n\n &-disabled &-selection__choice__remove {\n color: @disabled-color;\n cursor: default;\n &:hover {\n color: @disabled-color;\n }\n }\n\n &-search__field__wrap {\n position: relative;\n display: inline-block;\n }\n\n &-selection__placeholder,\n &-search__field__placeholder {\n // for TreeSelect compatibility\n position: absolute;\n top: 50%;\n right: 9px;\n left: 0;\n max-width: 100%;\n height: 20px;\n margin-top: -10px;\n overflow: hidden;\n color: @input-placeholder-color;\n line-height: 20px;\n white-space: nowrap;\n text-align: left;\n text-overflow: ellipsis;\n }\n\n &-search__field__placeholder {\n left: @control-padding-horizontal;\n }\n\n &-search__field__mirror {\n position: absolute;\n top: 0;\n left: 0;\n white-space: pre;\n opacity: 0;\n pointer-events: none;\n }\n\n &-search--inline {\n position: absolute;\n width: 100%;\n height: 100%;\n\n .@{select-prefix-cls}-search__field__wrap {\n width: 100%;\n height: 100%;\n }\n\n .@{select-prefix-cls}-search__field {\n width: 100%;\n height: 100%;\n font-size: 100%;\n line-height: 1;\n background: transparent;\n border-width: 0;\n border-radius: @border-radius-base;\n outline: 0;\n }\n\n > i {\n float: right;\n }\n }\n\n &-selection--multiple {\n min-height: @input-height-base;\n padding-bottom: 3px;\n cursor: text;\n .clearfix;\n\n .@{select-prefix-cls}-search--inline {\n position: static;\n float: left;\n width: auto;\n max-width: 100%;\n padding: 0;\n .@{select-prefix-cls}-search__field {\n width: 0.75em;\n max-width: 100%;\n }\n }\n\n .@{select-prefix-cls}-selection__rendered {\n height: auto;\n margin-bottom: -3px;\n margin-left: 5px;\n }\n\n .@{select-prefix-cls}-selection__placeholder {\n margin-left: 6px;\n }\n\n > ul > li,\n .@{select-prefix-cls}-selection__rendered > ul > li {\n height: @input-height-base - 8px;\n // for tree-select\n margin-top: 3px;\n line-height: @input-height-base - 8px - 2px;\n }\n\n .@{select-prefix-cls}-selection__choice {\n position: relative;\n float: left;\n max-width: 99%;\n margin-right: 4px;\n padding: 0 20px 0 10px;\n overflow: hidden;\n color: @tag-default-color;\n background-color: @tag-default-bg;\n border: 1px solid @border-color-split;\n border-radius: @border-radius-sm;\n cursor: default;\n transition: padding 0.3s @ease-in-out;\n &__disabled {\n padding: 0 10px;\n }\n }\n\n .@{select-prefix-cls}-selection__choice__content {\n display: inline-block;\n max-width: 100%;\n overflow: hidden;\n white-space: nowrap;\n text-overflow: ellipsis;\n transition: margin 0.3s @ease-in-out;\n }\n\n .@{select-prefix-cls}-selection__choice__remove {\n .iconfont-mixin();\n\n position: absolute;\n right: 4px;\n display: inline-block;\n color: @text-color-secondary;\n font-weight: bold;\n font-size: @font-size-sm;\n line-height: inherit;\n cursor: pointer;\n transition: all 0.3s;\n .iconfont-size-under-12px(10px);\n &:hover {\n color: @icon-color-hover;\n }\n }\n\n .@{select-prefix-cls}-selection__clear,\n .@{select-prefix-cls}-arrow {\n top: @input-height-base / 2;\n }\n }\n\n &-allow-clear &-selection--single &-selection-selected-value {\n padding-right: 16px;\n }\n\n &-allow-clear &-selection--multiple &-selection__rendered,\n &-show-arrow &-selection--multiple &-selection__rendered {\n margin-right: 20px; // In case that clear button will overlap content\n }\n\n &-open {\n .@{select-prefix-cls}-arrow {\n &-icon svg {\n transform: rotate(180deg);\n }\n }\n .@{select-prefix-cls}-selection {\n .active();\n }\n }\n\n &-combobox {\n .@{select-prefix-cls}-arrow {\n display: none;\n }\n .@{select-prefix-cls}-search--inline {\n float: none;\n width: 100%;\n height: 100%;\n }\n .@{select-prefix-cls}-search__field__wrap {\n width: 100%;\n height: 100%;\n }\n .@{select-prefix-cls}-search__field {\n position: relative;\n z-index: 1;\n width: 100%;\n height: 100%;\n box-shadow: none;\n transition: all 0.3s @ease-in-out, height 0s;\n }\n }\n &-combobox&-allow-clear &-selection:hover &-selection__rendered,\n &-combobox&-show-arrow &-selection:hover &-selection__rendered {\n margin-right: 20px; // In case that clear button will overlap content\n }\n}\n\n.@{select-prefix-cls}-dropdown {\n .reset-component;\n\n position: absolute;\n top: -9999px;\n left: -9999px;\n z-index: @zindex-dropdown;\n box-sizing: border-box;\n font-size: @font-size-base;\n // Fix select render lag of long text in chrome\n // https://github.com/ant-design/ant-design/issues/11456\n // https://github.com/ant-design/ant-design/issues/11843\n font-variant: initial;\n background-color: @component-background;\n border-radius: @border-radius-base;\n outline: none;\n box-shadow: @box-shadow-base;\n\n &.slide-up-enter.slide-up-enter-active&-placement-bottomLeft,\n &.slide-up-appear.slide-up-appear-active&-placement-bottomLeft {\n animation-name: antSlideUpIn;\n }\n\n &.slide-up-enter.slide-up-enter-active&-placement-topLeft,\n &.slide-up-appear.slide-up-appear-active&-placement-topLeft {\n animation-name: antSlideDownIn;\n }\n\n &.slide-up-leave.slide-up-leave-active&-placement-bottomLeft {\n animation-name: antSlideUpOut;\n }\n\n &.slide-up-leave.slide-up-leave-active&-placement-topLeft {\n animation-name: antSlideDownOut;\n }\n\n &-hidden {\n display: none;\n }\n\n &-menu {\n max-height: 250px;\n margin-bottom: 0;\n padding-left: 0; // Override default ul/ol\n overflow: auto;\n list-style: none;\n outline: none;\n\n &-item-group-list {\n margin: 0;\n padding: 0;\n\n > .@{select-prefix-cls}-dropdown-menu-item {\n padding-left: 20px;\n }\n }\n\n &-item-group-title {\n height: 32px;\n padding: 0 @control-padding-horizontal;\n color: @text-color-secondary;\n font-size: @font-size-sm;\n line-height: 32px;\n }\n\n &-item-group-list &-item:first-child:not(:last-child),\n &-item-group:not(:last-child) &-item-group-list &-item:last-child {\n border-radius: 0;\n }\n\n &-item {\n position: relative;\n display: block;\n padding: 5px @control-padding-horizontal;\n overflow: hidden;\n color: @text-color;\n font-weight: normal;\n line-height: 22px;\n white-space: nowrap;\n text-overflow: ellipsis;\n cursor: pointer;\n transition: background 0.3s ease;\n\n &:hover:not(&-disabled) {\n background-color: @item-hover-bg;\n }\n\n &:first-child {\n border-radius: @border-radius-base @border-radius-base 0 0;\n }\n\n &:last-child {\n border-radius: 0 0 @border-radius-base @border-radius-base;\n }\n\n &-selected {\n color: @text-color;\n font-weight: @select-item-selected-font-weight;\n background-color: @background-color-light;\n }\n\n &-disabled {\n color: @disabled-color;\n cursor: not-allowed;\n\n &:hover {\n color: @disabled-color;\n cursor: not-allowed;\n }\n }\n\n &-active:not(&-disabled) {\n background-color: @item-active-bg;\n }\n\n &-divider {\n height: 1px;\n margin: 1px 0;\n overflow: hidden;\n line-height: 0;\n background-color: @border-color-split;\n }\n }\n }\n\n &&--multiple {\n .@{select-prefix-cls}-dropdown-menu-item {\n padding-right: @control-padding-horizontal + 20;\n & .@{select-prefix-cls}-selected-icon {\n position: absolute;\n top: 50%;\n right: @control-padding-horizontal;\n color: transparent;\n font-weight: bold;\n font-size: 12px;\n text-shadow: 0 0.1px 0, 0.1px 0 0, 0 -0.1px 0, -0.1px 0;\n transform: translateY(-50%);\n transition: all 0.2s;\n }\n\n &:hover .@{select-prefix-cls}-selected-icon {\n color: fade(@black, 87%);\n }\n\n &-disabled .@{select-prefix-cls}-selected-icon {\n display: none;\n }\n\n &-selected .@{select-prefix-cls}-selected-icon,\n &-selected:hover .@{select-prefix-cls}-selected-icon {\n display: inline-block;\n color: @primary-color;\n }\n }\n }\n\n // Patch for popup adjust\n // https://github.com/ant-design/ant-design/issues/14422\n &--empty&--multiple &-menu-item {\n padding-right: @control-padding-horizontal;\n }\n\n &-container-open,\n &-open {\n .@{select-prefix-cls}-dropdown {\n display: block;\n }\n }\n}\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n\n@empty-prefix-cls: ~'@{ant-prefix}-empty';\n\n.@{empty-prefix-cls} {\n margin: 0 8px;\n font-size: @empty-font-size;\n line-height: 22px;\n text-align: center;\n\n &-image {\n height: 100px;\n margin-bottom: 8px;\n\n img {\n height: 100%;\n }\n }\n\n &-description {\n margin: 0;\n }\n\n &-footer {\n margin-top: 16px;\n }\n\n // antd internal empty style\n &-normal {\n margin: 32px 0;\n color: @disabled-color;\n\n .@{empty-prefix-cls}-image {\n height: 40px;\n }\n }\n\n &-small {\n margin: 8px 0;\n color: @disabled-color;\n\n .@{empty-prefix-cls}-image {\n height: 35px;\n }\n }\n}\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n@import './mixin';\n\n@btn-prefix-cls: ~'@{ant-prefix}-btn';\n\n// for compatible\n@btn-ghost-color: @text-color;\n@btn-ghost-bg: transparent;\n@btn-ghost-border: @border-color-base;\n\n// Button styles\n// -----------------------------\n.@{btn-prefix-cls} {\n // Fixing https://github.com/ant-design/ant-design/issues/12978\n // It is a render problem of chrome, which is only happened in the codesandbox demo\n // 0.001px solution works and I don't why\n line-height: @line-height-base - 0.001;\n .btn;\n .btn-default;\n\n // Make sure that the target of Button's click event always be `button`\n // Ref: https://github.com/ant-design/ant-design/issues/7034\n > i,\n > span {\n display: inline-block;\n pointer-events: none;\n }\n\n &-primary {\n .btn-primary;\n\n .@{btn-prefix-cls}-group &:not(:first-child):not(:last-child) {\n border-right-color: @btn-group-border;\n border-left-color: @btn-group-border;\n\n &:disabled {\n border-color: @btn-default-border;\n }\n }\n\n .@{btn-prefix-cls}-group &:first-child {\n &:not(:last-child) {\n border-right-color: @btn-group-border;\n &[disabled] {\n border-right-color: @btn-default-border;\n }\n }\n }\n\n .@{btn-prefix-cls}-group &:last-child:not(:first-child),\n .@{btn-prefix-cls}-group & + & {\n border-left-color: @btn-group-border;\n &[disabled] {\n border-left-color: @btn-default-border;\n }\n }\n }\n\n &-ghost {\n .btn-ghost;\n }\n\n &-dashed {\n .btn-dashed;\n }\n\n &-danger {\n .btn-danger;\n }\n\n &-link {\n .btn-link;\n }\n\n &-round {\n .btn-round(@btn-prefix-cls);\n }\n\n &-circle,\n &-circle-outline {\n .btn-circle(@btn-prefix-cls);\n }\n\n &::before {\n position: absolute;\n top: -1px;\n right: -1px;\n bottom: -1px;\n left: -1px;\n z-index: 1;\n display: none;\n background: @component-background;\n border-radius: inherit;\n opacity: 0.35;\n transition: opacity 0.2s;\n content: '';\n pointer-events: none;\n }\n\n .@{iconfont-css-prefix} {\n transition: margin-left 0.3s @ease-in-out;\n\n // Follow icon blur under windows. Change the render.\n // https://github.com/ant-design/ant-design/issues/13924\n &.@{iconfont-css-prefix}-plus,\n &.@{iconfont-css-prefix}-minus {\n > svg {\n shape-rendering: optimizeSpeed;\n }\n }\n }\n\n &&-loading {\n position: relative;\n pointer-events: none;\n }\n\n &&-loading::before {\n display: block;\n }\n\n &&-loading:not(&-circle):not(&-circle-outline):not(&-icon-only) {\n padding-left: 29px;\n .@{iconfont-css-prefix}:not(:last-child) {\n margin-left: -14px;\n }\n }\n\n &-sm&-loading:not(&-circle):not(&-circle-outline):not(&-icon-only) {\n padding-left: 24px;\n .@{iconfont-css-prefix} {\n margin-left: -17px;\n }\n }\n\n &-group {\n .btn-group(@btn-prefix-cls);\n }\n\n &:not(&-circle):not(&-circle-outline)&-icon-only {\n padding-right: 8px;\n padding-left: 8px;\n }\n\n // http://stackoverflow.com/a/21281554/3040605\n &:focus > span,\n &:active > span {\n position: relative;\n }\n\n // To ensure that a space will be placed between character and `Icon`.\n > .@{iconfont-css-prefix} + span,\n > span + .@{iconfont-css-prefix} {\n margin-left: 8px;\n }\n\n &-background-ghost {\n color: @component-background;\n background: transparent !important;\n border-color: @component-background;\n }\n\n &-background-ghost&-primary {\n .button-variant-ghost(@btn-primary-bg);\n }\n\n &-background-ghost&-danger {\n .button-variant-ghost(@btn-danger-color);\n }\n\n &-background-ghost&-link {\n .button-variant-ghost(@link-color; transparent);\n\n color: @component-background;\n }\n\n &-two-chinese-chars::first-letter {\n letter-spacing: 0.34em;\n }\n\n &-two-chinese-chars > *:not(.@{iconfont-css-prefix}) {\n margin-right: -0.34em;\n letter-spacing: 0.34em;\n }\n\n &-block {\n width: 100%;\n }\n\n // https://github.com/ant-design/ant-design/issues/12681\n &:empty {\n vertical-align: top;\n }\n}\n\na.@{btn-prefix-cls} {\n line-height: @btn-height-base - 2px;\n &-lg {\n line-height: @btn-height-lg - 2px;\n }\n &-sm {\n line-height: @btn-height-sm - 2px;\n }\n}\n\n@primary-color: hsl(200, 100%, 50%);@white: #000;@black: #fff;@primary-1: fade(@primary-color, 50%);@primary-2: fade(@primary-color, 40%);@body-background: hsl(200, 10%, 20%);@component-background: @body-background;@text-color: hsl(200, 10%, 90%);@text-color-secondary: hsl(200, 20%, 80%);@text-color-dark: fade(white, 85%);@text-color-secondary-dark: fade(white, 65%);@heading-color: fade(@black, 85%);@border-radius-base: 2px;@border-radius-sm: 0px;@background-color-light: lighten(@component-background, 20%);@background-color-base: fade(@primary-color, 20%);@item-active-bg: fade(@primary-color, 20%);@item-hover-bg: fade(@primary-color, 10%);@border-color-base: lighten(@component-background, 20%);@border-color-split: lighten(@component-background, 10%);@disabled-color: fade(#fff, 50%);@animation-duration-slow: 0.1s;@animation-duration-base: 0.066s;@animation-duration-fast: 0.033s;@input-bg: darken(@component-background, 5%);@btn-default-bg: lighten(@component-background, 10%);@modal-mask-bg: fade(black, 80%);@table-selected-row-bg: @item-active-bg;@table-row-hover-bg: @item-hover-bg;@menu-dark-bg: @component-background;","// mixins for button\n// ------------------------\n.button-size(@height; @padding; @font-size; @border-radius) {\n height: @height;\n padding: @padding;\n font-size: @font-size;\n border-radius: @border-radius;\n}\n\n.button-disabled(@color: @btn-disable-color; @background: @btn-disable-bg; @border: @btn-disable-border) {\n &-disabled,\n &.disabled,\n &[disabled] {\n &,\n &:hover,\n &:focus,\n &:active,\n &.active {\n .button-color(@color; @background; @border);\n\n text-shadow: none;\n box-shadow: none;\n }\n }\n}\n\n.button-variant-primary(@color; @background) {\n .button-color(@color; @background; @background);\n\n text-shadow: @btn-text-shadow;\n box-shadow: @btn-primary-shadow;\n\n &:hover,\n &:focus {\n .button-color(\n @color; ~`colorPalette('@{background}', 5) `; ~`colorPalette('@{background}', 5) `\n );\n }\n\n &:active,\n &.active {\n .button-color(\n @color; ~`colorPalette('@{background}', 7) `; ~`colorPalette('@{background}', 7) `\n );\n }\n\n .button-disabled();\n}\n\n.button-variant-other(@color; @background; @border) {\n .button-color(@color; @background; @border);\n\n &:hover,\n &:focus {\n .button-color(\n ~`colorPalette('@{btn-primary-bg}', 5) `; @background; ~`colorPalette('@{btn-primary-bg}', 5)\n `\n );\n }\n &:active,\n &.active {\n .button-color(\n ~`colorPalette('@{btn-primary-bg}', 7) `; @background; ~`colorPalette('@{btn-primary-bg}', 7)\n `\n );\n }\n .button-disabled();\n}\n.button-variant-danger(@color; @background; @border) {\n .button-color(@color; @background; @border);\n &:hover {\n .button-color(\n @btn-primary-color; ~`colorPalette('@{color}', 5) `; ~`colorPalette('@{color}', 5) `\n );\n }\n &:focus {\n .button-color(\n ~`colorPalette('@{color}', 5) `; @component-background; ~`colorPalette('@{color}', 5) `\n );\n }\n &:active,\n &.active {\n .button-color(\n @btn-primary-color; ~`colorPalette('@{color}', 7) `; ~`colorPalette('@{color}', 7) `\n );\n }\n .button-disabled();\n}\n.button-variant-ghost(@color; @border: @color) {\n .button-color(@color; transparent; @border);\n text-shadow: none;\n &:hover,\n &:focus {\n & when (@border = transparent) {\n .button-color(~`colorPalette('@{color}', 5) `; transparent; transparent);\n }\n & when not(@border = transparent) {\n .button-color(~`colorPalette('@{color}', 5) `; transparent; ~`colorPalette('@{color}', 5) `);\n }\n }\n &:active,\n &.active {\n & when (@border = transparent) {\n .button-color(~`colorPalette('@{color}', 7) `; transparent; transparent);\n }\n & when not(@border = transparent) {\n .button-color(~`colorPalette('@{color}', 7) `; transparent; ~`colorPalette('@{color}', 7) `);\n }\n }\n .button-disabled();\n}\n.button-color(@color; @background; @border) {\n color: @color;\n background-color: @background;\n border-color: @border;\n // a inside Button which only work in Chrome\n // http://stackoverflow.com/a/17253457\n > a:only-child {\n color: currentColor;\n &::after {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n background: transparent;\n content: '';\n }\n }\n}\n.button-group-base(@btnClassName) {\n position: relative;\n display: inline-block;\n > .@{btnClassName},\n > span > .@{btnClassName} {\n position: relative;\n &:hover,\n &:focus,\n &:active,\n &.active {\n z-index: 2;\n }\n &:disabled {\n z-index: 0;\n }\n }\n // size\n &-lg > .@{btnClassName},\n &-lg > span > .@{btnClassName} {\n .button-size(@btn-height-lg; @btn-padding-lg; @btn-font-size-lg; 0);\n line-height: @btn-height-lg - 2px;\n }\n &-sm > .@{btnClassName},\n &-sm > span > .@{btnClassName} {\n .button-size(@btn-height-sm; @btn-padding-sm; @font-size-base; 0);\n line-height: @btn-height-sm - 2px;\n > .@{iconfont-css-prefix} {\n font-size: @font-size-base;\n }\n }\n}\n// Base styles of buttons\n// --------------------------------------------------\n.btn() {\n position: relative;\n display: inline-block;\n font-weight: @btn-font-weight;\n white-space: nowrap;\n text-align: center;\n background-image: none;\n border: @btn-border-width @btn-border-style transparent;\n box-shadow: @btn-shadow;\n cursor: pointer;\n transition: all 0.3s @ease-in-out;\n user-select: none;\n touch-action: manipulation;\n .button-size(@btn-height-base; @btn-padding-base; @font-size-base; @btn-border-radius-base);\n > .@{iconfont-css-prefix} {\n line-height: 1;\n }\n &,\n &:active,\n &:focus {\n outline: 0;\n }\n &:not([disabled]):hover {\n text-decoration: none;\n }\n &:not([disabled]):active {\n outline: 0;\n box-shadow: none;\n }\n &.disabled,\n &[disabled] {\n cursor: not-allowed;\n > * {\n pointer-events: none;\n }\n }\n &-lg {\n .button-size(@btn-height-lg; @btn-padding-lg; @btn-font-size-lg; @btn-border-radius-base);\n }\n &-sm {\n .button-size(@btn-height-sm; @btn-padding-sm; @btn-font-size-sm; @btn-border-radius-sm);\n }\n}\n// primary button style\n.btn-primary() {\n .button-variant-primary(@btn-primary-color; @btn-primary-bg);\n}\n// default button style\n.btn-default() {\n .button-variant-other(@btn-default-color; @btn-default-bg; @btn-default-border);\n &:hover,\n &:focus,\n &:active,\n &.active {\n text-decoration: none;\n background: @btn-default-bg;\n }\n}\n// ghost button style\n.btn-ghost() {\n .button-variant-other(@btn-ghost-color, @btn-ghost-bg, @btn-ghost-border);\n}\n// dashed button style\n.btn-dashed() {\n .button-variant-other(@btn-default-color, @btn-default-bg, @btn-default-border);\n border-style: dashed;\n}\n// danger button style\n.btn-danger() {\n .button-variant-danger(@btn-danger-color, @btn-danger-bg, @btn-danger-border);\n}\n// link button style\n.btn-link() {\n .button-variant-other(@link-color, transparent, transparent);\n box-shadow: none;\n &:hover,\n &:focus,\n &:active {\n border-color: transparent;\n }\n .button-disabled(@disabled-color; transparent; transparent);\n}\n// round button\n.btn-round(@btnClassName: btn) {\n .button-size(@btn-circle-size; 0 @btn-circle-size / 2; @font-size-base + 2px; @btn-circle-size);\n &.@{btnClassName}-lg {\n .button-size(\n @btn-circle-size-lg; 0 @btn-circle-size-lg / 2; @btn-font-size-lg + 2px; @btn-circle-size-lg\n );\n }\n &.@{btnClassName}-sm {\n .button-size(\n @btn-circle-size-sm; 0 @btn-circle-size-sm / 2; @font-size-base; @btn-circle-size-sm\n );\n }\n}\n// circle button: the content only contains icon\n.btn-circle(@btnClassName: btn) {\n .square(@btn-circle-size);\n .button-size(@btn-circle-size; 0; @font-size-base + 2px; 50%);\n &.@{btnClassName}-lg {\n .square(@btn-circle-size-lg);\n .button-size(@btn-circle-size-lg; 0; @btn-font-size-lg + 2px; 50%);\n }\n &.@{btnClassName}-sm {\n .square(@btn-circle-size-sm);\n .button-size(@btn-circle-size-sm; 0; @font-size-base; 50%);\n }\n}\n// Horizontal button groups style\n// --------------------------------------------------\n.btn-group(@btnClassName: btn) {\n .button-group-base(@btnClassName);\n .@{btnClassName} + .@{btnClassName},\n .@{btnClassName} + &,\n span + .@{btnClassName},\n .@{btnClassName} + span,\n > span + span,\n & + .@{btnClassName},\n & + & {\n margin-left: -1px;\n }\n .@{btnClassName}-primary + .@{btnClassName}:not(.@{btnClassName}-primary):not([disabled]) {\n border-left-color: transparent;\n }\n .@{btnClassName} {\n border-radius: 0;\n }\n > .@{btnClassName}:first-child,\n > span:first-child > .@{btnClassName} {\n margin-left: 0;\n }\n > .@{btnClassName}:only-child {\n border-radius: @btn-border-radius-base;\n }\n > span:only-child > .@{btnClassName} {\n border-radius: @btn-border-radius-base;\n }\n > .@{btnClassName}:first-child:not(:last-child),\n > span:first-child:not(:last-child) > .@{btnClassName} {\n border-top-left-radius: @btn-border-radius-base;\n border-bottom-left-radius: @btn-border-radius-base;\n }\n > .@{btnClassName}:last-child:not(:first-child),\n > span:last-child:not(:first-child) > .@{btnClassName} {\n border-top-right-radius: @btn-border-radius-base;\n border-bottom-right-radius: @btn-border-radius-base;\n }\n &-sm {\n > .@{btnClassName}:only-child {\n border-radius: @btn-border-radius-sm;\n }\n > span:only-child > .@{btnClassName} {\n border-radius: @btn-border-radius-sm;\n }\n > .@{btnClassName}:first-child:not(:last-child),\n > span:first-child:not(:last-child) > .@{btnClassName} {\n border-top-left-radius: @btn-border-radius-sm;\n border-bottom-left-radius: @btn-border-radius-sm;\n }\n > .@{btnClassName}:last-child:not(:first-child),\n > span:last-child:not(:first-child) > .@{btnClassName} {\n border-top-right-radius: @btn-border-radius-sm;\n border-bottom-right-radius: @btn-border-radius-sm;\n }\n }\n & > & {\n float: left;\n }\n & > &:not(:first-child):not(:last-child) > .@{btnClassName} {\n border-radius: 0;\n }\n & > &:first-child:not(:last-child) {\n > .@{btnClassName}:last-child {\n padding-right: 8px;\n border-top-right-radius: 0;\n border-bottom-right-radius: 0;\n }\n }\n & > &:last-child:not(:first-child) > .@{btnClassName}:first-child {\n padding-left: 8px;\n border-top-left-radius: 0;\n border-bottom-left-radius: 0;\n }\n}\n","@dialog-prefix-cls: ~'@{ant-prefix}-modal';\n@table-prefix-cls: ~'@{ant-prefix}-table';\n@modal-footer-padding-vertical: 10px;\n@modal-footer-padding-horizontal: 16px;\n\n.@{dialog-prefix-cls} {\n .reset-component;\n\n position: relative;\n top: 100px;\n width: auto;\n margin: 0 auto;\n padding-bottom: 24px;\n\n &-wrap {\n position: fixed;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n z-index: @zindex-modal;\n overflow: auto;\n outline: 0;\n -webkit-overflow-scrolling: touch;\n }\n\n &-title {\n margin: 0;\n color: @heading-color;\n font-weight: 500;\n font-size: @font-size-lg;\n line-height: 22px;\n word-wrap: break-word;\n }\n\n &-content {\n position: relative;\n background-color: @component-background;\n background-clip: padding-box;\n border: 0;\n border-radius: @border-radius-base;\n box-shadow: @shadow-2;\n }\n\n &-close {\n position: absolute;\n top: 0;\n right: 0;\n z-index: @zindex-popup-close;\n padding: 0;\n color: @text-color-secondary;\n font-weight: 700;\n line-height: 1;\n text-decoration: none;\n background: transparent;\n border: 0;\n outline: 0;\n cursor: pointer;\n transition: color 0.3s;\n\n &-x {\n display: block;\n width: 56px;\n height: 56px;\n font-size: @font-size-lg;\n font-style: normal;\n line-height: 56px;\n text-align: center;\n text-transform: none;\n text-rendering: auto;\n }\n\n &:focus,\n &:hover {\n color: @icon-color-hover;\n text-decoration: none;\n }\n }\n\n &-header {\n padding: 16px 24px;\n color: @text-color;\n background: @modal-header-bg;\n border-bottom: @border-width-base @border-style-base @border-color-split;\n border-radius: @border-radius-base @border-radius-base 0 0;\n }\n\n &-body {\n padding: @modal-body-padding;\n font-size: @font-size-base;\n line-height: @line-height-base;\n word-wrap: break-word;\n }\n\n &-footer {\n padding: @modal-footer-padding-vertical @modal-footer-padding-horizontal;\n text-align: right;\n background: @modal-footer-bg;\n border-top: @border-width-base @border-style-base @border-color-split;\n border-radius: 0 0 @border-radius-base @border-radius-base;\n button + button {\n margin-bottom: 0;\n margin-left: 8px;\n }\n }\n\n &.zoom-enter,\n &.zoom-appear {\n transform: none; // reset scale avoid mousePosition bug\n opacity: 0;\n animation-duration: @animation-duration-slow;\n user-select: none; // https://github.com/ant-design/ant-design/issues/11777\n }\n\n &-mask {\n position: fixed;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n z-index: @zindex-modal-mask;\n height: 100%;\n background-color: @modal-mask-bg;\n filter: ~'alpha(opacity=50)';\n\n &-hidden {\n display: none;\n }\n }\n\n &-open {\n overflow: hidden;\n }\n}\n\n.@{dialog-prefix-cls}-centered {\n text-align: center;\n &::before {\n display: inline-block;\n width: 0;\n height: 100%;\n vertical-align: middle;\n content: '';\n }\n .@{dialog-prefix-cls} {\n top: 0;\n display: inline-block;\n text-align: left;\n vertical-align: middle;\n }\n}\n\n@media (max-width: @screen-sm-max) {\n .@{dialog-prefix-cls} {\n max-width: calc(100vw - 16px);\n margin: 8px auto;\n }\n .@{dialog-prefix-cls}-centered {\n .@{dialog-prefix-cls} {\n flex: 1;\n }\n }\n}\n","@import '../../style/mixins/index';\n\n@confirm-prefix-cls: ~'@{ant-prefix}-modal-confirm';\n\n.@{confirm-prefix-cls} {\n .@{ant-prefix}-modal-header {\n display: none;\n }\n\n .@{ant-prefix}-modal-close {\n display: none;\n }\n\n .@{ant-prefix}-modal-body {\n padding: 32px 32px 24px;\n }\n\n &-body-wrapper {\n .clearfix();\n }\n\n &-body {\n .@{confirm-prefix-cls}-title {\n display: block;\n // create BFC to avoid\n // https://user-images.githubusercontent.com/507615/37702510-ba844e06-2d2d-11e8-9b67-8e19be57f445.png\n overflow: hidden;\n color: @heading-color;\n font-weight: 500;\n font-size: @font-size-lg;\n line-height: 1.4;\n }\n\n .@{confirm-prefix-cls}-content {\n margin-top: 8px;\n color: @text-color;\n font-size: @font-size-base;\n }\n\n > .@{iconfont-css-prefix} {\n float: left;\n margin-right: 16px;\n font-size: 22px;\n\n // `content` after `icon` should set marginLeft\n + .@{confirm-prefix-cls}-title + .@{confirm-prefix-cls}-content {\n margin-left: 38px;\n }\n }\n }\n\n .@{confirm-prefix-cls}-btns {\n float: right;\n margin-top: 24px;\n\n button + button {\n margin-bottom: 0;\n margin-left: 8px;\n }\n }\n\n &-error &-body > .@{iconfont-css-prefix} {\n color: @error-color;\n }\n\n &-warning &-body > .@{iconfont-css-prefix},\n &-confirm &-body > .@{iconfont-css-prefix} {\n color: @warning-color;\n }\n\n &-info &-body > .@{iconfont-css-prefix} {\n color: @info-color;\n }\n\n &-success &-body > .@{iconfont-css-prefix} {\n color: @success-color;\n }\n}\n","@import '../../input/style/mixin';\n\n.form-control-validation(@text-color: @input-color; @border-color: @input-border-color; @background-color: @input-bg) {\n .@{ant-prefix}-form-explain,\n .@{ant-prefix}-form-split {\n color: @text-color;\n }\n // 输入框的不同校验状态\n .@{ant-prefix}-input {\n &,\n &:hover {\n border-color: @border-color;\n }\n\n &:focus {\n .active(@border-color);\n }\n\n &:not([disabled]):hover {\n border-color: @border-color;\n }\n }\n\n .@{ant-prefix}-calendar-picker-open .@{ant-prefix}-calendar-picker-input {\n .active(@border-color);\n }\n\n // Input prefix\n .@{ant-prefix}-input-affix-wrapper {\n .@{ant-prefix}-input {\n &,\n &:hover {\n background-color: @background-color;\n border-color: @border-color;\n }\n\n &:focus {\n .active(@border-color);\n }\n }\n\n &:hover .@{ant-prefix}-input:not(.@{ant-prefix}-input-disabled) {\n border-color: @border-color;\n }\n }\n\n .@{ant-prefix}-input-prefix {\n color: @text-color;\n }\n\n .@{ant-prefix}-input-group-addon {\n color: @text-color;\n background-color: @background-color;\n border-color: @border-color;\n }\n\n .has-feedback {\n color: @text-color;\n }\n}\n\n// Reset form styles\n// -----------------------------\n// Based on Bootstrap framework\n.reset-form() {\n legend {\n display: block;\n width: 100%;\n margin-bottom: 20px;\n padding: 0;\n color: @text-color-secondary;\n font-size: @font-size-lg;\n line-height: inherit;\n border: 0;\n border-bottom: @border-width-base @border-style-base @border-color-base;\n }\n\n label {\n font-size: @font-size-base;\n }\n\n input[type='search'] {\n box-sizing: border-box;\n }\n\n // Position radios and checkboxes better\n input[type='radio'],\n input[type='checkbox'] {\n line-height: normal;\n }\n\n input[type='file'] {\n display: block;\n }\n\n // Make range inputs behave like textual form controls\n input[type='range'] {\n display: block;\n width: 100%;\n }\n\n // Make multiple select elements height not fixed\n select[multiple],\n select[size] {\n height: auto;\n }\n\n // Focus for file, radio, and checkbox\n input[type='file']:focus,\n input[type='radio']:focus,\n input[type='checkbox']:focus {\n outline: thin dotted;\n outline: 5px auto -webkit-focus-ring-color;\n outline-offset: -2px;\n }\n\n // Adjust output element\n output {\n display: block;\n padding-top: 15px;\n color: @input-color;\n font-size: @font-size-base;\n line-height: @line-height-base;\n }\n}\n","@import '../../style/themes/index';\n@import '../../style/mixins/index';\n@import '../../input/style/mixin';\n@import '../../button/style/mixin';\n@import '../../grid/style/mixin';\n@import './mixin';\n\n@form-prefix-cls: ~'@{ant-prefix}-form';\n@form-component-height: @input-height-base;\n@form-component-max-height: @input-height-lg;\n@form-feedback-icon-size: @font-size-base;\n@form-help-margin-top: (@form-component-height - @form-component-max-height) / 2 + 2px;\n@form-explain-font-size: @font-size-base;\n// Extends additional 1px to fix precision issue.\n// https://github.com/ant-design/ant-design/issues/12803\n// https://github.com/ant-design/ant-design/issues/8220\n@form-explain-precision: 1px;\n@form-explain-height: floor(@form-explain-font-size * @line-height-base);\n\n.@{form-prefix-cls} {\n .reset-component;\n .reset-form;\n}\n\n.@{form-prefix-cls}-item-required::before {\n display: inline-block;\n margin-right: 4px;\n color: @label-required-color;\n font-size: @font-size-base;\n font-family: SimSun, sans-serif;\n line-height: 1;\n content: '*';\n .@{form-prefix-cls}-hide-required-mark & {\n display: none;\n }\n}\n\n.@{form-prefix-cls}-item-label > label {\n color: @label-color;\n\n &::after {\n & when (@form-item-trailing-colon=true) {\n content: ':';\n }\n & when not (@form-item-trailing-colon=true) {\n content: ' ';\n }\n\n position: relative;\n top: -0.5px;\n margin: 0 8px 0 2px;\n }\n\n &.@{form-prefix-cls}-item-no-colon::after {\n content: ' ';\n }\n}\n\n// Radio && Checkbox\ninput[type='radio'],\ninput[type='checkbox'] {\n &[disabled],\n &.disabled {\n cursor: not-allowed;\n }\n}\n\n// These classes are used directly on