diff --git a/.env.example b/.env.example deleted file mode 100644 index 4a67f7c..0000000 --- a/.env.example +++ /dev/null @@ -1,18 +0,0 @@ -# Base URL of the Django REST backend. -# -# - '/api' same-origin. The Vite dev server proxies -# this prefix to VITE_API_PROXY_TARGET (see -# below); in production your web server -# routes /api to Django. -# - 'http://localhost:8000/api' talk to Django directly. CORS must be -# enabled there. No proxy is registered. -VITE_API_BASE_URL=/api - -# Where the dev server should proxy API requests when VITE_API_BASE_URL is a -# relative path. Ignored for absolute URLs or when mocking. -VITE_API_PROXY_TARGET=http://localhost:8000 - -# When set to 'true', the Vite dev server serves a fake backend from -# mocks/vitePlugin.ts. No Django required. The real backend (if any) is -# ignored in that case. -VITE_USE_MOCK_API=false diff --git a/.gitignore b/.gitignore index 15dd615..3b462cb 100644 --- a/.gitignore +++ b/.gitignore @@ -21,7 +21,3 @@ Thumbs.db # Vite vite.config.js.timestamp-* vite.config.ts.timestamp-* - -# AI tools -.claude -tmpclaude* diff --git a/.prettierrc b/.prettierrc deleted file mode 100644 index f27d159..0000000 --- a/.prettierrc +++ /dev/null @@ -1,8 +0,0 @@ -{ - "tabWidth": 4, - "endOfLine": "lf", - "printWidth": 120, - "useTabs": true, - "htmlWhitespaceSensitivity": "ignore", - "bracketSameLine": true -} diff --git a/README.md b/README.md index a561a1a..b5b2950 100644 --- a/README.md +++ b/README.md @@ -1,44 +1,38 @@ -# leaflet_svelte +# sv -Weather-balloon trajectory planner. Static SvelteKit SPA that talks to a Django -REST backend. +Everything you need to build a Svelte project, powered by [`sv`](https://github.com/sveltejs/cli). -## Quick start +## Creating a project + +If you're seeing this, you've probably already done this step. Congrats! ```bash -npm install +# create a new project in the current directory +npx sv create -# dev against a local Django on :8000 (see .env.example) +# create a new project in my-app +npx sv create my-app +``` + +## Developing + +Once you've created a project and installed dependencies with `npm install` (or `pnpm install` or `yarn`), start a development server: + +```bash npm run dev -# dev with a fake backend (no Django needed) -VITE_USE_MOCK_API=true npm run dev +# or start the server and open the app in a new browser tab +npm run dev -- --open +``` -# type-check + production build (emits static files to ./build) -npm run check +## Building + +To create a production version of your app: + +```bash npm run build ``` -Serve `build/` from any static host. Route fallback is `index.html`. +You can preview the production build with `npm run preview`. -## Documentation - -- [`docs/ARCHITECTURE.md`](docs/ARCHITECTURE.md) — module layout and data flow. -- [`docs/CONVENTIONS.md`](docs/CONVENTIONS.md) — naming, styling, component patterns. -- [`docs/ADDING_A_FEATURE.md`](docs/ADDING_A_FEATURE.md) — walkthrough for adding - a new panel/feature. -- [`docs/TESTING.md`](docs/TESTING.md) — e2e tests against the real Django + - predictor stack. -- [`mocks/`](mocks/) — Vite dev-server mock backend. -- [`scripts/run-stack.sh`](scripts/run-stack.sh) — boot Vite + Django + - fake predictor in one command. - -## Stack - -- **SvelteKit + Vite** (TypeScript, Svelte 5 runes) — built as a pure SPA with - `@sveltejs/adapter-static`. -- **MapLibre GL JS** via the `$map` abstraction — the app never imports - `maplibre-gl` directly outside `src/lib/map/`. -- **Sveltestrap** + Bootstrap 5 for UI chrome. -- **Chart.js** for the ascent/descent profile editor. -- **@vincjo/datatables** for editor tables. +> To deploy your app, you may need to install an [adapter](https://svelte.dev/docs/kit/adapters) for your target environment. diff --git a/build.js b/build.js new file mode 100644 index 0000000..266cd5a --- /dev/null +++ b/build.js @@ -0,0 +1,390 @@ +const compile = require( 'svelte/compiler' ).compile + +const chokidar = require( 'chokidar' ); +const esbuild = require( 'esbuild' ); +const {readdirSync, statSync, existsSync, writeFileSync, readFileSync} = require( 'fs' ); +const {join, basename, resolve, dirname, relative} = require( 'path' ); +const sveltePlugin = require( 'esbuild-svelte' ); +const {sum} = require( 'lodash' ); +const parse5 = require( 'parse5' ); +const notifier = require('node-notifier'); + +process.on('uncaughtException', error => { + notifier.notify({ + title: 'Error occurs', + message: `${error}` + }); +}); + +const [watch, serve, minify, debug, logVars] = ['--watch', '--serve', '--minify', '--debug', '--log-vars'].map( s => + process.argv.includes( s ) +); +const debug_console_log = ( args, returnIndex = 0 ) => (debug && console.log( ...args ), args[ returnIndex ]); + +const ignorePath = new Set( [ + 'node_modules', + '.vscode', + '.idea', + '.git', + '.gitignore', + 'build.js', + 'package-lock.json', + 'package.json', + 'README.md', + 'build.js', + 'pullpush.sh', +] ); + +// find page candidates +function findPages( dir = '.', sink = [] ) { + if( ignorePath.has( dir.replace( './', '' ).replace( '.\\', '' ) ) ) { + debug && console.log( 'skip:', dir ); + return; + } + + const files = readdirSync( dir ).filter( f => f[ 0 ]!=='_' ); + const svelteFiles = files.filter( f => f.endsWith( '.svelte' ) && statSync( join( dir, f ) ).isFile() ); + svelteFiles.forEach( f => sink.push( join( dir, f ) ) ); + + files + .filter( f => !svelteFiles.includes( f ) ) + .map( f => join( dir, f ) ) + .filter( f => statSync( f ).isDirectory() ) + .forEach( f => findPages( f, sink ) ); + return sink; +} + +const _zId_prefix = `z_placeholder_${Math.floor( Math.random() * 1000000000 ).toString( 16 )}_`; +const _zReplacer = s => debug_console_log( ['z-replace:', s, `"${_zId_prefix}${Buffer.from( s ).toString( 'base64' )}"`], 2 ); + +const zPlaceholderReplacer = content => + + content?.replace( /\#\{\s*\w+\s*\}/gs, _zReplacer ) // #{ key } + .replace( /\/\*\!\s*\w+\s*\*\//gs, _zReplacer ) // map /*! mapKey */ + .replace( /\[\s*\/\*\s*\w+\s*\*\/\s*\]/gs, _zReplacer ) // map [/* mapKey */] + .replace( /\{\s*\/\*\s*\w+\s*\*\/\s*\}/gs, _zReplacer ); // map {/* mapKey */} + +global.zPlaceholderReplacer = zPlaceholderReplacer; + +const zPlaceholderRestore = ( content, sink ) => + content?.replace( new RegExp( `("|')${_zId_prefix}(\\w+=*)\\1`, 'g' ), ( _, _2, s ) => { + s = Buffer.from( s, 'base64' ).toString( 'ascii' ); + sink.push( s ); + debug && console.log( 'z-restore', _, s ); + return s; + } ); + +const svelteJsPathResolver = { + name: 'svelteJsPathResolver', + setup( build ) { + const options = {filter: /\.svelte\.(ts)$/}; + + build.onResolve( options, ( {path, resolveDir} ) => ({path: join( resolveDir, path )}) ); + build.onLoad( options, ( {path} ) => { + return { + contents: ` + import App from "./${basename( path ).replace( /\.ts$/, '' )}"; + export const app = new App({ target: document.getElementById("app") }); + `, + loader: 'ts', + resolveDir: dirname( path ), + }; + } ); + }, +}; + +function createBuilder( entryPoints ) { + console.log( 'pages:', entryPoints ); + + return esbuild.build( { + entryPoints: entryPoints.map( s => s + '.ts' ), + bundle: true, + outdir: '.', + write: false, + plugins: [svelteJsPathResolver, sveltePlugin( require( './svelte.config' ) )], + incremental: !!watch, + sourcemap: false, + minify, + } ) +} + +function layoutFor( path, content = {} ) { + path = (() => { + let temp = join( path, '..', '_layout.html' ); + + while( true ) { + if( existsSync( temp ) ) return temp; + if( resolve( __dirname )===resolve( dirname( temp ) ) ) return; + + temp = join( temp, '../..', '_layout.html' ); + } + })(); + + layoutFor.cache = layoutFor.cache || {}; + + const defaultKey = '_DEFAULT_LAYOUT'; + if( !path && layoutFor.cache[ defaultKey ] ) return layoutFor.cache[ defaultKey ]; + + let cache = layoutFor.cache[ path ]; + const m = statSync( path ).mtimeMs; + if( cache && m===cache.m ) return cache; + + const tree = parse5.parse( + path + ? readFileSync( path, 'utf-8' ) + : ` + + + #{title} + + +

layout not found, please create _layout.html

+ + +` + ); + + let slot = null; + let body = null; + + let stack = [...tree.childNodes]; + + while( stack.length && (slot==null || body==null) ) { + const t = stack.pop(); + if( t.nodeName==='body' ) body = t; + if( t.nodeName==='slot' || (t.nodeName==='#comment' && t.data?.trim()==='content_goes_here') ) slot = t; + + t.childNodes && (stack = [...stack, ...t.childNodes]); + } + + if( !body ) throw new Error( 'body not found' ); + + const appKEY = `${Math.random()}-APP-${Math.random()}`; + if( slot ) { + slot.nodeName = 'main'; + slot.tagName = 'main'; + delete slot.data; + slot.attrs = [ + {name: 'id', value: 'app'}, + ...(slot.attrs || [])?.filter( t => t.name!=='id' ) + ]; + slot.childNodes = [{nodeName: '#text', value: appKEY}] + } else { + body.childNodes.push( { + nodeName: 'main', + tagName: 'main', + attrs: [{name: 'id', value: 'app'}], + childNodes: [{nodeName: '#text', value: appKEY}], + namespaceURI: body.namespaceURI, + } ); + } + + // Remove main content + // content showing only for seo friendly + body.childNodes.push( { + nodeName: 'script', + tagName: 'script', + attrs: [], + childNodes: [{nodeName: '#text', value: "document.getElementById('app').innerHTML=''"}], + namespaceURI: body.namespaceURI, + } ) + + const jsKEY = `${Math.random()}-JS-${Math.random()}`; + const cssKEY = `${Math.random()}-CSS-${Math.random()}`; + const comments = { + nodeName: '#comment', + data: '', + }; + const cssVarsComments = { + nodeName: '#comment', + data: '', + }; + const jsVarsComments = { + nodeName: '#comment', + data: '', + }; + + body.childNodes = [ + ...body.childNodes, + comments, + { + nodeName: '#text', + value: '\n', + }, + logVars && cssVarsComments, + { + nodeName: '#text', + value: '\n', + }, + logVars && jsVarsComments, + { + nodeName: '#text', + value: '\n', + }, + { + nodeName: 'style', + tagName: 'style', + attrs: [], + childNodes: [{nodeName: '#text', value: cssKEY}], + namespaceURI: body.namespaceURI, + }, + { + nodeName: '#text', + value: '\n', + }, + { + nodeName: 'script', + tagName: 'script', + attrs: [], + childNodes: [{nodeName: '#text', value: jsKEY}], + namespaceURI: body.namespaceURI, + }, + { + nodeName: '#text', + value: '\n', + }, + ]; + + debug && console.log( 'build layout for:', path || defaultKey ); + + return (layoutFor.cache[ path || defaultKey ] = ( {js, css} ) => { + const cssVars = [], + jsVars = []; + js = zPlaceholderRestore( js, jsVars ) || ''; + css = zPlaceholderRestore( css, cssVars ) || ''; + + //comments.data = `BUILD TIME: ${new Date().toISOString()}`; + cssVarsComments.data = cssVars.length ? `--- CSS z-vars --- \n${cssVars.join( '\n' )}` : ''; + jsVarsComments.data = jsVars.length ? `--- JS z-vars --- \n${jsVars.join( '\n' )}` : ''; + + let html = content.html || ''; + const innerCss = (content.css || {}).code || ''; + + return parse5.serialize( tree ). + replace( cssKEY, css + innerCss ). + replace( jsKEY, js ). + replace( appKEY, html ). + replaceAll("fakecss:"+__dirname,'fakecss:.'); + }); +} + +(async () => { + let watcherReady = false; + + watch && console.log( 'first build start' ); + let pages = findPages(); + let builder = await createBuilder( pages ); + + const compiledFiles = new Set(); + let cache = {}; + + function saveFiles( files = builder, layoutChanged = false ) { + const output = {}; + let unchanged = 0; + // path = bla.svelte.js or bla.svelte.css + for( const {path, text} of files.outputFiles ) { + const ext = /\.(\w+)$/.exec( path )?.[ 1 ]; + if( ext!=='css' && ext!=='js' ) throw new Error( 'unknown ext:' + ext ); + + // bla.js or bla.css + const key = path.replace( /\.svelte\.\w+$/, '' ); + + output[ key ] = output[ key ] || {}; + output[ key ][ ext ] = text + + if( cache[ path ]===text && !layoutChanged ) { + unchanged += 1; + continue; + } + cache[ path ] = text; + } + + // do nothing if nothing's changed + if( unchanged===files.outputFiles.length ) return; + + if( Object.keys( output ).length===0 ) return console.log( 'no changes' ); + + // for each .html files need to be generated + Object.entries( output ).forEach( ( [path, data] ) => { + const renderedSvelte = compile( path + '.svelte' ); + + const content = layoutFor( path, renderedSvelte )( data ); + + path = resolve( path + '.html' ); + compiledFiles.add( path ); + console.log( 'compiled:', relative( resolve( __dirname ), path ) ); + writeFileSync( path, content ); + } ); + } + + saveFiles(); + watch && console.log( 'first build end' ); + + if( watch ) { + const pagesPaths = new Set( pages.map( p => resolve( p ) ) ); + + let timeRef = null; + + function changeListener( path, stats, type, watcher ) { + switch (type) { + case 'change': + notifier.notify({ + title: 'Change occurs', + message: `Change occurs in "${path}"` + }); + break; + case 'add': + notifier.notify({ + title: 'File added', + message: `Added file "${path}"` + }); + break; + case 'unlink': + notifier.notify({ + title: 'File remove', + message: `Removed file "${path}"` + }); + break; + } + + if( compiledFiles.has( resolve( path ) ) ) return; + console.log( type + ':', path.replace( __dirname, '' ) ); + + const svelteFile = path[ 0 ]!=='_' && path.endsWith( '.svelte' ); + + let pagesChanged = true; + if( svelteFile && type==='add' ) pagesPaths.add( resolve( path ) ); + else if( svelteFile && type==='unlink' ) pagesPaths.delete( resolve( path ) ); + else pagesChanged = false; + + let layoutChanged = path.endsWith( '_layout.html' ); + + if( timeRef ) clearTimeout( timeRef ); + timeRef = setTimeout( async () => { + pagesChanged + ? saveFiles( (builder = await createBuilder( Array.from( pagesPaths, p => relative( __dirname, p ) ) )), layoutChanged ) + : saveFiles( await builder.rebuild(), layoutChanged ); + }, 200 ); + } + + const watcher = chokidar + .watch( '.', {ignored: s => ignorePath.has( s ) || ignorePath.has( join( './', s ) ), ignoreInitial: true} ) + .on( 'change', ( path, stats ) => changeListener( path, stats, 'change', watcher ) ) + .on( 'add', ( path, stats ) => changeListener( path, stats, 'add', watcher ) ) + .on( 'unlink', ( path, stats ) => changeListener( path, stats, 'unlink', watcher ) ) + .on( 'ready', () => { + console.log( `watching ${sum( Object.values( watcher.getWatched() ).map( t => t.length ) )} files/dirs for changes` ); + watcherReady = true; + } ) + .on( 'error', err => console.log( 'ERROR:', err ) ); + } + + const FiveServer = require( 'five-server' ).default; + serve && + (await new FiveServer().start( { + open: true, + workspace: __dirname, + ignore: [...ignorePath, /\.(js|ts|svelte)$/, /\_layout\.html$/], + wait: 500, + } )); +})(); diff --git a/docs/ADDING_A_FEATURE.md b/docs/ADDING_A_FEATURE.md deleted file mode 100644 index 5211257..0000000 --- a/docs/ADDING_A_FEATURE.md +++ /dev/null @@ -1,152 +0,0 @@ -# Adding a feature - -Walk-through for building a new side-panel feature. We'll use a hypothetical -"NOTAMs" panel (no-fly-zone overlays) as the example. - -## 1. Pick a feature folder - -``` -src/lib/features/notams/ -├── index.ts -├── store.ts -├── types.ts -├── NotamsPanel.svelte -└── NotamRenderer.svelte # if you draw on the map -``` - -Re-export the public surface from `index.ts`. - -## 2. Define the domain - -If the feature introduces a durable type (something saved to a server or -localStorage), put the type in `src/lib/domain/notam.ts` and re-export from -`src/lib/domain/index.ts`. Ephemeral state types can live in the feature's -`types.ts`. - -```ts -// src/lib/domain/notam.ts -export interface Notam { - id: string; - code: string; - polygon: LatLngTuple[]; - validFrom: string; - validTo: string; -} -``` - -## 3. Add state - -Feature-scoped stores go in the feature folder. Use `persisted()` if the user -should not lose the state on reload, plain `writable()` otherwise. - -```ts -// src/lib/features/notams/store.ts -import { persisted } from '$state'; -import type { Notam } from '$domain'; - -export const notamsStore = persisted('notams', []); -``` - -## 4. Add API (if needed) - -```ts -// src/lib/api/notams.ts -import { api } from './client'; -import type { Notam } from '$domain'; -export const notamsApi = { list: () => api.get('/notams/') }; -``` - -And export from `src/lib/api/index.ts`. - -## 5. Build the panel - -Re-use UI primitives. Wrap everything in `CollapsibleCard` so it lands in the -side panel. - -```svelte - - - - - {#each $notamsStore as notam (notam.id)} -
{notam.code}
- {/each} -
-``` - -## 6. Add i18n keys - -Append to `src/lib/i18n/locales/ru.json` and `en.json`. The lookup is -dot-separated: - -```json -{ - "notams": { - "title": "NOTAMs", - "empty": "Нет активных NOTAMs" - } -} -``` - -## 7. Draw on the map (optional) - -Every on-map drawing goes through a Scene. One scene per feature, or per sub- -entity if the feature manages many independent overlays (like workspaces). - -```svelte - - -``` - -Add `` as a child of `` in the predict route. - -## 8. Wire into the route - -```svelte - - - - - - - {#if rightTab === 'notams'} - - {/if} - - -``` - -## 9. Add a settings entry (optional) - -If the feature should be toggleable, extend `src/lib/features/settings/store.ts` -and `schema.ts`. Keep the field declarative (kind + labelKey + path) — the -SettingsPanel renders it automatically. - -## 10. Checklist before you open a PR - -- [ ] `npm run check` passes -- [ ] `npm run build` passes -- [ ] All user-visible strings routed through `$t` -- [ ] Feature only depends on shared modules (`$domain`, `$state`, `$api`, - `$ui`, `$map`) — not on other features' internals -- [ ] `index.ts` exports only what the outside world needs -- [ ] Panel matches existing visual language (CollapsibleCard, Bootstrap sm - form controls, same spacing) diff --git a/docs/ARCHITECTURE.md b/docs/ARCHITECTURE.md deleted file mode 100644 index 6973d74..0000000 --- a/docs/ARCHITECTURE.md +++ /dev/null @@ -1,105 +0,0 @@ -# Architecture - -## Layer overview - -``` -src/ -├── routes/ # SvelteKit pages (thin; they compose features) -├── app.html # plain SPA shell -├── app.css # global styles (imports Bootstrap + bootstrap-icons) -└── lib/ - ├── domain/ # pure types + pure functions (no Svelte, no fetch) - ├── state/ # store factories (persisted, broadcast across tabs) - ├── api/ # HTTP client + resource modules (points, scenarios…) - ├── auth/ # auth store, login/logout, requireAuthenticated() - ├── i18n/ # t() store, locale loading, message dictionaries - ├── map/ # IMap interface + MapLibre impl + plot helpers + tools - ├── ui/ # library-agnostic primitives (panel, tab, editor…) - └── features/ # vertical slices — one folder per feature - ├── auth/ # Navbar, LoginForm - ├── footer/ - ├── prediction/ # ControlPanel, ScenarioPanel, Point/Scenario/Curve editors - ├── tracking/ # TelemetryPanel - ├── workspaces/ # multi-layer workspace system - ├── timeline/ # global playback clock - └── settings/ # schema-driven settings panel -mocks/ # dev-server API mock (enabled via VITE_USE_MOCK_API) -``` - -**Rules of thumb** - -- `domain/` depends on nothing. -- `state/`, `i18n/`, `api/` depend only on `domain/`. -- `map/` depends on `domain/` (and Maplibre, internally). -- `ui/` depends on `domain/`, `state/`, `i18n/` at most. -- `features/*` depend on everything in `lib/` but never on each other directly - — cross-feature coordination happens through stores (e.g. `workspacesStore` - is shared by prediction + timeline). -- `routes/` compose features. Pages stay thin. - -## Data flow - -### Stores -`$state` → persisted via `persisted()` (localStorage + BroadcastChannel for -cross-tab sync). `workspaces`, `settings`, and `auth` state live here. - -### API calls -All go through `$api` (`src/lib/api/client.ts`) which: -- prepends `VITE_API_BASE_URL` -- reads the `csrftoken` cookie (priming it via `/api/csrf/` if missing) -- serializes JSON bodies and parses structured DRF errors into `ApiError` -- delegates 401s to the auth layer (`setUnauthorizedHandler`) - -### Auth -`authStore` is the single source of truth for who is logged in. Pages that -require auth call `requireAuthenticated()` from `$auth` in `onMount` — this -refreshes state and redirects to `/login` if anonymous. The API client triggers -the same redirect on 401s. - -### Map -The map library is behind the `IMap` interface. Every plot operation (line, -marker, circle) runs through a **Scene** — a named collection of layers owned -by a feature. When a feature is done with its layers, `scene.clear()` or -`map.disposeScene(name)` removes them all at once. This is how each workspace -paints an independent trajectory without stepping on its neighbors. - -Swapping MapLibre for another library means implementing `IMap` once (see -`src/lib/map/maplibre.ts`) and updating `createMapLibreMap()` at one -call-site inside `Map.svelte`. - -### Workspaces -A workspace is an independently-configured prediction layer. The user creates -several, edits their flight parameters separately, runs each one, and sees all -their trajectories overlaid on the same map with their own colors and -opacities. `workspacesStore.run(id)` calls the prediction API and stores the -result in-memory on that workspace. - -`WorkspaceRenderer.svelte` listens to the workspaces store and repaints layers -when workspaces change. It also drives the **global timeline**: whenever -workspaces change it recomputes the global `[min, max]` time range (the union -of every visible workspace’s trajectory span) and updates the timeline store. - -### Timeline -`timelineStore` is a plain writable backed by a `requestAnimationFrame` loop. -It exposes `play`/`pause`/`seek`/`setSpeed`. `WorkspaceRenderer` subscribes to -`time` and draws a cursor marker on each visible workspace at that timestamp. -Workspaces stay in sync because they all sample the same clock. - -### i18n -`initI18n()` (called from the root layout) loads the saved locale from -localStorage, imports the JSON dictionary, and stores it. Components use the -`$t` store from `$i18n`: - -```svelte -
{$t('panel.workspaces')}
-``` - -Adding a locale: drop `xx.json` under `src/lib/i18n/locales/`, register it in -`loaders` and in the `Locale` type. - -## Build & deploy - -- `@sveltejs/adapter-static` with `fallback: 'index.html'` produces a pure SPA. -- SSR is disabled in the root `+layout.ts` (`ssr = false`, `prerender = false`). -- CSS is imported once in `+layout.svelte` (which loads `app.css`). -- No server-side routes. All data comes from the Django REST backend. diff --git a/docs/CONVENTIONS.md b/docs/CONVENTIONS.md deleted file mode 100644 index c9b545e..0000000 --- a/docs/CONVENTIONS.md +++ /dev/null @@ -1,132 +0,0 @@ -# Conventions - -## TypeScript - -- Every file is `.ts` or ` - -
-
- Logo -

{$t('app.title')}

-
-
-
-
-
-
{$t('login.heading')}
- - {#if error} - - {/if} - -
-
- - -
-
- - -
- - {$t('login.back')} -
-
-
-
-
-
diff --git a/src/lib/features/auth/Navbar.svelte b/src/lib/features/auth/Navbar.svelte deleted file mode 100644 index b94612b..0000000 --- a/src/lib/features/auth/Navbar.svelte +++ /dev/null @@ -1,80 +0,0 @@ - - - - - Logo - - (isOpen = !isOpen)} /> - - - - - diff --git a/src/lib/features/auth/index.ts b/src/lib/features/auth/index.ts deleted file mode 100644 index 5368bb0..0000000 --- a/src/lib/features/auth/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export { default as Navbar } from './Navbar.svelte'; -export { default as LoginForm } from './LoginForm.svelte'; diff --git a/src/lib/features/footer/Footer.svelte b/src/lib/features/footer/Footer.svelte deleted file mode 100644 index 9a5fa95..0000000 --- a/src/lib/features/footer/Footer.svelte +++ /dev/null @@ -1,25 +0,0 @@ - - -
-
-
-
-
- - {$t('app.company')} - -
-
-
-
-
-
-
-
-
Copyright © 2024 {$t('app.company')}
-
-
-
-
diff --git a/src/lib/features/footer/index.ts b/src/lib/features/footer/index.ts deleted file mode 100644 index 02db901..0000000 --- a/src/lib/features/footer/index.ts +++ /dev/null @@ -1 +0,0 @@ -export { default as Footer } from './Footer.svelte'; diff --git a/src/lib/features/prediction/ControlPanel.svelte b/src/lib/features/prediction/ControlPanel.svelte deleted file mode 100644 index 9d17c17..0000000 --- a/src/lib/features/prediction/ControlPanel.svelte +++ /dev/null @@ -1,342 +0,0 @@ - - - - {#if !active} -
{$t('workspaces.empty')}
- {:else} -
- - - - workspacesStore.patch(active!.id, { - launchTime: (e.currentTarget as HTMLInputElement).value, - })} /> - - - - - workspacesStore.patch(active!.id, { - launchDate: (e.currentTarget as HTMLInputElement).value, - })} /> - -
- - - - - - patchActive({ - profile: (e.currentTarget as HTMLSelectElement).value as ProfileIdentifier, - })}> - {#each PROFILE_IDENTIFIERS as id} - - {/each} - - - - - - - - ({ - value: p.id, - label: `${p.name}${p.id === selectedPointId && isPointDirty ? ` (${$t('scenario.modified')})` : ''}`, - }))} - placeholder={$t('conditions.pointPlaceholder')} - searchPlaceholder={$t('conditions.pointSearchPlaceholder')} - clearable={true} /> - - - - - - - - - patchActive({ - launch_latitude: parseFloat((e.currentTarget as HTMLInputElement).value), - })} /> - / - - patchActive({ - launch_longitude: parseFloat((e.currentTarget as HTMLInputElement).value), - })} /> - - - - -
- -
- -
- - - - patchActive({ - launch_altitude: parseFloat((e.currentTarget as HTMLInputElement).value), - })} /> - - - - - patchActive({ - burst_altitude: parseFloat((e.currentTarget as HTMLInputElement).value), - })} /> - -
- - {#if params.profile !== 'custom_profile'} -
- - - - patchActive({ - ascent_rate: parseFloat((e.currentTarget as HTMLInputElement).value), - })} /> - - - - - patchActive({ - descent_rate: parseFloat((e.currentTarget as HTMLInputElement).value), - })} /> - -
- {:else} - - -
- - - -
- -
- - - -
- -
- {/if} - -
- -
- {/if} -
- - - handlePointSelection(p?.id ?? -1)} /> diff --git a/src/lib/features/prediction/CurveChart.svelte b/src/lib/features/prediction/CurveChart.svelte deleted file mode 100644 index 568e8ac..0000000 --- a/src/lib/features/prediction/CurveChart.svelte +++ /dev/null @@ -1,144 +0,0 @@ - - -
- -
diff --git a/src/lib/features/prediction/CurveEditor.svelte b/src/lib/features/prediction/CurveEditor.svelte deleted file mode 100644 index 2831173..0000000 --- a/src/lib/features/prediction/CurveEditor.svelte +++ /dev/null @@ -1,338 +0,0 @@ - - - - - - - - (isConfirmationVisible = false)}> -

Delete curve "{selectedCurve?.name}"?

-
diff --git a/src/lib/features/prediction/PointEditor.svelte b/src/lib/features/prediction/PointEditor.svelte deleted file mode 100644 index 4cf8f34..0000000 --- a/src/lib/features/prediction/PointEditor.svelte +++ /dev/null @@ -1,140 +0,0 @@ - - - onClose()} - onSave={(p) => onSave(p)} - onSelect={(p) => onSelectPoint(p)}> - {#snippet tableHeader()} - - {$t('points.name')} - {$t('points.lat')} - {$t('points.lon')} - {$t('points.alt')} - - - {/snippet} - - {#snippet tableRow({ row })} - {row.name} - {row.lat.toFixed(5)} ° - {row.lon.toFixed(5)} ° - {row.alt} м - {/snippet} - - {#snippet formFields({ item })} -
- - -
-
- - - - {$t('points.degrees')} - - - - - {$t('points.degrees')} - - - - - {$t('points.metersAsl')} - -
- {/snippet} -
diff --git a/src/lib/features/prediction/ScenarioEditor.svelte b/src/lib/features/prediction/ScenarioEditor.svelte deleted file mode 100644 index 10f72b0..0000000 --- a/src/lib/features/prediction/ScenarioEditor.svelte +++ /dev/null @@ -1,83 +0,0 @@ - - - - - - diff --git a/src/lib/features/prediction/ScenarioPanel.svelte b/src/lib/features/prediction/ScenarioPanel.svelte deleted file mode 100644 index 05d9d5d..0000000 --- a/src/lib/features/prediction/ScenarioPanel.svelte +++ /dev/null @@ -1,189 +0,0 @@ - - - - - - - ({ - value: s.id, - label: `${s.name}${s.id === selectedScenarioId && scenarioUnsaved ? ` (${$t('scenario.modified')})` : ''}`, - }))} - bind:selected={selectedScenarioId} - placeholder={$t('scenario.placeholder')} - searchPlaceholder={$t('scenario.searchPlaceholder')} - clearable={true} - onChange={() => { - if (!scenarioUnsaved) handleApplySelected(false); - }} /> - - - - -
- - -
- -
- - {#if active} - - - - - {#each PREDICTION_MODES as mode} - - {/each} - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - {/if} -
- - diff --git a/src/lib/features/prediction/index.ts b/src/lib/features/prediction/index.ts deleted file mode 100644 index 3fe21d0..0000000 --- a/src/lib/features/prediction/index.ts +++ /dev/null @@ -1,7 +0,0 @@ -export { default as ControlPanel } from './ControlPanel.svelte'; -export { default as ScenarioPanel } from './ScenarioPanel.svelte'; -export { default as PointEditor } from './PointEditor.svelte'; -export { default as ScenarioEditor } from './ScenarioEditor.svelte'; -export { default as CurveEditor } from './CurveEditor.svelte'; -export { default as CurveChart } from './CurveChart.svelte'; -export { pointsStore, profilesStore, scenariosStore } from './pointsStore'; diff --git a/src/lib/features/prediction/pointsStore.ts b/src/lib/features/prediction/pointsStore.ts deleted file mode 100644 index 63974d3..0000000 --- a/src/lib/features/prediction/pointsStore.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { writable } from 'svelte/store'; -import type { SavedPoint, SavedFlightProfile, SavedScenario } from '$domain'; - -/** - * Session-scoped caches for the user's saved points, profiles, and scenarios. - * Persisting these would fight the server as the source of truth, so they - * are plain in-memory writables hydrated from the API on mount. - */ -export const pointsStore = writable([]); -export const profilesStore = writable([]); -export const scenariosStore = writable([]); diff --git a/src/lib/features/settings/SettingsPanel.svelte b/src/lib/features/settings/SettingsPanel.svelte deleted file mode 100644 index 1d03e64..0000000 --- a/src/lib/features/settings/SettingsPanel.svelte +++ /dev/null @@ -1,70 +0,0 @@ - - - - {#each SETTINGS_SCHEMA as section} -
{$t(section.titleKey)}
- - {#each section.fields as field} - {@const current = getPath($settingsStore, field.path)} - - - {#if field.kind !== 'boolean'} - - {/if} - - {#if field.kind === 'boolean'} -
- - applyChange(field.path, (e.currentTarget as HTMLInputElement).checked)} /> - -
- {:else if field.kind === 'select'} - applyChange(field.path, (e.currentTarget as HTMLSelectElement).value)}> - {#each field.options as option} - - {/each} - - {:else if field.kind === 'number'} - - applyChange(field.path, parseFloat((e.currentTarget as HTMLInputElement).value))} /> - {:else if field.kind === 'string'} - applyChange(field.path, (e.currentTarget as HTMLInputElement).value)} /> - {/if} -
- {/each} - {/each} -
diff --git a/src/lib/features/settings/index.ts b/src/lib/features/settings/index.ts deleted file mode 100644 index e78744c..0000000 --- a/src/lib/features/settings/index.ts +++ /dev/null @@ -1,5 +0,0 @@ -export { settingsStore, DEFAULT_SETTINGS } from './store'; -export type { AppSettings, MapSettings, UnitsSettings } from './store'; -export { default as SettingsPanel } from './SettingsPanel.svelte'; -export { SETTINGS_SCHEMA } from './schema'; -export type { SettingsField, SettingsSection } from './schema'; diff --git a/src/lib/features/settings/schema.ts b/src/lib/features/settings/schema.ts deleted file mode 100644 index 7fb4446..0000000 --- a/src/lib/features/settings/schema.ts +++ /dev/null @@ -1,86 +0,0 @@ -/** - * Declarative settings schema. Each `SettingsField` describes a single setting - * that renders as a labeled form control in the Settings panel. Keep this - * independent of Svelte so the same schema can drive future serializers - * (export/import settings, URL state, etc.). - */ - -export type FieldKind = 'boolean' | 'select' | 'number' | 'string'; - -export interface BaseField { - kind: K; - /** Dot-separated path into AppSettings (e.g. `'map.baseLayer'`). */ - path: string; - labelKey: string; - descriptionKey?: string; -} - -export interface BooleanField extends BaseField<'boolean'> {} - -export interface NumberField extends BaseField<'number'> { - min?: number; - max?: number; - step?: number; -} - -export interface StringField extends BaseField<'string'> { - placeholder?: string; -} - -export interface SelectField extends BaseField<'select'> { - options: { value: string; labelKey: string }[]; -} - -export type SettingsField = BooleanField | NumberField | StringField | SelectField; - -export interface SettingsSection { - titleKey: string; - fields: SettingsField[]; -} - -export const SETTINGS_SCHEMA: SettingsSection[] = [ - { - titleKey: 'settings.language', - fields: [ - { - kind: 'select', - path: 'locale', - labelKey: 'settings.language', - options: [ - { value: 'ru', labelKey: 'common.yes' }, - { value: 'en', labelKey: 'common.yes' }, - ], - }, - ], - }, - { - titleKey: 'settings.map', - fields: [ - { - kind: 'select', - path: 'map.baseLayer', - labelKey: 'settings.baseLayer', - options: [ - { value: 'osm', labelKey: 'settings.baseLayer' }, - { value: 'satellite', labelKey: 'settings.baseLayer' }, - ], - }, - { kind: 'boolean', path: 'map.showScale', labelKey: 'settings.showScale' }, - { kind: 'boolean', path: 'map.showNavigation', labelKey: 'settings.showNavigation' }, - ], - }, - { - titleKey: 'settings.units', - fields: [ - { - kind: 'select', - path: 'units.system', - labelKey: 'settings.units', - options: [ - { value: 'metric', labelKey: 'settings.metric' }, - { value: 'imperial', labelKey: 'settings.imperial' }, - ], - }, - ], - }, -]; diff --git a/src/lib/features/settings/store.ts b/src/lib/features/settings/store.ts deleted file mode 100644 index 41e5cc2..0000000 --- a/src/lib/features/settings/store.ts +++ /dev/null @@ -1,50 +0,0 @@ -import { persisted } from '$state'; -import type { Locale } from '$i18n'; - -export interface MapSettings { - baseLayer: 'osm' | 'satellite'; - showScale: boolean; - showNavigation: boolean; -} - -export interface UnitsSettings { - system: 'metric' | 'imperial'; -} - -export interface AppSettings { - locale: Locale; - map: MapSettings; - units: UnitsSettings; -} - -export const DEFAULT_SETTINGS: AppSettings = { - locale: 'ru', - map: { baseLayer: 'osm', showScale: true, showNavigation: true }, - units: { system: 'metric' }, -}; - -export const settingsStore = persisted('settings', DEFAULT_SETTINGS); - -/** Resolve `'a.b.c'` to `obj.a.b.c`. Used by the schema-driven settings form. */ -export function getPath(obj: unknown, path: string): unknown { - return path.split('.').reduce((acc, key) => { - if (acc && typeof acc === 'object' && key in acc) { - return (acc as Record)[key]; - } - return undefined; - }, obj); -} - -export function setPath(obj: T, path: string, value: unknown): T { - const keys = path.split('.'); - const next: Record = { ...(obj as Record) }; - let cursor: Record = next; - for (let i = 0; i < keys.length - 1; i++) { - const k = keys[i]; - const existing = cursor[k]; - cursor[k] = { ...((existing as Record | undefined) ?? {}) }; - cursor = cursor[k] as Record; - } - cursor[keys[keys.length - 1]] = value; - return next as T; -} diff --git a/src/lib/features/timeline/TimeLine.svelte b/src/lib/features/timeline/TimeLine.svelte deleted file mode 100644 index 8f4018d..0000000 --- a/src/lib/features/timeline/TimeLine.svelte +++ /dev/null @@ -1,117 +0,0 @@ - - -
-
-
-
- - {#if $timelineStore.playing} - - {:else} - - {/if} - -
- -
- -
- {fmtHms(elapsed)} - {fmtHms(duration)} -
-
-
-
-
- - diff --git a/src/lib/features/timeline/index.ts b/src/lib/features/timeline/index.ts deleted file mode 100644 index d7ab97b..0000000 --- a/src/lib/features/timeline/index.ts +++ /dev/null @@ -1,3 +0,0 @@ -export { timelineStore } from './store'; -export type { TimelineState } from './store'; -export { default as TimeLine } from './TimeLine.svelte'; diff --git a/src/lib/features/timeline/store.ts b/src/lib/features/timeline/store.ts deleted file mode 100644 index 50f08c3..0000000 --- a/src/lib/features/timeline/store.ts +++ /dev/null @@ -1,93 +0,0 @@ -import { writable } from 'svelte/store'; - -/** - * Global playback clock. - * - * `time` is an absolute timestamp in ms (UTC). Each layer/workspace samples - * its own trajectory against this clock, so multiple workspaces stay in sync. - * - * Consumers should treat the range `[min, max]` as the current domain; if - * they own a trajectory spanning a different interval they can clamp - * locally, but the UI slider always operates over the global range. - */ - -export interface TimelineState { - time: number; - min: number; - max: number; - speed: number; - playing: boolean; -} - -const initial: TimelineState = { - time: 0, - min: 0, - max: 0, - speed: 1, - playing: false, -}; - -function createTimeline() { - const store = writable(initial); - let frame: number | null = null; - let lastTick = 0; - - function tick(ts: number) { - store.update((s) => { - if (!s.playing) return s; - if (!lastTick) lastTick = ts; - const dt = (ts - lastTick) * s.speed; - lastTick = ts; - let next = s.time + dt; - if (next >= s.max) { - next = s.max; - frame = null; - return { ...s, time: next, playing: false }; - } - frame = requestAnimationFrame(tick); - return { ...s, time: next }; - }); - } - - function play() { - store.update((s) => { - if (s.max <= s.min) return s; - if (s.time >= s.max) return { ...s, time: s.min, playing: true }; - return { ...s, playing: true }; - }); - lastTick = 0; - frame = requestAnimationFrame(tick); - } - - function pause() { - if (frame !== null) cancelAnimationFrame(frame); - frame = null; - store.update((s) => ({ ...s, playing: false })); - } - - function reset() { - pause(); - store.update((s) => ({ ...s, time: s.min })); - } - - function seek(time: number) { - store.update((s) => ({ ...s, time: Math.max(s.min, Math.min(s.max, time)) })); - } - - function setSpeed(speed: number) { - store.update((s) => ({ ...s, speed })); - } - - function setRange(min: number, max: number) { - store.update((s) => { - const nextMin = Number.isFinite(min) ? min : s.min; - const nextMax = Number.isFinite(max) ? max : s.max; - const t = Math.max(nextMin, Math.min(nextMax, s.time)); - return { ...s, min: nextMin, max: nextMax, time: t }; - }); - } - - return { subscribe: store.subscribe, play, pause, reset, seek, setSpeed, setRange }; -} - -export const timelineStore = createTimeline(); diff --git a/src/lib/features/tracking/TelemetryPanel.svelte b/src/lib/features/tracking/TelemetryPanel.svelte deleted file mode 100644 index 3bb5428..0000000 --- a/src/lib/features/tracking/TelemetryPanel.svelte +++ /dev/null @@ -1,36 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/src/lib/features/tracking/index.ts b/src/lib/features/tracking/index.ts deleted file mode 100644 index 17dbca6..0000000 --- a/src/lib/features/tracking/index.ts +++ /dev/null @@ -1 +0,0 @@ -export { default as TelemetryPanel } from './TelemetryPanel.svelte'; diff --git a/src/lib/features/workspaces/WorkspaceRenderer.svelte b/src/lib/features/workspaces/WorkspaceRenderer.svelte deleted file mode 100644 index 39a9d1c..0000000 --- a/src/lib/features/workspaces/WorkspaceRenderer.svelte +++ /dev/null @@ -1,132 +0,0 @@ - diff --git a/src/lib/features/workspaces/WorkspacesPanel.svelte b/src/lib/features/workspaces/WorkspacesPanel.svelte deleted file mode 100644 index 1290c45..0000000 --- a/src/lib/features/workspaces/WorkspacesPanel.svelte +++ /dev/null @@ -1,159 +0,0 @@ - - - -
- -
- - {#if $workspacesStore.items.length === 0} -
{$t('workspaces.empty')}
- {/if} - -
- {#each $workspacesStore.items as w (w.id)} - {@const isActive = $workspacesStore.activeId === w.id} -
-
- - handleRename(w, (e.currentTarget as HTMLInputElement).value)} /> - -
- -
- {$t('workspaces.color')} - handleColor(w, (e.currentTarget as HTMLInputElement).value)} /> - {$t('workspaces.opacity')} - handleOpacity(w, parseFloat((e.currentTarget as HTMLInputElement).value))} /> -
- - - - - - - {#if w.lastRunError} -
{w.lastRunError}
- {/if} -
- {/each} -
-
- - (toDelete = null)}> - {#if toDelete} -

{$t('workspaces.deleteConfirm', { name: toDelete.name })}

- {/if} -
- - diff --git a/src/lib/features/workspaces/index.ts b/src/lib/features/workspaces/index.ts deleted file mode 100644 index 5cdc25c..0000000 --- a/src/lib/features/workspaces/index.ts +++ /dev/null @@ -1,4 +0,0 @@ -export { workspacesStore, getActiveWorkspace } from './store'; -export type { Workspace, WorkspaceInit } from './types'; -export { default as WorkspacesPanel } from './WorkspacesPanel.svelte'; -export { default as WorkspaceRenderer } from './WorkspaceRenderer.svelte'; diff --git a/src/lib/features/workspaces/store.ts b/src/lib/features/workspaces/store.ts deleted file mode 100644 index 98d82f4..0000000 --- a/src/lib/features/workspaces/store.ts +++ /dev/null @@ -1,128 +0,0 @@ -import { get } from 'svelte/store'; -import { persisted } from '$state'; -import { DEFAULT_FLIGHT_PARAMETERS, type FlightParameters } from '$domain'; -import type { Prediction } from '$domain'; -import { predictionsApi } from '$api'; -import { parsePrediction } from '$domain'; -import { buildLaunchDateTime } from '$api'; -import type { Workspace, WorkspaceInit } from './types'; - -const STORAGE_KEY = 'workspaces'; - -const DEFAULT_COLORS = [ - '#0d6efd', - '#dc3545', - '#198754', - '#fd7e14', - '#6f42c1', - '#20c997', - '#d63384', - '#0dcaf0', -]; - -function todayDate(): string { - return new Date().toISOString().split('T')[0]; -} - -function makeWorkspace(init: WorkspaceInit = {}, index = 0): Workspace { - return { - id: crypto.randomUUID(), - name: init.name ?? `Рабочая область ${index + 1}`, - color: init.color ?? DEFAULT_COLORS[index % DEFAULT_COLORS.length], - opacity: 1, - visible: true, - flightParameters: init.flightParameters ?? { ...DEFAULT_FLIGHT_PARAMETERS }, - launchDate: init.launchDate ?? todayDate(), - launchTime: init.launchTime ?? '12:00:00', - result: null, - }; -} - -export interface WorkspaceSlice { - items: Workspace[]; - activeId: string | null; -} - -const initial: WorkspaceSlice = { items: [], activeId: null }; - -/** - * Don't persist prediction results — they're large, transient, and can be - * re-fetched. The serializer strips `result` before writing to localStorage. - */ -const workspacesPersisted = persisted(STORAGE_KEY, initial, { - serializer: { - stringify: (value) => { - const lean: WorkspaceSlice = { - ...value, - items: value.items.map((w) => ({ ...w, result: null, lastRunError: undefined })), - }; - return JSON.stringify(lean); - }, - parse: (raw) => JSON.parse(raw) as WorkspaceSlice, - }, -}); - -function update(fn: (s: WorkspaceSlice) => WorkspaceSlice): void { - workspacesPersisted.update(fn); -} - -export const workspacesStore = { - subscribe: workspacesPersisted.subscribe, - - add(init: WorkspaceInit = {}): Workspace { - let created: Workspace | null = null; - update((s) => { - const w = makeWorkspace(init, s.items.length); - created = w; - return { items: [...s.items, w], activeId: w.id }; - }); - return created!; - }, - - remove(id: string): void { - update((s) => { - const items = s.items.filter((w) => w.id !== id); - const activeId = s.activeId === id ? (items[0]?.id ?? null) : s.activeId; - return { items, activeId }; - }); - }, - - patch(id: string, patch: Partial): void { - update((s) => ({ - ...s, - items: s.items.map((w) => (w.id === id ? { ...w, ...patch } : w)), - })); - }, - - setActive(id: string | null): void { - update((s) => ({ ...s, activeId: id })); - }, - - setFlightParameters(id: string, params: FlightParameters): void { - workspacesStore.patch(id, { flightParameters: params }); - }, - - setResult(id: string, result: Prediction | null, error?: string): void { - workspacesStore.patch(id, { result, lastRunError: error }); - }, - - async run(id: string): Promise { - const slice = get(workspacesPersisted); - const w = slice.items.find((x) => x.id === id); - if (!w) return; - try { - const launchDatetime = buildLaunchDateTime(w.launchDate, w.launchTime); - const response = await predictionsApi.run(w.flightParameters, launchDatetime); - const prediction = parsePrediction(response.result.prediction); - workspacesStore.setResult(id, prediction); - } catch (err: unknown) { - workspacesStore.setResult(id, null, (err as Error).message); - throw err; - } - }, -}; - -export function getActiveWorkspace(slice: WorkspaceSlice): Workspace | null { - if (!slice.activeId) return slice.items[0] ?? null; - return slice.items.find((w) => w.id === slice.activeId) ?? null; -} diff --git a/src/lib/features/workspaces/types.ts b/src/lib/features/workspaces/types.ts deleted file mode 100644 index 1ffe281..0000000 --- a/src/lib/features/workspaces/types.ts +++ /dev/null @@ -1,23 +0,0 @@ -import type { FlightParameters, Prediction } from '$domain'; - -/** A "workspace" is an independently-configured prediction layer on the map. */ -export interface Workspace { - id: string; - name: string; - color: string; - opacity: number; - visible: boolean; - flightParameters: FlightParameters; - launchDate: string; - launchTime: string; - result: Prediction | null; - lastRunError?: string; -} - -export interface WorkspaceInit { - name?: string; - color?: string; - flightParameters?: FlightParameters; - launchDate?: string; - launchTime?: string; -} diff --git a/src/lib/i18n/index.ts b/src/lib/i18n/index.ts deleted file mode 100644 index 3a65826..0000000 --- a/src/lib/i18n/index.ts +++ /dev/null @@ -1,87 +0,0 @@ -import { derived, writable, type Readable } from 'svelte/store'; -import { browser } from '$app/environment'; - -/** - * Minimal i18n layer — no external deps. - * - * Lookup is key-based (`t('panel.title')`) against flat or nested JSON - * dictionaries. Interpolation uses `{name}` placeholders. Missing keys fall - * back to the key itself so screens remain functional during translation. - * - * Adding a locale - * --------------- - * 1. Drop a JSON file into `src/lib/i18n/locales/.json`. - * 2. Register it in `loaders` below. - * 3. Expose it in the `Locale` type. - */ - -export type Locale = 'ru' | 'en'; - -export const DEFAULT_LOCALE: Locale = 'ru'; -export const SUPPORTED_LOCALES: Locale[] = ['ru', 'en']; -const STORAGE_KEY = 'locale'; - -type Messages = Record; - -const loaders: Record Promise<{ default: Messages }>> = { - ru: () => import('./locales/ru.json'), - en: () => import('./locales/en.json'), -}; - -const messages = writable>({} as Record); -const locale = writable(DEFAULT_LOCALE); - -async function loadLocale(code: Locale): Promise { - const mod = await loaders[code](); - messages.update((m) => ({ ...m, [code]: mod.default })); -} - -export async function setLocale(code: Locale): Promise { - if (!SUPPORTED_LOCALES.includes(code)) return; - await loadLocale(code); - locale.set(code); - if (browser) localStorage.setItem(STORAGE_KEY, code); -} - -export async function initI18n(): Promise { - const stored = browser ? (localStorage.getItem(STORAGE_KEY) as Locale | null) : null; - const next = stored && SUPPORTED_LOCALES.includes(stored) ? stored : DEFAULT_LOCALE; - await setLocale(next); -} - -function lookup(dict: Messages, key: string): string | undefined { - const parts = key.split('.'); - let node: unknown = dict; - for (const p of parts) { - if (node && typeof node === 'object' && p in (node as Messages)) { - node = (node as Messages)[p]; - } else { - return undefined; - } - } - return typeof node === 'string' ? node : undefined; -} - -function interpolate(template: string, values: Record): string { - return template.replace(/\{(\w+)\}/g, (_, name) => - name in values ? String(values[name]) : `{${name}}`, - ); -} - -export interface Translator { - (key: string, values?: Record): string; -} - -export const t: Readable = derived( - [locale, messages], - ([$locale, $messages]) => { - const dict = $messages[$locale]; - return (key: string, values?: Record) => { - const str = dict ? lookup(dict, key) : undefined; - if (str === undefined) return key; - return values ? interpolate(str, values) : str; - }; - }, -); - -export const currentLocale: Readable = { subscribe: locale.subscribe }; diff --git a/src/lib/i18n/locales/en.json b/src/lib/i18n/locales/en.json deleted file mode 100644 index fcd9967..0000000 --- a/src/lib/i18n/locales/en.json +++ /dev/null @@ -1,172 +0,0 @@ -{ - "app": { - "title": "Stratospheric Flights | YKS Ltd.", - "company": "Yakutsk Space Systems Ltd." - }, - "nav": { - "predict": "Predict", - "track": "Track", - "login": "Log in", - "logout": "Log out", - "account": "Account", - "scenarios": "Saved scenarios", - "predictionHistory": "Prediction history", - "trackingHistory": "Tracking history", - "user": "User" - }, - "login": { - "heading": "Sign in to your account", - "username": "Username", - "password": "Password", - "submit": "Sign in", - "submitting": "Signing in...", - "back": "Back", - "invalidCredentials": "Invalid credentials", - "fieldsRequired": "Please enter a username and password" - }, - "panel": { - "scenario": "Scenario", - "conditions": "Conditions", - "about": "About", - "layers": "Layers", - "results": "Results", - "settings": "Settings", - "workspaces": "Workspaces" - }, - "scenario": { - "title": "Prediction scenario", - "select": "Scenario", - "placeholder": "New scenario...", - "searchPlaceholder": "Search scenarios...", - "apply": "Apply scenario", - "applied": "Scenario applied", - "appliedBody": "Scenario \"{name}\" successfully applied.", - "notFound": "Scenario not found", - "notFoundBody": "The selected scenario does not exist.", - "updated": "Scenario updated", - "updatedBody": "Scenario \"{name}\" successfully updated.", - "updateError": "Scenario update error", - "updateErrorBody": "Error updating scenario: {error}", - "save": "Save scenario", - "update": "Update scenario", - "all": "All scenarios", - "mode": "Scenario mode", - "model": "Atmospheric model", - "dataset": "Dataset", - "datasetAuto": "Pick automatically", - "modified": "modified", - "export": "Export result", - "exportBtn": "Export" - }, - "predictionMode": { - "single": "Single", - "hourly": "Hourly", - "ensemble": "Ensemble" - }, - "profile": { - "standard_profile": "Standard", - "float_profile": "Float", - "reverse_profile": "Reverse", - "custom_profile": "Custom" - }, - "conditions": { - "title": "Prediction conditions", - "startTime": "Launch time (UTC)", - "startDate": "Launch date", - "flightProfile": "Flight profile", - "startPoint": "Launch point", - "pointPlaceholder": "New point...", - "pointSearchPlaceholder": "Search points...", - "latLng": "Latitude/Longitude", - "save": "Save", - "launchAlt": "Launch altitude (m)", - "burstAlt": "Burst altitude (m)", - "ascentRate": "Ascent rate (m/s)", - "descentRate": "Descent rate (m/s)", - "run": "Run prediction", - "profileEdit": "Ascent/descent profiles", - "ascentStage": "Ascent stage", - "descentStage": "Descent stage", - "stageNone": "None", - "stageStandard": "Standard", - "stageCustom": "Custom", - "openCurveEditor": "Open curve editor" - }, - "workspaces": { - "title": "Workspaces", - "empty": "No workspaces yet.", - "add": "Add", - "rename": "Rename", - "delete": "Delete", - "visible": "Show/hide", - "color": "Color", - "opacity": "Opacity", - "run": "Run", - "running": "Running...", - "edit": "Edit", - "defaultName": "Workspace {n}", - "deleteConfirm": "Delete workspace \"{name}\"?", - "runError": "Run error: {error}" - }, - "timeline": { - "title": "Flight timeline", - "time": "Time", - "altitude": "Altitude", - "position": "Position", - "play": "Play", - "pause": "Pause", - "stop": "Reset", - "speed": "Speed" - }, - "settings": { - "title": "Settings", - "language": "Language", - "map": "Map", - "baseLayer": "Base layer", - "showScale": "Show scale", - "showNavigation": "Navigation controls", - "units": "Units", - "metric": "Metric", - "imperial": "Imperial", - "saved": "Settings saved" - }, - "editor": { - "add": "Add", - "edit": "Edit", - "save": "Save", - "update": "Update", - "delete": "Delete", - "cancel": "Cancel", - "close": "Close", - "searchPlaceholder": "Search..." - }, - "points": { - "item": "point", - "itemGenitive": "point", - "items": "points", - "name": "Point name", - "lat": "Latitude", - "lon": "Longitude", - "alt": "Altitude", - "degrees": "Degrees", - "metersAsl": "Meters ASL" - }, - "common": { - "error": "Error", - "success": "Success", - "warning": "Warning", - "info": "Info", - "yes": "Yes", - "no": "No" - }, - "selection": { - "header": "Coordinate select mode", - "body": "Click the map to pick coordinates" - }, - "forecast": { - "success": "Forecast request", - "successBody": "Forecast request successful!", - "error": "Forecast error", - "errorBody": "Error running forecast: {error}" - } -} diff --git a/src/lib/i18n/locales/ru.json b/src/lib/i18n/locales/ru.json deleted file mode 100644 index eeb2149..0000000 --- a/src/lib/i18n/locales/ru.json +++ /dev/null @@ -1,172 +0,0 @@ -{ - "app": { - "title": "Стратосферные полеты | ООО ЯКС", - "company": "ООО «Якутские Космические Системы»" - }, - "nav": { - "predict": "Прогнозирование", - "track": "Слежение", - "login": "Войти", - "logout": "Выйти", - "account": "Учетная запись", - "scenarios": "Сохраненные сценарии", - "predictionHistory": "История прогнозов", - "trackingHistory": "История слежения", - "user": "Пользователь" - }, - "login": { - "heading": "Вход в учетную запись", - "username": "Имя пользователя", - "password": "Пароль", - "submit": "Войти", - "submitting": "Вход...", - "back": "Назад", - "invalidCredentials": "Неверные учетные данные", - "fieldsRequired": "Пожалуйста, введите имя пользователя и пароль" - }, - "panel": { - "scenario": "Сценарий", - "conditions": "Условия", - "about": "О проекте", - "layers": "Слои", - "results": "Результаты", - "settings": "Настройки", - "workspaces": "Рабочие области" - }, - "scenario": { - "title": "Сценарий прогнозирования", - "select": "Сценарий", - "placeholder": "Новый сценарий...", - "searchPlaceholder": "Поиск сценариев...", - "apply": "Применить сценарий", - "applied": "Сценарий применен", - "appliedBody": "Сценарий \"{name}\" успешно применен.", - "notFound": "Сценарий не найден", - "notFoundBody": "Выбранный сценарий не существует.", - "updated": "Сценарий обновлен", - "updatedBody": "Сценарий \"{name}\" успешно обновлен.", - "updateError": "Ошибка обновления сценария", - "updateErrorBody": "Ошибка при обновлении сценария: {error}", - "save": "Сохранить сценарий", - "update": "Обновить сценарий", - "all": "Все сценарии", - "mode": "Режим сценария", - "model": "Модель атмосферы", - "dataset": "Набор данных", - "datasetAuto": "Выбрать автоматически", - "modified": "изменено", - "export": "Экспортировать результат", - "exportBtn": "Экспорт" - }, - "predictionMode": { - "single": "Разовый", - "hourly": "Почасовой", - "ensemble": "Ансамблевый" - }, - "profile": { - "standard_profile": "Обычный", - "float_profile": "Дрейф", - "reverse_profile": "Реверсивный", - "custom_profile": "Пользовательский" - }, - "conditions": { - "title": "Условия прогнозирования", - "startTime": "Время старта (UTC)", - "startDate": "Дата старта", - "flightProfile": "Профиль полета", - "startPoint": "Точка старта", - "pointPlaceholder": "Новая точка...", - "pointSearchPlaceholder": "Поиск по точкам...", - "latLng": "Широта/Долгота", - "save": "Сохранить", - "launchAlt": "Высота старта (м)", - "burstAlt": "Высота разрыва (м)", - "ascentRate": "Скорость подъема (м/с)", - "descentRate": "Скорость спуска (м/с)", - "run": "Выполнить прогнозирование", - "profileEdit": "Профили подъема и спуска", - "ascentStage": "Стадия подъема", - "descentStage": "Стадия спуска", - "stageNone": "Нет", - "stageStandard": "Стандартная", - "stageCustom": "Пользовательская", - "openCurveEditor": "Открыть редактор кривых" - }, - "workspaces": { - "title": "Рабочие области", - "empty": "Нет созданных областей.", - "add": "Добавить", - "rename": "Переименовать", - "delete": "Удалить", - "visible": "Показать/скрыть", - "color": "Цвет", - "opacity": "Прозрачность", - "run": "Рассчитать", - "running": "Расчет...", - "edit": "Редактировать", - "defaultName": "Рабочая область {n}", - "deleteConfirm": "Удалить рабочую область \"{name}\"?", - "runError": "Ошибка расчета: {error}" - }, - "timeline": { - "title": "Временная шкала", - "time": "Время", - "altitude": "Высота", - "position": "Позиция", - "play": "Воспроизвести", - "pause": "Пауза", - "stop": "Сбросить", - "speed": "Скорость" - }, - "settings": { - "title": "Настройки", - "language": "Язык", - "map": "Карта", - "baseLayer": "Базовый слой", - "showScale": "Показывать масштаб", - "showNavigation": "Навигационные элементы", - "units": "Единицы измерения", - "metric": "Метрические", - "imperial": "Имперские", - "saved": "Настройки сохранены" - }, - "editor": { - "add": "Добавить", - "edit": "Редактировать", - "save": "Сохранить", - "update": "Обновить", - "delete": "Удалить", - "cancel": "Отмена", - "close": "Закрыть", - "searchPlaceholder": "Поиск..." - }, - "points": { - "item": "точка", - "itemGenitive": "точки", - "items": "точки", - "name": "Название точки", - "lat": "Широта", - "lon": "Долгота", - "alt": "Высота", - "degrees": "Градусы", - "metersAsl": "Метры над ур. моря" - }, - "common": { - "error": "Ошибка", - "success": "Успех", - "warning": "Предупреждение", - "info": "Информация", - "yes": "Да", - "no": "Нет" - }, - "selection": { - "header": "Режим выбора координат", - "body": "Кликните на карту, чтобы выбрать координаты" - }, - "forecast": { - "success": "Запрос прогноза", - "successBody": "Запрос прогноза успешно выполнен!", - "error": "Ошибка прогноза", - "errorBody": "Ошибка при получении прогноза: {error}" - } -} diff --git a/src/lib/index.js b/src/lib/index.js new file mode 100644 index 0000000..856f2b6 --- /dev/null +++ b/src/lib/index.js @@ -0,0 +1 @@ +// place files you want to import through the `$lib` alias in this folder. diff --git a/src/lib/map/Map.svelte b/src/lib/map/Map.svelte deleted file mode 100644 index 9b9f223..0000000 --- a/src/lib/map/Map.svelte +++ /dev/null @@ -1,77 +0,0 @@ - - -
- {#if ready && map} - {@render children?.()} - {/if} -
diff --git a/src/lib/map/context.ts b/src/lib/map/context.ts deleted file mode 100644 index 9c71eab..0000000 --- a/src/lib/map/context.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { getContext, setContext } from 'svelte'; -import type { IMap } from './core'; - -const KEY = Symbol('lsv-map'); - -/** Child components fetch the `IMap` instance via `getMap()`. */ -export interface MapContext { - get(): IMap | null; -} - -export function setMapContext(getter: () => IMap | null): void { - setContext(KEY, { get: getter }); -} - -export function getMap(): IMap | null { - return getContext(KEY)?.get() ?? null; -} diff --git a/src/lib/map/core.ts b/src/lib/map/core.ts deleted file mode 100644 index 37ab14b..0000000 --- a/src/lib/map/core.ts +++ /dev/null @@ -1,121 +0,0 @@ -import type { LatLngTuple, LngLatTuple } from '$domain'; - -/** - * Map abstraction. - * - * Goals: - * - Isolate all MapLibre-specific types inside src/lib/map/maplibre.ts. - * - Expose a small, map-library-agnostic vocabulary (markers, polylines, - * icons, events) so features (workspaces, timeline, tools) can be tested - * against the interface alone. - * - Support "scenes" — named collections of layers owned by a feature so - * each workspace/tool can add/remove everything it owns atomically. - * - * If another library ever replaces MapLibre, implementing IMap is the only - * file that changes. - */ - -export type MapEvent = 'click' | 'mousemove' | 'move' | 'zoom' | 'load'; - -export interface MapClickEvent { - lngLat: { lat: number; lng: number }; - originalEvent: MouseEvent; -} - -export type MapEventPayload = { - click: MapClickEvent; - mousemove: MapClickEvent; - move: { center: LngLatTuple; zoom: number }; - zoom: { zoom: number }; - load: undefined; -}; - -export type MapEventHandler = (e: MapEventPayload[E]) => void; - -export interface MarkerOptions { - lngLat: LngLatTuple; - iconUrl?: string; - iconSize?: [number, number]; - className?: string; - /** Optional HTML shown in a popup on hover. */ - popupHtml?: string; -} - -export interface LineOptions { - coords: LatLngTuple[]; - color?: string; - width?: number; - opacity?: number; - dashArray?: [number, number]; -} - -export interface CircleOptions { - center: LngLatTuple; - /** Radius in pixels (visual; screen-space, matches MapLibre circle layers). */ - radiusPx?: number; - color?: string; - opacity?: number; - strokeColor?: string; - strokeWidth?: number; -} - -export interface Marker { - setLngLat(pos: LngLatTuple): void; - remove(): void; -} - -export interface MapLayer { - readonly id: string; - remove(): void; -} - -export interface Scene { - readonly name: string; - addLine(id: string, options: LineOptions): MapLayer; - addCircle(id: string, options: CircleOptions): MapLayer; - addMarker(id: string, options: MarkerOptions): Marker; - /** Remove an individual layer in this scene by its id. */ - remove(id: string): void; - /** Remove everything added to this scene. */ - clear(): void; - /** Called by IMap.dispose() to release resources. */ - dispose(): void; -} - -export interface IMap { - ready: Promise; - - on(event: E, handler: MapEventHandler): () => void; - - setCenter(pos: LngLatTuple, zoom?: number): void; - panTo(pos: LngLatTuple, durationMs?: number): void; - fitBounds(coords: LatLngTuple[], paddingPx?: number): void; - getZoom(): number; - setZoom(zoom: number): void; - - setCursor(cursor: string | null): void; - - /** - * Get or create a named scene. Scenes are the unit of layer ownership: - * a feature adds all its layers through a scene and calls `.clear()` to - * remove them in one step. - */ - scene(name: string): Scene; - disposeScene(name: string): void; - - /** Underlying implementation instance, for code that needs library-specific APIs. Use sparingly. */ - getRawInstance(): unknown; - - dispose(): void; -} - -export interface MapInit { - container: HTMLElement; - center: LngLatTuple; - zoom: number; - baseLayer?: 'osm' | 'satellite'; - showNavigationControl?: boolean; - showScaleControl?: boolean; -} - -export type MapFactory = (init: MapInit) => IMap; diff --git a/src/lib/map/index.ts b/src/lib/map/index.ts deleted file mode 100644 index a57319a..0000000 --- a/src/lib/map/index.ts +++ /dev/null @@ -1,9 +0,0 @@ -export * from './core'; -export { createMapLibreMap } from './maplibre'; -export { plotPrediction, plotTelemetry, plotAnimatedMarker } from './layers'; -export type { TrajectoryStyle } from './layers'; -export { startCoordinateSelection } from './tools/selection'; -export { startMeasure } from './tools/measure'; -export type { MeasureHandle, MeasureOptions } from './tools/measure'; -export { setMapContext, getMap } from './context'; -export { default as Map } from './Map.svelte'; diff --git a/src/lib/map/layers.ts b/src/lib/map/layers.ts deleted file mode 100644 index 2a03e31..0000000 --- a/src/lib/map/layers.ts +++ /dev/null @@ -1,124 +0,0 @@ -import type { Prediction, Telemetry } from '$domain'; -import { toLngLat } from '$domain'; -import type { IMap, Scene } from './core'; - -/** - * Plot helpers for high-level domain objects. These live outside MapLibreMap - * so they can be reused against any IMap implementation. - * - * Icons are served from /static; pass explicit overrides if a workspace - * should use custom markers. - */ - -export interface TrajectoryStyle { - color?: string; - width?: number; - opacity?: number; - launchIcon?: string; - landingIcon?: string; - burstIcon?: string; - iconSize?: [number, number]; -} - -const DEFAULT_STYLE: Required> & - Pick = { - color: '#000000', - width: 3, - opacity: 1, - iconSize: [12, 12], - launchIcon: '/target-blue.png', - landingIcon: '/target-red.png', - burstIcon: '/pop-marker.png', -}; - -export function plotPrediction( - scene: Scene, - prediction: Prediction, - style: TrajectoryStyle = {}, -): void { - const s = { ...DEFAULT_STYLE, ...style }; - scene.clear(); - - scene.addLine('path', { - coords: prediction.flight_path, - color: s.color, - width: s.width, - opacity: s.opacity, - }); - - scene.addMarker('launch', { - lngLat: toLngLat(prediction.launch.latlng), - iconUrl: s.launchIcon, - iconSize: s.iconSize, - popupHtml: `Launch
${prediction.launch.latlng.lat.toFixed(6)}, ${prediction.launch.latlng.lng.toFixed(6)}`, - }); - - scene.addMarker('landing', { - lngLat: toLngLat(prediction.landing.latlng), - iconUrl: s.landingIcon, - iconSize: s.iconSize, - popupHtml: `Landing
${prediction.landing.latlng.lat.toFixed(6)}, ${prediction.landing.latlng.lng.toFixed(6)}`, - }); - - scene.addMarker('burst', { - lngLat: toLngLat(prediction.burst.latlng), - iconUrl: s.burstIcon, - iconSize: [s.iconSize[0] + 4, s.iconSize[1] + 4], - popupHtml: `Burst
${prediction.burst.latlng.lat.toFixed(6)}, ${prediction.burst.latlng.lng.toFixed(6)}`, - }); -} - -export function plotTelemetry( - map: IMap, - scene: Scene, - telemetry: Telemetry, - style: TrajectoryStyle = {}, -): void { - const s = { ...DEFAULT_STYLE, ...style }; - scene.clear(); - - scene.addLine('path', { - coords: telemetry.flight_path, - color: s.color, - width: s.width, - opacity: s.opacity, - }); - - scene.addMarker('launch', { - lngLat: toLngLat(telemetry.launch.latlng), - iconUrl: s.launchIcon, - iconSize: s.iconSize, - }); - - telemetry.datapoints.forEach((p, i) => { - scene.addMarker(`point-${i}`, { - lngLat: [p.longitude, p.latitude], - iconUrl: '/marker-sm-red.png', - iconSize: [8, 8], - popupHtml: `${p.datetime}
${p.latitude.toFixed(6)}, ${p.longitude.toFixed(6)}`, - }); - }); - - if (telemetry.flight_path.length > 0) { - map.fitBounds(telemetry.flight_path, 50); - } -} - -export function plotAnimatedMarker(scene: Scene, lng: number, lat: number): void { - scene.clear(); - scene.addCircle('marker-ring', { - center: [lng, lat], - radiusPx: 14, - color: '#FF6B6B', - opacity: 0.3, - strokeColor: '#FF1744', - strokeWidth: 0, - }); - scene.addCircle('marker-core', { - center: [lng, lat], - radiusPx: 6, - color: '#FF1744', - strokeColor: '#ffffff', - strokeWidth: 2, - }); -} diff --git a/src/lib/map/maplibre.ts b/src/lib/map/maplibre.ts deleted file mode 100644 index 32adfd5..0000000 --- a/src/lib/map/maplibre.ts +++ /dev/null @@ -1,316 +0,0 @@ -import maplibregl, { - type Map as MLMap, - type LngLatLike, - type MarkerOptions as MLMarkerOptions, -} from 'maplibre-gl'; -import 'maplibre-gl/dist/maplibre-gl.css'; - -import type { - CircleOptions, - IMap, - LineOptions, - MapEvent, - MapEventHandler, - MapEventPayload, - MapInit, - MapLayer, - Marker, - MarkerOptions, - Scene, -} from './core'; -import type { LatLngTuple, LngLatTuple } from '$domain'; - -/** Map common base-layer names to MapLibre style JSON. */ -const BASE_STYLES: Record, maplibregl.StyleSpecification> = { - osm: { - version: 8, - sources: { - osm: { - type: 'raster', - tiles: ['https://a.tile.openstreetmap.org/{z}/{x}/{y}.png'], - tileSize: 256, - attribution: '© OpenStreetMap', - }, - }, - layers: [{ id: 'osm', type: 'raster', source: 'osm', minzoom: 0, maxzoom: 19 }], - }, - satellite: { - version: 8, - sources: { - sat: { - type: 'raster', - tiles: [ - 'https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}', - ], - tileSize: 256, - attribution: 'Tiles © Esri', - }, - }, - layers: [{ id: 'sat', type: 'raster', source: 'sat', minzoom: 0, maxzoom: 19 }], - }, -}; - -class MapLibreScene implements Scene { - private sources = new Set(); - private layers = new Set(); - private markers = new Map(); - - constructor( - public readonly name: string, - private map: MLMap, - ) {} - - private scopeId(id: string): string { - return `${this.name}__${id}`; - } - - addLine(id: string, options: LineOptions): MapLayer { - const layerId = this.scopeId(id); - const coords = options.coords.map<[number, number]>((c) => [c[1], c[0]]); - - if (this.map.getSource(layerId)) this.remove(id); - - this.map.addSource(layerId, { - type: 'geojson', - data: { - type: 'Feature', - properties: {}, - geometry: { type: 'LineString', coordinates: coords }, - }, - }); - this.map.addLayer({ - id: layerId, - type: 'line', - source: layerId, - layout: { 'line-join': 'round', 'line-cap': 'round' }, - paint: { - 'line-color': options.color ?? '#000', - 'line-width': options.width ?? 3, - 'line-opacity': options.opacity ?? 1, - ...(options.dashArray ? { 'line-dasharray': options.dashArray } : {}), - }, - }); - this.sources.add(layerId); - this.layers.add(layerId); - - return { - id: layerId, - remove: () => this.remove(id), - }; - } - - addCircle(id: string, options: CircleOptions): MapLayer { - const layerId = this.scopeId(id); - if (this.map.getSource(layerId)) this.remove(id); - - this.map.addSource(layerId, { - type: 'geojson', - data: { - type: 'Feature', - properties: {}, - geometry: { type: 'Point', coordinates: options.center }, - }, - }); - this.map.addLayer({ - id: layerId, - type: 'circle', - source: layerId, - paint: { - 'circle-radius': options.radiusPx ?? 6, - 'circle-color': options.color ?? '#0b5ed7', - 'circle-opacity': options.opacity ?? 1, - 'circle-stroke-color': options.strokeColor ?? '#ffffff', - 'circle-stroke-width': options.strokeWidth ?? 2, - }, - }); - this.sources.add(layerId); - this.layers.add(layerId); - - return { - id: layerId, - remove: () => this.remove(id), - }; - } - - addMarker(id: string, options: MarkerOptions): Marker { - const scoped = this.scopeId(id); - const existing = this.markers.get(scoped); - if (existing) existing.remove(); - - let mlOptions: MLMarkerOptions | undefined; - if (options.iconUrl) { - const el = document.createElement('div'); - el.className = options.className ?? 'lsv-marker'; - el.style.backgroundImage = `url(${options.iconUrl})`; - const [w, h] = options.iconSize ?? [12, 12]; - el.style.width = `${w}px`; - el.style.height = `${h}px`; - el.style.backgroundSize = '100%'; - mlOptions = { element: el }; - } - - const marker = new maplibregl.Marker(mlOptions).setLngLat(options.lngLat as LngLatLike); - - if (options.popupHtml) { - const popup = new maplibregl.Popup({ offset: 16, closeButton: false }).setHTML( - options.popupHtml, - ); - marker.setPopup(popup); - marker.getElement().addEventListener('mouseenter', () => marker.togglePopup()); - marker.getElement().addEventListener('mouseleave', () => marker.togglePopup()); - } - - marker.addTo(this.map); - this.markers.set(scoped, marker); - - return { - setLngLat: (pos) => marker.setLngLat(pos as LngLatLike), - remove: () => this.remove(id), - }; - } - - remove(id: string): void { - const scoped = this.scopeId(id); - if (this.map.getLayer(scoped)) this.map.removeLayer(scoped); - if (this.map.getSource(scoped)) this.map.removeSource(scoped); - this.layers.delete(scoped); - this.sources.delete(scoped); - const marker = this.markers.get(scoped); - if (marker) { - marker.remove(); - this.markers.delete(scoped); - } - } - - clear(): void { - for (const id of Array.from(this.layers)) { - if (this.map.getLayer(id)) this.map.removeLayer(id); - } - for (const id of Array.from(this.sources)) { - if (this.map.getSource(id)) this.map.removeSource(id); - } - for (const marker of this.markers.values()) marker.remove(); - this.layers.clear(); - this.sources.clear(); - this.markers.clear(); - } - - dispose(): void { - this.clear(); - } -} - -export class MapLibreMap implements IMap { - private map: MLMap; - private scenes = new Map(); - public readonly ready: Promise; - - constructor(init: MapInit) { - this.map = new maplibregl.Map({ - container: init.container, - style: BASE_STYLES[init.baseLayer ?? 'osm'], - center: init.center, - zoom: init.zoom, - }); - - if (init.showNavigationControl !== false) { - this.map.addControl(new maplibregl.NavigationControl(), 'bottom-left'); - } - if (init.showScaleControl !== false) { - this.map.addControl(new maplibregl.ScaleControl({ maxWidth: 100, unit: 'metric' }), 'bottom-right'); - } - - this.ready = new Promise((resolve) => this.map.once('load', () => resolve())); - } - - on(event: E, handler: MapEventHandler): () => void { - const wrapped = (e: unknown) => { - switch (event) { - case 'click': - case 'mousemove': { - const ev = e as maplibregl.MapMouseEvent; - (handler as MapEventHandler<'click'>)({ - lngLat: { lat: ev.lngLat.lat, lng: ev.lngLat.lng }, - originalEvent: ev.originalEvent, - }); - break; - } - case 'move': - (handler as MapEventHandler<'move'>)({ - center: [this.map.getCenter().lng, this.map.getCenter().lat], - zoom: this.map.getZoom(), - }); - break; - case 'zoom': - (handler as MapEventHandler<'zoom'>)({ zoom: this.map.getZoom() }); - break; - case 'load': - (handler as MapEventHandler<'load'>)(undefined as MapEventPayload['load']); - break; - } - }; - this.map.on(event as 'click', wrapped); - return () => this.map.off(event as 'click', wrapped); - } - - setCenter(pos: LngLatTuple, zoom?: number): void { - this.map.setCenter(pos); - if (zoom !== undefined) this.map.setZoom(zoom); - } - - panTo(pos: LngLatTuple, durationMs?: number): void { - this.map.panTo(pos, durationMs ? { duration: durationMs } : undefined); - } - - fitBounds(coords: LatLngTuple[], paddingPx = 50): void { - if (coords.length === 0) return; - const first: [number, number] = [coords[0][1], coords[0][0]]; - const bounds = coords.reduce( - (b, c) => b.extend([c[1], c[0]] as [number, number]), - new maplibregl.LngLatBounds(first, first), - ); - this.map.fitBounds(bounds, { padding: paddingPx }); - } - - getZoom(): number { - return this.map.getZoom(); - } - - setZoom(zoom: number): void { - this.map.setZoom(zoom); - } - - setCursor(cursor: string | null): void { - this.map.getCanvas().style.cursor = cursor ?? ''; - } - - scene(name: string): Scene { - let scene = this.scenes.get(name); - if (!scene) { - scene = new MapLibreScene(name, this.map); - this.scenes.set(name, scene); - } - return scene; - } - - disposeScene(name: string): void { - const scene = this.scenes.get(name); - if (!scene) return; - scene.dispose(); - this.scenes.delete(name); - } - - getRawInstance(): MLMap { - return this.map; - } - - dispose(): void { - for (const s of this.scenes.values()) s.dispose(); - this.scenes.clear(); - this.map.remove(); - } -} - -export function createMapLibreMap(init: MapInit): IMap { - return new MapLibreMap(init); -} diff --git a/src/lib/map/tools/measure.ts b/src/lib/map/tools/measure.ts deleted file mode 100644 index 17a6acf..0000000 --- a/src/lib/map/tools/measure.ts +++ /dev/null @@ -1,75 +0,0 @@ -import type { IMap } from '../core'; -import type { LatLngTuple } from '$domain'; -import { distHaversine } from '$domain'; - -/** - * Library-agnostic measure tool. Click to drop points; each new point draws - * a segment and reports the running total distance in kilometers. - * - * Returns a disposer that removes all overlays and deactivates listeners. - */ -export interface MeasureHandle { - dispose(): void; - reset(): void; -} - -export interface MeasureOptions { - onUpdate?: (totalKm: number, points: LatLngTuple[]) => void; - color?: string; - width?: number; - sceneName?: string; -} - -export function startMeasure(map: IMap, options: MeasureOptions = {}): MeasureHandle { - const scene = map.scene(options.sceneName ?? 'measure'); - const points: LatLngTuple[] = []; - let segmentCounter = 0; - let pointCounter = 0; - let total = 0; - - map.setCursor('crosshair'); - - const off = map.on('click', (e) => { - const latlng: LatLngTuple = [e.lngLat.lat, e.lngLat.lng]; - points.push(latlng); - - scene.addMarker(`p-${pointCounter++}`, { - lngLat: [e.lngLat.lng, e.lngLat.lat], - iconSize: [8, 8], - className: 'lsv-measure-dot', - }); - - if (points.length > 1) { - const prev = points[points.length - 2]; - total += distHaversine( - { lat: prev[0], lng: prev[1] }, - { lat: latlng[0], lng: latlng[1] }, - ); - scene.addLine(`seg-${segmentCounter++}`, { - coords: [prev, latlng], - color: options.color ?? '#2563eb', - width: options.width ?? 2, - }); - } - - options.onUpdate?.(total, [...points]); - }); - - function reset() { - scene.clear(); - points.length = 0; - total = 0; - segmentCounter = 0; - pointCounter = 0; - options.onUpdate?.(0, []); - } - - return { - dispose() { - off(); - map.setCursor(null); - map.disposeScene(options.sceneName ?? 'measure'); - }, - reset, - }; -} diff --git a/src/lib/map/tools/selection.ts b/src/lib/map/tools/selection.ts deleted file mode 100644 index e34ca9a..0000000 --- a/src/lib/map/tools/selection.ts +++ /dev/null @@ -1,23 +0,0 @@ -import type { IMap } from '../core'; - -/** - * One-shot coordinate selection tool: next click reports a lat/lng and the - * tool disarms itself. - */ -export function startCoordinateSelection( - map: IMap, - onSelect: (coords: { lat: number; lng: number }) => void, -): () => void { - map.setCursor('crosshair'); - - const off = map.on('click', (e) => { - map.setCursor(null); - off(); - onSelect(e.lngLat); - }); - - return () => { - map.setCursor(null); - off(); - }; -} diff --git a/src/lib/state/index.ts b/src/lib/state/index.ts deleted file mode 100644 index 9e93089..0000000 --- a/src/lib/state/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export { persisted } from './persisted'; -export type { Serializer, PersistedOptions } from './persisted'; diff --git a/src/lib/state/persisted.ts b/src/lib/state/persisted.ts deleted file mode 100644 index 6f4a037..0000000 --- a/src/lib/state/persisted.ts +++ /dev/null @@ -1,131 +0,0 @@ -import { writable, type Writable, type Updater } from 'svelte/store'; -import { browser } from '$app/environment'; - -/** - * Writable store backed by localStorage, synced across tabs via - * BroadcastChannel. The store initializes with a snapshot from storage if - * present and falls back to `initial` otherwise. Subsequent writes are - * serialized to localStorage and broadcast to other tabs. - * - * Notes - * ----- - * - Serialization uses JSON by default; pass custom `serializer` for Dates etc. - * - Reads on the server (or before hydration) return `initial` — no storage - * access happens until the `browser` flag is true. - * - Removing the key from storage (via another tab or devtools) resets the - * store to `initial` on the next notification. - * - A per-store "suppress rebroadcast" flag prevents a ping-pong loop when a - * remote update arrives: the incoming `set` must not trigger a new - * broadcast or we'd echo forever. - */ - -export interface Serializer { - parse: (raw: string) => T; - stringify: (value: T) => string; -} - -const json: Serializer = { parse: JSON.parse, stringify: JSON.stringify }; - -export interface PersistedOptions { - serializer?: Serializer; - /** - * Skip cross-tab sync if multiple stores must not share a channel or if a - * store's value is too large to broadcast efficiently. - */ - syncTabs?: boolean; -} - -const CHANNEL = 'leaflet-svelte:store'; - -let channel: BroadcastChannel | null = null; -function getChannel(): BroadcastChannel | null { - if (!browser) return null; - if (typeof BroadcastChannel === 'undefined') return null; - channel ??= new BroadcastChannel(CHANNEL); - return channel; -} - -export function persisted( - key: string, - initial: T, - options: PersistedOptions = {}, -): Writable { - const serializer = (options.serializer ?? (json as Serializer)) as Serializer; - const syncTabs = options.syncTabs ?? true; - - const readFromStorage = (): T => { - if (!browser) return initial; - const raw = localStorage.getItem(key); - if (raw === null) return initial; - try { - return serializer.parse(raw); - } catch (e) { - console.warn(`[persisted] failed to parse "${key}":`, e); - return initial; - } - }; - - const store = writable(readFromStorage()); - - const write = (value: T) => { - if (!browser) return; - try { - localStorage.setItem(key, serializer.stringify(value)); - } catch (e) { - console.warn(`[persisted] failed to write "${key}":`, e); - } - }; - - let firstTick = true; - let remoteUpdate = false; - - if (browser) { - store.subscribe((value) => { - // The first subscribe fires synchronously with the initial value. We - // already read it from storage, so re-writing it is a no-op; skipping - // also avoids a spurious broadcast to other tabs on every page load. - if (firstTick) { - firstTick = false; - return; - } - write(value); - if (syncTabs && !remoteUpdate) { - getChannel()?.postMessage({ key, value }); - } - }); - - if (syncTabs) { - getChannel()?.addEventListener('message', (event) => { - if (event.data?.key !== key) return; - remoteUpdate = true; - try { - store.set(event.data.value); - } finally { - remoteUpdate = false; - } - }); - } - - window.addEventListener('storage', (event) => { - if (event.storageArea !== localStorage || event.key !== key) return; - remoteUpdate = true; - try { - if (event.newValue === null) { - store.set(initial); - } else { - store.set(serializer.parse(event.newValue)); - } - } catch { - // ignore malformed writes from another tab - } finally { - remoteUpdate = false; - } - }); - } - - return { - subscribe: store.subscribe, - set: store.set, - update: (fn: Updater) => store.update(fn), - }; -} diff --git a/src/lib/ui/CollapsibleCard.svelte b/src/lib/ui/CollapsibleCard.svelte deleted file mode 100644 index 651229c..0000000 --- a/src/lib/ui/CollapsibleCard.svelte +++ /dev/null @@ -1,60 +0,0 @@ - - - - - - - - - {#if !collapsed} - - {@render children?.()} - - {/if} - diff --git a/src/lib/ui/ConfirmationPrompt.svelte b/src/lib/ui/ConfirmationPrompt.svelte deleted file mode 100644 index 31376a1..0000000 --- a/src/lib/ui/ConfirmationPrompt.svelte +++ /dev/null @@ -1,49 +0,0 @@ - - - - {title} - - {#if children}{@render children()}{/if} - - - - - - diff --git a/src/lib/ui/EditableCell.svelte b/src/lib/ui/EditableCell.svelte deleted file mode 100644 index db405fc..0000000 --- a/src/lib/ui/EditableCell.svelte +++ /dev/null @@ -1,56 +0,0 @@ - - - - {#if editing} - - {:else if value === emptyValue || value === null || value === undefined} - {emptyPlaceholder} - {:else} - {valuePrefix}{value}{valueSuffix} - {/if} - diff --git a/src/lib/ui/LabelGroup.svelte b/src/lib/ui/LabelGroup.svelte deleted file mode 100644 index 811bc2b..0000000 --- a/src/lib/ui/LabelGroup.svelte +++ /dev/null @@ -1,33 +0,0 @@ - - -
- - -
- {@render children?.()} -
-
- - diff --git a/src/lib/ui/PanelContainer.svelte b/src/lib/ui/PanelContainer.svelte deleted file mode 100644 index 3087d42..0000000 --- a/src/lib/ui/PanelContainer.svelte +++ /dev/null @@ -1,38 +0,0 @@ - - - - - diff --git a/src/lib/ui/SelectSearchable.svelte b/src/lib/ui/SelectSearchable.svelte deleted file mode 100644 index 6056cfc..0000000 --- a/src/lib/ui/SelectSearchable.svelte +++ /dev/null @@ -1,191 +0,0 @@ - - - - -
e.key === 'Enter' && toggle()} - {...restProps}> - {selectedLabel || placeholder} - - {#if clearable && selected != null} - - {/if} - - {#if isOpen} - - {/if} -
- - diff --git a/src/lib/ui/SpoilerGroup.svelte b/src/lib/ui/SpoilerGroup.svelte deleted file mode 100644 index 53bff0e..0000000 --- a/src/lib/ui/SpoilerGroup.svelte +++ /dev/null @@ -1,56 +0,0 @@ - - -
- - - {#if expanded} -
- {@render children?.()} -
- {:else} -
- {/if} -
- - diff --git a/src/lib/ui/TabBar.svelte b/src/lib/ui/TabBar.svelte deleted file mode 100644 index 74e1511..0000000 --- a/src/lib/ui/TabBar.svelte +++ /dev/null @@ -1,48 +0,0 @@ - - -
- {#each tabs as tab (tab.id)} - - {/each} -
- - diff --git a/src/lib/ui/Toast.svelte b/src/lib/ui/Toast.svelte deleted file mode 100644 index 6c4dd27..0000000 --- a/src/lib/ui/Toast.svelte +++ /dev/null @@ -1,40 +0,0 @@ - - -
- {#each $toasts as toast (toast.id)} - removeToast(toast.id)}> - removeToast(toast.id)} class={`text-${toast.color}`}> - - {toast.header} - - - {toast.body} - - - {/each} -
- - diff --git a/src/lib/ui/editor/ListEditor.svelte b/src/lib/ui/editor/ListEditor.svelte deleted file mode 100644 index 6832fa4..0000000 --- a/src/lib/ui/editor/ListEditor.svelte +++ /dev/null @@ -1,338 +0,0 @@ - - - - - - - - - - (isConfirmationVisible = false)}> -

Delete "{selectedItem?.name}"?

-
diff --git a/src/lib/ui/index.ts b/src/lib/ui/index.ts deleted file mode 100644 index 818b54e..0000000 --- a/src/lib/ui/index.ts +++ /dev/null @@ -1,13 +0,0 @@ -export { default as CollapsibleCard } from './CollapsibleCard.svelte'; -export { default as PanelContainer } from './PanelContainer.svelte'; -export { default as TabBar } from './TabBar.svelte'; -export { default as Toast } from './Toast.svelte'; -export { default as ConfirmationPrompt } from './ConfirmationPrompt.svelte'; -export { default as SelectSearchable } from './SelectSearchable.svelte'; -export { default as SpoilerGroup } from './SpoilerGroup.svelte'; -export { default as LabelGroup } from './LabelGroup.svelte'; -export { default as EditableCell } from './EditableCell.svelte'; -export { default as ListEditor } from './editor/ListEditor.svelte'; -export type { ListEditorConfig, ListEditorApi } from './editor/ListEditor.svelte'; -export { addToast, removeToast, toasts } from './toasts'; -export type { Toast as ToastMessage, ToastColor, ToastInit } from './toasts'; diff --git a/src/lib/ui/toasts.ts b/src/lib/ui/toasts.ts deleted file mode 100644 index efb1bda..0000000 --- a/src/lib/ui/toasts.ts +++ /dev/null @@ -1,54 +0,0 @@ -import { writable } from 'svelte/store'; - -export type ToastColor = - | 'primary' - | 'secondary' - | 'success' - | 'danger' - | 'warning' - | 'info' - | 'light' - | 'dark'; - -export interface Toast { - id: string; - header: string; - body: string; - color: ToastColor; - persistent: boolean; - onRemove?: (id: string) => void; -} - -export interface ToastInit { - header: string; - body: string; - color?: ToastColor; - persistent?: boolean; - onRemove?: (id: string) => void; -} - -export const toasts = writable([]); - -export function addToast(init: ToastInit): string { - const id = crypto.randomUUID(); - toasts.update((all) => [ - ...all, - { - id, - header: init.header, - body: init.body, - color: init.color ?? 'info', - persistent: init.persistent ?? false, - onRemove: init.onRemove, - }, - ]); - return id; -} - -export function removeToast(id: string): void { - toasts.update((all) => { - const t = all.find((x) => x.id === id); - t?.onRemove?.(id); - return all.filter((x) => x.id !== id); - }); -} diff --git a/src/routes/+layout.svelte b/src/routes/+layout.svelte deleted file mode 100644 index 2d80679..0000000 --- a/src/routes/+layout.svelte +++ /dev/null @@ -1,22 +0,0 @@ - - -{#if ready} - {@render children?.()} -{/if} - diff --git a/src/routes/+layout.ts b/src/routes/+layout.ts deleted file mode 100644 index eae1350..0000000 --- a/src/routes/+layout.ts +++ /dev/null @@ -1,5 +0,0 @@ -// Pure SPA: all rendering happens in the browser against a Django REST backend. -// Disable SSR and prerender globally so every route is just a client-side chunk. -export const ssr = false; -export const prerender = false; -export const trailingSlash = 'ignore'; diff --git a/src/routes/+page.svelte b/src/routes/+page.svelte index 9beb424..1e03cf2 100644 --- a/src/routes/+page.svelte +++ b/src/routes/+page.svelte @@ -1,22 +1,7 @@ - -
-
- Loading... -
-
+
+ +
\ No newline at end of file diff --git a/src/routes/+page.ts b/src/routes/+page.ts new file mode 100644 index 0000000..b2463b2 --- /dev/null +++ b/src/routes/+page.ts @@ -0,0 +1 @@ +export const ssr =false; \ No newline at end of file diff --git a/src/routes/leaflet.svelte b/src/routes/leaflet.svelte new file mode 100644 index 0000000..380b25b --- /dev/null +++ b/src/routes/leaflet.svelte @@ -0,0 +1,548 @@ + + +
+
+
+ Lat: {mouseLat}, Long: {mouseLng} +
+
+
Launch Point
+ +
+ Start Point: + +
+ +
+ Latitude/Longitude: +
+ +

/

+ + +
+
+ +
+ +
+ +
+ Launch Height (m): + +
+ +
+ Launch Time (UTC): + +
+ +
+ Launch Date: + +
+ +
+ Ascent Rate (m/s): + +
+ +
+ Burst/Drift Altitude (m): + +
+ +
+ Flight Profile: + +
+ +
+ + + +
+ +
+ Descent Rate (m/s): + +
+ +
+ Forecast Mode (help): +
+ + +
+
+ +
+ + +
+
+
+ + +{#if showBurstCalculator} + +{/if} + + \ No newline at end of file diff --git a/src/routes/login/+page.svelte b/src/routes/login/+page.svelte deleted file mode 100644 index 186bfed..0000000 --- a/src/routes/login/+page.svelte +++ /dev/null @@ -1,5 +0,0 @@ - - - diff --git a/src/routes/predict/+page.svelte b/src/routes/predict/+page.svelte deleted file mode 100644 index eb06ae4..0000000 --- a/src/routes/predict/+page.svelte +++ /dev/null @@ -1,113 +0,0 @@ - - -
- -
- - - - - -
- {#if leftTab === 'scenario'} - - {:else if leftTab === 'conditions'} - - {/if} -
-
- - - -
- {#if rightTab === 'workspaces'} - - {:else if rightTab === 'settings'} - - {/if} -
-
- - -
-
diff --git a/src/routes/track/+page.svelte b/src/routes/track/+page.svelte deleted file mode 100644 index e513ea3..0000000 --- a/src/routes/track/+page.svelte +++ /dev/null @@ -1,22 +0,0 @@ - - -
- -
- - - - - -
diff --git a/src/routes/user/account/+page.svelte b/src/routes/user/account/+page.svelte deleted file mode 100644 index 1a4d9b5..0000000 --- a/src/routes/user/account/+page.svelte +++ /dev/null @@ -1,84 +0,0 @@ - - -
- -
-
-
- - -
- -
{$t('nav.account')}
- - - - - -
- -
-
-
-
-
-
-
-
- - (showConfirm = false)}> -

{confirmBody}

-
diff --git a/src/routes/user/flights/+page.svelte b/src/routes/user/flights/+page.svelte deleted file mode 100644 index 9ef900e..0000000 --- a/src/routes/user/flights/+page.svelte +++ /dev/null @@ -1,36 +0,0 @@ - - -
- -
-
-
- -
- -
{$t('nav.trackingHistory')}
- -

TODO: wire to tracking history endpoint.

-
-
-
-
-
-
-
diff --git a/src/routes/user/predictions/+page.svelte b/src/routes/user/predictions/+page.svelte deleted file mode 100644 index 6778b78..0000000 --- a/src/routes/user/predictions/+page.svelte +++ /dev/null @@ -1,36 +0,0 @@ - - -
- -
-
-
- -
- -
{$t('nav.predictionHistory')}
- -

TODO: wire to prediction history endpoint.

-
-
-
-
-
-
-
diff --git a/src/routes/user/templates/+page.svelte b/src/routes/user/templates/+page.svelte deleted file mode 100644 index 80e7007..0000000 --- a/src/routes/user/templates/+page.svelte +++ /dev/null @@ -1,145 +0,0 @@ - - -
- -
-
-
- - -
- -
{$t('points.items')}
- - search.set()} /> -
- - - - - - - - - - - - {#each table.rows as row} - - - - - - - - {/each} - -
{$t('points.name')}{$t('points.lat')}{$t('points.lon')}{$t('points.alt')}
{row.name}{row.lat.toFixed(4)} °{row.lon.toFixed(4)} °{row.alt} м - - -
-
- - - table.setPage('previous')} /> - - {#each table.pagesWithEllipsis as page} - - table.setPage(page)}>{page} - - {/each} - - table.setPage('next')} /> - - -
-
-
-
-
-
-
- - (toDelete = null)}> - {#if toDelete} -

Delete "{toDelete.name}"?

- {/if} -
- - (editPoint = null)} /> diff --git a/static/css/bootstrap.min.css b/static/css/bootstrap.min.css deleted file mode 100644 index b4bb94f..0000000 --- a/static/css/bootstrap.min.css +++ /dev/null @@ -1,7 +0,0 @@ -@charset "UTF-8";/*! - * Bootstrap v5.2.3 (https://getbootstrap.com/) - * Copyright 2011-2022 The Bootstrap Authors - * Copyright 2011-2022 Twitter, Inc. - * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) - */:root{--bs-blue:#457aab;--bs-indigo:#285580;--bs-purple:#8d5fbf;--bs-pink:#d63384;--bs-red:#c42526;--bs-orange:#ff9100;--bs-yellow:#ffc107;--bs-green:#2a8256;--bs-teal:#009691;--bs-cyan:#0068b3;--bs-black:#222;--bs-white:#fff;--bs-gray:#a1a3a9;--bs-gray-dark:#454545;--bs-gray-100:#f2f2f2;--bs-gray-200:#e2e2e3;--bs-gray-300:#d4d4d4;--bs-gray-400:#cdcdcd;--bs-gray-500:#abb3bb;--bs-gray-600:#a1a3a9;--bs-gray-700:#5d6d75;--bs-gray-800:#454545;--bs-gray-900:#2a2a2a;--bs-primary:#457aab;--bs-secondary:#5d6d75;--bs-success:#2a8256;--bs-info:#0068b3;--bs-warning:#ffc107;--bs-danger:#c42526;--bs-light:#f2f2f2;--bs-dark:#2a2a2a;--bs-primary-rgb:69,122,171;--bs-secondary-rgb:93,109,117;--bs-success-rgb:42,130,86;--bs-info-rgb:0,104,179;--bs-warning-rgb:255,193,7;--bs-danger-rgb:196,37,38;--bs-light-rgb:242,242,242;--bs-dark-rgb:42,42,42;--bs-white-rgb:255,255,255;--bs-black-rgb:34,34,34;--bs-body-color-rgb:42,42,42;--bs-body-bg-rgb:255,255,255;--bs-font-sans-serif:Arial,sans-serif,"Helvetica Neue","Liberation Sans",system-ui,-apple-system,"Segoe UI",Roboto,"Noto Sans","Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--bs-font-monospace:Consolas,"Liberation Mono","Courier New",monospace,SFMono-Regular,Menlo,Monaco;--bs-gradient:linear-gradient(180deg, rgba(255, 255, 255, 0.15), rgba(255, 255, 255, 0));--bs-body-font-family:var(--bs-font-sans-serif);--bs-body-font-size:1rem;--bs-body-font-weight:400;--bs-body-line-height:1.5;--bs-body-color:#2a2a2a;--bs-body-bg:#fff;--bs-border-width:1px;--bs-border-style:solid;--bs-border-color:#d4d4d4;--bs-border-color-translucent:rgba(34, 34, 34, 0.175);--bs-border-radius:0;--bs-border-radius-sm:0;--bs-border-radius-lg:0;--bs-border-radius-xl:0;--bs-border-radius-2xl:0;--bs-border-radius-pill:0;--bs-link-color:#457aab;--bs-link-hover-color:#233d56;--bs-code-color:#d63384;--bs-highlight-bg:#fff3cd}*,::after,::before{box-sizing:border-box}@media (prefers-reduced-motion:no-preference){:root{scroll-behavior:smooth}}body{margin:0;font-family:var(--bs-body-font-family);font-size:var(--bs-body-font-size);font-weight:var(--bs-body-font-weight);line-height:var(--bs-body-line-height);color:var(--bs-body-color);text-align:var(--bs-body-text-align);background-color:var(--bs-body-bg);-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:rgba(34,34,34,0)}hr{margin:1rem 0;color:inherit;border:0;border-top:1px solid;opacity:.25}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem;font-weight:600;line-height:1.2}.h1,h1{font-size:calc(1.325rem + .9vw)}@media (min-width:1200px){.h1,h1{font-size:2rem}}.h2,h2{font-size:calc(1.3rem + .6vw)}@media (min-width:1200px){.h2,h2{font-size:1.75rem}}.h3,h3{font-size:calc(1.275rem + .3vw)}@media (min-width:1200px){.h3,h3{font-size:1.5rem}}.h4,h4{font-size:1.25rem}.h5,h5{font-size:1.15rem}.h6,h6{font-size:1rem}p{margin-top:0;margin-bottom:1rem}abbr[title]{-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}ol,ul{padding-left:2rem}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}.small,small{font-size:.875em}.mark,mark{padding:.1875em;background-color:var(--bs-highlight-bg)}sub,sup{position:relative;font-size:.75em;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:var(--bs-link-color);text-decoration:underline}a:hover{color:var(--bs-link-hover-color)}a:not([href]):not([class]),a:not([href]):not([class]):hover{color:inherit;text-decoration:none}code,kbd,pre,samp{font-family:var(--bs-font-monospace);font-size:1em}pre{display:block;margin-top:0;margin-bottom:1rem;overflow:auto;font-size:.875em}pre code{font-size:inherit;color:inherit;word-break:normal}code{font-size:.875em;color:var(--bs-code-color);word-wrap:break-word}a>code{color:inherit}kbd{padding:.1875rem .25rem;font-size:.875em;color:var(--bs-body-bg);background-color:var(--bs-body-color);border-radius:0}kbd kbd{padding:0;font-size:1em}figure{margin:0 0 1rem}img,svg{vertical-align:middle}table{caption-side:bottom;border-collapse:collapse}caption{padding-top:.3rem;padding-bottom:.3rem;color:#a1a3a9;text-align:left}th{text-align:inherit;text-align:-webkit-match-parent}tbody,td,tfoot,th,thead,tr{border-color:inherit;border-style:solid;border-width:0}label{display:inline-block}button{border-radius:0}button:focus:not(:focus-visible){outline:0}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,select{text-transform:none}[role=button]{cursor:pointer}select{word-wrap:normal}select:disabled{opacity:1}[list]:not([type=date]):not([type=datetime-local]):not([type=month]):not([type=week]):not([type=time])::-webkit-calendar-picker-indicator{display:none!important}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}::-moz-focus-inner{padding:0;border-style:none}textarea{resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{float:left;width:100%;padding:0;margin-bottom:.5rem;font-size:calc(1.275rem + .3vw);line-height:inherit}@media (min-width:1200px){legend{font-size:1.5rem}}legend+*{clear:left}::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-fields-wrapper,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-text,::-webkit-datetime-edit-year-field{padding:0}::-webkit-inner-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:textfield}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-color-swatch-wrapper{padding:0}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}::file-selector-button{font:inherit;-webkit-appearance:button}output{display:inline-block}iframe{border:0}summary{display:list-item;cursor:pointer}progress{vertical-align:baseline}[hidden]{display:none!important}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:calc(1.625rem + 4.5vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-1{font-size:5rem}}.display-2{font-size:calc(1.575rem + 3.9vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-2{font-size:4.5rem}}.display-3{font-size:calc(1.525rem + 3.3vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-3{font-size:4rem}}.display-4{font-size:calc(1.475rem + 2.7vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-4{font-size:3.5rem}}.display-5{font-size:calc(1.425rem + 2.1vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-5{font-size:3rem}}.display-6{font-size:calc(1.375rem + 1.5vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-6{font-size:2.5rem}}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:.875em;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote>:last-child{margin-bottom:0}.blockquote-footer{margin-top:-1rem;margin-bottom:1rem;font-size:.875em;color:#a1a3a9}.blockquote-footer::before{content:"— "}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:#fff;border:1px solid var(--bs-border-color);border-radius:0;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:.875em;color:#a1a3a9}.container,.container-fluid,.container-lg,.container-md,.container-sm,.container-xl,.container-xxl{--bs-gutter-x:1.5rem;--bs-gutter-y:0;width:100%;padding-right:calc(var(--bs-gutter-x) * .5);padding-left:calc(var(--bs-gutter-x) * .5);margin-right:auto;margin-left:auto}@media (min-width:576px){.container,.container-sm{max-width:540px}}@media (min-width:768px){.container,.container-md,.container-sm{max-width:720px}}@media (min-width:992px){.container,.container-lg,.container-md,.container-sm{max-width:960px}}@media (min-width:1200px){.container,.container-lg,.container-md,.container-sm,.container-xl{max-width:1140px}}@media (min-width:1400px){.container,.container-lg,.container-md,.container-sm,.container-xl,.container-xxl{max-width:1320px}}.row{--bs-gutter-x:1.5rem;--bs-gutter-y:0;display:flex;flex-wrap:wrap;margin-top:calc(-1 * var(--bs-gutter-y));margin-right:calc(-.5 * var(--bs-gutter-x));margin-left:calc(-.5 * var(--bs-gutter-x))}.row>*{flex-shrink:0;width:100%;max-width:100%;padding-right:calc(var(--bs-gutter-x) * .5);padding-left:calc(var(--bs-gutter-x) * .5);margin-top:var(--bs-gutter-y)}.col{flex:1 0 0%}.row-cols-auto>*{flex:0 0 auto;width:auto}.row-cols-1>*{flex:0 0 auto;width:100%}.row-cols-2>*{flex:0 0 auto;width:50%}.row-cols-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-4>*{flex:0 0 auto;width:25%}.row-cols-5>*{flex:0 0 auto;width:20%}.row-cols-6>*{flex:0 0 auto;width:16.6666666667%}.col-auto{flex:0 0 auto;width:auto}.col-1{flex:0 0 auto;width:8.33333333%}.col-2{flex:0 0 auto;width:16.66666667%}.col-3{flex:0 0 auto;width:25%}.col-4{flex:0 0 auto;width:33.33333333%}.col-5{flex:0 0 auto;width:41.66666667%}.col-6{flex:0 0 auto;width:50%}.col-7{flex:0 0 auto;width:58.33333333%}.col-8{flex:0 0 auto;width:66.66666667%}.col-9{flex:0 0 auto;width:75%}.col-10{flex:0 0 auto;width:83.33333333%}.col-11{flex:0 0 auto;width:91.66666667%}.col-12{flex:0 0 auto;width:100%}.offset-1{margin-left:8.33333333%}.offset-2{margin-left:16.66666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.33333333%}.offset-5{margin-left:41.66666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.33333333%}.offset-8{margin-left:66.66666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.33333333%}.offset-11{margin-left:91.66666667%}.g-0,.gx-0{--bs-gutter-x:0}.g-0,.gy-0{--bs-gutter-y:0}.g-1,.gx-1{--bs-gutter-x:0.25rem}.g-1,.gy-1{--bs-gutter-y:0.25rem}.g-2,.gx-2{--bs-gutter-x:0.5rem}.g-2,.gy-2{--bs-gutter-y:0.5rem}.g-3,.gx-3{--bs-gutter-x:1rem}.g-3,.gy-3{--bs-gutter-y:1rem}.g-4,.gx-4{--bs-gutter-x:1.5rem}.g-4,.gy-4{--bs-gutter-y:1.5rem}.g-5,.gx-5{--bs-gutter-x:3rem}.g-5,.gy-5{--bs-gutter-y:3rem}@media (min-width:576px){.col-sm{flex:1 0 0%}.row-cols-sm-auto>*{flex:0 0 auto;width:auto}.row-cols-sm-1>*{flex:0 0 auto;width:100%}.row-cols-sm-2>*{flex:0 0 auto;width:50%}.row-cols-sm-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-sm-4>*{flex:0 0 auto;width:25%}.row-cols-sm-5>*{flex:0 0 auto;width:20%}.row-cols-sm-6>*{flex:0 0 auto;width:16.6666666667%}.col-sm-auto{flex:0 0 auto;width:auto}.col-sm-1{flex:0 0 auto;width:8.33333333%}.col-sm-2{flex:0 0 auto;width:16.66666667%}.col-sm-3{flex:0 0 auto;width:25%}.col-sm-4{flex:0 0 auto;width:33.33333333%}.col-sm-5{flex:0 0 auto;width:41.66666667%}.col-sm-6{flex:0 0 auto;width:50%}.col-sm-7{flex:0 0 auto;width:58.33333333%}.col-sm-8{flex:0 0 auto;width:66.66666667%}.col-sm-9{flex:0 0 auto;width:75%}.col-sm-10{flex:0 0 auto;width:83.33333333%}.col-sm-11{flex:0 0 auto;width:91.66666667%}.col-sm-12{flex:0 0 auto;width:100%}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.33333333%}.offset-sm-2{margin-left:16.66666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.33333333%}.offset-sm-5{margin-left:41.66666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.33333333%}.offset-sm-8{margin-left:66.66666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.33333333%}.offset-sm-11{margin-left:91.66666667%}.g-sm-0,.gx-sm-0{--bs-gutter-x:0}.g-sm-0,.gy-sm-0{--bs-gutter-y:0}.g-sm-1,.gx-sm-1{--bs-gutter-x:0.25rem}.g-sm-1,.gy-sm-1{--bs-gutter-y:0.25rem}.g-sm-2,.gx-sm-2{--bs-gutter-x:0.5rem}.g-sm-2,.gy-sm-2{--bs-gutter-y:0.5rem}.g-sm-3,.gx-sm-3{--bs-gutter-x:1rem}.g-sm-3,.gy-sm-3{--bs-gutter-y:1rem}.g-sm-4,.gx-sm-4{--bs-gutter-x:1.5rem}.g-sm-4,.gy-sm-4{--bs-gutter-y:1.5rem}.g-sm-5,.gx-sm-5{--bs-gutter-x:3rem}.g-sm-5,.gy-sm-5{--bs-gutter-y:3rem}}@media (min-width:768px){.col-md{flex:1 0 0%}.row-cols-md-auto>*{flex:0 0 auto;width:auto}.row-cols-md-1>*{flex:0 0 auto;width:100%}.row-cols-md-2>*{flex:0 0 auto;width:50%}.row-cols-md-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-md-4>*{flex:0 0 auto;width:25%}.row-cols-md-5>*{flex:0 0 auto;width:20%}.row-cols-md-6>*{flex:0 0 auto;width:16.6666666667%}.col-md-auto{flex:0 0 auto;width:auto}.col-md-1{flex:0 0 auto;width:8.33333333%}.col-md-2{flex:0 0 auto;width:16.66666667%}.col-md-3{flex:0 0 auto;width:25%}.col-md-4{flex:0 0 auto;width:33.33333333%}.col-md-5{flex:0 0 auto;width:41.66666667%}.col-md-6{flex:0 0 auto;width:50%}.col-md-7{flex:0 0 auto;width:58.33333333%}.col-md-8{flex:0 0 auto;width:66.66666667%}.col-md-9{flex:0 0 auto;width:75%}.col-md-10{flex:0 0 auto;width:83.33333333%}.col-md-11{flex:0 0 auto;width:91.66666667%}.col-md-12{flex:0 0 auto;width:100%}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.33333333%}.offset-md-2{margin-left:16.66666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.33333333%}.offset-md-5{margin-left:41.66666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.33333333%}.offset-md-8{margin-left:66.66666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.33333333%}.offset-md-11{margin-left:91.66666667%}.g-md-0,.gx-md-0{--bs-gutter-x:0}.g-md-0,.gy-md-0{--bs-gutter-y:0}.g-md-1,.gx-md-1{--bs-gutter-x:0.25rem}.g-md-1,.gy-md-1{--bs-gutter-y:0.25rem}.g-md-2,.gx-md-2{--bs-gutter-x:0.5rem}.g-md-2,.gy-md-2{--bs-gutter-y:0.5rem}.g-md-3,.gx-md-3{--bs-gutter-x:1rem}.g-md-3,.gy-md-3{--bs-gutter-y:1rem}.g-md-4,.gx-md-4{--bs-gutter-x:1.5rem}.g-md-4,.gy-md-4{--bs-gutter-y:1.5rem}.g-md-5,.gx-md-5{--bs-gutter-x:3rem}.g-md-5,.gy-md-5{--bs-gutter-y:3rem}}@media (min-width:992px){.col-lg{flex:1 0 0%}.row-cols-lg-auto>*{flex:0 0 auto;width:auto}.row-cols-lg-1>*{flex:0 0 auto;width:100%}.row-cols-lg-2>*{flex:0 0 auto;width:50%}.row-cols-lg-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-lg-4>*{flex:0 0 auto;width:25%}.row-cols-lg-5>*{flex:0 0 auto;width:20%}.row-cols-lg-6>*{flex:0 0 auto;width:16.6666666667%}.col-lg-auto{flex:0 0 auto;width:auto}.col-lg-1{flex:0 0 auto;width:8.33333333%}.col-lg-2{flex:0 0 auto;width:16.66666667%}.col-lg-3{flex:0 0 auto;width:25%}.col-lg-4{flex:0 0 auto;width:33.33333333%}.col-lg-5{flex:0 0 auto;width:41.66666667%}.col-lg-6{flex:0 0 auto;width:50%}.col-lg-7{flex:0 0 auto;width:58.33333333%}.col-lg-8{flex:0 0 auto;width:66.66666667%}.col-lg-9{flex:0 0 auto;width:75%}.col-lg-10{flex:0 0 auto;width:83.33333333%}.col-lg-11{flex:0 0 auto;width:91.66666667%}.col-lg-12{flex:0 0 auto;width:100%}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.33333333%}.offset-lg-2{margin-left:16.66666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.33333333%}.offset-lg-5{margin-left:41.66666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.33333333%}.offset-lg-8{margin-left:66.66666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.33333333%}.offset-lg-11{margin-left:91.66666667%}.g-lg-0,.gx-lg-0{--bs-gutter-x:0}.g-lg-0,.gy-lg-0{--bs-gutter-y:0}.g-lg-1,.gx-lg-1{--bs-gutter-x:0.25rem}.g-lg-1,.gy-lg-1{--bs-gutter-y:0.25rem}.g-lg-2,.gx-lg-2{--bs-gutter-x:0.5rem}.g-lg-2,.gy-lg-2{--bs-gutter-y:0.5rem}.g-lg-3,.gx-lg-3{--bs-gutter-x:1rem}.g-lg-3,.gy-lg-3{--bs-gutter-y:1rem}.g-lg-4,.gx-lg-4{--bs-gutter-x:1.5rem}.g-lg-4,.gy-lg-4{--bs-gutter-y:1.5rem}.g-lg-5,.gx-lg-5{--bs-gutter-x:3rem}.g-lg-5,.gy-lg-5{--bs-gutter-y:3rem}}@media (min-width:1200px){.col-xl{flex:1 0 0%}.row-cols-xl-auto>*{flex:0 0 auto;width:auto}.row-cols-xl-1>*{flex:0 0 auto;width:100%}.row-cols-xl-2>*{flex:0 0 auto;width:50%}.row-cols-xl-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-xl-4>*{flex:0 0 auto;width:25%}.row-cols-xl-5>*{flex:0 0 auto;width:20%}.row-cols-xl-6>*{flex:0 0 auto;width:16.6666666667%}.col-xl-auto{flex:0 0 auto;width:auto}.col-xl-1{flex:0 0 auto;width:8.33333333%}.col-xl-2{flex:0 0 auto;width:16.66666667%}.col-xl-3{flex:0 0 auto;width:25%}.col-xl-4{flex:0 0 auto;width:33.33333333%}.col-xl-5{flex:0 0 auto;width:41.66666667%}.col-xl-6{flex:0 0 auto;width:50%}.col-xl-7{flex:0 0 auto;width:58.33333333%}.col-xl-8{flex:0 0 auto;width:66.66666667%}.col-xl-9{flex:0 0 auto;width:75%}.col-xl-10{flex:0 0 auto;width:83.33333333%}.col-xl-11{flex:0 0 auto;width:91.66666667%}.col-xl-12{flex:0 0 auto;width:100%}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.33333333%}.offset-xl-2{margin-left:16.66666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.33333333%}.offset-xl-5{margin-left:41.66666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.33333333%}.offset-xl-8{margin-left:66.66666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.33333333%}.offset-xl-11{margin-left:91.66666667%}.g-xl-0,.gx-xl-0{--bs-gutter-x:0}.g-xl-0,.gy-xl-0{--bs-gutter-y:0}.g-xl-1,.gx-xl-1{--bs-gutter-x:0.25rem}.g-xl-1,.gy-xl-1{--bs-gutter-y:0.25rem}.g-xl-2,.gx-xl-2{--bs-gutter-x:0.5rem}.g-xl-2,.gy-xl-2{--bs-gutter-y:0.5rem}.g-xl-3,.gx-xl-3{--bs-gutter-x:1rem}.g-xl-3,.gy-xl-3{--bs-gutter-y:1rem}.g-xl-4,.gx-xl-4{--bs-gutter-x:1.5rem}.g-xl-4,.gy-xl-4{--bs-gutter-y:1.5rem}.g-xl-5,.gx-xl-5{--bs-gutter-x:3rem}.g-xl-5,.gy-xl-5{--bs-gutter-y:3rem}}@media (min-width:1400px){.col-xxl{flex:1 0 0%}.row-cols-xxl-auto>*{flex:0 0 auto;width:auto}.row-cols-xxl-1>*{flex:0 0 auto;width:100%}.row-cols-xxl-2>*{flex:0 0 auto;width:50%}.row-cols-xxl-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-xxl-4>*{flex:0 0 auto;width:25%}.row-cols-xxl-5>*{flex:0 0 auto;width:20%}.row-cols-xxl-6>*{flex:0 0 auto;width:16.6666666667%}.col-xxl-auto{flex:0 0 auto;width:auto}.col-xxl-1{flex:0 0 auto;width:8.33333333%}.col-xxl-2{flex:0 0 auto;width:16.66666667%}.col-xxl-3{flex:0 0 auto;width:25%}.col-xxl-4{flex:0 0 auto;width:33.33333333%}.col-xxl-5{flex:0 0 auto;width:41.66666667%}.col-xxl-6{flex:0 0 auto;width:50%}.col-xxl-7{flex:0 0 auto;width:58.33333333%}.col-xxl-8{flex:0 0 auto;width:66.66666667%}.col-xxl-9{flex:0 0 auto;width:75%}.col-xxl-10{flex:0 0 auto;width:83.33333333%}.col-xxl-11{flex:0 0 auto;width:91.66666667%}.col-xxl-12{flex:0 0 auto;width:100%}.offset-xxl-0{margin-left:0}.offset-xxl-1{margin-left:8.33333333%}.offset-xxl-2{margin-left:16.66666667%}.offset-xxl-3{margin-left:25%}.offset-xxl-4{margin-left:33.33333333%}.offset-xxl-5{margin-left:41.66666667%}.offset-xxl-6{margin-left:50%}.offset-xxl-7{margin-left:58.33333333%}.offset-xxl-8{margin-left:66.66666667%}.offset-xxl-9{margin-left:75%}.offset-xxl-10{margin-left:83.33333333%}.offset-xxl-11{margin-left:91.66666667%}.g-xxl-0,.gx-xxl-0{--bs-gutter-x:0}.g-xxl-0,.gy-xxl-0{--bs-gutter-y:0}.g-xxl-1,.gx-xxl-1{--bs-gutter-x:0.25rem}.g-xxl-1,.gy-xxl-1{--bs-gutter-y:0.25rem}.g-xxl-2,.gx-xxl-2{--bs-gutter-x:0.5rem}.g-xxl-2,.gy-xxl-2{--bs-gutter-y:0.5rem}.g-xxl-3,.gx-xxl-3{--bs-gutter-x:1rem}.g-xxl-3,.gy-xxl-3{--bs-gutter-y:1rem}.g-xxl-4,.gx-xxl-4{--bs-gutter-x:1.5rem}.g-xxl-4,.gy-xxl-4{--bs-gutter-y:1.5rem}.g-xxl-5,.gx-xxl-5{--bs-gutter-x:3rem}.g-xxl-5,.gy-xxl-5{--bs-gutter-y:3rem}}.table{--bs-table-color:var(--bs-body-color);--bs-table-bg:transparent;--bs-table-border-color:var(--bs-border-color);--bs-table-accent-bg:transparent;--bs-table-striped-color:var(--bs-body-color);--bs-table-striped-bg:rgba(34, 34, 34, 0.05);--bs-table-active-color:var(--bs-body-color);--bs-table-active-bg:rgba(34, 34, 34, 0.1);--bs-table-hover-color:var(--bs-body-color);--bs-table-hover-bg:rgba(34, 34, 34, 0.075);width:100%;margin-bottom:1rem;color:var(--bs-table-color);vertical-align:top;border-color:var(--bs-table-border-color)}.table>:not(caption)>*>*{padding:.3rem .3rem;background-color:var(--bs-table-bg);border-bottom-width:1px;box-shadow:inset 0 0 0 9999px var(--bs-table-accent-bg)}.table>tbody{vertical-align:inherit}.table>thead{vertical-align:bottom}.table-group-divider{border-top:2px solid currentcolor}.caption-top{caption-side:top}.table-sm>:not(caption)>*>*{padding:.15rem .15rem}.table-bordered>:not(caption)>*{border-width:1px 0}.table-bordered>:not(caption)>*>*{border-width:0 1px}.table-borderless>:not(caption)>*>*{border-bottom-width:0}.table-borderless>:not(:first-child){border-top-width:0}.table-striped>tbody>tr:nth-of-type(odd)>*{--bs-table-accent-bg:var(--bs-table-striped-bg);color:var(--bs-table-striped-color)}.table-striped-columns>:not(caption)>tr>:nth-child(2n){--bs-table-accent-bg:var(--bs-table-striped-bg);color:var(--bs-table-striped-color)}.table-active{--bs-table-accent-bg:var(--bs-table-active-bg);color:var(--bs-table-active-color)}.table-hover>tbody>tr:hover>*{--bs-table-accent-bg:var(--bs-table-hover-bg);color:var(--bs-table-hover-color)}.table-primary{--bs-table-color:#fff;--bs-table-bg:#457aab;--bs-table-border-color:#5887b3;--bs-table-striped-bg:#4e81af;--bs-table-striped-color:#fff;--bs-table-active-bg:#5887b3;--bs-table-active-color:#222;--bs-table-hover-bg:#5384b1;--bs-table-hover-color:#222;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-secondary{--bs-table-color:#fff;--bs-table-bg:#5d6d75;--bs-table-border-color:#6d7c83;--bs-table-striped-bg:#65747c;--bs-table-striped-color:#fff;--bs-table-active-bg:#6d7c83;--bs-table-active-color:#fff;--bs-table-hover-bg:#69787f;--bs-table-hover-color:#fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-success{--bs-table-color:#fff;--bs-table-bg:#2a8256;--bs-table-border-color:#3f8f67;--bs-table-striped-bg:#35885e;--bs-table-striped-color:#fff;--bs-table-active-bg:#3f8f67;--bs-table-active-color:#222;--bs-table-hover-bg:#3a8b63;--bs-table-hover-color:#fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-info{--bs-table-color:#fff;--bs-table-bg:#0068b3;--bs-table-border-color:#1a77bb;--bs-table-striped-bg:#0d70b7;--bs-table-striped-color:#fff;--bs-table-active-bg:#1a77bb;--bs-table-active-color:#fff;--bs-table-hover-bg:#1373b9;--bs-table-hover-color:#fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-warning{--bs-table-color:#222;--bs-table-bg:#ffc107;--bs-table-border-color:#e9b10a;--bs-table-striped-bg:#f4b908;--bs-table-striped-color:#222;--bs-table-active-bg:#e9b10a;--bs-table-active-color:#222;--bs-table-hover-bg:#eeb509;--bs-table-hover-color:#222;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-danger{--bs-table-color:#fff;--bs-table-bg:#c42526;--bs-table-border-color:#ca3b3c;--bs-table-striped-bg:#c73031;--bs-table-striped-color:#fff;--bs-table-active-bg:#ca3b3c;--bs-table-active-color:#fff;--bs-table-hover-bg:#c83536;--bs-table-hover-color:#fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-light{--bs-table-color:#222;--bs-table-bg:#f2f2f2;--bs-table-border-color:#dddddd;--bs-table-striped-bg:#e8e8e8;--bs-table-striped-color:#222;--bs-table-active-bg:#dddddd;--bs-table-active-color:#222;--bs-table-hover-bg:#e2e2e2;--bs-table-hover-color:#222;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-dark{--bs-table-color:#fff;--bs-table-bg:#2a2a2a;--bs-table-border-color:#3f3f3f;--bs-table-striped-bg:#353535;--bs-table-striped-color:#fff;--bs-table-active-bg:#3f3f3f;--bs-table-active-color:#fff;--bs-table-hover-bg:#3a3a3a;--bs-table-hover-color:#fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-responsive{overflow-x:auto;-webkit-overflow-scrolling:touch}@media (max-width:575.98px){.table-responsive-sm{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:767.98px){.table-responsive-md{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:991.98px){.table-responsive-lg{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:1199.98px){.table-responsive-xl{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:1399.98px){.table-responsive-xxl{overflow-x:auto;-webkit-overflow-scrolling:touch}}.form-label{margin-bottom:.375rem}.col-form-label{padding-top:calc(.375rem + 1px);padding-bottom:calc(.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + 1px);padding-bottom:calc(.5rem + 1px);font-size:1.25rem}.col-form-label-sm{padding-top:calc(.25rem + 1px);padding-bottom:calc(.25rem + 1px);font-size:.875rem}.form-text{margin-top:.25rem;font-size:.875em;color:#a1a3a9}.form-control{display:block;width:100%;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#2a2a2a;background-color:#fff;background-clip:padding-box;border:1px solid #cdcdcd;-webkit-appearance:none;-moz-appearance:none;appearance:none;border-radius:0}.form-control[type=file]{overflow:hidden}.form-control[type=file]:not(:disabled):not([readonly]){cursor:pointer}.form-control:focus{color:#2a2a2a;background-color:#fff;border-color:#a2bdd5;outline:0;box-shadow:0 0 0 .25rem rgba(69,122,171,.25)}.form-control::-webkit-date-and-time-value{height:1.5em}.form-control::-moz-placeholder{color:#a1a3a9;opacity:1}.form-control::placeholder{color:#a1a3a9;opacity:1}.form-control:disabled{background-color:#e2e2e3;opacity:1}.form-control::-webkit-file-upload-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:#2a2a2a;background-color:#e2e2e3;pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:1px;border-radius:0}.form-control::file-selector-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:#2a2a2a;background-color:#e2e2e3;pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:1px;border-radius:0}.form-control:hover:not(:disabled):not([readonly])::-webkit-file-upload-button{background-color:#d7d7d8}.form-control:hover:not(:disabled):not([readonly])::file-selector-button{background-color:#d7d7d8}.form-control-plaintext{display:block;width:100%;padding:.375rem 0;margin-bottom:0;line-height:1.5;color:#2a2a2a;background-color:transparent;border:solid transparent;border-width:1px 0}.form-control-plaintext:focus{outline:0}.form-control-plaintext.form-control-lg,.form-control-plaintext.form-control-sm{padding-right:0;padding-left:0}.form-control-sm{min-height:calc(1.5em + .5rem + 2px);padding:.25rem .5rem;font-size:.875rem;border-radius:0}.form-control-sm::-webkit-file-upload-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-sm::file-selector-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-lg{min-height:calc(1.5em + 1rem + 2px);padding:.5rem 1rem;font-size:1.25rem;border-radius:0}.form-control-lg::-webkit-file-upload-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}.form-control-lg::file-selector-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}textarea.form-control{min-height:calc(1.5em + .75rem + 2px)}textarea.form-control-sm{min-height:calc(1.5em + .5rem + 2px)}textarea.form-control-lg{min-height:calc(1.5em + 1rem + 2px)}.form-control-color{width:3rem;height:calc(1.5em + .75rem + 2px);padding:.375rem}.form-control-color:not(:disabled):not([readonly]){cursor:pointer}.form-control-color::-moz-color-swatch{border:0!important;border-radius:0}.form-control-color::-webkit-color-swatch{border-radius:0}.form-control-color.form-control-sm{height:calc(1.5em + .5rem + 2px)}.form-control-color.form-control-lg{height:calc(1.5em + 1rem + 2px)}.form-select{display:block;width:100%;padding:.375rem 2.25rem .375rem .75rem;-moz-padding-start:calc(0.75rem - 3px);font-size:1rem;font-weight:400;line-height:1.5;color:#2a2a2a;background-color:#fff;background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23454545' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right .75rem center;background-size:16px 12px;border:1px solid #cdcdcd;border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none}.form-select:focus{border-color:#a2bdd5;outline:0;box-shadow:0 0 0 .25rem rgba(69,122,171,.25)}.form-select[multiple],.form-select[size]:not([size="1"]){padding-right:.75rem;background-image:none}.form-select:disabled{background-color:#e2e2e3}.form-select:-moz-focusring{color:transparent;text-shadow:0 0 0 #2a2a2a}.form-select-sm{padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:.875rem;border-radius:0}.form-select-lg{padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem;border-radius:0}.form-check{display:block;min-height:1.5rem;padding-left:1.5em;margin-bottom:.125rem}.form-check .form-check-input{float:left;margin-left:-1.5em}.form-check-reverse{padding-right:1.5em;padding-left:0;text-align:right}.form-check-reverse .form-check-input{float:right;margin-right:-1.5em;margin-left:0}.form-check-input{width:1em;height:1em;margin-top:.25em;vertical-align:top;background-color:#fff;background-repeat:no-repeat;background-position:center;background-size:contain;border:1px solid rgba(34,34,34,.25);-webkit-appearance:none;-moz-appearance:none;appearance:none;-webkit-print-color-adjust:exact;color-adjust:exact;print-color-adjust:exact}.form-check-input[type=checkbox]{border-radius:0}.form-check-input[type=radio]{border-radius:50%}.form-check-input:active{filter:brightness(90%)}.form-check-input:focus{border-color:#a2bdd5;outline:0;box-shadow:0 0 0 .25rem rgba(69,122,171,.25)}.form-check-input:checked{background-color:#457aab;border-color:#457aab}.form-check-input:checked[type=checkbox]{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23fff' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='m6 10 3 3 6-6'/%3e%3c/svg%3e")}.form-check-input:checked[type=radio]{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='2' fill='%23fff'/%3e%3c/svg%3e")}.form-check-input[type=checkbox]:indeterminate{background-color:#457aab;border-color:#457aab;background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23fff' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='M6 10h8'/%3e%3c/svg%3e")}.form-check-input:disabled{pointer-events:none;filter:none;opacity:.5}.form-check-input:disabled~.form-check-label,.form-check-input[disabled]~.form-check-label{cursor:default;opacity:.5}.form-switch{padding-left:2.5em}.form-switch .form-check-input{width:2em;margin-left:-2.5em;background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3crect width='6' height='6' x='-3' y='-3' fill='rgba%2834, 34, 34, 0.25%29'/%3e%3c/svg%3e");background-position:left center;border-radius:0}.form-switch .form-check-input:focus{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3crect width='6' height='6' x='-3' y='-3' fill='%23a2bdd5'/%3e%3c/svg%3e")}.form-switch .form-check-input:checked{background-position:right center;background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3crect width='6' height='6' x='-3' y='-3' fill='%23fff'/%3e%3c/svg%3e")}.form-switch.form-check-reverse{padding-right:2.5em;padding-left:0}.form-switch.form-check-reverse .form-check-input{margin-right:-2.5em;margin-left:0}.form-check-inline{display:inline-block;margin-right:1rem}.btn-check{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.btn-check:disabled+.btn,.btn-check[disabled]+.btn{pointer-events:none;filter:none;opacity:.65}.form-range{width:100%;height:1.5rem;padding:0;background-color:transparent;-webkit-appearance:none;-moz-appearance:none;appearance:none}.form-range:focus{outline:0}.form-range:focus::-webkit-slider-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(69,122,171,.25)}.form-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(69,122,171,.25)}.form-range::-moz-focus-outer{border:0}.form-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-.25rem;background-color:#457aab;border:0;border-radius:1rem;-webkit-appearance:none;appearance:none}.form-range::-webkit-slider-thumb:active{background-color:#c7d7e6}.form-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#d4d4d4;border-color:transparent;border-radius:1rem}.form-range::-moz-range-thumb{width:1rem;height:1rem;background-color:#457aab;border:0;border-radius:1rem;-moz-appearance:none;appearance:none}.form-range::-moz-range-thumb:active{background-color:#c7d7e6}.form-range::-moz-range-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#d4d4d4;border-color:transparent;border-radius:1rem}.form-range:disabled{pointer-events:none}.form-range:disabled::-webkit-slider-thumb{background-color:#abb3bb}.form-range:disabled::-moz-range-thumb{background-color:#abb3bb}.form-floating{position:relative}.form-floating>.form-control,.form-floating>.form-control-plaintext,.form-floating>.form-select{height:calc(3.5rem + 2px);line-height:1.25}.form-floating>label{position:absolute;top:0;left:0;width:100%;height:100%;padding:1rem .75rem;overflow:hidden;text-align:start;text-overflow:ellipsis;white-space:nowrap;pointer-events:none;border:1px solid transparent;transform-origin:0 0}.form-floating>.form-control,.form-floating>.form-control-plaintext{padding:1rem .75rem}.form-floating>.form-control-plaintext::-moz-placeholder,.form-floating>.form-control::-moz-placeholder{color:transparent}.form-floating>.form-control-plaintext::placeholder,.form-floating>.form-control::placeholder{color:transparent}.form-floating>.form-control-plaintext:not(:-moz-placeholder-shown),.form-floating>.form-control:not(:-moz-placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control-plaintext:focus,.form-floating>.form-control-plaintext:not(:placeholder-shown),.form-floating>.form-control:focus,.form-floating>.form-control:not(:placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control-plaintext:-webkit-autofill,.form-floating>.form-control:-webkit-autofill{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-select{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control:not(:-moz-placeholder-shown)~label{opacity:.65;transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control-plaintext~label,.form-floating>.form-control:focus~label,.form-floating>.form-control:not(:placeholder-shown)~label,.form-floating>.form-select~label{opacity:.65;transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control:-webkit-autofill~label{opacity:.65;transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control-plaintext~label{border-width:1px 0}.input-group{position:relative;display:flex;flex-wrap:wrap;align-items:stretch;width:100%}.input-group>.form-control,.input-group>.form-floating,.input-group>.form-select{position:relative;flex:1 1 auto;width:1%;min-width:0}.input-group>.form-control:focus,.input-group>.form-floating:focus-within,.input-group>.form-select:focus{z-index:5}.input-group .btn{position:relative;z-index:2}.input-group .btn:focus{z-index:5}.input-group-text{display:flex;align-items:center;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#2a2a2a;text-align:center;white-space:nowrap;background-color:#e2e2e3;border:1px solid #cdcdcd;border-radius:0}.input-group-lg>.btn,.input-group-lg>.form-control,.input-group-lg>.form-select,.input-group-lg>.input-group-text{padding:.5rem 1rem;font-size:1.25rem;border-radius:0}.input-group-sm>.btn,.input-group-sm>.form-control,.input-group-sm>.form-select,.input-group-sm>.input-group-text{padding:.25rem .5rem;font-size:.875rem;border-radius:0}.input-group-lg>.form-select,.input-group-sm>.form-select{padding-right:3rem}.input-group:not(.has-validation)>.dropdown-toggle:nth-last-child(n+3),.input-group:not(.has-validation)>.form-floating:not(:last-child)>.form-control,.input-group:not(.has-validation)>.form-floating:not(:last-child)>.form-select,.input-group:not(.has-validation)>:not(:last-child):not(.dropdown-toggle):not(.dropdown-menu):not(.form-floating){border-top-right-radius:0;border-bottom-right-radius:0}.input-group.has-validation>.dropdown-toggle:nth-last-child(n+4),.input-group.has-validation>.form-floating:nth-last-child(n+3)>.form-control,.input-group.has-validation>.form-floating:nth-last-child(n+3)>.form-select,.input-group.has-validation>:nth-last-child(n+3):not(.dropdown-toggle):not(.dropdown-menu):not(.form-floating){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>:not(:first-child):not(.dropdown-menu):not(.valid-tooltip):not(.valid-feedback):not(.invalid-tooltip):not(.invalid-feedback){margin-left:-1px;border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.form-floating:not(:first-child)>.form-control,.input-group>.form-floating:not(:first-child)>.form-select{border-top-left-radius:0;border-bottom-left-radius:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:#2a8256}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;color:#fff;background-color:rgba(42,130,86,.9);border-radius:0}.is-valid~.valid-feedback,.is-valid~.valid-tooltip,.was-validated :valid~.valid-feedback,.was-validated :valid~.valid-tooltip{display:block}.form-control.is-valid,.was-validated .form-control:valid{border-color:#2a8256;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%232a8256' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-valid:focus,.was-validated .form-control:valid:focus{border-color:#2a8256;box-shadow:0 0 0 .25rem rgba(42,130,86,.25)}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.form-select.is-valid,.was-validated .form-select:valid{border-color:#2a8256}.form-select.is-valid:not([multiple]):not([size]),.form-select.is-valid:not([multiple])[size="1"],.was-validated .form-select:valid:not([multiple]):not([size]),.was-validated .form-select:valid:not([multiple])[size="1"]{padding-right:4.125rem;background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23454545' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e"),url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%232a8256' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(.75em + .375rem) calc(.75em + .375rem)}.form-select.is-valid:focus,.was-validated .form-select:valid:focus{border-color:#2a8256;box-shadow:0 0 0 .25rem rgba(42,130,86,.25)}.form-control-color.is-valid,.was-validated .form-control-color:valid{width:calc(3rem + calc(1.5em + .75rem))}.form-check-input.is-valid,.was-validated .form-check-input:valid{border-color:#2a8256}.form-check-input.is-valid:checked,.was-validated .form-check-input:valid:checked{background-color:#2a8256}.form-check-input.is-valid:focus,.was-validated .form-check-input:valid:focus{box-shadow:0 0 0 .25rem rgba(42,130,86,.25)}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:#2a8256}.form-check-inline .form-check-input~.valid-feedback{margin-left:.5em}.input-group>.form-control:not(:focus).is-valid,.input-group>.form-floating:not(:focus-within).is-valid,.input-group>.form-select:not(:focus).is-valid,.was-validated .input-group>.form-control:not(:focus):valid,.was-validated .input-group>.form-floating:not(:focus-within):valid,.was-validated .input-group>.form-select:not(:focus):valid{z-index:3}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:#c42526}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;color:#fff;background-color:rgba(196,37,38,.9);border-radius:0}.is-invalid~.invalid-feedback,.is-invalid~.invalid-tooltip,.was-validated :invalid~.invalid-feedback,.was-validated :invalid~.invalid-tooltip{display:block}.form-control.is-invalid,.was-validated .form-control:invalid{border-color:#c42526;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23c42526'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23c42526' stroke='none'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-invalid:focus,.was-validated .form-control:invalid:focus{border-color:#c42526;box-shadow:0 0 0 .25rem rgba(196,37,38,.25)}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.form-select.is-invalid,.was-validated .form-select:invalid{border-color:#c42526}.form-select.is-invalid:not([multiple]):not([size]),.form-select.is-invalid:not([multiple])[size="1"],.was-validated .form-select:invalid:not([multiple]):not([size]),.was-validated .form-select:invalid:not([multiple])[size="1"]{padding-right:4.125rem;background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23454545' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e"),url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23c42526'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23c42526' stroke='none'/%3e%3c/svg%3e");background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(.75em + .375rem) calc(.75em + .375rem)}.form-select.is-invalid:focus,.was-validated .form-select:invalid:focus{border-color:#c42526;box-shadow:0 0 0 .25rem rgba(196,37,38,.25)}.form-control-color.is-invalid,.was-validated .form-control-color:invalid{width:calc(3rem + calc(1.5em + .75rem))}.form-check-input.is-invalid,.was-validated .form-check-input:invalid{border-color:#c42526}.form-check-input.is-invalid:checked,.was-validated .form-check-input:invalid:checked{background-color:#c42526}.form-check-input.is-invalid:focus,.was-validated .form-check-input:invalid:focus{box-shadow:0 0 0 .25rem rgba(196,37,38,.25)}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:#c42526}.form-check-inline .form-check-input~.invalid-feedback{margin-left:.5em}.input-group>.form-control:not(:focus).is-invalid,.input-group>.form-floating:not(:focus-within).is-invalid,.input-group>.form-select:not(:focus).is-invalid,.was-validated .input-group>.form-control:not(:focus):invalid,.was-validated .input-group>.form-floating:not(:focus-within):invalid,.was-validated .input-group>.form-select:not(:focus):invalid{z-index:4}.btn{--bs-btn-padding-x:0.75rem;--bs-btn-padding-y:0.375rem;--bs-btn-font-family: ;--bs-btn-font-size:1rem;--bs-btn-font-weight:400;--bs-btn-line-height:1.5;--bs-btn-color:#2a2a2a;--bs-btn-bg:transparent;--bs-btn-border-width:1px;--bs-btn-border-color:transparent;--bs-btn-border-radius:0;--bs-btn-hover-border-color:transparent;--bs-btn-box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.15),0 1px 1px rgba(34, 34, 34, 0.075);--bs-btn-disabled-opacity:0.65;--bs-btn-focus-box-shadow:0 0 0 0.25rem rgba(var(--bs-btn-focus-shadow-rgb), .5);display:inline-block;padding:var(--bs-btn-padding-y) var(--bs-btn-padding-x);font-family:var(--bs-btn-font-family);font-size:var(--bs-btn-font-size);font-weight:var(--bs-btn-font-weight);line-height:var(--bs-btn-line-height);color:var(--bs-btn-color);text-align:center;text-decoration:none;vertical-align:middle;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none;border:var(--bs-btn-border-width) solid var(--bs-btn-border-color);border-radius:var(--bs-btn-border-radius);background-color:var(--bs-btn-bg)}.btn:hover{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color)}.btn-check+.btn:hover{color:var(--bs-btn-color);background-color:var(--bs-btn-bg);border-color:var(--bs-btn-border-color)}.btn:focus-visible{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:focus-visible+.btn{border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:checked+.btn,.btn.active,.btn.show,.btn:first-child:active,:not(.btn-check)+.btn:active{color:var(--bs-btn-active-color);background-color:var(--bs-btn-active-bg);border-color:var(--bs-btn-active-border-color)}.btn-check:checked+.btn:focus-visible,.btn.active:focus-visible,.btn.show:focus-visible,.btn:first-child:active:focus-visible,:not(.btn-check)+.btn:active:focus-visible{box-shadow:var(--bs-btn-focus-box-shadow)}.btn.disabled,.btn:disabled,fieldset:disabled .btn{color:var(--bs-btn-disabled-color);pointer-events:none;background-color:var(--bs-btn-disabled-bg);border-color:var(--bs-btn-disabled-border-color);opacity:var(--bs-btn-disabled-opacity)}.btn-primary{--bs-btn-color:#fff;--bs-btn-bg:#457aab;--bs-btn-border-color:#457aab;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#3b6891;--bs-btn-hover-border-color:#376289;--bs-btn-focus-shadow-rgb:97,142,184;--bs-btn-active-color:#fff;--bs-btn-active-bg:#376289;--bs-btn-active-border-color:#345c80;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#457aab;--bs-btn-disabled-border-color:#457aab}.btn-secondary{--bs-btn-color:#fff;--bs-btn-bg:#5d6d75;--bs-btn-border-color:#5d6d75;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#4f5d63;--bs-btn-hover-border-color:#4a575e;--bs-btn-focus-shadow-rgb:117,131,138;--bs-btn-active-color:#fff;--bs-btn-active-bg:#4a575e;--bs-btn-active-border-color:#465258;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#5d6d75;--bs-btn-disabled-border-color:#5d6d75}.btn-success{--bs-btn-color:#fff;--bs-btn-bg:#2a8256;--bs-btn-border-color:#2a8256;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#246f49;--bs-btn-hover-border-color:#226845;--bs-btn-focus-shadow-rgb:74,149,111;--bs-btn-active-color:#fff;--bs-btn-active-bg:#226845;--bs-btn-active-border-color:#206241;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#2a8256;--bs-btn-disabled-border-color:#2a8256}.btn-info{--bs-btn-color:#fff;--bs-btn-bg:#0068b3;--bs-btn-border-color:#0068b3;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#005898;--bs-btn-hover-border-color:#00538f;--bs-btn-focus-shadow-rgb:38,127,190;--bs-btn-active-color:#fff;--bs-btn-active-bg:#00538f;--bs-btn-active-border-color:#004e86;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#0068b3;--bs-btn-disabled-border-color:#0068b3}.btn-warning{--bs-btn-color:#222;--bs-btn-bg:#ffc107;--bs-btn-border-color:#ffc107;--bs-btn-hover-color:#222;--bs-btn-hover-bg:#ffca2c;--bs-btn-hover-border-color:#ffc720;--bs-btn-focus-shadow-rgb:222,169,11;--bs-btn-active-color:#222;--bs-btn-active-bg:#ffcd39;--bs-btn-active-border-color:#ffc720;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#222;--bs-btn-disabled-bg:#ffc107;--bs-btn-disabled-border-color:#ffc107}.btn-danger{--bs-btn-color:#fff;--bs-btn-bg:#c42526;--bs-btn-border-color:#c42526;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#a71f20;--bs-btn-hover-border-color:#9d1e1e;--bs-btn-focus-shadow-rgb:205,70,71;--bs-btn-active-color:#fff;--bs-btn-active-bg:#9d1e1e;--bs-btn-active-border-color:#931c1d;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#c42526;--bs-btn-disabled-border-color:#c42526}.btn-light{--bs-btn-color:#222;--bs-btn-bg:#f2f2f2;--bs-btn-border-color:#f2f2f2;--bs-btn-hover-color:#222;--bs-btn-hover-bg:#cecece;--bs-btn-hover-border-color:#c2c2c2;--bs-btn-focus-shadow-rgb:211,211,211;--bs-btn-active-color:#222;--bs-btn-active-bg:#c2c2c2;--bs-btn-active-border-color:#b6b6b6;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#222;--bs-btn-disabled-bg:#f2f2f2;--bs-btn-disabled-border-color:#f2f2f2}.btn-dark{--bs-btn-color:#fff;--bs-btn-bg:#2a2a2a;--bs-btn-border-color:#2a2a2a;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#4a4a4a;--bs-btn-hover-border-color:#3f3f3f;--bs-btn-focus-shadow-rgb:74,74,74;--bs-btn-active-color:#fff;--bs-btn-active-bg:#555555;--bs-btn-active-border-color:#3f3f3f;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#2a2a2a;--bs-btn-disabled-border-color:#2a2a2a}.btn-outline-primary{--bs-btn-color:#457aab;--bs-btn-border-color:#457aab;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#457aab;--bs-btn-hover-border-color:#457aab;--bs-btn-focus-shadow-rgb:69,122,171;--bs-btn-active-color:#fff;--bs-btn-active-bg:#457aab;--bs-btn-active-border-color:#457aab;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#457aab;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#457aab;--bs-gradient:none}.btn-outline-secondary{--bs-btn-color:#5d6d75;--bs-btn-border-color:#5d6d75;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#5d6d75;--bs-btn-hover-border-color:#5d6d75;--bs-btn-focus-shadow-rgb:93,109,117;--bs-btn-active-color:#fff;--bs-btn-active-bg:#5d6d75;--bs-btn-active-border-color:#5d6d75;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#5d6d75;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#5d6d75;--bs-gradient:none}.btn-outline-success{--bs-btn-color:#2a8256;--bs-btn-border-color:#2a8256;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#2a8256;--bs-btn-hover-border-color:#2a8256;--bs-btn-focus-shadow-rgb:42,130,86;--bs-btn-active-color:#fff;--bs-btn-active-bg:#2a8256;--bs-btn-active-border-color:#2a8256;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#2a8256;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#2a8256;--bs-gradient:none}.btn-outline-info{--bs-btn-color:#0068b3;--bs-btn-border-color:#0068b3;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#0068b3;--bs-btn-hover-border-color:#0068b3;--bs-btn-focus-shadow-rgb:0,104,179;--bs-btn-active-color:#fff;--bs-btn-active-bg:#0068b3;--bs-btn-active-border-color:#0068b3;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#0068b3;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#0068b3;--bs-gradient:none}.btn-outline-warning{--bs-btn-color:#ffc107;--bs-btn-border-color:#ffc107;--bs-btn-hover-color:#222;--bs-btn-hover-bg:#ffc107;--bs-btn-hover-border-color:#ffc107;--bs-btn-focus-shadow-rgb:255,193,7;--bs-btn-active-color:#222;--bs-btn-active-bg:#ffc107;--bs-btn-active-border-color:#ffc107;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#ffc107;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#ffc107;--bs-gradient:none}.btn-outline-danger{--bs-btn-color:#c42526;--bs-btn-border-color:#c42526;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#c42526;--bs-btn-hover-border-color:#c42526;--bs-btn-focus-shadow-rgb:196,37,38;--bs-btn-active-color:#fff;--bs-btn-active-bg:#c42526;--bs-btn-active-border-color:#c42526;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#c42526;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#c42526;--bs-gradient:none}.btn-outline-light{--bs-btn-color:#f2f2f2;--bs-btn-border-color:#f2f2f2;--bs-btn-hover-color:#222;--bs-btn-hover-bg:#f2f2f2;--bs-btn-hover-border-color:#f2f2f2;--bs-btn-focus-shadow-rgb:242,242,242;--bs-btn-active-color:#222;--bs-btn-active-bg:#f2f2f2;--bs-btn-active-border-color:#f2f2f2;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#f2f2f2;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#f2f2f2;--bs-gradient:none}.btn-outline-dark{--bs-btn-color:#2a2a2a;--bs-btn-border-color:#2a2a2a;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#2a2a2a;--bs-btn-hover-border-color:#2a2a2a;--bs-btn-focus-shadow-rgb:42,42,42;--bs-btn-active-color:#fff;--bs-btn-active-bg:#2a2a2a;--bs-btn-active-border-color:#2a2a2a;--bs-btn-active-shadow:inset 0 3px 5px rgba(34, 34, 34, 0.125);--bs-btn-disabled-color:#2a2a2a;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#2a2a2a;--bs-gradient:none}.btn-link{--bs-btn-font-weight:400;--bs-btn-color:var(--bs-link-color);--bs-btn-bg:transparent;--bs-btn-border-color:transparent;--bs-btn-hover-color:var(--bs-link-hover-color);--bs-btn-hover-border-color:transparent;--bs-btn-active-color:var(--bs-link-hover-color);--bs-btn-active-border-color:transparent;--bs-btn-disabled-color:#a1a3a9;--bs-btn-disabled-border-color:transparent;--bs-btn-box-shadow:none;--bs-btn-focus-shadow-rgb:97,142,184;text-decoration:underline}.btn-link:focus-visible{color:var(--bs-btn-color)}.btn-link:hover{color:var(--bs-btn-hover-color)}.btn-group-lg>.btn,.btn-lg{--bs-btn-padding-y:0.5rem;--bs-btn-padding-x:1rem;--bs-btn-font-size:1.25rem;--bs-btn-border-radius:0}.btn-group-sm>.btn,.btn-sm{--bs-btn-padding-y:0.25rem;--bs-btn-padding-x:0.5rem;--bs-btn-font-size:0.875rem;--bs-btn-border-radius:0}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{height:0;overflow:hidden}.collapsing.collapse-horizontal{width:0;height:auto}.dropdown,.dropdown-center,.dropend,.dropstart,.dropup,.dropup-center{position:relative}.dropdown-toggle{white-space:nowrap}.dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid transparent;border-bottom:0;border-left:.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{--bs-dropdown-zindex:1000;--bs-dropdown-min-width:10rem;--bs-dropdown-padding-x:0;--bs-dropdown-padding-y:0.5rem;--bs-dropdown-spacer:0.125rem;--bs-dropdown-font-size:1rem;--bs-dropdown-color:#2a2a2a;--bs-dropdown-bg:#fff;--bs-dropdown-border-color:var(--bs-border-color-translucent);--bs-dropdown-border-radius:0;--bs-dropdown-border-width:1px;--bs-dropdown-inner-border-radius:-1px;--bs-dropdown-divider-bg:var(--bs-border-color-translucent);--bs-dropdown-divider-margin-y:0.5rem;--bs-dropdown-box-shadow:0 0.5rem 1rem rgba(34, 34, 34, 0.15);--bs-dropdown-link-color:#2a2a2a;--bs-dropdown-link-hover-color:#262626;--bs-dropdown-link-hover-bg:#e2e2e3;--bs-dropdown-link-active-color:#fff;--bs-dropdown-link-active-bg:#457aab;--bs-dropdown-link-disabled-color:#abb3bb;--bs-dropdown-item-padding-x:1rem;--bs-dropdown-item-padding-y:0.25rem;--bs-dropdown-header-color:#a1a3a9;--bs-dropdown-header-padding-x:1rem;--bs-dropdown-header-padding-y:0.5rem;position:absolute;z-index:var(--bs-dropdown-zindex);display:none;min-width:var(--bs-dropdown-min-width);padding:var(--bs-dropdown-padding-y) var(--bs-dropdown-padding-x);margin:0;font-size:var(--bs-dropdown-font-size);color:var(--bs-dropdown-color);text-align:left;list-style:none;background-color:var(--bs-dropdown-bg);background-clip:padding-box;border:var(--bs-dropdown-border-width) solid var(--bs-dropdown-border-color);border-radius:var(--bs-dropdown-border-radius)}.dropdown-menu[data-bs-popper]{top:100%;left:0;margin-top:var(--bs-dropdown-spacer)}.dropdown-menu-start{--bs-position:start}.dropdown-menu-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-end{--bs-position:end}.dropdown-menu-end[data-bs-popper]{right:0;left:auto}@media (min-width:576px){.dropdown-menu-sm-start{--bs-position:start}.dropdown-menu-sm-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-sm-end{--bs-position:end}.dropdown-menu-sm-end[data-bs-popper]{right:0;left:auto}}@media (min-width:768px){.dropdown-menu-md-start{--bs-position:start}.dropdown-menu-md-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-md-end{--bs-position:end}.dropdown-menu-md-end[data-bs-popper]{right:0;left:auto}}@media (min-width:992px){.dropdown-menu-lg-start{--bs-position:start}.dropdown-menu-lg-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-lg-end{--bs-position:end}.dropdown-menu-lg-end[data-bs-popper]{right:0;left:auto}}@media (min-width:1200px){.dropdown-menu-xl-start{--bs-position:start}.dropdown-menu-xl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xl-end{--bs-position:end}.dropdown-menu-xl-end[data-bs-popper]{right:0;left:auto}}@media (min-width:1400px){.dropdown-menu-xxl-start{--bs-position:start}.dropdown-menu-xxl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xxl-end{--bs-position:end}.dropdown-menu-xxl-end[data-bs-popper]{right:0;left:auto}}.dropup .dropdown-menu[data-bs-popper]{top:auto;bottom:100%;margin-top:0;margin-bottom:var(--bs-dropdown-spacer)}.dropup .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid transparent;border-bottom:.3em solid;border-left:.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-menu[data-bs-popper]{top:0;right:auto;left:100%;margin-top:0;margin-left:var(--bs-dropdown-spacer)}.dropend .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:0;border-bottom:.3em solid transparent;border-left:.3em solid}.dropend .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-toggle::after{vertical-align:0}.dropstart .dropdown-menu[data-bs-popper]{top:0;right:100%;left:auto;margin-top:0;margin-right:var(--bs-dropdown-spacer)}.dropstart .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:""}.dropstart .dropdown-toggle::after{display:none}.dropstart .dropdown-toggle::before{display:inline-block;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:.3em solid;border-bottom:.3em solid transparent}.dropstart .dropdown-toggle:empty::after{margin-left:0}.dropstart .dropdown-toggle::before{vertical-align:0}.dropdown-divider{height:0;margin:var(--bs-dropdown-divider-margin-y) 0;overflow:hidden;border-top:1px solid var(--bs-dropdown-divider-bg);opacity:1}.dropdown-item{display:block;width:100%;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);clear:both;font-weight:400;color:var(--bs-dropdown-link-color);text-align:inherit;text-decoration:none;white-space:nowrap;background-color:transparent;border:0}.dropdown-item:focus,.dropdown-item:hover{color:var(--bs-dropdown-link-hover-color);background-color:var(--bs-dropdown-link-hover-bg)}.dropdown-item.active,.dropdown-item:active{color:var(--bs-dropdown-link-active-color);text-decoration:none;background-color:var(--bs-dropdown-link-active-bg)}.dropdown-item.disabled,.dropdown-item:disabled{color:var(--bs-dropdown-link-disabled-color);pointer-events:none;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:var(--bs-dropdown-header-padding-y) var(--bs-dropdown-header-padding-x);margin-bottom:0;font-size:.875rem;color:var(--bs-dropdown-header-color);white-space:nowrap}.dropdown-item-text{display:block;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);color:var(--bs-dropdown-link-color)}.dropdown-menu-dark{--bs-dropdown-color:#d4d4d4;--bs-dropdown-bg:#454545;--bs-dropdown-border-color:var(--bs-border-color-translucent);--bs-dropdown-box-shadow: ;--bs-dropdown-link-color:#d4d4d4;--bs-dropdown-link-hover-color:#fff;--bs-dropdown-divider-bg:var(--bs-border-color-translucent);--bs-dropdown-link-hover-bg:rgba(255, 255, 255, 0.15);--bs-dropdown-link-active-color:#fff;--bs-dropdown-link-active-bg:#457aab;--bs-dropdown-link-disabled-color:#abb3bb;--bs-dropdown-header-color:#abb3bb}.btn-group,.btn-group-vertical{position:relative;display:inline-flex;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;flex:1 1 auto}.btn-group-vertical>.btn-check:checked+.btn,.btn-group-vertical>.btn-check:focus+.btn,.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group-vertical>.btn:hover,.btn-group>.btn-check:checked+.btn,.btn-group>.btn-check:focus+.btn,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus,.btn-group>.btn:hover{z-index:1}.btn-toolbar{display:flex;flex-wrap:wrap;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group{border-radius:0}.btn-group>.btn-group:not(:first-child),.btn-group>:not(.btn-check:first-child)+.btn{margin-left:-1px}.btn-group>.btn-group:not(:last-child)>.btn,.btn-group>.btn.dropdown-toggle-split:first-child,.btn-group>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:not(:first-child)>.btn,.btn-group>.btn:nth-child(n+3),.btn-group>:not(.btn-check)+.btn{border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropend .dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after{margin-left:0}.dropstart .dropdown-toggle-split::before{margin-right:0}.btn-group-sm>.btn+.dropdown-toggle-split,.btn-sm+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-group-lg>.btn+.dropdown-toggle-split,.btn-lg+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{flex-direction:column;align-items:flex-start;justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn-group:not(:first-child),.btn-group-vertical>.btn:not(:first-child){margin-top:-1px}.btn-group-vertical>.btn-group:not(:last-child)>.btn,.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle){border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:not(:first-child)>.btn,.btn-group-vertical>.btn~.btn{border-top-left-radius:0;border-top-right-radius:0}.nav{--bs-nav-link-padding-x:1rem;--bs-nav-link-padding-y:0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color:var(--bs-link-color);--bs-nav-link-hover-color:var(--bs-link-hover-color);--bs-nav-link-disabled-color:#a1a3a9;display:flex;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:var(--bs-nav-link-padding-y) var(--bs-nav-link-padding-x);font-size:var(--bs-nav-link-font-size);font-weight:var(--bs-nav-link-font-weight);color:var(--bs-nav-link-color);text-decoration:none}.nav-link:focus,.nav-link:hover{color:var(--bs-nav-link-hover-color)}.nav-link.disabled{color:var(--bs-nav-link-disabled-color);pointer-events:none;cursor:default}.nav-tabs{--bs-nav-tabs-border-width:1px;--bs-nav-tabs-border-color:#d4d4d4;--bs-nav-tabs-border-radius:0;--bs-nav-tabs-link-hover-border-color:#e2e2e3 #e2e2e3 #d4d4d4;--bs-nav-tabs-link-active-color:#5d6d75;--bs-nav-tabs-link-active-bg:#fff;--bs-nav-tabs-link-active-border-color:#d4d4d4 #d4d4d4 #fff;border-bottom:var(--bs-nav-tabs-border-width) solid var(--bs-nav-tabs-border-color)}.nav-tabs .nav-link{margin-bottom:calc(-1 * var(--bs-nav-tabs-border-width));background:0 0;border:var(--bs-nav-tabs-border-width) solid transparent;border-top-left-radius:var(--bs-nav-tabs-border-radius);border-top-right-radius:var(--bs-nav-tabs-border-radius)}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{isolation:isolate;border-color:var(--bs-nav-tabs-link-hover-border-color)}.nav-tabs .nav-link.disabled,.nav-tabs .nav-link:disabled{color:var(--bs-nav-link-disabled-color);background-color:transparent;border-color:transparent}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:var(--bs-nav-tabs-link-active-color);background-color:var(--bs-nav-tabs-link-active-bg);border-color:var(--bs-nav-tabs-link-active-border-color)}.nav-tabs .dropdown-menu{margin-top:calc(-1 * var(--bs-nav-tabs-border-width));border-top-left-radius:0;border-top-right-radius:0}.nav-pills{--bs-nav-pills-border-radius:0;--bs-nav-pills-link-active-color:#fff;--bs-nav-pills-link-active-bg:#457aab}.nav-pills .nav-link{background:0 0;border:0;border-radius:var(--bs-nav-pills-border-radius)}.nav-pills .nav-link:disabled{color:var(--bs-nav-link-disabled-color);background-color:transparent;border-color:transparent}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:var(--bs-nav-pills-link-active-color);background-color:var(--bs-nav-pills-link-active-bg)}.nav-fill .nav-item,.nav-fill>.nav-link{flex:1 1 auto;text-align:center}.nav-justified .nav-item,.nav-justified>.nav-link{flex-basis:0;flex-grow:1;text-align:center}.nav-fill .nav-item .nav-link,.nav-justified .nav-item .nav-link{width:100%}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{--bs-navbar-padding-x:0;--bs-navbar-padding-y:0.5rem;--bs-navbar-color:rgba(34, 34, 34, 0.55);--bs-navbar-hover-color:rgba(34, 34, 34, 0.7);--bs-navbar-disabled-color:rgba(34, 34, 34, 0.3);--bs-navbar-active-color:rgba(34, 34, 34, 0.9);--bs-navbar-brand-padding-y:0.3125rem;--bs-navbar-brand-margin-end:1rem;--bs-navbar-brand-font-size:1.25rem;--bs-navbar-brand-color:rgba(34, 34, 34, 0.9);--bs-navbar-brand-hover-color:rgba(34, 34, 34, 0.9);--bs-navbar-nav-link-padding-x:0.5rem;--bs-navbar-toggler-padding-y:0.25rem;--bs-navbar-toggler-padding-x:0.75rem;--bs-navbar-toggler-font-size:1.25rem;--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%2834, 34, 34, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e");--bs-navbar-toggler-border-color:rgba(34, 34, 34, 0.1);--bs-navbar-toggler-border-radius:0;--bs-navbar-toggler-focus-width:0.25rem;--bs-navbar-toggler-transition:box-shadow 0.15s ease-in-out;position:relative;display:flex;flex-wrap:wrap;align-items:center;justify-content:space-between;padding:var(--bs-navbar-padding-y) var(--bs-navbar-padding-x)}.navbar>.container,.navbar>.container-fluid,.navbar>.container-lg,.navbar>.container-md,.navbar>.container-sm,.navbar>.container-xl,.navbar>.container-xxl{display:flex;flex-wrap:inherit;align-items:center;justify-content:space-between}.navbar-brand{padding-top:var(--bs-navbar-brand-padding-y);padding-bottom:var(--bs-navbar-brand-padding-y);margin-right:var(--bs-navbar-brand-margin-end);font-size:var(--bs-navbar-brand-font-size);color:var(--bs-navbar-brand-color);text-decoration:none;white-space:nowrap}.navbar-brand:focus,.navbar-brand:hover{color:var(--bs-navbar-brand-hover-color)}.navbar-nav{--bs-nav-link-padding-x:0;--bs-nav-link-padding-y:0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color:var(--bs-navbar-color);--bs-nav-link-hover-color:var(--bs-navbar-hover-color);--bs-nav-link-disabled-color:var(--bs-navbar-disabled-color);display:flex;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link.active,.navbar-nav .show>.nav-link{color:var(--bs-navbar-active-color)}.navbar-nav .dropdown-menu{position:static}.navbar-text{padding-top:.5rem;padding-bottom:.5rem;color:var(--bs-navbar-color)}.navbar-text a,.navbar-text a:focus,.navbar-text a:hover{color:var(--bs-navbar-active-color)}.navbar-collapse{flex-basis:100%;flex-grow:1;align-items:center}.navbar-toggler{padding:var(--bs-navbar-toggler-padding-y) var(--bs-navbar-toggler-padding-x);font-size:var(--bs-navbar-toggler-font-size);line-height:1;color:var(--bs-navbar-color);background-color:transparent;border:var(--bs-border-width) solid var(--bs-navbar-toggler-border-color);border-radius:var(--bs-navbar-toggler-border-radius)}.navbar-toggler:hover{text-decoration:none}.navbar-toggler:focus{text-decoration:none;outline:0;box-shadow:0 0 0 var(--bs-navbar-toggler-focus-width)}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;background-image:var(--bs-navbar-toggler-icon-bg);background-repeat:no-repeat;background-position:center;background-size:100%}.navbar-nav-scroll{max-height:var(--bs-scroll-height,75vh);overflow-y:auto}@media (min-width:576px){.navbar-expand-sm{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-sm .navbar-nav{flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-sm .navbar-nav-scroll{overflow:visible}.navbar-expand-sm .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}.navbar-expand-sm .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important}.navbar-expand-sm .offcanvas .offcanvas-header{display:none}.navbar-expand-sm .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:768px){.navbar-expand-md{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-md .navbar-nav{flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-md .navbar-nav-scroll{overflow:visible}.navbar-expand-md .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}.navbar-expand-md .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important}.navbar-expand-md .offcanvas .offcanvas-header{display:none}.navbar-expand-md .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:992px){.navbar-expand-lg{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-lg .navbar-nav{flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-lg .navbar-nav-scroll{overflow:visible}.navbar-expand-lg .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}.navbar-expand-lg .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important}.navbar-expand-lg .offcanvas .offcanvas-header{display:none}.navbar-expand-lg .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:1200px){.navbar-expand-xl{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-xl .navbar-nav{flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xl .navbar-nav-scroll{overflow:visible}.navbar-expand-xl .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}.navbar-expand-xl .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important}.navbar-expand-xl .offcanvas .offcanvas-header{display:none}.navbar-expand-xl .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:1400px){.navbar-expand-xxl{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-xxl .navbar-nav{flex-direction:row}.navbar-expand-xxl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xxl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xxl .navbar-nav-scroll{overflow:visible}.navbar-expand-xxl .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-xxl .navbar-toggler{display:none}.navbar-expand-xxl .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important}.navbar-expand-xxl .offcanvas .offcanvas-header{display:none}.navbar-expand-xxl .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}.navbar-expand{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand .navbar-nav{flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand .navbar-nav-scroll{overflow:visible}.navbar-expand .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-expand .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important}.navbar-expand .offcanvas .offcanvas-header{display:none}.navbar-expand .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}.navbar-dark{--bs-navbar-color:rgba(255, 255, 255, 0.55);--bs-navbar-hover-color:rgba(255, 255, 255, 0.75);--bs-navbar-disabled-color:rgba(255, 255, 255, 0.25);--bs-navbar-active-color:#fff;--bs-navbar-brand-color:#fff;--bs-navbar-brand-hover-color:#fff;--bs-navbar-toggler-border-color:rgba(255, 255, 255, 0.1);--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.card{--bs-card-spacer-y:1rem;--bs-card-spacer-x:1rem;--bs-card-title-spacer-y:0.5rem;--bs-card-border-width:1px;--bs-card-border-color:var(--bs-border-color-translucent);--bs-card-border-radius:0;--bs-card-box-shadow: ;--bs-card-inner-border-radius:-1px;--bs-card-cap-padding-y:0.5rem;--bs-card-cap-padding-x:1rem;--bs-card-cap-bg:rgba(34, 34, 34, 0.03);--bs-card-cap-color: ;--bs-card-height: ;--bs-card-color: ;--bs-card-bg:#fff;--bs-card-img-overlay-padding:1rem;--bs-card-group-margin:0.75rem;position:relative;display:flex;flex-direction:column;min-width:0;height:var(--bs-card-height);word-wrap:break-word;background-color:var(--bs-card-bg);background-clip:border-box;border:var(--bs-card-border-width) solid var(--bs-card-border-color);border-radius:var(--bs-card-border-radius)}.card>hr{margin-right:0;margin-left:0}.card>.list-group{border-top:inherit;border-bottom:inherit}.card>.list-group:first-child{border-top-width:0;border-top-left-radius:var(--bs-card-inner-border-radius);border-top-right-radius:var(--bs-card-inner-border-radius)}.card>.list-group:last-child{border-bottom-width:0;border-bottom-right-radius:var(--bs-card-inner-border-radius);border-bottom-left-radius:var(--bs-card-inner-border-radius)}.card>.card-header+.list-group,.card>.list-group+.card-footer{border-top:0}.card-body{flex:1 1 auto;padding:var(--bs-card-spacer-y) var(--bs-card-spacer-x);color:var(--bs-card-color)}.card-title{margin-bottom:var(--bs-card-title-spacer-y)}.card-subtitle{margin-top:calc(-.5 * var(--bs-card-title-spacer-y));margin-bottom:0}.card-text:last-child{margin-bottom:0}.card-link+.card-link{margin-left:var(--bs-card-spacer-x)}.card-header{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);margin-bottom:0;color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-bottom:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-header:first-child{border-radius:var(--bs-card-inner-border-radius) var(--bs-card-inner-border-radius) 0 0}.card-footer{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-top:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-footer:last-child{border-radius:0 0 var(--bs-card-inner-border-radius) var(--bs-card-inner-border-radius)}.card-header-tabs{margin-right:calc(-.5 * var(--bs-card-cap-padding-x));margin-bottom:calc(-1 * var(--bs-card-cap-padding-y));margin-left:calc(-.5 * var(--bs-card-cap-padding-x));border-bottom:0}.card-header-tabs .nav-link.active{background-color:var(--bs-card-bg);border-bottom-color:var(--bs-card-bg)}.card-header-pills{margin-right:calc(-.5 * var(--bs-card-cap-padding-x));margin-left:calc(-.5 * var(--bs-card-cap-padding-x))}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:var(--bs-card-img-overlay-padding);border-radius:var(--bs-card-inner-border-radius)}.card-img,.card-img-bottom,.card-img-top{width:100%}.card-img,.card-img-top{border-top-left-radius:var(--bs-card-inner-border-radius);border-top-right-radius:var(--bs-card-inner-border-radius)}.card-img,.card-img-bottom{border-bottom-right-radius:var(--bs-card-inner-border-radius);border-bottom-left-radius:var(--bs-card-inner-border-radius)}.card-group>.card{margin-bottom:var(--bs-card-group-margin)}@media (min-width:576px){.card-group{display:flex;flex-flow:row wrap}.card-group>.card{flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:not(:last-child) .card-header,.card-group>.card:not(:last-child) .card-img-top{border-top-right-radius:0}.card-group>.card:not(:last-child) .card-footer,.card-group>.card:not(:last-child) .card-img-bottom{border-bottom-right-radius:0}.card-group>.card:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:not(:first-child) .card-header,.card-group>.card:not(:first-child) .card-img-top{border-top-left-radius:0}.card-group>.card:not(:first-child) .card-footer,.card-group>.card:not(:first-child) .card-img-bottom{border-bottom-left-radius:0}}.accordion{--bs-accordion-color:#2a2a2a;--bs-accordion-bg:#fff;--bs-accordion-transition:color 0.15s ease-in-out,background-color 0.15s ease-in-out,border-color 0.15s ease-in-out,box-shadow 0.15s ease-in-out,border-radius 0.15s ease;--bs-accordion-border-color:var(--bs-border-color);--bs-accordion-border-width:1px;--bs-accordion-border-radius:0;--bs-accordion-inner-border-radius:-1px;--bs-accordion-btn-padding-x:1.25rem;--bs-accordion-btn-padding-y:1rem;--bs-accordion-btn-color:#2a2a2a;--bs-accordion-btn-bg:var(--bs-accordion-bg);--bs-accordion-btn-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%232a2a2a'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-icon-width:1.25rem;--bs-accordion-btn-icon-transform:rotate(-180deg);--bs-accordion-btn-icon-transition:transform 0.2s ease-in-out;--bs-accordion-btn-active-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='black'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-focus-border-color:#a2bdd5;--bs-accordion-btn-focus-box-shadow:0 0 0 0.25rem rgba(69, 122, 171, 0.25);--bs-accordion-body-padding-x:1.25rem;--bs-accordion-body-padding-y:1rem;--bs-accordion-active-color:black;--bs-accordion-active-bg:#ecf2f7}.accordion-button{position:relative;display:flex;align-items:center;width:100%;padding:var(--bs-accordion-btn-padding-y) var(--bs-accordion-btn-padding-x);font-size:1rem;color:var(--bs-accordion-btn-color);text-align:left;background-color:var(--bs-accordion-btn-bg);border:0;border-radius:0;overflow-anchor:none}.accordion-button:not(.collapsed){color:var(--bs-accordion-active-color);background-color:var(--bs-accordion-active-bg);box-shadow:inset 0 calc(-1 * var(--bs-accordion-border-width)) 0 var(--bs-accordion-border-color)}.accordion-button:not(.collapsed)::after{background-image:var(--bs-accordion-btn-active-icon);transform:var(--bs-accordion-btn-icon-transform)}.accordion-button::after{flex-shrink:0;width:var(--bs-accordion-btn-icon-width);height:var(--bs-accordion-btn-icon-width);margin-left:auto;content:"";background-image:var(--bs-accordion-btn-icon);background-repeat:no-repeat;background-size:var(--bs-accordion-btn-icon-width)}.accordion-button:hover{z-index:2}.accordion-button:focus{z-index:3;border-color:var(--bs-accordion-btn-focus-border-color);outline:0;box-shadow:var(--bs-accordion-btn-focus-box-shadow)}.accordion-header{margin-bottom:0}.accordion-item{color:var(--bs-accordion-color);background-color:var(--bs-accordion-bg);border:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.accordion-item:first-of-type{border-top-left-radius:var(--bs-accordion-border-radius);border-top-right-radius:var(--bs-accordion-border-radius)}.accordion-item:first-of-type .accordion-button{border-top-left-radius:var(--bs-accordion-inner-border-radius);border-top-right-radius:var(--bs-accordion-inner-border-radius)}.accordion-item:not(:first-of-type){border-top:0}.accordion-item:last-of-type{border-bottom-right-radius:var(--bs-accordion-border-radius);border-bottom-left-radius:var(--bs-accordion-border-radius)}.accordion-item:last-of-type .accordion-button.collapsed{border-bottom-right-radius:var(--bs-accordion-inner-border-radius);border-bottom-left-radius:var(--bs-accordion-inner-border-radius)}.accordion-item:last-of-type .accordion-collapse{border-bottom-right-radius:var(--bs-accordion-border-radius);border-bottom-left-radius:var(--bs-accordion-border-radius)}.accordion-body{padding:var(--bs-accordion-body-padding-y) var(--bs-accordion-body-padding-x)}.accordion-flush .accordion-collapse{border-width:0}.accordion-flush .accordion-item{border-right:0;border-left:0;border-radius:0}.accordion-flush .accordion-item:first-child{border-top:0}.accordion-flush .accordion-item:last-child{border-bottom:0}.accordion-flush .accordion-item .accordion-button,.accordion-flush .accordion-item .accordion-button.collapsed{border-radius:0}.breadcrumb{--bs-breadcrumb-padding-x:0;--bs-breadcrumb-padding-y:0;--bs-breadcrumb-margin-bottom:1rem;--bs-breadcrumb-bg: ;--bs-breadcrumb-border-radius: ;--bs-breadcrumb-divider-color:#a1a3a9;--bs-breadcrumb-item-padding-x:0.5rem;--bs-breadcrumb-item-active-color:#a1a3a9;display:flex;flex-wrap:wrap;padding:var(--bs-breadcrumb-padding-y) var(--bs-breadcrumb-padding-x);margin-bottom:var(--bs-breadcrumb-margin-bottom);font-size:var(--bs-breadcrumb-font-size);list-style:none;background-color:var(--bs-breadcrumb-bg);border-radius:var(--bs-breadcrumb-border-radius)}.breadcrumb-item+.breadcrumb-item{padding-left:var(--bs-breadcrumb-item-padding-x)}.breadcrumb-item+.breadcrumb-item::before{float:left;padding-right:var(--bs-breadcrumb-item-padding-x);color:var(--bs-breadcrumb-divider-color);content:var(--bs-breadcrumb-divider, "/")}.breadcrumb-item.active{color:var(--bs-breadcrumb-item-active-color)}.pagination{--bs-pagination-padding-x:0.75rem;--bs-pagination-padding-y:0.375rem;--bs-pagination-font-size:1rem;--bs-pagination-color:var(--bs-link-color);--bs-pagination-bg:#fff;--bs-pagination-border-width:1px;--bs-pagination-border-color:#d4d4d4;--bs-pagination-border-radius:0;--bs-pagination-hover-color:var(--bs-link-hover-color);--bs-pagination-hover-bg:#e2e2e3;--bs-pagination-hover-border-color:#d4d4d4;--bs-pagination-focus-color:var(--bs-link-hover-color);--bs-pagination-focus-bg:#e2e2e3;--bs-pagination-focus-box-shadow:0 0 0 0.25rem rgba(69, 122, 171, 0.25);--bs-pagination-active-color:#fff;--bs-pagination-active-bg:#457aab;--bs-pagination-active-border-color:#457aab;--bs-pagination-disabled-color:#a1a3a9;--bs-pagination-disabled-bg:#fff;--bs-pagination-disabled-border-color:#d4d4d4;display:flex;padding-left:0;list-style:none}.page-link{position:relative;display:block;padding:var(--bs-pagination-padding-y) var(--bs-pagination-padding-x);font-size:var(--bs-pagination-font-size);color:var(--bs-pagination-color);text-decoration:none;background-color:var(--bs-pagination-bg);border:var(--bs-pagination-border-width) solid var(--bs-pagination-border-color)}.page-link:hover{z-index:2;color:var(--bs-pagination-hover-color);background-color:var(--bs-pagination-hover-bg);border-color:var(--bs-pagination-hover-border-color)}.page-link:focus{z-index:3;color:var(--bs-pagination-focus-color);background-color:var(--bs-pagination-focus-bg);outline:0;box-shadow:var(--bs-pagination-focus-box-shadow)}.active>.page-link,.page-link.active{z-index:3;color:var(--bs-pagination-active-color);background-color:var(--bs-pagination-active-bg);border-color:var(--bs-pagination-active-border-color)}.disabled>.page-link,.page-link.disabled{color:var(--bs-pagination-disabled-color);pointer-events:none;background-color:var(--bs-pagination-disabled-bg);border-color:var(--bs-pagination-disabled-border-color)}.page-item:not(:first-child) .page-link{margin-left:-1px}.page-item:first-child .page-link{border-top-left-radius:var(--bs-pagination-border-radius);border-bottom-left-radius:var(--bs-pagination-border-radius)}.page-item:last-child .page-link{border-top-right-radius:var(--bs-pagination-border-radius);border-bottom-right-radius:var(--bs-pagination-border-radius)}.pagination-lg{--bs-pagination-padding-x:1.5rem;--bs-pagination-padding-y:0.75rem;--bs-pagination-font-size:1.25rem;--bs-pagination-border-radius:0}.pagination-sm{--bs-pagination-padding-x:0.5rem;--bs-pagination-padding-y:0.25rem;--bs-pagination-font-size:0.875rem;--bs-pagination-border-radius:0}.badge{--bs-badge-padding-x:0.65em;--bs-badge-padding-y:0.35em;--bs-badge-font-size:0.75em;--bs-badge-font-weight:700;--bs-badge-color:#fff;--bs-badge-border-radius:0;display:inline-block;padding:var(--bs-badge-padding-y) var(--bs-badge-padding-x);font-size:var(--bs-badge-font-size);font-weight:var(--bs-badge-font-weight);line-height:1;color:var(--bs-badge-color);text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:var(--bs-badge-border-radius)}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.alert{--bs-alert-bg:transparent;--bs-alert-padding-x:1rem;--bs-alert-padding-y:1rem;--bs-alert-margin-bottom:1rem;--bs-alert-color:inherit;--bs-alert-border-color:transparent;--bs-alert-border:1px solid var(--bs-alert-border-color);--bs-alert-border-radius:0;position:relative;padding:var(--bs-alert-padding-y) var(--bs-alert-padding-x);margin-bottom:var(--bs-alert-margin-bottom);color:var(--bs-alert-color);background-color:var(--bs-alert-bg);border:var(--bs-alert-border);border-radius:var(--bs-alert-border-radius)}.alert-heading{color:inherit}.alert-link{font-weight:400}.alert-dismissible{padding-right:3rem}.alert-dismissible .btn-close{position:absolute;top:0;right:0;z-index:2;padding:1.25rem 1rem}.alert-primary{--bs-alert-color:white;--bs-alert-bg:#457aab;--bs-alert-border-color:#457aab}.alert-primary .alert-link{color:#ccc}.alert-secondary{--bs-alert-color:white;--bs-alert-bg:#5d6d75;--bs-alert-border-color:#5d6d75}.alert-secondary .alert-link{color:#ccc}.alert-success{--bs-alert-color:white;--bs-alert-bg:#2a8256;--bs-alert-border-color:#2a8256}.alert-success .alert-link{color:#ccc}.alert-info{--bs-alert-color:white;--bs-alert-bg:#0068b3;--bs-alert-border-color:#0068b3}.alert-info .alert-link{color:#ccc}.alert-warning{--bs-alert-color:#222222;--bs-alert-bg:#ffc107;--bs-alert-border-color:#ffc107}.alert-warning .alert-link{color:#1b1b1b}.alert-danger{--bs-alert-color:white;--bs-alert-bg:#c42526;--bs-alert-border-color:#c42526}.alert-danger .alert-link{color:#ccc}.alert-light{--bs-alert-color:#222222;--bs-alert-bg:#f2f2f2;--bs-alert-border-color:#f2f2f2}.alert-light .alert-link{color:#1b1b1b}.alert-dark{--bs-alert-color:white;--bs-alert-bg:#2a2a2a;--bs-alert-border-color:#2a2a2a}.alert-dark .alert-link{color:#ccc}.progress{--bs-progress-height:1rem;--bs-progress-font-size:0.75rem;--bs-progress-bg:#e2e2e3;--bs-progress-border-radius:0;--bs-progress-box-shadow:inset 0 1px 2px rgba(34, 34, 34, 0.075);--bs-progress-bar-color:#fff;--bs-progress-bar-bg:#457aab;--bs-progress-bar-transition:width 0.6s ease;display:flex;height:var(--bs-progress-height);overflow:hidden;font-size:var(--bs-progress-font-size);background-color:var(--bs-progress-bg);border-radius:var(--bs-progress-border-radius)}.progress-bar{display:flex;flex-direction:column;justify-content:center;overflow:hidden;color:var(--bs-progress-bar-color);text-align:center;white-space:nowrap;background-color:var(--bs-progress-bar-bg)}.progress-bar-striped{background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-size:var(--bs-progress-height) var(--bs-progress-height)}.list-group{--bs-list-group-color:#2a2a2a;--bs-list-group-bg:#fff;--bs-list-group-border-color:rgba(34, 34, 34, 0.125);--bs-list-group-border-width:1px;--bs-list-group-border-radius:0;--bs-list-group-item-padding-x:1rem;--bs-list-group-item-padding-y:0.5rem;--bs-list-group-action-color:#5d6d75;--bs-list-group-action-hover-color:#5d6d75;--bs-list-group-action-hover-bg:#f2f2f2;--bs-list-group-action-active-color:#2a2a2a;--bs-list-group-action-active-bg:#e2e2e3;--bs-list-group-disabled-color:#a1a3a9;--bs-list-group-disabled-bg:#fff;--bs-list-group-active-color:#fff;--bs-list-group-active-bg:#457aab;--bs-list-group-active-border-color:#457aab;display:flex;flex-direction:column;padding-left:0;margin-bottom:0;border-radius:var(--bs-list-group-border-radius)}.list-group-numbered{list-style-type:none;counter-reset:section}.list-group-numbered>.list-group-item::before{content:counters(section, ".") ". ";counter-increment:section}.list-group-item-action{width:100%;color:var(--bs-list-group-action-color);text-align:inherit}.list-group-item-action:focus,.list-group-item-action:hover{z-index:1;color:var(--bs-list-group-action-hover-color);text-decoration:none;background-color:var(--bs-list-group-action-hover-bg)}.list-group-item-action:active{color:var(--bs-list-group-action-active-color);background-color:var(--bs-list-group-action-active-bg)}.list-group-item{position:relative;display:block;padding:var(--bs-list-group-item-padding-y) var(--bs-list-group-item-padding-x);color:var(--bs-list-group-color);text-decoration:none;background-color:var(--bs-list-group-bg);border:var(--bs-list-group-border-width) solid var(--bs-list-group-border-color)}.list-group-item:first-child{border-top-left-radius:inherit;border-top-right-radius:inherit}.list-group-item:last-child{border-bottom-right-radius:inherit;border-bottom-left-radius:inherit}.list-group-item.disabled,.list-group-item:disabled{color:var(--bs-list-group-disabled-color);pointer-events:none;background-color:var(--bs-list-group-disabled-bg)}.list-group-item.active{z-index:2;color:var(--bs-list-group-active-color);background-color:var(--bs-list-group-active-bg);border-color:var(--bs-list-group-active-border-color)}.list-group-item+.list-group-item{border-top-width:0}.list-group-item+.list-group-item.active{margin-top:calc(-1 * var(--bs-list-group-border-width));border-top-width:var(--bs-list-group-border-width)}.list-group-horizontal{flex-direction:row}.list-group-horizontal>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal>.list-group-item.active{margin-top:0}.list-group-horizontal>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}@media (min-width:576px){.list-group-horizontal-sm{flex-direction:row}.list-group-horizontal-sm>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-sm>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-sm>.list-group-item.active{margin-top:0}.list-group-horizontal-sm>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-sm>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:768px){.list-group-horizontal-md{flex-direction:row}.list-group-horizontal-md>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-md>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-md>.list-group-item.active{margin-top:0}.list-group-horizontal-md>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-md>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:992px){.list-group-horizontal-lg{flex-direction:row}.list-group-horizontal-lg>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-lg>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-lg>.list-group-item.active{margin-top:0}.list-group-horizontal-lg>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-lg>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:1200px){.list-group-horizontal-xl{flex-direction:row}.list-group-horizontal-xl>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-xl>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-xl>.list-group-item.active{margin-top:0}.list-group-horizontal-xl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xl>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:1400px){.list-group-horizontal-xxl{flex-direction:row}.list-group-horizontal-xxl>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-xxl>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-xxl>.list-group-item.active{margin-top:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}.list-group-flush{border-radius:0}.list-group-flush>.list-group-item{border-width:0 0 var(--bs-list-group-border-width)}.list-group-flush>.list-group-item:last-child{border-bottom-width:0}.list-group-item-primary{color:#fff;background-color:#457aab}.list-group-item-primary.list-group-item-action:focus,.list-group-item-primary.list-group-item-action:hover{color:#fff;background-color:#3e6e9a}.list-group-item-primary.list-group-item-action.active{color:#fff;background-color:#fff;border-color:#fff}.list-group-item-secondary{color:#fff;background-color:#5d6d75}.list-group-item-secondary.list-group-item-action:focus,.list-group-item-secondary.list-group-item-action:hover{color:#fff;background-color:#546269}.list-group-item-secondary.list-group-item-action.active{color:#fff;background-color:#fff;border-color:#fff}.list-group-item-success{color:#fff;background-color:#2a8256}.list-group-item-success.list-group-item-action:focus,.list-group-item-success.list-group-item-action:hover{color:#fff;background-color:#26754d}.list-group-item-success.list-group-item-action.active{color:#fff;background-color:#fff;border-color:#fff}.list-group-item-info{color:#fff;background-color:#0068b3}.list-group-item-info.list-group-item-action:focus,.list-group-item-info.list-group-item-action:hover{color:#fff;background-color:#005ea1}.list-group-item-info.list-group-item-action.active{color:#fff;background-color:#fff;border-color:#fff}.list-group-item-warning{color:#222;background-color:#ffc107}.list-group-item-warning.list-group-item-action:focus,.list-group-item-warning.list-group-item-action:hover{color:#222;background-color:#e6ae06}.list-group-item-warning.list-group-item-action.active{color:#fff;background-color:#222;border-color:#222}.list-group-item-danger{color:#fff;background-color:#c42526}.list-group-item-danger.list-group-item-action:focus,.list-group-item-danger.list-group-item-action:hover{color:#fff;background-color:#b02122}.list-group-item-danger.list-group-item-action.active{color:#fff;background-color:#fff;border-color:#fff}.list-group-item-light{color:#222;background-color:#f2f2f2}.list-group-item-light.list-group-item-action:focus,.list-group-item-light.list-group-item-action:hover{color:#222;background-color:#dadada}.list-group-item-light.list-group-item-action.active{color:#fff;background-color:#222;border-color:#222}.list-group-item-dark{color:#fff;background-color:#2a2a2a}.list-group-item-dark.list-group-item-action:focus,.list-group-item-dark.list-group-item-action:hover{color:#fff;background-color:#262626}.list-group-item-dark.list-group-item-action.active{color:#fff;background-color:#fff;border-color:#fff}.btn-close{box-sizing:content-box;width:1em;height:1em;padding:.25em .25em;color:#222;background:transparent url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23222'%3e%3cpath d='M.293.293a1 1 0 0 1 1.414 0L8 6.586 14.293.293a1 1 0 1 1 1.414 1.414L9.414 8l6.293 6.293a1 1 0 0 1-1.414 1.414L8 9.414l-6.293 6.293a1 1 0 0 1-1.414-1.414L6.586 8 .293 1.707a1 1 0 0 1 0-1.414z'/%3e%3c/svg%3e") center/1em auto no-repeat;border:0;border-radius:0;opacity:.5}.btn-close:hover{color:#222;text-decoration:none;opacity:.75}.btn-close:focus{outline:0;box-shadow:0 0 0 .25rem rgba(69,122,171,.25);opacity:1}.btn-close.disabled,.btn-close:disabled{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none;opacity:.25}.btn-close-white{filter:invert(1) grayscale(100%) brightness(200%)}.toast{--bs-toast-zindex:1090;--bs-toast-padding-x:0.75rem;--bs-toast-padding-y:0.5rem;--bs-toast-spacing:1.5rem;--bs-toast-max-width:350px;--bs-toast-font-size:0.875rem;--bs-toast-color: ;--bs-toast-bg:rgba(255, 255, 255, 0.85);--bs-toast-border-width:1px;--bs-toast-border-color:var(--bs-border-color-translucent);--bs-toast-border-radius:0;--bs-toast-box-shadow:0 0.5rem 1rem rgba(34, 34, 34, 0.15);--bs-toast-header-color:#a1a3a9;--bs-toast-header-bg:rgba(255, 255, 255, 0.85);--bs-toast-header-border-color:rgba(34, 34, 34, 0.05);width:var(--bs-toast-max-width);max-width:100%;font-size:var(--bs-toast-font-size);color:var(--bs-toast-color);pointer-events:auto;background-color:var(--bs-toast-bg);background-clip:padding-box;border:var(--bs-toast-border-width) solid var(--bs-toast-border-color);box-shadow:var(--bs-toast-box-shadow);border-radius:var(--bs-toast-border-radius)}.toast.showing{opacity:0}.toast:not(.show){display:none}.toast-container{--bs-toast-zindex:1090;position:absolute;z-index:var(--bs-toast-zindex);width:-webkit-max-content;width:-moz-max-content;width:max-content;max-width:100%;pointer-events:none}.toast-container>:not(:last-child){margin-bottom:var(--bs-toast-spacing)}.toast-header{display:flex;align-items:center;padding:var(--bs-toast-padding-y) var(--bs-toast-padding-x);color:var(--bs-toast-header-color);background-color:var(--bs-toast-header-bg);background-clip:padding-box;border-bottom:var(--bs-toast-border-width) solid var(--bs-toast-header-border-color);border-top-left-radius:calc(var(--bs-toast-border-radius) - var(--bs-toast-border-width));border-top-right-radius:calc(var(--bs-toast-border-radius) - var(--bs-toast-border-width))}.toast-header .btn-close{margin-right:calc(-.5 * var(--bs-toast-padding-x));margin-left:var(--bs-toast-padding-x)}.toast-body{padding:var(--bs-toast-padding-x);word-wrap:break-word}.modal{--bs-modal-zindex:1055;--bs-modal-width:500px;--bs-modal-padding:1rem;--bs-modal-margin:0.5rem;--bs-modal-color: ;--bs-modal-bg:#fff;--bs-modal-border-color:var(--bs-border-color-translucent);--bs-modal-border-width:1px;--bs-modal-border-radius:0;--bs-modal-box-shadow:0 0.125rem 0.25rem rgba(34, 34, 34, 0.075);--bs-modal-inner-border-radius:-1px;--bs-modal-header-padding-x:1rem;--bs-modal-header-padding-y:1rem;--bs-modal-header-padding:1rem 1rem;--bs-modal-header-border-color:var(--bs-border-color);--bs-modal-header-border-width:1px;--bs-modal-title-line-height:1.5;--bs-modal-footer-gap:0.5rem;--bs-modal-footer-bg: ;--bs-modal-footer-border-color:var(--bs-border-color);--bs-modal-footer-border-width:1px;position:fixed;top:0;left:0;z-index:var(--bs-modal-zindex);display:none;width:100%;height:100%;overflow-x:hidden;overflow-y:auto;outline:0}.modal-dialog{position:relative;width:auto;margin:var(--bs-modal-margin);pointer-events:none}.modal.fade .modal-dialog{transform:translate(0,-50px)}.modal.show .modal-dialog{transform:none}.modal.modal-static .modal-dialog{transform:scale(1.02)}.modal-dialog-scrollable{height:calc(100% - var(--bs-modal-margin) * 2)}.modal-dialog-scrollable .modal-content{max-height:100%;overflow:hidden}.modal-dialog-scrollable .modal-body{overflow-y:auto}.modal-dialog-centered{display:flex;align-items:center;min-height:calc(100% - var(--bs-modal-margin) * 2)}.modal-content{position:relative;display:flex;flex-direction:column;width:100%;color:var(--bs-modal-color);pointer-events:auto;background-color:var(--bs-modal-bg);background-clip:padding-box;border:var(--bs-modal-border-width) solid var(--bs-modal-border-color);border-radius:var(--bs-modal-border-radius);outline:0}.modal-backdrop{--bs-backdrop-zindex:1050;--bs-backdrop-bg:#222;--bs-backdrop-opacity:0.5;position:fixed;top:0;left:0;z-index:var(--bs-backdrop-zindex);width:100vw;height:100vh;background-color:var(--bs-backdrop-bg)}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:var(--bs-backdrop-opacity)}.modal-header{display:flex;flex-shrink:0;align-items:center;justify-content:space-between;padding:var(--bs-modal-header-padding);border-bottom:var(--bs-modal-header-border-width) solid var(--bs-modal-header-border-color);border-top-left-radius:var(--bs-modal-inner-border-radius);border-top-right-radius:var(--bs-modal-inner-border-radius)}.modal-header .btn-close{padding:calc(var(--bs-modal-header-padding-y) * .5) calc(var(--bs-modal-header-padding-x) * .5);margin:calc(-.5 * var(--bs-modal-header-padding-y)) calc(-.5 * var(--bs-modal-header-padding-x)) calc(-.5 * var(--bs-modal-header-padding-y)) auto}.modal-title{margin-bottom:0;line-height:var(--bs-modal-title-line-height)}.modal-body{position:relative;flex:1 1 auto;padding:var(--bs-modal-padding)}.modal-footer{display:flex;flex-shrink:0;flex-wrap:wrap;align-items:center;justify-content:flex-end;padding:calc(var(--bs-modal-padding) - var(--bs-modal-footer-gap) * .5);background-color:var(--bs-modal-footer-bg);border-top:var(--bs-modal-footer-border-width) solid var(--bs-modal-footer-border-color);border-bottom-right-radius:var(--bs-modal-inner-border-radius);border-bottom-left-radius:var(--bs-modal-inner-border-radius)}.modal-footer>*{margin:calc(var(--bs-modal-footer-gap) * .5)}@media (min-width:576px){.modal{--bs-modal-margin:1.75rem;--bs-modal-box-shadow:0 0.5rem 1rem rgba(34, 34, 34, 0.15)}.modal-dialog{max-width:var(--bs-modal-width);margin-right:auto;margin-left:auto}.modal-sm{--bs-modal-width:300px}}@media (min-width:992px){.modal-lg,.modal-xl{--bs-modal-width:800px}}@media (min-width:1200px){.modal-xl{--bs-modal-width:1140px}}.modal-fullscreen{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen .modal-footer,.modal-fullscreen .modal-header{border-radius:0}.modal-fullscreen .modal-body{overflow-y:auto}@media (max-width:575.98px){.modal-fullscreen-sm-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-sm-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-sm-down .modal-footer,.modal-fullscreen-sm-down .modal-header{border-radius:0}.modal-fullscreen-sm-down .modal-body{overflow-y:auto}}@media (max-width:767.98px){.modal-fullscreen-md-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-md-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-md-down .modal-footer,.modal-fullscreen-md-down .modal-header{border-radius:0}.modal-fullscreen-md-down .modal-body{overflow-y:auto}}@media (max-width:991.98px){.modal-fullscreen-lg-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-lg-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-lg-down .modal-footer,.modal-fullscreen-lg-down .modal-header{border-radius:0}.modal-fullscreen-lg-down .modal-body{overflow-y:auto}}@media (max-width:1199.98px){.modal-fullscreen-xl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xl-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-xl-down .modal-footer,.modal-fullscreen-xl-down .modal-header{border-radius:0}.modal-fullscreen-xl-down .modal-body{overflow-y:auto}}@media (max-width:1399.98px){.modal-fullscreen-xxl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xxl-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-xxl-down .modal-footer,.modal-fullscreen-xxl-down .modal-header{border-radius:0}.modal-fullscreen-xxl-down .modal-body{overflow-y:auto}}.tooltip{--bs-tooltip-zindex:1080;--bs-tooltip-max-width:200px;--bs-tooltip-padding-x:0.5rem;--bs-tooltip-padding-y:0.25rem;--bs-tooltip-margin: ;--bs-tooltip-font-size:0.875rem;--bs-tooltip-color:#fff;--bs-tooltip-bg:#222;--bs-tooltip-border-radius:0;--bs-tooltip-opacity:0.9;--bs-tooltip-arrow-width:0.8rem;--bs-tooltip-arrow-height:0.4rem;z-index:var(--bs-tooltip-zindex);display:block;padding:var(--bs-tooltip-arrow-height);margin:var(--bs-tooltip-margin);font-family:var(--bs-font-sans-serif);font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-tooltip-font-size);word-wrap:break-word;opacity:0}.tooltip.show{opacity:var(--bs-tooltip-opacity)}.tooltip .tooltip-arrow{display:block;width:var(--bs-tooltip-arrow-width);height:var(--bs-tooltip-arrow-height)}.tooltip .tooltip-arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow,.bs-tooltip-top .tooltip-arrow{bottom:0}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow::before,.bs-tooltip-top .tooltip-arrow::before{top:-1px;border-width:var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width) * .5) 0;border-top-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow,.bs-tooltip-end .tooltip-arrow{left:0;width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow::before,.bs-tooltip-end .tooltip-arrow::before{right:-1px;border-width:calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width) * .5) 0;border-right-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow,.bs-tooltip-bottom .tooltip-arrow{top:0}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow::before,.bs-tooltip-bottom .tooltip-arrow::before{bottom:-1px;border-width:0 calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height);border-bottom-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow,.bs-tooltip-start .tooltip-arrow{right:0;width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow::before,.bs-tooltip-start .tooltip-arrow::before{left:-1px;border-width:calc(var(--bs-tooltip-arrow-width) * .5) 0 calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height);border-left-color:var(--bs-tooltip-bg)}.tooltip-inner{max-width:var(--bs-tooltip-max-width);padding:var(--bs-tooltip-padding-y) var(--bs-tooltip-padding-x);color:var(--bs-tooltip-color);text-align:center;background-color:var(--bs-tooltip-bg);border-radius:var(--bs-tooltip-border-radius)}.popover{--bs-popover-zindex:1070;--bs-popover-max-width:276px;--bs-popover-font-size:0.875rem;--bs-popover-bg:#fff;--bs-popover-border-width:1px;--bs-popover-border-color:var(--bs-border-color-translucent);--bs-popover-border-radius:0;--bs-popover-inner-border-radius:-1px;--bs-popover-box-shadow:0 0.5rem 1rem rgba(34, 34, 34, 0.15);--bs-popover-header-padding-x:1rem;--bs-popover-header-padding-y:0.5rem;--bs-popover-header-font-size:1rem;--bs-popover-header-color: ;--bs-popover-header-bg:#f0f0f0;--bs-popover-body-padding-x:1rem;--bs-popover-body-padding-y:1rem;--bs-popover-body-color:#2a2a2a;--bs-popover-arrow-width:1rem;--bs-popover-arrow-height:0.5rem;--bs-popover-arrow-border:var(--bs-popover-border-color);z-index:var(--bs-popover-zindex);display:block;max-width:var(--bs-popover-max-width);font-family:var(--bs-font-sans-serif);font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-popover-font-size);word-wrap:break-word;background-color:var(--bs-popover-bg);background-clip:padding-box;border:var(--bs-popover-border-width) solid var(--bs-popover-border-color);border-radius:var(--bs-popover-border-radius)}.popover .popover-arrow{display:block;width:var(--bs-popover-arrow-width);height:var(--bs-popover-arrow-height)}.popover .popover-arrow::after,.popover .popover-arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid;border-width:0}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow,.bs-popover-top>.popover-arrow{bottom:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::after,.bs-popover-top>.popover-arrow::before{border-width:var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width) * .5) 0}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::before{bottom:0;border-top-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after,.bs-popover-top>.popover-arrow::after{bottom:var(--bs-popover-border-width);border-top-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow,.bs-popover-end>.popover-arrow{left:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::after,.bs-popover-end>.popover-arrow::before{border-width:calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width) * .5) 0}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::before{left:0;border-right-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after,.bs-popover-end>.popover-arrow::after{left:var(--bs-popover-border-width);border-right-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow,.bs-popover-bottom>.popover-arrow{top:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::after,.bs-popover-bottom>.popover-arrow::before{border-width:0 calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::before{top:0;border-bottom-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after,.bs-popover-bottom>.popover-arrow::after{top:var(--bs-popover-border-width);border-bottom-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=bottom] .popover-header::before,.bs-popover-bottom .popover-header::before{position:absolute;top:0;left:50%;display:block;width:var(--bs-popover-arrow-width);margin-left:calc(-.5 * var(--bs-popover-arrow-width));content:"";border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-header-bg)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow,.bs-popover-start>.popover-arrow{right:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::after,.bs-popover-start>.popover-arrow::before{border-width:calc(var(--bs-popover-arrow-width) * .5) 0 calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::before{right:0;border-left-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after,.bs-popover-start>.popover-arrow::after{right:var(--bs-popover-border-width);border-left-color:var(--bs-popover-bg)}.popover-header{padding:var(--bs-popover-header-padding-y) var(--bs-popover-header-padding-x);margin-bottom:0;font-size:var(--bs-popover-header-font-size);color:var(--bs-popover-header-color);background-color:var(--bs-popover-header-bg);border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-border-color);border-top-left-radius:var(--bs-popover-inner-border-radius);border-top-right-radius:var(--bs-popover-inner-border-radius)}.popover-header:empty{display:none}.popover-body{padding:var(--bs-popover-body-padding-y) var(--bs-popover-body-padding-x);color:var(--bs-popover-body-color)}.carousel{position:relative}.carousel.pointer-event{touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;-webkit-backface-visibility:hidden;backface-visibility:hidden}.carousel-item-next,.carousel-item-prev,.carousel-item.active{display:block}.active.carousel-item-end,.carousel-item-next:not(.carousel-item-start){transform:translateX(100%)}.active.carousel-item-start,.carousel-item-prev:not(.carousel-item-end){transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;transition-property:opacity;transform:none}.carousel-fade .carousel-item-next.carousel-item-start,.carousel-fade .carousel-item-prev.carousel-item-end,.carousel-fade .carousel-item.active{z-index:1;opacity:1}.carousel-fade .active.carousel-item-end,.carousel-fade .active.carousel-item-start{z-index:0;opacity:0}.carousel-control-next,.carousel-control-prev{position:absolute;top:0;bottom:0;z-index:1;display:flex;align-items:center;justify-content:center;width:15%;padding:0;color:#fff;text-align:center;background:0 0;border:0;opacity:.5}.carousel-control-next:focus,.carousel-control-next:hover,.carousel-control-prev:focus,.carousel-control-prev:hover{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-next-icon,.carousel-control-prev-icon{display:inline-block;width:2rem;height:2rem;background-repeat:no-repeat;background-position:50%;background-size:100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M11.354 1.646a.5.5 0 0 1 0 .708L5.707 8l5.647 5.646a.5.5 0 0 1-.708.708l-6-6a.5.5 0 0 1 0-.708l6-6a.5.5 0 0 1 .708 0z'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M4.646 1.646a.5.5 0 0 1 .708 0l6 6a.5.5 0 0 1 0 .708l-6 6a.5.5 0 0 1-.708-.708L10.293 8 4.646 2.354a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:2;display:flex;justify-content:center;padding:0;margin-right:15%;margin-bottom:1rem;margin-left:15%;list-style:none}.carousel-indicators [data-bs-target]{box-sizing:content-box;flex:0 1 auto;width:30px;height:3px;padding:0;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:#fff;background-clip:padding-box;border:0;border-top:10px solid transparent;border-bottom:10px solid transparent;opacity:.5}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:1.25rem;left:15%;padding-top:1.25rem;padding-bottom:1.25rem;color:#fff;text-align:center}.carousel-dark .carousel-control-next-icon,.carousel-dark .carousel-control-prev-icon{filter:invert(1) grayscale(100)}.carousel-dark .carousel-indicators [data-bs-target]{background-color:#222}.carousel-dark .carousel-caption{color:#222}.spinner-border,.spinner-grow{display:inline-block;width:var(--bs-spinner-width);height:var(--bs-spinner-height);vertical-align:var(--bs-spinner-vertical-align);border-radius:50%;animation:var(--bs-spinner-animation-speed) linear infinite var(--bs-spinner-animation-name)}@keyframes spinner-border{to{transform:rotate(360deg)}}.spinner-border{--bs-spinner-width:2rem;--bs-spinner-height:2rem;--bs-spinner-vertical-align:-0.125em;--bs-spinner-border-width:0.25em;--bs-spinner-animation-speed:0.75s;--bs-spinner-animation-name:spinner-border;border:var(--bs-spinner-border-width) solid currentcolor;border-right-color:transparent}.spinner-border-sm{--bs-spinner-width:1rem;--bs-spinner-height:1rem;--bs-spinner-border-width:0.2em}@keyframes spinner-grow{0%{transform:scale(0)}50%{opacity:1;transform:none}}.spinner-grow{--bs-spinner-width:2rem;--bs-spinner-height:2rem;--bs-spinner-vertical-align:-0.125em;--bs-spinner-animation-speed:0.75s;--bs-spinner-animation-name:spinner-grow;background-color:currentcolor;opacity:0}.spinner-grow-sm{--bs-spinner-width:1rem;--bs-spinner-height:1rem}@media (prefers-reduced-motion:reduce){.spinner-border,.spinner-grow{--bs-spinner-animation-speed:1.5s}}.offcanvas,.offcanvas-lg,.offcanvas-md,.offcanvas-sm,.offcanvas-xl,.offcanvas-xxl{--bs-offcanvas-zindex:1045;--bs-offcanvas-width:400px;--bs-offcanvas-height:30vh;--bs-offcanvas-padding-x:1rem;--bs-offcanvas-padding-y:1rem;--bs-offcanvas-color: ;--bs-offcanvas-bg:#fff;--bs-offcanvas-border-width:1px;--bs-offcanvas-border-color:var(--bs-border-color-translucent);--bs-offcanvas-box-shadow:0 0.125rem 0.25rem rgba(34, 34, 34, 0.075)}@media (max-width:575.98px){.offcanvas-sm{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0}.offcanvas-sm.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-sm.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-sm.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-sm.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-sm.show:not(.hiding),.offcanvas-sm.showing{transform:none}.offcanvas-sm.hiding,.offcanvas-sm.show,.offcanvas-sm.showing{visibility:visible}}@media (min-width:576px){.offcanvas-sm{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-sm .offcanvas-header{display:none}.offcanvas-sm .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:767.98px){.offcanvas-md{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0}.offcanvas-md.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-md.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-md.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-md.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-md.show:not(.hiding),.offcanvas-md.showing{transform:none}.offcanvas-md.hiding,.offcanvas-md.show,.offcanvas-md.showing{visibility:visible}}@media (min-width:768px){.offcanvas-md{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-md .offcanvas-header{display:none}.offcanvas-md .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:991.98px){.offcanvas-lg{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0}.offcanvas-lg.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-lg.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-lg.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-lg.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-lg.show:not(.hiding),.offcanvas-lg.showing{transform:none}.offcanvas-lg.hiding,.offcanvas-lg.show,.offcanvas-lg.showing{visibility:visible}}@media (min-width:992px){.offcanvas-lg{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-lg .offcanvas-header{display:none}.offcanvas-lg .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:1199.98px){.offcanvas-xl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0}.offcanvas-xl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xl.show:not(.hiding),.offcanvas-xl.showing{transform:none}.offcanvas-xl.hiding,.offcanvas-xl.show,.offcanvas-xl.showing{visibility:visible}}@media (min-width:1200px){.offcanvas-xl{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-xl .offcanvas-header{display:none}.offcanvas-xl .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:1399.98px){.offcanvas-xxl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0}.offcanvas-xxl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xxl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xxl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xxl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xxl.show:not(.hiding),.offcanvas-xxl.showing{transform:none}.offcanvas-xxl.hiding,.offcanvas-xxl.show,.offcanvas-xxl.showing{visibility:visible}}@media (min-width:1400px){.offcanvas-xxl{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-xxl .offcanvas-header{display:none}.offcanvas-xxl .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}.offcanvas{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0}.offcanvas.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas.show:not(.hiding),.offcanvas.showing{transform:none}.offcanvas.hiding,.offcanvas.show,.offcanvas.showing{visibility:visible}.offcanvas-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#222}.offcanvas-backdrop.fade{opacity:0}.offcanvas-backdrop.show{opacity:.5}.offcanvas-header{display:flex;align-items:center;justify-content:space-between;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x)}.offcanvas-header .btn-close{padding:calc(var(--bs-offcanvas-padding-y) * .5) calc(var(--bs-offcanvas-padding-x) * .5);margin-top:calc(-.5 * var(--bs-offcanvas-padding-y));margin-right:calc(-.5 * var(--bs-offcanvas-padding-x));margin-bottom:calc(-.5 * var(--bs-offcanvas-padding-y))}.offcanvas-title{margin-bottom:0;line-height:1.5}.offcanvas-body{flex-grow:1;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x);overflow-y:auto}.placeholder{display:inline-block;min-height:1em;vertical-align:middle;cursor:wait;background-color:currentcolor;opacity:.5}.placeholder.btn::before{display:inline-block;content:""}.placeholder-xs{min-height:.6em}.placeholder-sm{min-height:.8em}.placeholder-lg{min-height:1.2em}.placeholder-glow .placeholder{animation:placeholder-glow 2s ease-in-out infinite}@keyframes placeholder-glow{50%{opacity:.2}}.placeholder-wave{-webkit-mask-image:linear-gradient(130deg,#222 55%,rgba(0,0,0,0.8) 75%,#222 95%);mask-image:linear-gradient(130deg,#222 55%,rgba(0,0,0,0.8) 75%,#222 95%);-webkit-mask-size:200% 100%;mask-size:200% 100%;animation:placeholder-wave 2s linear infinite}@keyframes placeholder-wave{100%{-webkit-mask-position:-200% 0%;mask-position:-200% 0%}}.clearfix::after{display:block;clear:both;content:""}.text-bg-primary{color:#fff!important;background-color:RGBA(69,122,171,var(--bs-bg-opacity,1))!important}.text-bg-secondary{color:#fff!important;background-color:RGBA(93,109,117,var(--bs-bg-opacity,1))!important}.text-bg-success{color:#fff!important;background-color:RGBA(42,130,86,var(--bs-bg-opacity,1))!important}.text-bg-info{color:#fff!important;background-color:RGBA(0,104,179,var(--bs-bg-opacity,1))!important}.text-bg-warning{color:#222!important;background-color:RGBA(255,193,7,var(--bs-bg-opacity,1))!important}.text-bg-danger{color:#fff!important;background-color:RGBA(196,37,38,var(--bs-bg-opacity,1))!important}.text-bg-light{color:#222!important;background-color:RGBA(242,242,242,var(--bs-bg-opacity,1))!important}.text-bg-dark{color:#fff!important;background-color:RGBA(42,42,42,var(--bs-bg-opacity,1))!important}.link-primary{color:#457aab!important}.link-primary:focus,.link-primary:hover{color:#233d56!important}.link-secondary{color:#5d6d75!important}.link-secondary:focus,.link-secondary:hover{color:#2f373b!important}.link-success{color:#2a8256!important}.link-success:focus,.link-success:hover{color:#15412b!important}.link-info{color:#0068b3!important}.link-info:focus,.link-info:hover{color:#00345a!important}.link-warning{color:#ffc107!important}.link-warning:focus,.link-warning:hover{color:#ffe083!important}.link-danger{color:#c42526!important}.link-danger:focus,.link-danger:hover{color:#621313!important}.link-light{color:#f2f2f2!important}.link-light:focus,.link-light:hover{color:#f9f9f9!important}.link-dark{color:#2a2a2a!important}.link-dark:focus,.link-dark:hover{color:#151515!important}.ratio{position:relative;width:100%}.ratio::before{display:block;padding-top:var(--bs-aspect-ratio);content:""}.ratio>*{position:absolute;top:0;left:0;width:100%;height:100%}.ratio-1x1{--bs-aspect-ratio:100%}.ratio-4x3{--bs-aspect-ratio:75%}.ratio-16x9{--bs-aspect-ratio:56.25%}.ratio-21x9{--bs-aspect-ratio:42.8571428571%}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}@media (min-width:576px){.sticky-sm-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-sm-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:768px){.sticky-md-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-md-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:992px){.sticky-lg-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-lg-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:1200px){.sticky-xl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-xl-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:1400px){.sticky-xxl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-xxl-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}.hstack{display:flex;flex-direction:row;align-items:center;align-self:stretch}.vstack{display:flex;flex:1 1 auto;flex-direction:column;align-self:stretch}.visually-hidden,.visually-hidden-focusable:not(:focus):not(:focus-within){position:absolute!important;width:1px!important;height:1px!important;padding:0!important;margin:-1px!important;overflow:hidden!important;clip:rect(0,0,0,0)!important;white-space:nowrap!important;border:0!important}.stretched-link::after{position:absolute;top:0;right:0;bottom:0;left:0;z-index:1;content:""}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.vr{display:inline-block;align-self:stretch;width:1px;min-height:1em;background-color:currentcolor;opacity:.25}.align-baseline{vertical-align:baseline!important}.align-top{vertical-align:top!important}.align-middle{vertical-align:middle!important}.align-bottom{vertical-align:bottom!important}.align-text-bottom{vertical-align:text-bottom!important}.align-text-top{vertical-align:text-top!important}.float-start{float:left!important}.float-end{float:right!important}.float-none{float:none!important}.opacity-0{opacity:0!important}.opacity-25{opacity:.25!important}.opacity-50{opacity:.5!important}.opacity-75{opacity:.75!important}.opacity-100{opacity:1!important}.overflow-auto{overflow:auto!important}.overflow-hidden{overflow:hidden!important}.overflow-visible{overflow:visible!important}.overflow-scroll{overflow:scroll!important}.d-inline{display:inline!important}.d-inline-block{display:inline-block!important}.d-block{display:block!important}.d-grid{display:grid!important}.d-table{display:table!important}.d-table-row{display:table-row!important}.d-table-cell{display:table-cell!important}.d-flex{display:flex!important}.d-inline-flex{display:inline-flex!important}.d-none{display:none!important}.shadow{box-shadow:0 .5rem 1rem rgba(34,34,34,.15)!important}.shadow-sm{box-shadow:0 .125rem .25rem rgba(34,34,34,.075)!important}.shadow-lg{box-shadow:0 1rem 3rem rgba(34,34,34,.175)!important}.shadow-none{box-shadow:none!important}.position-static{position:static!important}.position-relative{position:relative!important}.position-absolute{position:absolute!important}.position-fixed{position:fixed!important}.position-sticky{position:-webkit-sticky!important;position:sticky!important}.top-0{top:0!important}.top-50{top:50%!important}.top-100{top:100%!important}.bottom-0{bottom:0!important}.bottom-50{bottom:50%!important}.bottom-100{bottom:100%!important}.start-0{left:0!important}.start-50{left:50%!important}.start-100{left:100%!important}.end-0{right:0!important}.end-50{right:50%!important}.end-100{right:100%!important}.translate-middle{transform:translate(-50%,-50%)!important}.translate-middle-x{transform:translateX(-50%)!important}.translate-middle-y{transform:translateY(-50%)!important}.border{border:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-0{border:0!important}.border-top{border-top:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-top-0{border-top:0!important}.border-end{border-right:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-end-0{border-right:0!important}.border-bottom{border-bottom:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-bottom-0{border-bottom:0!important}.border-start{border-left:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-start-0{border-left:0!important}.border-primary{--bs-border-opacity:1;border-color:rgba(var(--bs-primary-rgb),var(--bs-border-opacity))!important}.border-secondary{--bs-border-opacity:1;border-color:rgba(var(--bs-secondary-rgb),var(--bs-border-opacity))!important}.border-success{--bs-border-opacity:1;border-color:rgba(var(--bs-success-rgb),var(--bs-border-opacity))!important}.border-info{--bs-border-opacity:1;border-color:rgba(var(--bs-info-rgb),var(--bs-border-opacity))!important}.border-warning{--bs-border-opacity:1;border-color:rgba(var(--bs-warning-rgb),var(--bs-border-opacity))!important}.border-danger{--bs-border-opacity:1;border-color:rgba(var(--bs-danger-rgb),var(--bs-border-opacity))!important}.border-light{--bs-border-opacity:1;border-color:rgba(var(--bs-light-rgb),var(--bs-border-opacity))!important}.border-dark{--bs-border-opacity:1;border-color:rgba(var(--bs-dark-rgb),var(--bs-border-opacity))!important}.border-white{--bs-border-opacity:1;border-color:rgba(var(--bs-white-rgb),var(--bs-border-opacity))!important}.border-1{--bs-border-width:1px}.border-2{--bs-border-width:2px}.border-3{--bs-border-width:3px}.border-4{--bs-border-width:4px}.border-5{--bs-border-width:5px}.border-opacity-10{--bs-border-opacity:0.1}.border-opacity-25{--bs-border-opacity:0.25}.border-opacity-50{--bs-border-opacity:0.5}.border-opacity-75{--bs-border-opacity:0.75}.border-opacity-100{--bs-border-opacity:1}.w-25{width:25%!important}.w-50{width:50%!important}.w-75{width:75%!important}.w-100{width:100%!important}.w-auto{width:auto!important}.mw-100{max-width:100%!important}.vw-100{width:100vw!important}.min-vw-100{min-width:100vw!important}.h-25{height:25%!important}.h-50{height:50%!important}.h-75{height:75%!important}.h-100{height:100%!important}.h-auto{height:auto!important}.mh-100{max-height:100%!important}.vh-100{height:100vh!important}.min-vh-100{min-height:100vh!important}.flex-fill{flex:1 1 auto!important}.flex-row{flex-direction:row!important}.flex-column{flex-direction:column!important}.flex-row-reverse{flex-direction:row-reverse!important}.flex-column-reverse{flex-direction:column-reverse!important}.flex-grow-0{flex-grow:0!important}.flex-grow-1{flex-grow:1!important}.flex-shrink-0{flex-shrink:0!important}.flex-shrink-1{flex-shrink:1!important}.flex-wrap{flex-wrap:wrap!important}.flex-nowrap{flex-wrap:nowrap!important}.flex-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-start{justify-content:flex-start!important}.justify-content-end{justify-content:flex-end!important}.justify-content-center{justify-content:center!important}.justify-content-between{justify-content:space-between!important}.justify-content-around{justify-content:space-around!important}.justify-content-evenly{justify-content:space-evenly!important}.align-items-start{align-items:flex-start!important}.align-items-end{align-items:flex-end!important}.align-items-center{align-items:center!important}.align-items-baseline{align-items:baseline!important}.align-items-stretch{align-items:stretch!important}.align-content-start{align-content:flex-start!important}.align-content-end{align-content:flex-end!important}.align-content-center{align-content:center!important}.align-content-between{align-content:space-between!important}.align-content-around{align-content:space-around!important}.align-content-stretch{align-content:stretch!important}.align-self-auto{align-self:auto!important}.align-self-start{align-self:flex-start!important}.align-self-end{align-self:flex-end!important}.align-self-center{align-self:center!important}.align-self-baseline{align-self:baseline!important}.align-self-stretch{align-self:stretch!important}.order-first{order:-1!important}.order-0{order:0!important}.order-1{order:1!important}.order-2{order:2!important}.order-3{order:3!important}.order-4{order:4!important}.order-5{order:5!important}.order-last{order:6!important}.m-0{margin:0!important}.m-1{margin:.25rem!important}.m-2{margin:.5rem!important}.m-3{margin:1rem!important}.m-4{margin:1.5rem!important}.m-5{margin:3rem!important}.m-auto{margin:auto!important}.mx-0{margin-right:0!important;margin-left:0!important}.mx-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-3{margin-right:1rem!important;margin-left:1rem!important}.mx-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-5{margin-right:3rem!important;margin-left:3rem!important}.mx-auto{margin-right:auto!important;margin-left:auto!important}.my-0{margin-top:0!important;margin-bottom:0!important}.my-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-0{margin-top:0!important}.mt-1{margin-top:.25rem!important}.mt-2{margin-top:.5rem!important}.mt-3{margin-top:1rem!important}.mt-4{margin-top:1.5rem!important}.mt-5{margin-top:3rem!important}.mt-auto{margin-top:auto!important}.me-0{margin-right:0!important}.me-1{margin-right:.25rem!important}.me-2{margin-right:.5rem!important}.me-3{margin-right:1rem!important}.me-4{margin-right:1.5rem!important}.me-5{margin-right:3rem!important}.me-auto{margin-right:auto!important}.mb-0{margin-bottom:0!important}.mb-1{margin-bottom:.25rem!important}.mb-2{margin-bottom:.5rem!important}.mb-3{margin-bottom:1rem!important}.mb-4{margin-bottom:1.5rem!important}.mb-5{margin-bottom:3rem!important}.mb-auto{margin-bottom:auto!important}.ms-0{margin-left:0!important}.ms-1{margin-left:.25rem!important}.ms-2{margin-left:.5rem!important}.ms-3{margin-left:1rem!important}.ms-4{margin-left:1.5rem!important}.ms-5{margin-left:3rem!important}.ms-auto{margin-left:auto!important}.p-0{padding:0!important}.p-1{padding:.25rem!important}.p-2{padding:.5rem!important}.p-3{padding:1rem!important}.p-4{padding:1.5rem!important}.p-5{padding:3rem!important}.px-0{padding-right:0!important;padding-left:0!important}.px-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-3{padding-right:1rem!important;padding-left:1rem!important}.px-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-5{padding-right:3rem!important;padding-left:3rem!important}.py-0{padding-top:0!important;padding-bottom:0!important}.py-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-0{padding-top:0!important}.pt-1{padding-top:.25rem!important}.pt-2{padding-top:.5rem!important}.pt-3{padding-top:1rem!important}.pt-4{padding-top:1.5rem!important}.pt-5{padding-top:3rem!important}.pe-0{padding-right:0!important}.pe-1{padding-right:.25rem!important}.pe-2{padding-right:.5rem!important}.pe-3{padding-right:1rem!important}.pe-4{padding-right:1.5rem!important}.pe-5{padding-right:3rem!important}.pb-0{padding-bottom:0!important}.pb-1{padding-bottom:.25rem!important}.pb-2{padding-bottom:.5rem!important}.pb-3{padding-bottom:1rem!important}.pb-4{padding-bottom:1.5rem!important}.pb-5{padding-bottom:3rem!important}.ps-0{padding-left:0!important}.ps-1{padding-left:.25rem!important}.ps-2{padding-left:.5rem!important}.ps-3{padding-left:1rem!important}.ps-4{padding-left:1.5rem!important}.ps-5{padding-left:3rem!important}.gap-0{gap:0!important}.gap-1{gap:.25rem!important}.gap-2{gap:.5rem!important}.gap-3{gap:1rem!important}.gap-4{gap:1.5rem!important}.gap-5{gap:3rem!important}.font-monospace{font-family:var(--bs-font-monospace)!important}.fs-1{font-size:calc(1.325rem + .9vw)!important}.fs-2{font-size:calc(1.3rem + .6vw)!important}.fs-3{font-size:calc(1.275rem + .3vw)!important}.fs-4{font-size:1.25rem!important}.fs-5{font-size:1.15rem!important}.fs-6{font-size:1rem!important}.fst-italic{font-style:italic!important}.fst-normal{font-style:normal!important}.fw-light{font-weight:300!important}.fw-lighter{font-weight:lighter!important}.fw-normal{font-weight:400!important}.fw-bold{font-weight:700!important}.fw-semibold{font-weight:600!important}.fw-bolder{font-weight:bolder!important}.lh-1{line-height:1!important}.lh-sm{line-height:1.25!important}.lh-base{line-height:1.5!important}.lh-lg{line-height:2!important}.text-start{text-align:left!important}.text-end{text-align:right!important}.text-center{text-align:center!important}.text-decoration-none{text-decoration:none!important}.text-decoration-underline{text-decoration:underline!important}.text-decoration-line-through{text-decoration:line-through!important}.text-lowercase{text-transform:lowercase!important}.text-uppercase{text-transform:uppercase!important}.text-capitalize{text-transform:capitalize!important}.text-wrap{white-space:normal!important}.text-nowrap{white-space:nowrap!important}.text-break{word-wrap:break-word!important;word-break:break-word!important}.text-primary{--bs-text-opacity:1;color:rgba(var(--bs-primary-rgb),var(--bs-text-opacity))!important}.text-secondary{--bs-text-opacity:1;color:rgba(var(--bs-secondary-rgb),var(--bs-text-opacity))!important}.text-success{--bs-text-opacity:1;color:rgba(var(--bs-success-rgb),var(--bs-text-opacity))!important}.text-info{--bs-text-opacity:1;color:rgba(var(--bs-info-rgb),var(--bs-text-opacity))!important}.text-warning{--bs-text-opacity:1;color:rgba(var(--bs-warning-rgb),var(--bs-text-opacity))!important}.text-danger{--bs-text-opacity:1;color:rgba(var(--bs-danger-rgb),var(--bs-text-opacity))!important}.text-light{--bs-text-opacity:1;color:rgba(var(--bs-light-rgb),var(--bs-text-opacity))!important}.text-dark{--bs-text-opacity:1;color:rgba(var(--bs-dark-rgb),var(--bs-text-opacity))!important}.text-black{--bs-text-opacity:1;color:rgba(var(--bs-black-rgb),var(--bs-text-opacity))!important}.text-white{--bs-text-opacity:1;color:rgba(var(--bs-white-rgb),var(--bs-text-opacity))!important}.text-body{--bs-text-opacity:1;color:rgba(var(--bs-body-color-rgb),var(--bs-text-opacity))!important}.text-muted{--bs-text-opacity:1;color:#a1a3a9!important}.text-black-50{--bs-text-opacity:1;color:rgba(34,34,34,.5)!important}.text-white-50{--bs-text-opacity:1;color:rgba(255,255,255,.5)!important}.text-reset{--bs-text-opacity:1;color:inherit!important}.text-opacity-25{--bs-text-opacity:0.25}.text-opacity-50{--bs-text-opacity:0.5}.text-opacity-75{--bs-text-opacity:0.75}.text-opacity-100{--bs-text-opacity:1}.bg-primary{--bs-bg-opacity:1;background-color:rgba(var(--bs-primary-rgb),var(--bs-bg-opacity))!important}.bg-secondary{--bs-bg-opacity:1;background-color:rgba(var(--bs-secondary-rgb),var(--bs-bg-opacity))!important}.bg-success{--bs-bg-opacity:1;background-color:rgba(var(--bs-success-rgb),var(--bs-bg-opacity))!important}.bg-info{--bs-bg-opacity:1;background-color:rgba(var(--bs-info-rgb),var(--bs-bg-opacity))!important}.bg-warning{--bs-bg-opacity:1;background-color:rgba(var(--bs-warning-rgb),var(--bs-bg-opacity))!important}.bg-danger{--bs-bg-opacity:1;background-color:rgba(var(--bs-danger-rgb),var(--bs-bg-opacity))!important}.bg-light{--bs-bg-opacity:1;background-color:rgba(var(--bs-light-rgb),var(--bs-bg-opacity))!important}.bg-dark{--bs-bg-opacity:1;background-color:rgba(var(--bs-dark-rgb),var(--bs-bg-opacity))!important}.bg-black{--bs-bg-opacity:1;background-color:rgba(var(--bs-black-rgb),var(--bs-bg-opacity))!important}.bg-white{--bs-bg-opacity:1;background-color:rgba(var(--bs-white-rgb),var(--bs-bg-opacity))!important}.bg-body{--bs-bg-opacity:1;background-color:rgba(var(--bs-body-bg-rgb),var(--bs-bg-opacity))!important}.bg-transparent{--bs-bg-opacity:1;background-color:transparent!important}.bg-opacity-10{--bs-bg-opacity:0.1}.bg-opacity-25{--bs-bg-opacity:0.25}.bg-opacity-50{--bs-bg-opacity:0.5}.bg-opacity-75{--bs-bg-opacity:0.75}.bg-opacity-100{--bs-bg-opacity:1}.bg-gradient{background-image:var(--bs-gradient)!important}.user-select-all{-webkit-user-select:all!important;-moz-user-select:all!important;user-select:all!important}.user-select-auto{-webkit-user-select:auto!important;-moz-user-select:auto!important;user-select:auto!important}.user-select-none{-webkit-user-select:none!important;-moz-user-select:none!important;user-select:none!important}.pe-none{pointer-events:none!important}.pe-auto{pointer-events:auto!important}.rounded{border-radius:var(--bs-border-radius)!important}.rounded-0{border-radius:0!important}.rounded-1{border-radius:var(--bs-border-radius-sm)!important}.rounded-2{border-radius:var(--bs-border-radius)!important}.rounded-3{border-radius:var(--bs-border-radius-lg)!important}.rounded-4{border-radius:var(--bs-border-radius-xl)!important}.rounded-5{border-radius:var(--bs-border-radius-2xl)!important}.rounded-circle{border-radius:50%!important}.rounded-pill{border-radius:var(--bs-border-radius-pill)!important}.rounded-top{border-top-left-radius:var(--bs-border-radius)!important;border-top-right-radius:var(--bs-border-radius)!important}.rounded-end{border-top-right-radius:var(--bs-border-radius)!important;border-bottom-right-radius:var(--bs-border-radius)!important}.rounded-bottom{border-bottom-right-radius:var(--bs-border-radius)!important;border-bottom-left-radius:var(--bs-border-radius)!important}.rounded-start{border-bottom-left-radius:var(--bs-border-radius)!important;border-top-left-radius:var(--bs-border-radius)!important}.visible{visibility:visible!important}.invisible{visibility:hidden!important}@media (min-width:576px){.float-sm-start{float:left!important}.float-sm-end{float:right!important}.float-sm-none{float:none!important}.d-sm-inline{display:inline!important}.d-sm-inline-block{display:inline-block!important}.d-sm-block{display:block!important}.d-sm-grid{display:grid!important}.d-sm-table{display:table!important}.d-sm-table-row{display:table-row!important}.d-sm-table-cell{display:table-cell!important}.d-sm-flex{display:flex!important}.d-sm-inline-flex{display:inline-flex!important}.d-sm-none{display:none!important}.flex-sm-fill{flex:1 1 auto!important}.flex-sm-row{flex-direction:row!important}.flex-sm-column{flex-direction:column!important}.flex-sm-row-reverse{flex-direction:row-reverse!important}.flex-sm-column-reverse{flex-direction:column-reverse!important}.flex-sm-grow-0{flex-grow:0!important}.flex-sm-grow-1{flex-grow:1!important}.flex-sm-shrink-0{flex-shrink:0!important}.flex-sm-shrink-1{flex-shrink:1!important}.flex-sm-wrap{flex-wrap:wrap!important}.flex-sm-nowrap{flex-wrap:nowrap!important}.flex-sm-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-sm-start{justify-content:flex-start!important}.justify-content-sm-end{justify-content:flex-end!important}.justify-content-sm-center{justify-content:center!important}.justify-content-sm-between{justify-content:space-between!important}.justify-content-sm-around{justify-content:space-around!important}.justify-content-sm-evenly{justify-content:space-evenly!important}.align-items-sm-start{align-items:flex-start!important}.align-items-sm-end{align-items:flex-end!important}.align-items-sm-center{align-items:center!important}.align-items-sm-baseline{align-items:baseline!important}.align-items-sm-stretch{align-items:stretch!important}.align-content-sm-start{align-content:flex-start!important}.align-content-sm-end{align-content:flex-end!important}.align-content-sm-center{align-content:center!important}.align-content-sm-between{align-content:space-between!important}.align-content-sm-around{align-content:space-around!important}.align-content-sm-stretch{align-content:stretch!important}.align-self-sm-auto{align-self:auto!important}.align-self-sm-start{align-self:flex-start!important}.align-self-sm-end{align-self:flex-end!important}.align-self-sm-center{align-self:center!important}.align-self-sm-baseline{align-self:baseline!important}.align-self-sm-stretch{align-self:stretch!important}.order-sm-first{order:-1!important}.order-sm-0{order:0!important}.order-sm-1{order:1!important}.order-sm-2{order:2!important}.order-sm-3{order:3!important}.order-sm-4{order:4!important}.order-sm-5{order:5!important}.order-sm-last{order:6!important}.m-sm-0{margin:0!important}.m-sm-1{margin:.25rem!important}.m-sm-2{margin:.5rem!important}.m-sm-3{margin:1rem!important}.m-sm-4{margin:1.5rem!important}.m-sm-5{margin:3rem!important}.m-sm-auto{margin:auto!important}.mx-sm-0{margin-right:0!important;margin-left:0!important}.mx-sm-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-sm-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-sm-3{margin-right:1rem!important;margin-left:1rem!important}.mx-sm-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-sm-5{margin-right:3rem!important;margin-left:3rem!important}.mx-sm-auto{margin-right:auto!important;margin-left:auto!important}.my-sm-0{margin-top:0!important;margin-bottom:0!important}.my-sm-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-sm-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-sm-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-sm-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-sm-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-sm-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-sm-0{margin-top:0!important}.mt-sm-1{margin-top:.25rem!important}.mt-sm-2{margin-top:.5rem!important}.mt-sm-3{margin-top:1rem!important}.mt-sm-4{margin-top:1.5rem!important}.mt-sm-5{margin-top:3rem!important}.mt-sm-auto{margin-top:auto!important}.me-sm-0{margin-right:0!important}.me-sm-1{margin-right:.25rem!important}.me-sm-2{margin-right:.5rem!important}.me-sm-3{margin-right:1rem!important}.me-sm-4{margin-right:1.5rem!important}.me-sm-5{margin-right:3rem!important}.me-sm-auto{margin-right:auto!important}.mb-sm-0{margin-bottom:0!important}.mb-sm-1{margin-bottom:.25rem!important}.mb-sm-2{margin-bottom:.5rem!important}.mb-sm-3{margin-bottom:1rem!important}.mb-sm-4{margin-bottom:1.5rem!important}.mb-sm-5{margin-bottom:3rem!important}.mb-sm-auto{margin-bottom:auto!important}.ms-sm-0{margin-left:0!important}.ms-sm-1{margin-left:.25rem!important}.ms-sm-2{margin-left:.5rem!important}.ms-sm-3{margin-left:1rem!important}.ms-sm-4{margin-left:1.5rem!important}.ms-sm-5{margin-left:3rem!important}.ms-sm-auto{margin-left:auto!important}.p-sm-0{padding:0!important}.p-sm-1{padding:.25rem!important}.p-sm-2{padding:.5rem!important}.p-sm-3{padding:1rem!important}.p-sm-4{padding:1.5rem!important}.p-sm-5{padding:3rem!important}.px-sm-0{padding-right:0!important;padding-left:0!important}.px-sm-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-sm-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-sm-3{padding-right:1rem!important;padding-left:1rem!important}.px-sm-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-sm-5{padding-right:3rem!important;padding-left:3rem!important}.py-sm-0{padding-top:0!important;padding-bottom:0!important}.py-sm-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-sm-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-sm-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-sm-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-sm-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-sm-0{padding-top:0!important}.pt-sm-1{padding-top:.25rem!important}.pt-sm-2{padding-top:.5rem!important}.pt-sm-3{padding-top:1rem!important}.pt-sm-4{padding-top:1.5rem!important}.pt-sm-5{padding-top:3rem!important}.pe-sm-0{padding-right:0!important}.pe-sm-1{padding-right:.25rem!important}.pe-sm-2{padding-right:.5rem!important}.pe-sm-3{padding-right:1rem!important}.pe-sm-4{padding-right:1.5rem!important}.pe-sm-5{padding-right:3rem!important}.pb-sm-0{padding-bottom:0!important}.pb-sm-1{padding-bottom:.25rem!important}.pb-sm-2{padding-bottom:.5rem!important}.pb-sm-3{padding-bottom:1rem!important}.pb-sm-4{padding-bottom:1.5rem!important}.pb-sm-5{padding-bottom:3rem!important}.ps-sm-0{padding-left:0!important}.ps-sm-1{padding-left:.25rem!important}.ps-sm-2{padding-left:.5rem!important}.ps-sm-3{padding-left:1rem!important}.ps-sm-4{padding-left:1.5rem!important}.ps-sm-5{padding-left:3rem!important}.gap-sm-0{gap:0!important}.gap-sm-1{gap:.25rem!important}.gap-sm-2{gap:.5rem!important}.gap-sm-3{gap:1rem!important}.gap-sm-4{gap:1.5rem!important}.gap-sm-5{gap:3rem!important}.text-sm-start{text-align:left!important}.text-sm-end{text-align:right!important}.text-sm-center{text-align:center!important}}@media (min-width:768px){.float-md-start{float:left!important}.float-md-end{float:right!important}.float-md-none{float:none!important}.d-md-inline{display:inline!important}.d-md-inline-block{display:inline-block!important}.d-md-block{display:block!important}.d-md-grid{display:grid!important}.d-md-table{display:table!important}.d-md-table-row{display:table-row!important}.d-md-table-cell{display:table-cell!important}.d-md-flex{display:flex!important}.d-md-inline-flex{display:inline-flex!important}.d-md-none{display:none!important}.flex-md-fill{flex:1 1 auto!important}.flex-md-row{flex-direction:row!important}.flex-md-column{flex-direction:column!important}.flex-md-row-reverse{flex-direction:row-reverse!important}.flex-md-column-reverse{flex-direction:column-reverse!important}.flex-md-grow-0{flex-grow:0!important}.flex-md-grow-1{flex-grow:1!important}.flex-md-shrink-0{flex-shrink:0!important}.flex-md-shrink-1{flex-shrink:1!important}.flex-md-wrap{flex-wrap:wrap!important}.flex-md-nowrap{flex-wrap:nowrap!important}.flex-md-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-md-start{justify-content:flex-start!important}.justify-content-md-end{justify-content:flex-end!important}.justify-content-md-center{justify-content:center!important}.justify-content-md-between{justify-content:space-between!important}.justify-content-md-around{justify-content:space-around!important}.justify-content-md-evenly{justify-content:space-evenly!important}.align-items-md-start{align-items:flex-start!important}.align-items-md-end{align-items:flex-end!important}.align-items-md-center{align-items:center!important}.align-items-md-baseline{align-items:baseline!important}.align-items-md-stretch{align-items:stretch!important}.align-content-md-start{align-content:flex-start!important}.align-content-md-end{align-content:flex-end!important}.align-content-md-center{align-content:center!important}.align-content-md-between{align-content:space-between!important}.align-content-md-around{align-content:space-around!important}.align-content-md-stretch{align-content:stretch!important}.align-self-md-auto{align-self:auto!important}.align-self-md-start{align-self:flex-start!important}.align-self-md-end{align-self:flex-end!important}.align-self-md-center{align-self:center!important}.align-self-md-baseline{align-self:baseline!important}.align-self-md-stretch{align-self:stretch!important}.order-md-first{order:-1!important}.order-md-0{order:0!important}.order-md-1{order:1!important}.order-md-2{order:2!important}.order-md-3{order:3!important}.order-md-4{order:4!important}.order-md-5{order:5!important}.order-md-last{order:6!important}.m-md-0{margin:0!important}.m-md-1{margin:.25rem!important}.m-md-2{margin:.5rem!important}.m-md-3{margin:1rem!important}.m-md-4{margin:1.5rem!important}.m-md-5{margin:3rem!important}.m-md-auto{margin:auto!important}.mx-md-0{margin-right:0!important;margin-left:0!important}.mx-md-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-md-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-md-3{margin-right:1rem!important;margin-left:1rem!important}.mx-md-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-md-5{margin-right:3rem!important;margin-left:3rem!important}.mx-md-auto{margin-right:auto!important;margin-left:auto!important}.my-md-0{margin-top:0!important;margin-bottom:0!important}.my-md-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-md-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-md-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-md-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-md-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-md-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-md-0{margin-top:0!important}.mt-md-1{margin-top:.25rem!important}.mt-md-2{margin-top:.5rem!important}.mt-md-3{margin-top:1rem!important}.mt-md-4{margin-top:1.5rem!important}.mt-md-5{margin-top:3rem!important}.mt-md-auto{margin-top:auto!important}.me-md-0{margin-right:0!important}.me-md-1{margin-right:.25rem!important}.me-md-2{margin-right:.5rem!important}.me-md-3{margin-right:1rem!important}.me-md-4{margin-right:1.5rem!important}.me-md-5{margin-right:3rem!important}.me-md-auto{margin-right:auto!important}.mb-md-0{margin-bottom:0!important}.mb-md-1{margin-bottom:.25rem!important}.mb-md-2{margin-bottom:.5rem!important}.mb-md-3{margin-bottom:1rem!important}.mb-md-4{margin-bottom:1.5rem!important}.mb-md-5{margin-bottom:3rem!important}.mb-md-auto{margin-bottom:auto!important}.ms-md-0{margin-left:0!important}.ms-md-1{margin-left:.25rem!important}.ms-md-2{margin-left:.5rem!important}.ms-md-3{margin-left:1rem!important}.ms-md-4{margin-left:1.5rem!important}.ms-md-5{margin-left:3rem!important}.ms-md-auto{margin-left:auto!important}.p-md-0{padding:0!important}.p-md-1{padding:.25rem!important}.p-md-2{padding:.5rem!important}.p-md-3{padding:1rem!important}.p-md-4{padding:1.5rem!important}.p-md-5{padding:3rem!important}.px-md-0{padding-right:0!important;padding-left:0!important}.px-md-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-md-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-md-3{padding-right:1rem!important;padding-left:1rem!important}.px-md-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-md-5{padding-right:3rem!important;padding-left:3rem!important}.py-md-0{padding-top:0!important;padding-bottom:0!important}.py-md-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-md-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-md-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-md-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-md-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-md-0{padding-top:0!important}.pt-md-1{padding-top:.25rem!important}.pt-md-2{padding-top:.5rem!important}.pt-md-3{padding-top:1rem!important}.pt-md-4{padding-top:1.5rem!important}.pt-md-5{padding-top:3rem!important}.pe-md-0{padding-right:0!important}.pe-md-1{padding-right:.25rem!important}.pe-md-2{padding-right:.5rem!important}.pe-md-3{padding-right:1rem!important}.pe-md-4{padding-right:1.5rem!important}.pe-md-5{padding-right:3rem!important}.pb-md-0{padding-bottom:0!important}.pb-md-1{padding-bottom:.25rem!important}.pb-md-2{padding-bottom:.5rem!important}.pb-md-3{padding-bottom:1rem!important}.pb-md-4{padding-bottom:1.5rem!important}.pb-md-5{padding-bottom:3rem!important}.ps-md-0{padding-left:0!important}.ps-md-1{padding-left:.25rem!important}.ps-md-2{padding-left:.5rem!important}.ps-md-3{padding-left:1rem!important}.ps-md-4{padding-left:1.5rem!important}.ps-md-5{padding-left:3rem!important}.gap-md-0{gap:0!important}.gap-md-1{gap:.25rem!important}.gap-md-2{gap:.5rem!important}.gap-md-3{gap:1rem!important}.gap-md-4{gap:1.5rem!important}.gap-md-5{gap:3rem!important}.text-md-start{text-align:left!important}.text-md-end{text-align:right!important}.text-md-center{text-align:center!important}}@media (min-width:992px){.float-lg-start{float:left!important}.float-lg-end{float:right!important}.float-lg-none{float:none!important}.d-lg-inline{display:inline!important}.d-lg-inline-block{display:inline-block!important}.d-lg-block{display:block!important}.d-lg-grid{display:grid!important}.d-lg-table{display:table!important}.d-lg-table-row{display:table-row!important}.d-lg-table-cell{display:table-cell!important}.d-lg-flex{display:flex!important}.d-lg-inline-flex{display:inline-flex!important}.d-lg-none{display:none!important}.flex-lg-fill{flex:1 1 auto!important}.flex-lg-row{flex-direction:row!important}.flex-lg-column{flex-direction:column!important}.flex-lg-row-reverse{flex-direction:row-reverse!important}.flex-lg-column-reverse{flex-direction:column-reverse!important}.flex-lg-grow-0{flex-grow:0!important}.flex-lg-grow-1{flex-grow:1!important}.flex-lg-shrink-0{flex-shrink:0!important}.flex-lg-shrink-1{flex-shrink:1!important}.flex-lg-wrap{flex-wrap:wrap!important}.flex-lg-nowrap{flex-wrap:nowrap!important}.flex-lg-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-lg-start{justify-content:flex-start!important}.justify-content-lg-end{justify-content:flex-end!important}.justify-content-lg-center{justify-content:center!important}.justify-content-lg-between{justify-content:space-between!important}.justify-content-lg-around{justify-content:space-around!important}.justify-content-lg-evenly{justify-content:space-evenly!important}.align-items-lg-start{align-items:flex-start!important}.align-items-lg-end{align-items:flex-end!important}.align-items-lg-center{align-items:center!important}.align-items-lg-baseline{align-items:baseline!important}.align-items-lg-stretch{align-items:stretch!important}.align-content-lg-start{align-content:flex-start!important}.align-content-lg-end{align-content:flex-end!important}.align-content-lg-center{align-content:center!important}.align-content-lg-between{align-content:space-between!important}.align-content-lg-around{align-content:space-around!important}.align-content-lg-stretch{align-content:stretch!important}.align-self-lg-auto{align-self:auto!important}.align-self-lg-start{align-self:flex-start!important}.align-self-lg-end{align-self:flex-end!important}.align-self-lg-center{align-self:center!important}.align-self-lg-baseline{align-self:baseline!important}.align-self-lg-stretch{align-self:stretch!important}.order-lg-first{order:-1!important}.order-lg-0{order:0!important}.order-lg-1{order:1!important}.order-lg-2{order:2!important}.order-lg-3{order:3!important}.order-lg-4{order:4!important}.order-lg-5{order:5!important}.order-lg-last{order:6!important}.m-lg-0{margin:0!important}.m-lg-1{margin:.25rem!important}.m-lg-2{margin:.5rem!important}.m-lg-3{margin:1rem!important}.m-lg-4{margin:1.5rem!important}.m-lg-5{margin:3rem!important}.m-lg-auto{margin:auto!important}.mx-lg-0{margin-right:0!important;margin-left:0!important}.mx-lg-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-lg-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-lg-3{margin-right:1rem!important;margin-left:1rem!important}.mx-lg-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-lg-5{margin-right:3rem!important;margin-left:3rem!important}.mx-lg-auto{margin-right:auto!important;margin-left:auto!important}.my-lg-0{margin-top:0!important;margin-bottom:0!important}.my-lg-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-lg-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-lg-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-lg-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-lg-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-lg-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-lg-0{margin-top:0!important}.mt-lg-1{margin-top:.25rem!important}.mt-lg-2{margin-top:.5rem!important}.mt-lg-3{margin-top:1rem!important}.mt-lg-4{margin-top:1.5rem!important}.mt-lg-5{margin-top:3rem!important}.mt-lg-auto{margin-top:auto!important}.me-lg-0{margin-right:0!important}.me-lg-1{margin-right:.25rem!important}.me-lg-2{margin-right:.5rem!important}.me-lg-3{margin-right:1rem!important}.me-lg-4{margin-right:1.5rem!important}.me-lg-5{margin-right:3rem!important}.me-lg-auto{margin-right:auto!important}.mb-lg-0{margin-bottom:0!important}.mb-lg-1{margin-bottom:.25rem!important}.mb-lg-2{margin-bottom:.5rem!important}.mb-lg-3{margin-bottom:1rem!important}.mb-lg-4{margin-bottom:1.5rem!important}.mb-lg-5{margin-bottom:3rem!important}.mb-lg-auto{margin-bottom:auto!important}.ms-lg-0{margin-left:0!important}.ms-lg-1{margin-left:.25rem!important}.ms-lg-2{margin-left:.5rem!important}.ms-lg-3{margin-left:1rem!important}.ms-lg-4{margin-left:1.5rem!important}.ms-lg-5{margin-left:3rem!important}.ms-lg-auto{margin-left:auto!important}.p-lg-0{padding:0!important}.p-lg-1{padding:.25rem!important}.p-lg-2{padding:.5rem!important}.p-lg-3{padding:1rem!important}.p-lg-4{padding:1.5rem!important}.p-lg-5{padding:3rem!important}.px-lg-0{padding-right:0!important;padding-left:0!important}.px-lg-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-lg-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-lg-3{padding-right:1rem!important;padding-left:1rem!important}.px-lg-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-lg-5{padding-right:3rem!important;padding-left:3rem!important}.py-lg-0{padding-top:0!important;padding-bottom:0!important}.py-lg-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-lg-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-lg-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-lg-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-lg-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-lg-0{padding-top:0!important}.pt-lg-1{padding-top:.25rem!important}.pt-lg-2{padding-top:.5rem!important}.pt-lg-3{padding-top:1rem!important}.pt-lg-4{padding-top:1.5rem!important}.pt-lg-5{padding-top:3rem!important}.pe-lg-0{padding-right:0!important}.pe-lg-1{padding-right:.25rem!important}.pe-lg-2{padding-right:.5rem!important}.pe-lg-3{padding-right:1rem!important}.pe-lg-4{padding-right:1.5rem!important}.pe-lg-5{padding-right:3rem!important}.pb-lg-0{padding-bottom:0!important}.pb-lg-1{padding-bottom:.25rem!important}.pb-lg-2{padding-bottom:.5rem!important}.pb-lg-3{padding-bottom:1rem!important}.pb-lg-4{padding-bottom:1.5rem!important}.pb-lg-5{padding-bottom:3rem!important}.ps-lg-0{padding-left:0!important}.ps-lg-1{padding-left:.25rem!important}.ps-lg-2{padding-left:.5rem!important}.ps-lg-3{padding-left:1rem!important}.ps-lg-4{padding-left:1.5rem!important}.ps-lg-5{padding-left:3rem!important}.gap-lg-0{gap:0!important}.gap-lg-1{gap:.25rem!important}.gap-lg-2{gap:.5rem!important}.gap-lg-3{gap:1rem!important}.gap-lg-4{gap:1.5rem!important}.gap-lg-5{gap:3rem!important}.text-lg-start{text-align:left!important}.text-lg-end{text-align:right!important}.text-lg-center{text-align:center!important}}@media (min-width:1200px){.float-xl-start{float:left!important}.float-xl-end{float:right!important}.float-xl-none{float:none!important}.d-xl-inline{display:inline!important}.d-xl-inline-block{display:inline-block!important}.d-xl-block{display:block!important}.d-xl-grid{display:grid!important}.d-xl-table{display:table!important}.d-xl-table-row{display:table-row!important}.d-xl-table-cell{display:table-cell!important}.d-xl-flex{display:flex!important}.d-xl-inline-flex{display:inline-flex!important}.d-xl-none{display:none!important}.flex-xl-fill{flex:1 1 auto!important}.flex-xl-row{flex-direction:row!important}.flex-xl-column{flex-direction:column!important}.flex-xl-row-reverse{flex-direction:row-reverse!important}.flex-xl-column-reverse{flex-direction:column-reverse!important}.flex-xl-grow-0{flex-grow:0!important}.flex-xl-grow-1{flex-grow:1!important}.flex-xl-shrink-0{flex-shrink:0!important}.flex-xl-shrink-1{flex-shrink:1!important}.flex-xl-wrap{flex-wrap:wrap!important}.flex-xl-nowrap{flex-wrap:nowrap!important}.flex-xl-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-xl-start{justify-content:flex-start!important}.justify-content-xl-end{justify-content:flex-end!important}.justify-content-xl-center{justify-content:center!important}.justify-content-xl-between{justify-content:space-between!important}.justify-content-xl-around{justify-content:space-around!important}.justify-content-xl-evenly{justify-content:space-evenly!important}.align-items-xl-start{align-items:flex-start!important}.align-items-xl-end{align-items:flex-end!important}.align-items-xl-center{align-items:center!important}.align-items-xl-baseline{align-items:baseline!important}.align-items-xl-stretch{align-items:stretch!important}.align-content-xl-start{align-content:flex-start!important}.align-content-xl-end{align-content:flex-end!important}.align-content-xl-center{align-content:center!important}.align-content-xl-between{align-content:space-between!important}.align-content-xl-around{align-content:space-around!important}.align-content-xl-stretch{align-content:stretch!important}.align-self-xl-auto{align-self:auto!important}.align-self-xl-start{align-self:flex-start!important}.align-self-xl-end{align-self:flex-end!important}.align-self-xl-center{align-self:center!important}.align-self-xl-baseline{align-self:baseline!important}.align-self-xl-stretch{align-self:stretch!important}.order-xl-first{order:-1!important}.order-xl-0{order:0!important}.order-xl-1{order:1!important}.order-xl-2{order:2!important}.order-xl-3{order:3!important}.order-xl-4{order:4!important}.order-xl-5{order:5!important}.order-xl-last{order:6!important}.m-xl-0{margin:0!important}.m-xl-1{margin:.25rem!important}.m-xl-2{margin:.5rem!important}.m-xl-3{margin:1rem!important}.m-xl-4{margin:1.5rem!important}.m-xl-5{margin:3rem!important}.m-xl-auto{margin:auto!important}.mx-xl-0{margin-right:0!important;margin-left:0!important}.mx-xl-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-xl-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-xl-3{margin-right:1rem!important;margin-left:1rem!important}.mx-xl-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-xl-5{margin-right:3rem!important;margin-left:3rem!important}.mx-xl-auto{margin-right:auto!important;margin-left:auto!important}.my-xl-0{margin-top:0!important;margin-bottom:0!important}.my-xl-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-xl-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-xl-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-xl-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-xl-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-xl-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-xl-0{margin-top:0!important}.mt-xl-1{margin-top:.25rem!important}.mt-xl-2{margin-top:.5rem!important}.mt-xl-3{margin-top:1rem!important}.mt-xl-4{margin-top:1.5rem!important}.mt-xl-5{margin-top:3rem!important}.mt-xl-auto{margin-top:auto!important}.me-xl-0{margin-right:0!important}.me-xl-1{margin-right:.25rem!important}.me-xl-2{margin-right:.5rem!important}.me-xl-3{margin-right:1rem!important}.me-xl-4{margin-right:1.5rem!important}.me-xl-5{margin-right:3rem!important}.me-xl-auto{margin-right:auto!important}.mb-xl-0{margin-bottom:0!important}.mb-xl-1{margin-bottom:.25rem!important}.mb-xl-2{margin-bottom:.5rem!important}.mb-xl-3{margin-bottom:1rem!important}.mb-xl-4{margin-bottom:1.5rem!important}.mb-xl-5{margin-bottom:3rem!important}.mb-xl-auto{margin-bottom:auto!important}.ms-xl-0{margin-left:0!important}.ms-xl-1{margin-left:.25rem!important}.ms-xl-2{margin-left:.5rem!important}.ms-xl-3{margin-left:1rem!important}.ms-xl-4{margin-left:1.5rem!important}.ms-xl-5{margin-left:3rem!important}.ms-xl-auto{margin-left:auto!important}.p-xl-0{padding:0!important}.p-xl-1{padding:.25rem!important}.p-xl-2{padding:.5rem!important}.p-xl-3{padding:1rem!important}.p-xl-4{padding:1.5rem!important}.p-xl-5{padding:3rem!important}.px-xl-0{padding-right:0!important;padding-left:0!important}.px-xl-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-xl-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-xl-3{padding-right:1rem!important;padding-left:1rem!important}.px-xl-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-xl-5{padding-right:3rem!important;padding-left:3rem!important}.py-xl-0{padding-top:0!important;padding-bottom:0!important}.py-xl-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-xl-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-xl-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-xl-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-xl-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-xl-0{padding-top:0!important}.pt-xl-1{padding-top:.25rem!important}.pt-xl-2{padding-top:.5rem!important}.pt-xl-3{padding-top:1rem!important}.pt-xl-4{padding-top:1.5rem!important}.pt-xl-5{padding-top:3rem!important}.pe-xl-0{padding-right:0!important}.pe-xl-1{padding-right:.25rem!important}.pe-xl-2{padding-right:.5rem!important}.pe-xl-3{padding-right:1rem!important}.pe-xl-4{padding-right:1.5rem!important}.pe-xl-5{padding-right:3rem!important}.pb-xl-0{padding-bottom:0!important}.pb-xl-1{padding-bottom:.25rem!important}.pb-xl-2{padding-bottom:.5rem!important}.pb-xl-3{padding-bottom:1rem!important}.pb-xl-4{padding-bottom:1.5rem!important}.pb-xl-5{padding-bottom:3rem!important}.ps-xl-0{padding-left:0!important}.ps-xl-1{padding-left:.25rem!important}.ps-xl-2{padding-left:.5rem!important}.ps-xl-3{padding-left:1rem!important}.ps-xl-4{padding-left:1.5rem!important}.ps-xl-5{padding-left:3rem!important}.gap-xl-0{gap:0!important}.gap-xl-1{gap:.25rem!important}.gap-xl-2{gap:.5rem!important}.gap-xl-3{gap:1rem!important}.gap-xl-4{gap:1.5rem!important}.gap-xl-5{gap:3rem!important}.text-xl-start{text-align:left!important}.text-xl-end{text-align:right!important}.text-xl-center{text-align:center!important}}@media (min-width:1400px){.float-xxl-start{float:left!important}.float-xxl-end{float:right!important}.float-xxl-none{float:none!important}.d-xxl-inline{display:inline!important}.d-xxl-inline-block{display:inline-block!important}.d-xxl-block{display:block!important}.d-xxl-grid{display:grid!important}.d-xxl-table{display:table!important}.d-xxl-table-row{display:table-row!important}.d-xxl-table-cell{display:table-cell!important}.d-xxl-flex{display:flex!important}.d-xxl-inline-flex{display:inline-flex!important}.d-xxl-none{display:none!important}.flex-xxl-fill{flex:1 1 auto!important}.flex-xxl-row{flex-direction:row!important}.flex-xxl-column{flex-direction:column!important}.flex-xxl-row-reverse{flex-direction:row-reverse!important}.flex-xxl-column-reverse{flex-direction:column-reverse!important}.flex-xxl-grow-0{flex-grow:0!important}.flex-xxl-grow-1{flex-grow:1!important}.flex-xxl-shrink-0{flex-shrink:0!important}.flex-xxl-shrink-1{flex-shrink:1!important}.flex-xxl-wrap{flex-wrap:wrap!important}.flex-xxl-nowrap{flex-wrap:nowrap!important}.flex-xxl-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-xxl-start{justify-content:flex-start!important}.justify-content-xxl-end{justify-content:flex-end!important}.justify-content-xxl-center{justify-content:center!important}.justify-content-xxl-between{justify-content:space-between!important}.justify-content-xxl-around{justify-content:space-around!important}.justify-content-xxl-evenly{justify-content:space-evenly!important}.align-items-xxl-start{align-items:flex-start!important}.align-items-xxl-end{align-items:flex-end!important}.align-items-xxl-center{align-items:center!important}.align-items-xxl-baseline{align-items:baseline!important}.align-items-xxl-stretch{align-items:stretch!important}.align-content-xxl-start{align-content:flex-start!important}.align-content-xxl-end{align-content:flex-end!important}.align-content-xxl-center{align-content:center!important}.align-content-xxl-between{align-content:space-between!important}.align-content-xxl-around{align-content:space-around!important}.align-content-xxl-stretch{align-content:stretch!important}.align-self-xxl-auto{align-self:auto!important}.align-self-xxl-start{align-self:flex-start!important}.align-self-xxl-end{align-self:flex-end!important}.align-self-xxl-center{align-self:center!important}.align-self-xxl-baseline{align-self:baseline!important}.align-self-xxl-stretch{align-self:stretch!important}.order-xxl-first{order:-1!important}.order-xxl-0{order:0!important}.order-xxl-1{order:1!important}.order-xxl-2{order:2!important}.order-xxl-3{order:3!important}.order-xxl-4{order:4!important}.order-xxl-5{order:5!important}.order-xxl-last{order:6!important}.m-xxl-0{margin:0!important}.m-xxl-1{margin:.25rem!important}.m-xxl-2{margin:.5rem!important}.m-xxl-3{margin:1rem!important}.m-xxl-4{margin:1.5rem!important}.m-xxl-5{margin:3rem!important}.m-xxl-auto{margin:auto!important}.mx-xxl-0{margin-right:0!important;margin-left:0!important}.mx-xxl-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-xxl-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-xxl-3{margin-right:1rem!important;margin-left:1rem!important}.mx-xxl-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-xxl-5{margin-right:3rem!important;margin-left:3rem!important}.mx-xxl-auto{margin-right:auto!important;margin-left:auto!important}.my-xxl-0{margin-top:0!important;margin-bottom:0!important}.my-xxl-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-xxl-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-xxl-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-xxl-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-xxl-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-xxl-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-xxl-0{margin-top:0!important}.mt-xxl-1{margin-top:.25rem!important}.mt-xxl-2{margin-top:.5rem!important}.mt-xxl-3{margin-top:1rem!important}.mt-xxl-4{margin-top:1.5rem!important}.mt-xxl-5{margin-top:3rem!important}.mt-xxl-auto{margin-top:auto!important}.me-xxl-0{margin-right:0!important}.me-xxl-1{margin-right:.25rem!important}.me-xxl-2{margin-right:.5rem!important}.me-xxl-3{margin-right:1rem!important}.me-xxl-4{margin-right:1.5rem!important}.me-xxl-5{margin-right:3rem!important}.me-xxl-auto{margin-right:auto!important}.mb-xxl-0{margin-bottom:0!important}.mb-xxl-1{margin-bottom:.25rem!important}.mb-xxl-2{margin-bottom:.5rem!important}.mb-xxl-3{margin-bottom:1rem!important}.mb-xxl-4{margin-bottom:1.5rem!important}.mb-xxl-5{margin-bottom:3rem!important}.mb-xxl-auto{margin-bottom:auto!important}.ms-xxl-0{margin-left:0!important}.ms-xxl-1{margin-left:.25rem!important}.ms-xxl-2{margin-left:.5rem!important}.ms-xxl-3{margin-left:1rem!important}.ms-xxl-4{margin-left:1.5rem!important}.ms-xxl-5{margin-left:3rem!important}.ms-xxl-auto{margin-left:auto!important}.p-xxl-0{padding:0!important}.p-xxl-1{padding:.25rem!important}.p-xxl-2{padding:.5rem!important}.p-xxl-3{padding:1rem!important}.p-xxl-4{padding:1.5rem!important}.p-xxl-5{padding:3rem!important}.px-xxl-0{padding-right:0!important;padding-left:0!important}.px-xxl-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-xxl-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-xxl-3{padding-right:1rem!important;padding-left:1rem!important}.px-xxl-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-xxl-5{padding-right:3rem!important;padding-left:3rem!important}.py-xxl-0{padding-top:0!important;padding-bottom:0!important}.py-xxl-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-xxl-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-xxl-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-xxl-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-xxl-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-xxl-0{padding-top:0!important}.pt-xxl-1{padding-top:.25rem!important}.pt-xxl-2{padding-top:.5rem!important}.pt-xxl-3{padding-top:1rem!important}.pt-xxl-4{padding-top:1.5rem!important}.pt-xxl-5{padding-top:3rem!important}.pe-xxl-0{padding-right:0!important}.pe-xxl-1{padding-right:.25rem!important}.pe-xxl-2{padding-right:.5rem!important}.pe-xxl-3{padding-right:1rem!important}.pe-xxl-4{padding-right:1.5rem!important}.pe-xxl-5{padding-right:3rem!important}.pb-xxl-0{padding-bottom:0!important}.pb-xxl-1{padding-bottom:.25rem!important}.pb-xxl-2{padding-bottom:.5rem!important}.pb-xxl-3{padding-bottom:1rem!important}.pb-xxl-4{padding-bottom:1.5rem!important}.pb-xxl-5{padding-bottom:3rem!important}.ps-xxl-0{padding-left:0!important}.ps-xxl-1{padding-left:.25rem!important}.ps-xxl-2{padding-left:.5rem!important}.ps-xxl-3{padding-left:1rem!important}.ps-xxl-4{padding-left:1.5rem!important}.ps-xxl-5{padding-left:3rem!important}.gap-xxl-0{gap:0!important}.gap-xxl-1{gap:.25rem!important}.gap-xxl-2{gap:.5rem!important}.gap-xxl-3{gap:1rem!important}.gap-xxl-4{gap:1.5rem!important}.gap-xxl-5{gap:3rem!important}.text-xxl-start{text-align:left!important}.text-xxl-end{text-align:right!important}.text-xxl-center{text-align:center!important}}@media (min-width:1200px){.fs-1{font-size:2rem!important}.fs-2{font-size:1.75rem!important}.fs-3{font-size:1.5rem!important}}@media print{.d-print-inline{display:inline!important}.d-print-inline-block{display:inline-block!important}.d-print-block{display:block!important}.d-print-grid{display:grid!important}.d-print-table{display:table!important}.d-print-table-row{display:table-row!important}.d-print-table-cell{display:table-cell!important}.d-print-flex{display:flex!important}.d-print-inline-flex{display:inline-flex!important}.d-print-none{display:none!important}} -/*# sourceMappingURL=bootstrap.min.css.map */ \ No newline at end of file diff --git a/static/logo-full-ru-dark.svg b/static/logo-full-ru-dark.svg deleted file mode 100644 index a8a41c2..0000000 --- a/static/logo-full-ru-dark.svg +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/static/logo-lg.svg b/static/logo-lg.svg deleted file mode 100644 index 63a2208..0000000 --- a/static/logo-lg.svg +++ /dev/null @@ -1,8 +0,0 @@ - - - - - - - - diff --git a/static/logo.svg b/static/logo.svg deleted file mode 100644 index 741bbe4..0000000 --- a/static/logo.svg +++ /dev/null @@ -1,28 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/static/marker-sm-red.png b/static/marker-sm-red.png deleted file mode 100644 index 62806ab..0000000 Binary files a/static/marker-sm-red.png and /dev/null differ diff --git a/static/pop-marker.png b/static/pop-marker.png deleted file mode 100644 index a111bfe..0000000 Binary files a/static/pop-marker.png and /dev/null differ diff --git a/static/target-blue.png b/static/target-blue.png deleted file mode 100644 index 6ef9e5b..0000000 Binary files a/static/target-blue.png and /dev/null differ diff --git a/static/target-red.png b/static/target-red.png deleted file mode 100644 index c3222b8..0000000 Binary files a/static/target-red.png and /dev/null differ diff --git a/svelte.config.js b/svelte.config.js index b591c22..10c4eeb 100644 --- a/svelte.config.js +++ b/svelte.config.js @@ -1,34 +1,13 @@ -import adapter from '@sveltejs/adapter-static'; +import adapter from '@sveltejs/adapter-auto'; -/** - * This project is deployed as a pure static SPA against a Django REST backend. - * SSR/hydration are disabled; all routing is done client-side. - * - * `fallback: 'index.html'` makes every unknown path serve index.html so the - * SvelteKit client router can handle it. - * - * @type {import('@sveltejs/kit').Config} - */ +/** @type {import('@sveltejs/kit').Config} */ const config = { kit: { - adapter: adapter({ - pages: 'build', - assets: 'build', - fallback: 'index.html', - precompress: false, - strict: false, - }), - alias: { - $api: 'src/lib/api', - $auth: 'src/lib/auth', - $domain: 'src/lib/domain', - $map: 'src/lib/map', - $state: 'src/lib/state', - $ui: 'src/lib/ui', - $i18n: 'src/lib/i18n', - $features: 'src/lib/features', - }, - }, + // adapter-auto only supports some environments, see https://svelte.dev/docs/kit/adapter-auto for a list. + // If your environment is not supported, or you settled on a specific environment, switch out the adapter. + // See https://svelte.dev/docs/kit/adapters for more information about adapters. + adapter: adapter() + } }; export default config; diff --git a/test-results/.last-run.json b/test-results/.last-run.json deleted file mode 100644 index cbcc1fb..0000000 --- a/test-results/.last-run.json +++ /dev/null @@ -1,4 +0,0 @@ -{ - "status": "passed", - "failedTests": [] -} \ No newline at end of file diff --git a/test.json b/test.json new file mode 100644 index 0000000..fdbd53b --- /dev/null +++ b/test.json @@ -0,0 +1,920 @@ +{ + "metadata": { + "complete_datetime": "2025-03-31T11:21:22.334824Z", + "start_datetime": "2025-03-31T11:21:22.328084Z" + }, + "prediction": [ + { + "stage": "ascent", + "trajectory": [ + { + "altitude": 0.0, + "datetime": "2025-03-31T13:13:00Z", + "latitude": 56.4017, + "longitude": 38.7543 + }, + { + "altitude": 300.0, + "datetime": "2025-03-31T13:14:00Z", + "latitude": 56.40215054172678, + "longitude": 38.75157050170719 + }, + { + "altitude": 600.0, + "datetime": "2025-03-31T13:15:00Z", + "latitude": 56.40332498907658, + "longitude": 38.74615166519835 + }, + { + "altitude": 900.0, + "datetime": "2025-03-31T13:16:00Z", + "latitude": 56.404753828602864, + "longitude": 38.740408034014045 + }, + { + "altitude": 1200.0, + "datetime": "2025-03-31T13:17:00Z", + "latitude": 56.40632110694436, + "longitude": 38.73477312828467 + }, + { + "altitude": 1500.0, + "datetime": "2025-03-31T13:18:00Z", + "latitude": 56.408012192686506, + "longitude": 38.72933807024897 + }, + { + "altitude": 1800.0, + "datetime": "2025-03-31T13:19:00Z", + "latitude": 56.409860988769495, + "longitude": 38.72424201583512 + }, + { + "altitude": 2100.0, + "datetime": "2025-03-31T13:20:00Z", + "latitude": 56.411986735157, + "longitude": 38.71981897537694 + }, + { + "altitude": 2400.0, + "datetime": "2025-03-31T13:21:00Z", + "latitude": 56.41460314194559, + "longitude": 38.71651325684098 + }, + { + "altitude": 2700.0, + "datetime": "2025-03-31T13:22:00Z", + "latitude": 56.417445830879565, + "longitude": 38.71449460977531 + }, + { + "altitude": 3000.0, + "datetime": "2025-03-31T13:23:00Z", + "latitude": 56.420260980621514, + "longitude": 38.71353347807817 + }, + { + "altitude": 3300.0, + "datetime": "2025-03-31T13:24:00Z", + "latitude": 56.42300526643632, + "longitude": 38.71364047695437 + }, + { + "altitude": 3600.0, + "datetime": "2025-03-31T13:25:00Z", + "latitude": 56.42560244234647, + "longitude": 38.71500279488988 + }, + { + "altitude": 3900.0, + "datetime": "2025-03-31T13:26:00Z", + "latitude": 56.42800474919969, + "longitude": 38.71771218331113 + }, + { + "altitude": 4200.0, + "datetime": "2025-03-31T13:27:00Z", + "latitude": 56.43035247557277, + "longitude": 38.721704453343484 + }, + { + "altitude": 4500.0, + "datetime": "2025-03-31T13:28:00Z", + "latitude": 56.43294445852403, + "longitude": 38.726776262051736 + }, + { + "altitude": 4800.0, + "datetime": "2025-03-31T13:29:00Z", + "latitude": 56.4359630588216, + "longitude": 38.732677743925045 + }, + { + "altitude": 5100.0, + "datetime": "2025-03-31T13:30:00Z", + "latitude": 56.4393360045026, + "longitude": 38.73926876572009 + }, + { + "altitude": 5400.0, + "datetime": "2025-03-31T13:31:00Z", + "latitude": 56.44288055157443, + "longitude": 38.74658518806238 + }, + { + "altitude": 5700.0, + "datetime": "2025-03-31T13:32:00Z", + "latitude": 56.446434338278614, + "longitude": 38.754772972333114 + }, + { + "altitude": 6000.0, + "datetime": "2025-03-31T13:33:00Z", + "latitude": 56.44981674584807, + "longitude": 38.763758146134066 + }, + { + "altitude": 6300.0, + "datetime": "2025-03-31T13:34:00Z", + "latitude": 56.452990230581186, + "longitude": 38.77423265117657 + }, + { + "altitude": 6600.0, + "datetime": "2025-03-31T13:35:00Z", + "latitude": 56.45608905533306, + "longitude": 38.787242969809995 + }, + { + "altitude": 6900.0, + "datetime": "2025-03-31T13:36:00Z", + "latitude": 56.45942357381433, + "longitude": 38.80371893271402 + }, + { + "altitude": 7200.0, + "datetime": "2025-03-31T13:37:00Z", + "latitude": 56.463735777260574, + "longitude": 38.82278590796712 + }, + { + "altitude": 7500.0, + "datetime": "2025-03-31T13:38:00Z", + "latitude": 56.468933957558775, + "longitude": 38.84377272173889 + }, + { + "altitude": 7800.0, + "datetime": "2025-03-31T13:39:00Z", + "latitude": 56.47468060202706, + "longitude": 38.86605826280632 + }, + { + "altitude": 8100.0, + "datetime": "2025-03-31T13:40:00Z", + "latitude": 56.48044282662076, + "longitude": 38.88880384040967 + }, + { + "altitude": 8400.0, + "datetime": "2025-03-31T13:41:00Z", + "latitude": 56.486163231471835, + "longitude": 38.91210189676355 + }, + { + "altitude": 8700.0, + "datetime": "2025-03-31T13:42:00Z", + "latitude": 56.49192248431474, + "longitude": 38.936345306832436 + }, + { + "altitude": 9000.0, + "datetime": "2025-03-31T13:43:00Z", + "latitude": 56.49833631849528, + "longitude": 38.96241622178067 + }, + { + "altitude": 9300.0, + "datetime": "2025-03-31T13:44:00Z", + "latitude": 56.50571182964095, + "longitude": 38.990669274952154 + }, + { + "altitude": 9600.0, + "datetime": "2025-03-31T13:45:00Z", + "latitude": 56.51337025687285, + "longitude": 39.019049558659006 + }, + { + "altitude": 9900.0, + "datetime": "2025-03-31T13:46:00Z", + "latitude": 56.521125070840114, + "longitude": 39.04721760675422 + }, + { + "altitude": 10200.0, + "datetime": "2025-03-31T13:47:00Z", + "latitude": 56.528575230662696, + "longitude": 39.07522154566774 + }, + { + "altitude": 10500.0, + "datetime": "2025-03-31T13:48:00Z", + "latitude": 56.53564629115801, + "longitude": 39.10274223821664 + }, + { + "altitude": 10800.0, + "datetime": "2025-03-31T13:49:00Z", + "latitude": 56.542000436538515, + "longitude": 39.12758147402441 + }, + { + "altitude": 11100.0, + "datetime": "2025-03-31T13:50:00Z", + "latitude": 56.54762174602161, + "longitude": 39.149506604684426 + }, + { + "altitude": 11400.0, + "datetime": "2025-03-31T13:51:00Z", + "latitude": 56.552759311087236, + "longitude": 39.16853616202282 + }, + { + "altitude": 11700.0, + "datetime": "2025-03-31T13:52:00Z", + "latitude": 56.557505594951245, + "longitude": 39.18467406297027 + }, + { + "altitude": 12000.0, + "datetime": "2025-03-31T13:53:00Z", + "latitude": 56.561819700764815, + "longitude": 39.19878440563311 + }, + { + "altitude": 12300.0, + "datetime": "2025-03-31T13:54:00Z", + "latitude": 56.56562282293917, + "longitude": 39.21216690199406 + }, + { + "altitude": 12600.0, + "datetime": "2025-03-31T13:55:00Z", + "latitude": 56.56891800612863, + "longitude": 39.2248781006055 + }, + { + "altitude": 12900.0, + "datetime": "2025-03-31T13:56:00Z", + "latitude": 56.571963932626076, + "longitude": 39.236946605630145 + }, + { + "altitude": 13200.0, + "datetime": "2025-03-31T13:57:00Z", + "latitude": 56.57497942709497, + "longitude": 39.248392073486414 + }, + { + "altitude": 13500.0, + "datetime": "2025-03-31T13:58:00Z", + "latitude": 56.577963607974155, + "longitude": 39.259214803835036 + }, + { + "altitude": 13800.0, + "datetime": "2025-03-31T13:59:00Z", + "latitude": 56.58084348918746, + "longitude": 39.26981548803511 + }, + { + "altitude": 14100.0, + "datetime": "2025-03-31T14:00:00Z", + "latitude": 56.583465279162795, + "longitude": 39.28103438452099 + }, + { + "altitude": 14400.0, + "datetime": "2025-03-31T14:01:00Z", + "latitude": 56.585816798254065, + "longitude": 39.29291674080829 + }, + { + "altitude": 14700.0, + "datetime": "2025-03-31T14:02:00Z", + "latitude": 56.58789458571269, + "longitude": 39.305459331653864 + }, + { + "altitude": 15000.0, + "datetime": "2025-03-31T14:03:00Z", + "latitude": 56.58978065716326, + "longitude": 39.31841453898297 + }, + { + "altitude": 15300.0, + "datetime": "2025-03-31T14:04:00Z", + "latitude": 56.59159078709161, + "longitude": 39.331443835207125 + }, + { + "altitude": 15600.0, + "datetime": "2025-03-31T14:05:00Z", + "latitude": 56.59332852408948, + "longitude": 39.344533392286614 + }, + { + "altitude": 15900.0, + "datetime": "2025-03-31T14:06:00Z", + "latitude": 56.59499387347947, + "longitude": 39.35767928204484 + }, + { + "altitude": 16200.0, + "datetime": "2025-03-31T14:07:00Z", + "latitude": 56.5965884110844, + "longitude": 39.37087665489451 + }, + { + "altitude": 16500.0, + "datetime": "2025-03-31T14:08:00Z", + "latitude": 56.59814359766257, + "longitude": 39.38410721551844 + }, + { + "altitude": 16800.0, + "datetime": "2025-03-31T14:09:00Z", + "latitude": 56.599676993329204, + "longitude": 39.397363673908345 + }, + { + "altitude": 17100.0, + "datetime": "2025-03-31T14:10:00Z", + "latitude": 56.60118963841368, + "longitude": 39.41064760654967 + }, + { + "altitude": 17400.0, + "datetime": "2025-03-31T14:11:00Z", + "latitude": 56.60268255950781, + "longitude": 39.42396060064398 + }, + { + "altitude": 17700.0, + "datetime": "2025-03-31T14:12:00Z", + "latitude": 56.604156769185536, + "longitude": 39.437304255012414 + }, + { + "altitude": 18000.0, + "datetime": "2025-03-31T14:13:00Z", + "latitude": 56.60561326576004, + "longitude": 39.450680180910034 + }, + { + "altitude": 18300.0, + "datetime": "2025-03-31T14:14:00Z", + "latitude": 56.60705303307887, + "longitude": 39.46409000275508 + }, + { + "altitude": 18600.0, + "datetime": "2025-03-31T14:15:00Z", + "latitude": 56.60850213034999, + "longitude": 39.47764104013138 + }, + { + "altitude": 18900.0, + "datetime": "2025-03-31T14:16:00Z", + "latitude": 56.610034776290696, + "longitude": 39.491647799081484 + }, + { + "altitude": 19200.0, + "datetime": "2025-03-31T14:17:00Z", + "latitude": 56.61165551489344, + "longitude": 39.50613562484206 + }, + { + "altitude": 19500.0, + "datetime": "2025-03-31T14:18:00Z", + "latitude": 56.613363136267466, + "longitude": 39.52110682966359 + }, + { + "altitude": 19800.0, + "datetime": "2025-03-31T14:19:00Z", + "latitude": 56.61515661819169, + "longitude": 39.53656119291367 + }, + { + "altitude": 20100.0, + "datetime": "2025-03-31T14:20:00Z", + "latitude": 56.61703503353181, + "longitude": 39.552497932157735 + }, + { + "altitude": 20400.0, + "datetime": "2025-03-31T14:21:00Z", + "latitude": 56.61899750619939, + "longitude": 39.56891601522489 + }, + { + "altitude": 20700.0, + "datetime": "2025-03-31T14:22:00Z", + "latitude": 56.62107263491984, + "longitude": 39.585834731780864 + }, + { + "altitude": 21000.0, + "datetime": "2025-03-31T14:23:00Z", + "latitude": 56.62339157872715, + "longitude": 39.60334774628025 + }, + { + "altitude": 21300.0, + "datetime": "2025-03-31T14:24:00Z", + "latitude": 56.625962830157604, + "longitude": 39.621466648390324 + }, + { + "altitude": 21600.0, + "datetime": "2025-03-31T14:25:00Z", + "latitude": 56.628783261295915, + "longitude": 39.640195181039765 + }, + { + "altitude": 21900.0, + "datetime": "2025-03-31T14:26:00Z", + "latitude": 56.6318496870023, + "longitude": 39.65953716616418 + }, + { + "altitude": 22200.0, + "datetime": "2025-03-31T14:27:00Z", + "latitude": 56.63515887140614, + "longitude": 39.67949652491455 + }, + { + "altitude": 22500.0, + "datetime": "2025-03-31T14:28:00Z", + "latitude": 56.63870753453714, + "longitude": 39.70007729903087 + }, + { + "altitude": 22800.0, + "datetime": "2025-03-31T14:29:00Z", + "latitude": 56.64249235906931, + "longitude": 39.7212836733661 + }, + { + "altitude": 23100.0, + "datetime": "2025-03-31T14:30:00Z", + "latitude": 56.646509997154055, + "longitude": 39.743119999548064 + }, + { + "altitude": 23400.0, + "datetime": "2025-03-31T14:31:00Z", + "latitude": 56.65075707731941, + "longitude": 39.765590820770655 + }, + { + "altitude": 23700.0, + "datetime": "2025-03-31T14:32:00Z", + "latitude": 56.655230211413766, + "longitude": 39.78870089770936 + }, + { + "altitude": 24000.0, + "datetime": "2025-03-31T14:33:00Z", + "latitude": 56.659838417787675, + "longitude": 39.81183579308032 + }, + { + "altitude": 24300.0, + "datetime": "2025-03-31T14:34:00Z", + "latitude": 56.66437164857745, + "longitude": 39.833503371349664 + }, + { + "altitude": 24600.0, + "datetime": "2025-03-31T14:35:00Z", + "latitude": 56.66881997113293, + "longitude": 39.85359515134159 + }, + { + "altitude": 24900.0, + "datetime": "2025-03-31T14:36:00Z", + "latitude": 56.67318719704676, + "longitude": 39.8720995627054 + }, + { + "altitude": 25200.0, + "datetime": "2025-03-31T14:37:00Z", + "latitude": 56.67747715758913, + "longitude": 39.88900578369577 + }, + { + "altitude": 25500.0, + "datetime": "2025-03-31T14:38:00Z", + "latitude": 56.68169373540689, + "longitude": 39.9043038176483 + }, + { + "altitude": 25800.0, + "datetime": "2025-03-31T14:39:00Z", + "latitude": 56.68584089439761, + "longitude": 39.91798456828216 + }, + { + "altitude": 26100.0, + "datetime": "2025-03-31T14:40:00Z", + "latitude": 56.68992270784105, + "longitude": 39.9300399138291 + }, + { + "altitude": 26400.0, + "datetime": "2025-03-31T14:41:00Z", + "latitude": 56.69394338486571, + "longitude": 39.940462779983974 + }, + { + "altitude": 26700.0, + "datetime": "2025-03-31T14:42:00Z", + "latitude": 56.697953087921626, + "longitude": 39.949622480306985 + }, + { + "altitude": 27000.0, + "datetime": "2025-03-31T14:43:00Z", + "latitude": 56.70200483356251, + "longitude": 39.95793958501962 + }, + { + "altitude": 27300.0, + "datetime": "2025-03-31T14:44:00Z", + "latitude": 56.70609933367715, + "longitude": 39.96542104870997 + }, + { + "altitude": 27600.0, + "datetime": "2025-03-31T14:45:00Z", + "latitude": 56.7102366946848, + "longitude": 39.97206868514009 + }, + { + "altitude": 27900.0, + "datetime": "2025-03-31T14:46:00Z", + "latitude": 56.714417048642495, + "longitude": 39.97788425861655 + }, + { + "altitude": 28200.0, + "datetime": "2025-03-31T14:47:00Z", + "latitude": 56.71864055182835, + "longitude": 39.98286948375741 + }, + { + "altitude": 28500.0, + "datetime": "2025-03-31T14:48:00Z", + "latitude": 56.72290738342261, + "longitude": 39.987026024968785 + }, + { + "altitude": 28800.0, + "datetime": "2025-03-31T14:49:00Z", + "latitude": 56.727217744290414, + "longitude": 39.99035549562477 + }, + { + "altitude": 29100.0, + "datetime": "2025-03-31T14:50:00Z", + "latitude": 56.731571855869994, + "longitude": 39.992859456944316 + }, + { + "altitude": 29400.0, + "datetime": "2025-03-31T14:51:00Z", + "latitude": 56.73596995917022, + "longitude": 39.99453941655818 + }, + { + "altitude": 29700.0, + "datetime": "2025-03-31T14:52:00Z", + "latitude": 56.74041231388129, + "longitude": 39.995396826758856 + }, + { + "altitude": 29997.65625, + "datetime": "2025-03-31T14:52:59.53125Z", + "latitude": 56.7448641438232, + "longitude": 39.995432799178765 + } + ] + }, + { + "stage": "descent", + "trajectory": [ + { + "altitude": 29997.65625, + "datetime": "2025-03-31T14:52:59.53125Z", + "latitude": 56.7448641438232, + "longitude": 39.995432799178765 + }, + { + "altitude": 27715.54955745421, + "datetime": "2025-03-31T14:53:59.53125Z", + "latitude": 56.74918746167504, + "longitude": 39.99829024344756 + }, + { + "altitude": 25795.085730205752, + "datetime": "2025-03-31T14:54:59.53125Z", + "latitude": 56.75320322793879, + "longitude": 40.00716270128483 + }, + { + "altitude": 24151.23596308202, + "datetime": "2025-03-31T14:55:59.53125Z", + "latitude": 56.757239111558626, + "longitude": 40.02464968299965 + }, + { + "altitude": 22698.48614507021, + "datetime": "2025-03-31T14:56:59.53125Z", + "latitude": 56.761022050960314, + "longitude": 40.04741531593104 + }, + { + "altitude": 21394.57996312004, + "datetime": "2025-03-31T14:57:59.53125Z", + "latitude": 56.76385846990279, + "longitude": 40.06785990381932 + }, + { + "altitude": 20211.831974882156, + "datetime": "2025-03-31T14:58:59.53125Z", + "latitude": 56.76583138188944, + "longitude": 40.085797017117756 + }, + { + "altitude": 19129.628141004454, + "datetime": "2025-03-31T14:59:59.53125Z", + "latitude": 56.7673236305108, + "longitude": 40.10178875248437 + }, + { + "altitude": 18132.205400921866, + "datetime": "2025-03-31T15:00:59.53125Z", + "latitude": 56.768502125013455, + "longitude": 40.11618875110261 + }, + { + "altitude": 17207.239689521524, + "datetime": "2025-03-31T15:01:59.53125Z", + "latitude": 56.769554389916124, + "longitude": 40.13001705148342 + }, + { + "altitude": 16344.913025727472, + "datetime": "2025-03-31T15:02:59.53125Z", + "latitude": 56.77055513776312, + "longitude": 40.143668230793914 + }, + { + "altitude": 15537.276953067527, + "datetime": "2025-03-31T15:03:59.53125Z", + "latitude": 56.7715796623419, + "longitude": 40.15730197520101 + }, + { + "altitude": 14777.806080763798, + "datetime": "2025-03-31T15:04:59.53125Z", + "latitude": 56.77278062783376, + "longitude": 40.17125115046076 + }, + { + "altitude": 14061.077422573206, + "datetime": "2025-03-31T15:05:59.53125Z", + "latitude": 56.77438122970693, + "longitude": 40.18474630512388 + }, + { + "altitude": 13382.535258039723, + "datetime": "2025-03-31T15:06:59.53125Z", + "latitude": 56.77660857508063, + "longitude": 40.19701657087815 + }, + { + "altitude": 12738.315525430071, + "datetime": "2025-03-31T15:07:59.53125Z", + "latitude": 56.7791996664954, + "longitude": 40.20986555240374 + }, + { + "altitude": 12125.112528246149, + "datetime": "2025-03-31T15:08:59.53125Z", + "latitude": 56.78226243966738, + "longitude": 40.22380101779603 + }, + { + "altitude": 11540.076280910698, + "datetime": "2025-03-31T15:09:59.53125Z", + "latitude": 56.78617806262856, + "longitude": 40.23883353825766 + }, + { + "altitude": 10980.661699362267, + "datetime": "2025-03-31T15:10:59.53125Z", + "latitude": 56.79024566070442, + "longitude": 40.259069107136334 + }, + { + "altitude": 10442.283472314524, + "datetime": "2025-03-31T15:11:59.53125Z", + "latitude": 56.79483243601256, + "longitude": 40.28588402840369 + }, + { + "altitude": 9921.481976025381, + "datetime": "2025-03-31T15:12:59.53125Z", + "latitude": 56.80027033495069, + "longitude": 40.31726533051636 + }, + { + "altitude": 9416.90446611415, + "datetime": "2025-03-31T15:13:59.53125Z", + "latitude": 56.805870192884576, + "longitude": 40.34843151030903 + }, + { + "altitude": 8927.35702773637, + "datetime": "2025-03-31T15:14:59.53125Z", + "latitude": 56.8102067869637, + "longitude": 40.376358200351405 + }, + { + "altitude": 8451.780039188363, + "datetime": "2025-03-31T15:15:59.53125Z", + "latitude": 56.813475892246316, + "longitude": 40.402371203158886 + }, + { + "altitude": 7989.22826252314, + "datetime": "2025-03-31T15:16:59.53125Z", + "latitude": 56.81582358854548, + "longitude": 40.427024105269645 + }, + { + "altitude": 7538.854543045095, + "datetime": "2025-03-31T15:17:59.53125Z", + "latitude": 56.81784352575536, + "longitude": 40.450763823829746 + }, + { + "altitude": 7099.89635323502, + "datetime": "2025-03-31T15:18:59.53125Z", + "latitude": 56.81968565269125, + "longitude": 40.47334320480768 + }, + { + "altitude": 6671.664600272706, + "datetime": "2025-03-31T15:19:59.53125Z", + "latitude": 56.82117204336888, + "longitude": 40.49192582779129 + }, + { + "altitude": 6253.534250988108, + "datetime": "2025-03-31T15:20:59.53125Z", + "latitude": 56.82287520210138, + "longitude": 40.505309210746724 + }, + { + "altitude": 5844.936428040647, + "datetime": "2025-03-31T15:21:59.53125Z", + "latitude": 56.825325132733035, + "longitude": 40.515583328790065 + }, + { + "altitude": 5445.351706175651, + "datetime": "2025-03-31T15:22:59.53125Z", + "latitude": 56.82799885305469, + "longitude": 40.52511249438823 + }, + { + "altitude": 5054.304394337709, + "datetime": "2025-03-31T15:23:59.53125Z", + "latitude": 56.83059415931585, + "longitude": 40.533850407051375 + }, + { + "altitude": 4671.3576330263495, + "datetime": "2025-03-31T15:24:59.53125Z", + "latitude": 56.833044047883114, + "longitude": 40.541294657534934 + }, + { + "altitude": 4296.109169981099, + "datetime": "2025-03-31T15:25:59.53125Z", + "latitude": 56.83528265412473, + "longitude": 40.547560499099795 + }, + { + "altitude": 3928.1877035500443, + "datetime": "2025-03-31T15:26:59.53125Z", + "latitude": 56.837495890955594, + "longitude": 40.552510288020066 + }, + { + "altitude": 3567.2497037306716, + "datetime": "2025-03-31T15:27:59.53125Z", + "latitude": 56.839985198231744, + "longitude": 40.55590103377752 + }, + { + "altitude": 3212.976637201876, + "datetime": "2025-03-31T15:28:59.53125Z", + "latitude": 56.84283521857802, + "longitude": 40.55750680699751 + }, + { + "altitude": 2865.0725356798516, + "datetime": "2025-03-31T15:29:59.53125Z", + "latitude": 56.84592990753435, + "longitude": 40.55749510056533 + }, + { + "altitude": 2523.261857369623, + "datetime": "2025-03-31T15:30:59.53125Z", + "latitude": 56.84905318718447, + "longitude": 40.5561140156843 + }, + { + "altitude": 2187.287599709619, + "datetime": "2025-03-31T15:31:59.53125Z", + "latitude": 56.85186662452198, + "longitude": 40.553442343543466 + }, + { + "altitude": 1856.9096284470706, + "datetime": "2025-03-31T15:32:59.53125Z", + "latitude": 56.85403077813578, + "longitude": 40.549463220878415 + }, + { + "altitude": 1531.9031936662172, + "datetime": "2025-03-31T15:33:59.53125Z", + "latitude": 56.855646254620865, + "longitude": 40.544473443658276 + }, + { + "altitude": 1212.0576079739358, + "datetime": "2025-03-31T15:34:59.53125Z", + "latitude": 56.857181905639024, + "longitude": 40.538917093364624 + }, + { + "altitude": 897.1750658269347, + "datetime": "2025-03-31T15:35:59.53125Z", + "latitude": 56.85882278460045, + "longitude": 40.533054228275724 + }, + { + "altitude": 587.0695861165462, + "datetime": "2025-03-31T15:36:59.53125Z", + "latitude": 56.860545444702105, + "longitude": 40.5270461265674 + }, + { + "altitude": 281.56606273414195, + "datetime": "2025-03-31T15:37:59.53125Z", + "latitude": 56.862054342901594, + "longitude": 40.5213786095734 + }, + { + "altitude": 1.6681590385689375, + "datetime": "2025-03-31T15:38:55.3125Z", + "latitude": 56.86240115306377, + "longitude": 40.51842596571808 + } + ] + } + ], + "request": { + "ascent_rate": 5.0, + "burst_altitude": 30000.0, + "dataset": "2025-03-31T06:00:00Z", + "descent_rate": 5.0, + "format": "json", + "launch_altitude": 0.0, + "launch_datetime": "2025-03-31T13:13:00Z", + "launch_latitude": 56.4017, + "launch_longitude": 38.7543, + "profile": "standard_profile", + "version": 2 + }, + "warnings": {} +} \ No newline at end of file diff --git a/tests/e2e/auth.spec.ts b/tests/e2e/auth.spec.ts deleted file mode 100644 index 49777af..0000000 --- a/tests/e2e/auth.spec.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { test, expect, login, logout } from './fixtures'; - -test('anonymous users are redirected to /login from /predict', async ({ page, context }) => { - await logout(context); - await page.goto('/predict'); - await page.waitForURL('**/login', { timeout: 10_000 }); - await expect(page.getByRole('heading', { level: 5 }).first()).toContainText( - /Вход|Sign in/, - ); -}); - -test('submitting the login form authenticates and redirects', async ({ page, context }) => { - await logout(context); - await page.goto('/login'); - - await page.getByLabel(/Имя пользователя|Username/).fill('demo'); - await page.getByLabel(/Пароль|Password/).fill('demo'); - await page.getByRole('button', { name: /^(Войти|Sign in)$/ }).click(); - - await page.waitForURL('**/predict', { timeout: 15_000 }); -}); - -test('already-authenticated visit to / lands on /predict', async ({ page, context }) => { - await login(context); - await page.goto('/'); - await page.waitForURL('**/predict', { timeout: 15_000 }); -}); diff --git a/tests/e2e/fake_tawhiri.py b/tests/e2e/fake_tawhiri.py deleted file mode 100644 index 16cebcc..0000000 --- a/tests/e2e/fake_tawhiri.py +++ /dev/null @@ -1,104 +0,0 @@ -""" -Tiny stand-in for the Tawhiri prediction service. Responds to Tawhiri v2's -GET query-string endpoint with a synthetic trajectory so stratoflights has -something to forward back to leaflet_svelte during e2e tests. - -Start: - python3 /tmp/fake_tawhiri.py - -Then tell stratoflights to use it: - TAWHIRI_BASE_URL=http://localhost:8001/api/v2/ ... python3 manage.py runserver -""" -from http.server import BaseHTTPRequestHandler, HTTPServer -from urllib.parse import parse_qs, urlparse -from datetime import datetime, timedelta, timezone -import json - - -def _iso(dt: datetime) -> str: - return dt.isoformat().replace("+00:00", "Z") - - -def build_prediction(params): - try: - launch_dt = datetime.fromisoformat( - params.get("launch_datetime", "2026-05-01T12:00:00Z").replace("Z", "+00:00"), - ) - except Exception: - launch_dt = datetime.now(timezone.utc) - - launch_lat = float(params.get("launch_latitude", 62.0)) - launch_lng = float(params.get("launch_longitude", 129.0)) - launch_alt = float(params.get("launch_altitude", 0.0)) - burst_alt = float(params.get("burst_altitude", 30000.0)) - ascent_rate = float(params.get("ascent_rate", 5.0)) - descent_rate = float(params.get("descent_rate", 5.0)) - - ascent_duration_s = int((burst_alt - launch_alt) / max(0.1, ascent_rate)) - descent_duration_s = int(burst_alt / max(0.1, descent_rate)) - step = 30 - - ascent = [] - for t in range(0, ascent_duration_s + 1, step): - ascent.append({ - "altitude": launch_alt + t * ascent_rate, - "datetime": _iso(launch_dt + timedelta(seconds=t)), - "latitude": launch_lat + t * 0.00002, - "longitude": launch_lng + t * 0.00005, - }) - - descent = [] - burst_dt = launch_dt + timedelta(seconds=ascent_duration_s) - burst_lat = launch_lat + ascent_duration_s * 0.00002 - burst_lng = launch_lng + ascent_duration_s * 0.00005 - for t in range(0, descent_duration_s + 1, step): - alt = max(0.0, burst_alt - t * descent_rate) - descent.append({ - "altitude": alt, - "datetime": _iso(burst_dt + timedelta(seconds=t)), - "latitude": burst_lat + t * 0.00001, - "longitude": burst_lng + t * 0.00003, - }) - - return { - "metadata": { - "start_datetime": _iso(launch_dt - timedelta(hours=1)), - "complete_datetime": _iso(burst_dt + timedelta(seconds=descent_duration_s)), - }, - "prediction": [ - {"stage": "ascent", "trajectory": ascent}, - {"stage": "descent", "trajectory": descent}, - ], - "request": { - "dataset": "fake", - "launch_latitude": launch_lat, - "launch_longitude": launch_lng, - "launch_altitude": launch_alt, - }, - } - - -class Handler(BaseHTTPRequestHandler): - def do_GET(self): - parsed = urlparse(self.path) - if not parsed.path.rstrip("/").endswith("/api/v2"): - self.send_error(404) - return - params = {k: v[0] for k, v in parse_qs(parsed.query).items()} - body = json.dumps(build_prediction(params)).encode() - self.send_response(200) - self.send_header("Content-Type", "application/json") - self.send_header("Access-Control-Allow-Origin", "*") - self.end_headers() - self.wfile.write(body) - - def log_message(self, format, *args): - # Quiet default access log; keep stderr clean. - pass - - -if __name__ == "__main__": - port = 8001 - server = HTTPServer(("127.0.0.1", port), Handler) - print(f"fake-tawhiri listening on http://127.0.0.1:{port}/api/v2/") - server.serve_forever() diff --git a/tests/e2e/fixtures.ts b/tests/e2e/fixtures.ts deleted file mode 100644 index ec60836..0000000 --- a/tests/e2e/fixtures.ts +++ /dev/null @@ -1,74 +0,0 @@ -import { - test as base, - expect, - type Page, - type BrowserContext, -} from '@playwright/test'; - -/** - * Shared fixtures and helpers. - * - * Works against either the mock plugin (VITE_USE_MOCK_API=true) or the real - * Django backend (stratoflights). Mock accepts any credentials; Django - * expects demo/demo and enforces CSRF. - * - * The `TEST_USERNAME` / `TEST_PASSWORD` envs let CI override the default. - * - * Critical detail: login/logout must go through `page.context().request` so - * cookies are shared between API calls and the page. Using the plain - * `request` fixture gives you a separate cookie jar. - */ - -const USERNAME = process.env.TEST_USERNAME ?? 'demo'; -const PASSWORD = process.env.TEST_PASSWORD ?? 'demo'; - -export async function login(context: BrowserContext): Promise { - const request = context.request; - await request.get('/api/csrf/'); - const state = await context.storageState(); - const csrf = state.cookies.find((c) => c.name === 'csrftoken')?.value ?? ''; - const res = await request.post('/api/login/', { - data: { username: USERNAME, password: PASSWORD }, - headers: { 'X-CSRFToken': csrf, 'Content-Type': 'application/json' }, - }); - if (!res.ok()) { - throw new Error( - `Login failed: ${res.status()} ${await res.text()}. Is stratoflights running on :8000 with a user "${USERNAME}" / "${PASSWORD}"?`, - ); - } -} - -export async function logout(context: BrowserContext): Promise { - const state = await context.storageState(); - const csrf = state.cookies.find((c) => c.name === 'csrftoken')?.value ?? ''; - await context.request.post('/api/logout/', { - headers: { 'X-CSRFToken': csrf, 'Content-Type': 'application/json' }, - }); - // Clear cookies so the page navigator is fully anonymous next goto. - await context.clearCookies(); -} - -export const test = base.extend({ - page: async ({ page }, use) => { - const hardErrors: string[] = []; - page.on('pageerror', (e) => hardErrors.push(`[pageerror] ${e.message}`)); - page.on('console', (msg) => { - if (msg.type() === 'error') { - // eslint-disable-next-line no-console - console.log(' [page console.error]', msg.text()); - } - }); - await use(page); - if (hardErrors.length > 0) throw new Error('Page errors:\n' + hardErrors.join('\n')); - }, -}); - -export { expect }; - -export async function openPredict(page: Page) { - await page.goto('/predict'); - await page - .locator('.map-container canvas') - .first() - .waitFor({ state: 'attached', timeout: 15_000 }); -} diff --git a/tests/e2e/saved-points.spec.ts b/tests/e2e/saved-points.spec.ts deleted file mode 100644 index 05af2fd..0000000 --- a/tests/e2e/saved-points.spec.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { test, expect, openPredict, login } from './fixtures'; - -test.beforeEach(async ({ context, page }) => { - await login(context); - await page.goto('/'); - await page.evaluate(() => localStorage.removeItem('workspaces')); -}); - -test('can open the points library from the conditions panel', async ({ page }) => { - await openPredict(page); - - await page - .locator('.panel-container-left') - .getByRole('button', { name: /Условия|Conditions/ }) - .first() - .click(); - - const openBookBtn = page - .locator('.panel-container-left') - .getByRole('button') - .filter({ has: page.locator('i.bi-journal-bookmark-fill') }) - .first(); - await openBookBtn.click(); - - await expect(page.getByRole('dialog')).toBeVisible(); -}); diff --git a/tests/e2e/settings.spec.ts b/tests/e2e/settings.spec.ts deleted file mode 100644 index b358a7c..0000000 --- a/tests/e2e/settings.spec.ts +++ /dev/null @@ -1,24 +0,0 @@ -import { test, expect, openPredict, login } from './fixtures'; - -test.beforeEach(async ({ context }) => { - await login(context); -}); - -test('switching locale to English updates UI text', async ({ page }) => { - await openPredict(page); - - await page - .locator('.panel-container-right') - .getByRole('button', { name: /Настройки|Settings/ }) - .first() - .click(); - - const langSelect = page.locator('.panel-container-right select').first(); - await langSelect.selectOption('en'); - - await expect( - page.locator('.panel-container-right').getByRole('button', { name: /Workspaces/ }).first(), - ).toBeVisible({ timeout: 5_000 }); - - await langSelect.selectOption('ru'); -}); diff --git a/tests/e2e/smoke.spec.ts b/tests/e2e/smoke.spec.ts deleted file mode 100644 index 80345c3..0000000 --- a/tests/e2e/smoke.spec.ts +++ /dev/null @@ -1,31 +0,0 @@ -import { test, expect, login } from './fixtures'; - -test.beforeEach(async ({ context }) => { - await login(context); -}); - -test('predict page loads and mounts the MapLibre canvas', async ({ page }) => { - await page.goto('/predict'); - await expect(page.locator('.map-container canvas').first()).toBeAttached({ - timeout: 15_000, - }); -}); - -test('left and right panels are visible on predict', async ({ page }) => { - await page.goto('/predict'); - await expect(page.locator('.panel-container-left')).toBeVisible(); - await expect(page.locator('.panel-container-right')).toBeVisible(); -}); - -test('navbar shows current username dropdown when authenticated', async ({ page }) => { - await page.goto('/predict'); - await expect(page.getByRole('link', { name: /Прогноз|Predict/ }).first()).toBeVisible(); - await expect(page.getByText('demo').first()).toBeVisible(); -}); - -test('tracking page mounts its own map', async ({ page }) => { - await page.goto('/track'); - await expect(page.locator('.map-container canvas').first()).toBeAttached({ - timeout: 15_000, - }); -}); diff --git a/tests/e2e/workspaces.spec.ts b/tests/e2e/workspaces.spec.ts deleted file mode 100644 index 8bc8ac1..0000000 --- a/tests/e2e/workspaces.spec.ts +++ /dev/null @@ -1,68 +0,0 @@ -import { test, expect, openPredict, login } from './fixtures'; - -test.beforeEach(async ({ context, page }) => { - await login(context); - await page.goto('/'); - await page.evaluate(() => localStorage.removeItem('workspaces')); -}); - -function workspacesPanel(page: import('@playwright/test').Page) { - return page.locator('.panel-container-right'); -} - -test('a workspace is auto-created on first visit to /predict', async ({ page }) => { - await openPredict(page); - const panel = workspacesPanel(page); - await expect(panel.locator('.workspace-row').first()).toBeVisible({ timeout: 10_000 }); - await expect(panel.getByRole('button', { name: /Рассчитать|Run/ }).first()).toBeVisible(); -}); - -test('can add and remove workspaces', async ({ page }) => { - await openPredict(page); - const panel = workspacesPanel(page); - await expect(panel.locator('.workspace-row')).toHaveCount(1, { timeout: 10_000 }); - - const addBtn = panel.getByRole('button', { name: /Добавить|Add/ }).first(); - await addBtn.click(); - await expect(panel.locator('.workspace-row')).toHaveCount(2); - await addBtn.click(); - await expect(panel.locator('.workspace-row')).toHaveCount(3); - - // Delete the last workspace via its row's delete button. - const deleteBtn = panel.locator('.workspace-row').last().locator('button.btn-danger'); - await deleteBtn.click(); - // Confirm in the modal. - await page - .getByRole('dialog') - .getByRole('button', { name: /^(Удалить|Delete)$/ }) - .click(); - await expect(panel.locator('.workspace-row')).toHaveCount(2); -}); - -test('workspace render pipeline adds a map scene after a run', async ({ page }) => { - test.setTimeout(90_000); - await openPredict(page); - - // Wait for the Run button to be enabled before clicking. - const runBtn = workspacesPanel(page) - .locator('.workspace-row') - .first() - .getByRole('button', { name: /Рассчитать|Run/ }); - await runBtn.click(); - - // Wait for the prediction request to complete and layers to be added. - await expect - .poll( - async () => - page.evaluate(() => { - // eslint-disable-next-line @typescript-eslint/no-explicit-any - const map: any = (window as any)._lsvMap; - if (!map) return 0; - return map - .getStyle() - .layers.filter((l: { id: string }) => l.id.startsWith('ws/')).length; - }), - { timeout: 75_000, intervals: [1000, 2000, 3000] }, - ) - .toBeGreaterThan(0); -}); diff --git a/vite.config.js b/vite.config.js index 2e8746b..bbf8c7d 100644 --- a/vite.config.js +++ b/vite.config.js @@ -1,41 +1,6 @@ import { sveltekit } from '@sveltejs/kit/vite'; -import { defineConfig, loadEnv } from 'vite'; -import { mockApiPlugin } from './mocks/vitePlugin'; +import { defineConfig } from 'vite'; -/** - * Three dev modes are supported: - * - * 1. VITE_USE_MOCK_API=true - * → serve fake data from ./mocks. No backend needed. - * - * 2. VITE_API_BASE_URL starts with '/' - * → client makes same-origin requests; the dev server proxies - * VITE_API_BASE_URL to VITE_API_PROXY_TARGET (default - * http://localhost:8000). This matches production where the web - * server routes /api to Django. - * - * 3. VITE_API_BASE_URL is an absolute URL - * → client talks to it directly. CORS must be enabled on that host. - * No proxy is registered. - */ -export default defineConfig(({ mode }) => { - const env = loadEnv(mode, process.cwd(), ''); - const useMock = env.VITE_USE_MOCK_API === 'true'; - const base = env.VITE_API_BASE_URL ?? '/api'; - const proxyTarget = env.VITE_API_PROXY_TARGET ?? 'http://localhost:8000'; - const shouldProxy = !useMock && base.startsWith('/'); - - return { - plugins: [sveltekit(), ...(useMock ? [mockApiPlugin()] : [])], - server: { - proxy: shouldProxy - ? { - [base]: { - target: proxyTarget, - changeOrigin: true, - }, - } - : undefined, - }, - }; +export default defineConfig({ + plugins: [sveltekit()] });