diff --git a/.env.example b/.env.example new file mode 100644 index 0000000..4a67f7c --- /dev/null +++ b/.env.example @@ -0,0 +1,18 @@ +# 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 3b462cb..15dd615 100644 --- a/.gitignore +++ b/.gitignore @@ -21,3 +21,7 @@ Thumbs.db # Vite vite.config.js.timestamp-* vite.config.ts.timestamp-* + +# AI tools +.claude +tmpclaude* diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..f27d159 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,8 @@ +{ + "tabWidth": 4, + "endOfLine": "lf", + "printWidth": 120, + "useTabs": true, + "htmlWhitespaceSensitivity": "ignore", + "bracketSameLine": true +} diff --git a/README.md b/README.md index b5b2950..a561a1a 100644 --- a/README.md +++ b/README.md @@ -1,38 +1,44 @@ -# sv +# leaflet_svelte -Everything you need to build a Svelte project, powered by [`sv`](https://github.com/sveltejs/cli). +Weather-balloon trajectory planner. Static SvelteKit SPA that talks to a Django +REST backend. -## Creating a project - -If you're seeing this, you've probably already done this step. Congrats! +## Quick start ```bash -# create a new project in the current directory -npx sv create +npm install -# 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 +# dev against a local Django on :8000 (see .env.example) npm run dev -# or start the server and open the app in a new browser tab -npm run dev -- --open -``` +# dev with a fake backend (no Django needed) +VITE_USE_MOCK_API=true npm run dev -## Building - -To create a production version of your app: - -```bash +# type-check + production build (emits static files to ./build) +npm run check npm run build ``` -You can preview the production build with `npm run preview`. +Serve `build/` from any static host. Route fallback is `index.html`. -> To deploy your app, you may need to install an [adapter](https://svelte.dev/docs/kit/adapters) for your target environment. +## 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. diff --git a/build.js b/build.js deleted file mode 100644 index 266cd5a..0000000 --- a/build.js +++ /dev/null @@ -1,390 +0,0 @@ -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 new file mode 100644 index 0000000..5211257 --- /dev/null +++ b/docs/ADDING_A_FEATURE.md @@ -0,0 +1,152 @@ +# 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 new file mode 100644 index 0000000..6973d74 --- /dev/null +++ b/docs/ARCHITECTURE.md @@ -0,0 +1,105 @@ +# 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 new file mode 100644 index 0000000..c9b545e --- /dev/null +++ b/docs/CONVENTIONS.md @@ -0,0 +1,132 @@ +# 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 new file mode 100644 index 0000000..397b9c2 --- /dev/null +++ b/src/lib/features/auth/Navbar.svelte @@ -0,0 +1,147 @@ + + + { if (isDropdownOpen) isDropdownOpen = false; }} /> + + + + diff --git a/src/lib/features/auth/index.ts b/src/lib/features/auth/index.ts new file mode 100644 index 0000000..5368bb0 --- /dev/null +++ b/src/lib/features/auth/index.ts @@ -0,0 +1,2 @@ +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 new file mode 100644 index 0000000..9a5fa95 --- /dev/null +++ b/src/lib/features/footer/Footer.svelte @@ -0,0 +1,25 @@ + + +
+
+
+
+
+ + {$t('app.company')} + +
+
+
+
+
+
+
+
+
Copyright © 2024 {$t('app.company')}
+
+
+
+
diff --git a/src/lib/features/footer/index.ts b/src/lib/features/footer/index.ts new file mode 100644 index 0000000..02db901 --- /dev/null +++ b/src/lib/features/footer/index.ts @@ -0,0 +1 @@ +export { default as Footer } from './Footer.svelte'; diff --git a/src/lib/features/prediction/ControlPanel.svelte b/src/lib/features/prediction/ControlPanel.svelte new file mode 100644 index 0000000..9d17c17 --- /dev/null +++ b/src/lib/features/prediction/ControlPanel.svelte @@ -0,0 +1,342 @@ + + + + {#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 new file mode 100644 index 0000000..568e8ac --- /dev/null +++ b/src/lib/features/prediction/CurveChart.svelte @@ -0,0 +1,144 @@ + + +
+ +
diff --git a/src/lib/features/prediction/CurveEditor.svelte b/src/lib/features/prediction/CurveEditor.svelte new file mode 100644 index 0000000..2831173 --- /dev/null +++ b/src/lib/features/prediction/CurveEditor.svelte @@ -0,0 +1,338 @@ + + + + + + + + (isConfirmationVisible = false)}> +

Delete curve "{selectedCurve?.name}"?

+
diff --git a/src/lib/features/prediction/PointEditor.svelte b/src/lib/features/prediction/PointEditor.svelte new file mode 100644 index 0000000..4cf8f34 --- /dev/null +++ b/src/lib/features/prediction/PointEditor.svelte @@ -0,0 +1,140 @@ + + + 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 new file mode 100644 index 0000000..10f72b0 --- /dev/null +++ b/src/lib/features/prediction/ScenarioEditor.svelte @@ -0,0 +1,83 @@ + + + + + + diff --git a/src/lib/features/prediction/ScenarioPanel.svelte b/src/lib/features/prediction/ScenarioPanel.svelte new file mode 100644 index 0000000..05d9d5d --- /dev/null +++ b/src/lib/features/prediction/ScenarioPanel.svelte @@ -0,0 +1,189 @@ + + + + + + + ({ + 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 new file mode 100644 index 0000000..3fe21d0 --- /dev/null +++ b/src/lib/features/prediction/index.ts @@ -0,0 +1,7 @@ +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 new file mode 100644 index 0000000..63974d3 --- /dev/null +++ b/src/lib/features/prediction/pointsStore.ts @@ -0,0 +1,11 @@ +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 new file mode 100644 index 0000000..1d03e64 --- /dev/null +++ b/src/lib/features/settings/SettingsPanel.svelte @@ -0,0 +1,70 @@ + + + + {#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 new file mode 100644 index 0000000..1deb343 --- /dev/null +++ b/src/lib/features/settings/index.ts @@ -0,0 +1,5 @@ +export { settingsStore, DEFAULT_SETTINGS } from './store'; +export type { AppSettings, MapSettings, UnitsSettings, WindSettings } 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 new file mode 100644 index 0000000..22265ef --- /dev/null +++ b/src/lib/features/settings/schema.ts @@ -0,0 +1,172 @@ +/** + * 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' }, + ], + }, + ], + }, + { + titleKey: 'settings.wind', + fields: [ + { kind: 'boolean', path: 'wind.enabled', labelKey: 'settings.windEnabled' }, + { + kind: 'number', + path: 'wind.step', + labelKey: 'settings.windStep', + min: 0.25, + max: 10, + step: 0.25, + }, + { + kind: 'number', + path: 'wind.trajectoryStep', + labelKey: 'settings.windTrajectoryStep', + min: 0.25, + max: 5, + step: 0.25, + }, + { + kind: 'number', + path: 'wind.prefetchIntervalMinutes', + labelKey: 'settings.windPrefetchInterval', + min: 5, + max: 60, + step: 5, + }, + { + kind: 'number', + path: 'wind.maxFlightDurationHours', + labelKey: 'settings.windMaxDuration', + min: 1, + max: 8, + step: 0.5, + }, + { + kind: 'number', + path: 'wind.maxRegionDegrees', + labelKey: 'settings.windMaxRegion', + min: 5, + max: 60, + step: 5, + }, + { + kind: 'number', + path: 'wind.trajectoryMarginDegrees', + labelKey: 'settings.windMargin', + min: 0.5, + max: 5, + step: 0.5, + }, + { + kind: 'number', + path: 'wind.particleDensity', + labelKey: 'settings.windParticleDensity', + min: 0.25, + max: 3, + step: 0.25, + }, + { + kind: 'number', + path: 'wind.particleSpeed', + labelKey: 'settings.windParticleSpeed', + min: 0.25, + max: 4, + step: 0.25, + }, + { + kind: 'number', + path: 'wind.trailPersistence', + labelKey: 'settings.windTrailPersistence', + min: 0.7, + max: 0.98, + step: 0.02, + }, + { + kind: 'number', + path: 'wind.maxVelocity', + labelKey: 'settings.windMaxVelocity', + min: 10, + max: 80, + step: 5, + }, + ], + }, +]; diff --git a/src/lib/features/settings/store.ts b/src/lib/features/settings/store.ts new file mode 100644 index 0000000..c13a866 --- /dev/null +++ b/src/lib/features/settings/store.ts @@ -0,0 +1,55 @@ +import { persisted } from '$state'; +import type { Locale } from '$i18n'; +import { type WindSettings, DEFAULT_WIND_SETTINGS } from '$domain'; + +export type { WindSettings }; + +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; + wind: WindSettings; +} + +export const DEFAULT_SETTINGS: AppSettings = { + locale: 'ru', + map: { baseLayer: 'osm', showScale: true, showNavigation: true }, + units: { system: 'metric' }, + wind: { ...DEFAULT_WIND_SETTINGS }, +}; + +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 new file mode 100644 index 0000000..ba7cf9c --- /dev/null +++ b/src/lib/features/timeline/TimeLine.svelte @@ -0,0 +1,185 @@ + + +
+
+
+
+ + {#if $timelineStore.playing} + + {:else} + + {/if} + +
+ +
+
+ + {#if hasData && $timelineStore.markers.length > 0} +
+ {#each $timelineStore.markers as m} + {@const pct = $timelineStore.max > 0 ? m.time / $timelineStore.max : 0} + + {fmtHms(m.time)} + + {/each} +
+ {/if} +
+
+ {fmtHms(elapsed)} + {fmtHms(duration)} +
+
+
+
+
+ + diff --git a/src/lib/features/timeline/index.ts b/src/lib/features/timeline/index.ts new file mode 100644 index 0000000..4cb5edd --- /dev/null +++ b/src/lib/features/timeline/index.ts @@ -0,0 +1,3 @@ +export { timelineStore } from './store'; +export type { TimelineState, TimelineMarker } 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 new file mode 100644 index 0000000..6c3017f --- /dev/null +++ b/src/lib/features/timeline/store.ts @@ -0,0 +1,104 @@ +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 TimelineMarker { + time: number; + color: string; +} + +export interface TimelineState { + time: number; + min: number; + max: number; + speed: number; + playing: boolean; + markers: TimelineMarker[]; +} + +const initial: TimelineState = { + time: 0, + min: 0, + max: 0, + speed: 1, + playing: false, + markers: [], +}; + +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 }; + }); + } + + function setMarkers(markers: TimelineMarker[]) { + store.update((s) => ({ ...s, markers })); + } + + return { subscribe: store.subscribe, play, pause, reset, seek, setSpeed, setRange, setMarkers }; +} + +export const timelineStore = createTimeline(); diff --git a/src/lib/features/tracking/DeviationChart.svelte b/src/lib/features/tracking/DeviationChart.svelte new file mode 100644 index 0000000..06c8a77 --- /dev/null +++ b/src/lib/features/tracking/DeviationChart.svelte @@ -0,0 +1,203 @@ + + + + + +{#if !hasData} +

{$t('tracking.noData')}

+{/if} + + +
+

{$t('tracking.altProfile')}

+
+ +
+ + {#if !hasDeviation} +

{$t('tracking.selectPrediction')}

+ {/if} +
+ + +
+
+

{$t('tracking.horizontalDev')}

+
+ +
+ + {#if deviations && deviations.length > 0} + {@const maxDev = Math.max(...deviations.map((d) => d.horizontal))} + {@const last = deviations[deviations.length - 1]} +
+ + {$t('tracking.devMax')} {maxDev.toFixed(2)} км + + + {$t('tracking.devCurrent')} {last.horizontal.toFixed(2)} км + + + Δh: + {last.vertical > 0 ? '+' : ''}{last.vertical.toFixed(0)} м + + +
+ {/if} +
diff --git a/src/lib/features/tracking/TelemetryPanel.svelte b/src/lib/features/tracking/TelemetryPanel.svelte new file mode 100644 index 0000000..c4de83c --- /dev/null +++ b/src/lib/features/tracking/TelemetryPanel.svelte @@ -0,0 +1,100 @@ + + + + + +
+ + {#if telemetryStore.status === 'idle'} + + {:else} + + {/if} +
+
+ + +
+ + + {$t(`tracking.status_${telemetryStore.status}`)} + +
+ {#if telemetryStore.error} + {telemetryStore.error} + {/if} +
+ + {#if telemetryStore.latest} + + + + + + + + + + + + + + + + + + + + + + + {$t('tracking.packetCount', { count: telemetryStore.points.length })} + + {:else if telemetryStore.status !== 'idle'} + {$t('tracking.waitingData')} + {/if} +
diff --git a/src/lib/features/tracking/index.ts b/src/lib/features/tracking/index.ts new file mode 100644 index 0000000..4e5685c --- /dev/null +++ b/src/lib/features/tracking/index.ts @@ -0,0 +1,3 @@ +export { default as TelemetryPanel } from './TelemetryPanel.svelte'; +export { default as DeviationChart } from './DeviationChart.svelte'; +export { telemetryStore, type TrackingStatus } from './telemetryStore.svelte'; diff --git a/src/lib/features/tracking/telemetryStore.svelte.ts b/src/lib/features/tracking/telemetryStore.svelte.ts new file mode 100644 index 0000000..2d0f2b9 --- /dev/null +++ b/src/lib/features/tracking/telemetryStore.svelte.ts @@ -0,0 +1,99 @@ +import { telemetryApi, buildWsUrl, type RawTelemetryPacket } from '$api/telemetry'; +import { parseTelemetry, type TelemetryPoint, type Telemetry } from '$domain'; + +export type TrackingStatus = 'idle' | 'connecting' | 'connected' | 'error'; + +function toPoint(p: RawTelemetryPacket): TelemetryPoint { + return { + latitude: p.lat, + longitude: p.lon, + altitude: p.alt, + datetime: new Date(p.timestamp * 1000).toISOString(), + payload: JSON.stringify(p.payload), + }; +} + +class TelemetryStore { + satelliteId = $state(''); + status = $state('idle'); + error = $state(null); + points = $state([]); + + #ws: WebSocket | null = null; + + get latest(): TelemetryPoint | null { + return this.points[this.points.length - 1] ?? null; + } + + get telemetry(): Telemetry | null { + if (this.points.length === 0) return null; + try { + return parseTelemetry(this.points); + } catch { + return null; + } + } + + async connect(id: string): Promise { + const UUID_RE = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i; + if (!UUID_RE.test(id)) { + this.status = 'error'; + this.error = `Invalid satellite ID — expected a UUID (e.g. 550e8400-e29b-41d4-a716-446655440000)`; + return; + } + + this.disconnect(); + this.satelliteId = id; + this.status = 'connecting'; + this.error = null; + this.points = []; + + // Load historical packets first — non-fatal if it fails + try { + const history = await telemetryApi.fetchHistory(id); + // API returns newest-first; reverse to chronological order + this.points = [...history].reverse().map(toPoint); + } catch (e) { + console.warn('[telemetry] history fetch failed:', e); + } + + const ws = new WebSocket(buildWsUrl(id)); + this.#ws = ws; + + ws.onopen = () => { + this.status = 'connected'; + }; + + ws.onmessage = ({ data }) => { + try { + const packet = JSON.parse(data) as { error?: string } & RawTelemetryPacket; + if (!packet.error) { + this.points = [...this.points, toPoint(packet)]; + } + } catch { + // ignore malformed frames + } + }; + + ws.onerror = () => { + this.status = 'error'; + this.error = 'WebSocket connection failed'; + }; + + ws.onclose = () => { + if (this.status !== 'idle') this.status = 'idle'; + this.#ws = null; + }; + } + + disconnect(): void { + this.#ws?.close(); + this.#ws = null; + this.status = 'idle'; + this.satelliteId = ''; + this.points = []; + this.error = null; + } +} + +export const telemetryStore = new TelemetryStore(); diff --git a/src/lib/features/wind/ParticleField.ts b/src/lib/features/wind/ParticleField.ts new file mode 100644 index 0000000..3d47d44 --- /dev/null +++ b/src/lib/features/wind/ParticleField.ts @@ -0,0 +1,335 @@ +/** + * ParticleField — an animated wind-flow layer rendered to a 2D canvas + * overlaid on the MapLibre container, in the spirit of leaflet-velocity / + * cambecc's "earth". + * + * Particles live in CSS-pixel space. Each frame, every particle is unprojected + * to lng/lat, the wind [u, v] there is sampled, and that vector is pushed + * through the map projection's local Jacobian to obtain a pixel-space velocity + * (so motion is correct at any zoom/latitude). Trails are faded by compositing + * a translucent clear over the previous frame, leaving the basemap visible. + * + * The wind field can change every frame (the renderer interpolates between + * pre-fetched trajectory frames over time); only the lightweight interpolator + * closure is swapped, so particle motion stays continuous. See + * docs/wind-vis-math.tex §"Particle Advection". + */ + +import type { Map as MLMap } from 'maplibre-gl'; +import type { WindInterpolator } from '$domain'; + +export interface ParticleOptions { + /** Particles per screen pixel (scaled by the base multiplier). */ + density: number; + /** Advection speed multiplier. */ + speed: number; + /** Trail persistence in [0,1): fraction of the trail kept each frame. */ + trailPersistence: number; + /** Max frames a particle lives before it is respawned. */ + maxAge: number; + /** Trail line width (CSS px). */ + lineWidth: number; + /** Wind speed (m/s) at the bottom / top of the colour scale. */ + minVelocity: number; + maxVelocity: number; + /** Target frame rate (the field is re-evaluated at most this often). */ + frameRate: number; + /** Colour ramp from slow → fast wind. */ + colorScale: string[]; +} + +export const DEFAULT_COLOR_SCALE = [ + 'rgb(36,104,180)', + 'rgb(60,157,194)', + 'rgb(128,205,193)', + 'rgb(151,218,168)', + 'rgb(198,231,181)', + 'rgb(238,247,217)', + 'rgb(255,238,159)', + 'rgb(252,217,125)', + 'rgb(255,182,100)', + 'rgb(252,150,75)', + 'rgb(250,112,52)', + 'rgb(245,64,32)', + 'rgb(237,45,28)', + 'rgb(220,24,32)', + 'rgb(180,0,35)', +]; + +export const DEFAULT_PARTICLE_OPTIONS: ParticleOptions = { + density: 1.0, + speed: 1.0, + trailPersistence: 0.92, + maxAge: 100, + lineWidth: 1.4, + minVelocity: 0, + maxVelocity: 30, + frameRate: 30, + colorScale: DEFAULT_COLOR_SCALE, +}; + +/** Base particle count = pixels × this (kept modest for performance). */ +const PARTICLE_MULTIPLIER = 1 / 350; +const MAX_PARTICLES = 6000; + +interface Particle { + x: number; + y: number; + xt: number; + yt: number; + age: number; + speed: number; +} + +export class ParticleField { + private map: MLMap; + private host: HTMLElement; + private canvas: HTMLCanvasElement; + private ctx: CanvasRenderingContext2D; + private opts: ParticleOptions; + private interp: WindInterpolator | null = null; + + private particles: Particle[] = []; + private raf = 0; + private then = 0; + private moving = false; + private width = 0; + private height = 0; + private debugLogged = false; + + constructor(map: MLMap, opts: Partial = {}) { + this.map = map; + this.opts = { ...DEFAULT_PARTICLE_OPTIONS, ...opts }; + + // Mount inside the MapLibre canvas container so the overlay sits above + // the basemap but below the control container and the app's panels. + this.host = map.getCanvasContainer(); + + const canvas = document.createElement('canvas'); + canvas.className = 'wind-particles'; + canvas.style.position = 'absolute'; + canvas.style.top = '0'; + canvas.style.left = '0'; + canvas.style.pointerEvents = 'none'; + canvas.style.zIndex = '3'; + this.host.appendChild(canvas); + this.canvas = canvas; + this.ctx = canvas.getContext('2d')!; + + this.map.on('movestart', this.onMoveStart); + this.map.on('moveend', this.onMoveEnd); + this.map.on('resize', this.onResize); + this.resize(); + } + + setOptions(opts: Partial): void { + const densityChanged = opts.density !== undefined && opts.density !== this.opts.density; + this.opts = { ...this.opts, ...opts }; + if (densityChanged) this.seedParticles(); + } + + /** Swap the wind field. Pass null to clear the flow. */ + setField(interp: WindInterpolator | null): void { + this.interp = interp; + if (interp && this.particles.length === 0) this.seedParticles(); + } + + start(): void { + if (this.raf) return; + this.then = performance.now(); + this.raf = requestAnimationFrame(this.frame); + } + + stop(): void { + if (this.raf) cancelAnimationFrame(this.raf); + this.raf = 0; + this.clear(); + } + + destroy(): void { + this.stop(); + this.map.off('movestart', this.onMoveStart); + this.map.off('moveend', this.onMoveEnd); + this.map.off('resize', this.onResize); + this.canvas.remove(); + } + + // ── Internals ───────────────────────────────────────────────────────────── + + private onMoveStart = (): void => { + this.moving = true; + this.clear(); + }; + + private onMoveEnd = (): void => { + this.moving = false; + this.seedParticles(); + }; + + private onResize = (): void => { + this.resize(); + }; + + private resize(): void { + const dpr = window.devicePixelRatio || 1; + // Size from the gl canvas: it always reports the true viewport size, + // whereas the canvas-container wrapper can measure 0 in some layouts. + const glCanvas = this.map.getCanvas(); + const w = glCanvas.clientWidth || this.map.getContainer().clientWidth; + const h = glCanvas.clientHeight || this.map.getContainer().clientHeight; + if (!w || !h) return; + this.width = w; + this.height = h; + this.canvas.style.width = `${w}px`; + this.canvas.style.height = `${h}px`; + this.canvas.width = Math.round(w * dpr); + this.canvas.height = Math.round(h * dpr); + this.ctx.setTransform(dpr, 0, 0, dpr, 0, 0); // draw in CSS-pixel space + this.seedParticles(); + } + + private particleCount(): number { + const n = this.width * this.height * PARTICLE_MULTIPLIER * this.opts.density; + return Math.max(0, Math.min(MAX_PARTICLES, Math.round(n))); + } + + private seedParticles(): void { + const count = this.particleCount(); + this.particles = new Array(count); + for (let i = 0; i < count; i++) { + this.particles[i] = { x: 0, y: 0, xt: 0, yt: 0, age: 0, speed: 0 }; + this.respawn(this.particles[i]); + this.particles[i].age = Math.floor(Math.random() * this.opts.maxAge); + } + } + + /** Place a particle at a random pixel that has wind (a few retries). */ + private respawn(p: Particle): void { + for (let attempt = 0; attempt < 8; attempt++) { + const x = Math.random() * this.width; + const y = Math.random() * this.height; + if (!this.interp) { + p.x = p.xt = x; + p.y = p.yt = y; + break; + } + const ll = this.map.unproject([x, y]); + if (this.interp(ll.lng, ll.lat)) { + p.x = p.xt = x; + p.y = p.yt = y; + break; + } + p.x = p.xt = x; + p.y = p.yt = y; + } + p.age = 0; + p.speed = 0; + } + + private clear(): void { + this.ctx.clearRect(0, 0, this.width, this.height); + } + + private colorIndex(speed: number): number { + const { minVelocity, maxVelocity, colorScale } = this.opts; + const f = (speed - minVelocity) / (maxVelocity - minVelocity); + return Math.max(0, Math.min(colorScale.length - 1, Math.round(f * (colorScale.length - 1)))); + } + + private evolve(): void { + const interp = this.interp; + if (!interp) return; + const scale = 0.06 * this.opts.speed; // pixel velocity = Jacobian·wind·scale + const eps = 0.02; // degrees, for the projection Jacobian + + for (const p of this.particles) { + if (p.age >= this.opts.maxAge) { + this.respawn(p); + continue; + } + const ll = this.map.unproject([p.x, p.y]); + const wind = interp(ll.lng, ll.lat); + if (!wind) { + p.age = this.opts.maxAge; // escaped the field → respawn next tick + continue; + } + const [u, v] = wind; + + // Local projection Jacobian: pixel deltas per degree at this point. + const east = this.map.project([ll.lng + eps, ll.lat]); + const north = this.map.project([ll.lng, ll.lat + eps]); + const jxLng = (east.x - p.x) / eps; + const jyLng = (east.y - p.y) / eps; + const jxLat = (north.x - p.x) / eps; + const jyLat = (north.y - p.y) / eps; + + p.xt = p.x + (jxLng * u + jxLat * v) * scale; + p.yt = p.y + (jyLng * u + jyLat * v) * scale; + p.speed = Math.sqrt(u * u + v * v); + p.age += 1; + } + } + + private draw(): void { + const ctx = this.ctx; + + // Fade existing trails toward transparent (keeps the basemap visible). + ctx.globalCompositeOperation = 'destination-in'; + ctx.fillStyle = `rgba(0,0,0,${this.opts.trailPersistence})`; + ctx.fillRect(0, 0, this.width, this.height); + ctx.globalCompositeOperation = 'source-over'; + + // Draw new trail segments, grouped by colour bucket. + const { colorScale } = this.opts; + ctx.lineWidth = this.opts.lineWidth; + const buckets: Particle[][] = colorScale.map(() => []); + for (const p of this.particles) { + if (p.age >= this.opts.maxAge || p.speed === 0) continue; + buckets[this.colorIndex(p.speed)].push(p); + } + let drawn = 0; + for (let i = 0; i < buckets.length; i++) { + const bucket = buckets[i]; + if (bucket.length === 0) continue; + drawn += bucket.length; + ctx.strokeStyle = colorScale[i]; + ctx.beginPath(); + for (const p of bucket) { + ctx.moveTo(p.x, p.y); + ctx.lineTo(p.xt, p.yt); + } + ctx.stroke(); + } + + if (import.meta.env.DEV && !this.debugLogged) { + this.debugLogged = true; + // One-shot diagnostic: confirms field, canvas size, and that segments + // are actually being drawn. Remove once the layer is verified. + // eslint-disable-next-line no-console + console.debug('[wind] first draw', { + hasInterp: !!this.interp, + canvas: `${this.width}x${this.height}`, + backing: `${this.canvas.width}x${this.canvas.height}`, + particles: this.particles.length, + drawnSegments: drawn, + host: this.host.className, + }); + } + + // Advance positions for the next frame. + for (const p of this.particles) { + p.x = p.xt; + p.y = p.yt; + } + } + + private frame = (now: number): void => { + this.raf = requestAnimationFrame(this.frame); + if (this.moving || !this.interp) return; + const frameTime = 1000 / this.opts.frameRate; + if (now - this.then < frameTime) return; + this.then = now - ((now - this.then) % frameTime); + this.evolve(); + this.draw(); + }; +} diff --git a/src/lib/features/wind/WindRenderer.svelte b/src/lib/features/wind/WindRenderer.svelte new file mode 100644 index 0000000..1e3c15a --- /dev/null +++ b/src/lib/features/wind/WindRenderer.svelte @@ -0,0 +1,332 @@ + + +{#if windSettings.enabled && prefetchSkipReason} +
+ + {#if prefetchSkipReason === 'wind.skippedLong'} + Wind sync skipped: flight > {windSettings.maxFlightDurationHours}h + {:else} + Wind sync skipped: region > {windSettings.maxRegionDegrees}° + {/if} +
+{/if} + + diff --git a/src/lib/features/wind/index.ts b/src/lib/features/wind/index.ts new file mode 100644 index 0000000..b70b760 --- /dev/null +++ b/src/lib/features/wind/index.ts @@ -0,0 +1,3 @@ +export { default as WindRenderer } from './WindRenderer.svelte'; +export { windCache } from './store'; +export { ParticleField, DEFAULT_PARTICLE_OPTIONS, type ParticleOptions } from './ParticleField'; diff --git a/src/lib/features/wind/store.ts b/src/lib/features/wind/store.ts new file mode 100644 index 0000000..5e34494 --- /dev/null +++ b/src/lib/features/wind/store.ts @@ -0,0 +1,61 @@ +/** + * Thin cache layer for wind field responses. + * + * Each unique set of request parameters is keyed by a stable JSON string so + * that the same (time, altitude, bbox, step) combination is fetched only once + * per session even if multiple effects request it concurrently. The cache is + * intentionally never invalidated during a session — the predictor's dataset + * does not change while the user is working. + */ + +import { windApi, type WindFieldParams } from '$api'; +import type { WindField } from '$domain'; + +function cacheKey(params: WindFieldParams): string { + return JSON.stringify({ + altitude: params.altitude ?? null, + step: params.step ?? null, + time: params.time ?? null, + min_lat: params.min_lat ?? null, + max_lat: params.max_lat ?? null, + min_lng: params.min_lng ?? null, + max_lng: params.max_lng ?? null, + }); +} + +class WindCache { + private readonly hits = new Map(); + private readonly pending = new Map>(); + + fetch(params: WindFieldParams): Promise { + const key = cacheKey(params); + + const hit = this.hits.get(key); + if (hit) return Promise.resolve(hit); + + const existing = this.pending.get(key); + if (existing) return existing; + + const promise = windApi + .field(params) + .then((field) => { + this.hits.set(key, field); + this.pending.delete(key); + return field; + }) + .catch((err: unknown) => { + this.pending.delete(key); + throw err; + }); + + this.pending.set(key, promise); + return promise; + } + + clear(): void { + this.hits.clear(); + this.pending.clear(); + } +} + +export const windCache = new WindCache(); diff --git a/src/lib/features/workspaces/WorkspaceRenderer.svelte b/src/lib/features/workspaces/WorkspaceRenderer.svelte new file mode 100644 index 0000000..961d7d7 --- /dev/null +++ b/src/lib/features/workspaces/WorkspaceRenderer.svelte @@ -0,0 +1,168 @@ + diff --git a/src/lib/features/workspaces/WorkspacesPanel.svelte b/src/lib/features/workspaces/WorkspacesPanel.svelte new file mode 100644 index 0000000..1290c45 --- /dev/null +++ b/src/lib/features/workspaces/WorkspacesPanel.svelte @@ -0,0 +1,159 @@ + + + +
+ +
+ + {#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 new file mode 100644 index 0000000..5cdc25c --- /dev/null +++ b/src/lib/features/workspaces/index.ts @@ -0,0 +1,4 @@ +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 new file mode 100644 index 0000000..98d82f4 --- /dev/null +++ b/src/lib/features/workspaces/store.ts @@ -0,0 +1,128 @@ +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 new file mode 100644 index 0000000..1ffe281 --- /dev/null +++ b/src/lib/features/workspaces/types.ts @@ -0,0 +1,23 @@ +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 new file mode 100644 index 0000000..3a65826 --- /dev/null +++ b/src/lib/i18n/index.ts @@ -0,0 +1,87 @@ +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 new file mode 100644 index 0000000..16bd6a5 --- /dev/null +++ b/src/lib/i18n/locales/en.json @@ -0,0 +1,206 @@ +{ + "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", + "wind": "Wind visualization", + "windEnabled": "Show wind layer", + "windStep": "Grid resolution (°)", + "windTrajectoryStep": "Trajectory grid res. (°)", + "windPrefetchInterval": "Pre-fetch interval (min)", + "windMaxDuration": "Max sync duration (h)", + "windMaxRegion": "Max region size (°)", + "windMargin": "Trajectory margin (°)", + "windParticleDensity": "Particle density", + "windParticleSpeed": "Particle speed", + "windTrailPersistence": "Trail length", + "windMaxVelocity": "Max wind speed (m/s)" + }, + "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" + }, + "tracking": { + "satelliteId": "Satellite ID", + "satelliteIdPlaceholder": "Enter satellite UUID...", + "connect": "Connect", + "disconnect": "Disconnect", + "status": "Status", + "status_idle": "Idle", + "status_connecting": "Connecting", + "status_connected": "Connected", + "status_error": "Error", + "packetCount": "{count} packets received", + "waitingData": "Waiting for data...", + "deviation": "Compare with forecast", + "selectForecast": "Reference forecast", + "noForecast": "— No forecast —", + "noData": "No telemetry data", + "altProfile": "Altitude profile", + "selectPrediction": "Select a forecast to show deviations", + "horizontalDev": "Horizontal deviation", + "devMax": "Max:", + "devCurrent": "Current:" + }, + "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 new file mode 100644 index 0000000..6441dbb --- /dev/null +++ b/src/lib/i18n/locales/ru.json @@ -0,0 +1,206 @@ +{ + "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": "Настройки сохранены", + "wind": "Визуализация ветра", + "windEnabled": "Показывать слой ветра", + "windStep": "Шаг сетки (°)", + "windTrajectoryStep": "Шаг сетки по траектории (°)", + "windPrefetchInterval": "Интервал предзагрузки (мин)", + "windMaxDuration": "Макс. длительность синхронизации (ч)", + "windMaxRegion": "Макс. размер региона (°)", + "windMargin": "Отступ вокруг траектории (°)", + "windParticleDensity": "Плотность частиц", + "windParticleSpeed": "Скорость частиц", + "windTrailPersistence": "Длина следа", + "windMaxVelocity": "Макс. скорость ветра (м/с)" + }, + "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": "Кликните на карту, чтобы выбрать координаты" + }, + "tracking": { + "satelliteId": "ID спутника", + "satelliteIdPlaceholder": "Введите UUID спутника...", + "connect": "Подключиться", + "disconnect": "Отключиться", + "status": "Статус", + "status_idle": "Ожидание", + "status_connecting": "Подключение", + "status_connected": "Подключено", + "status_error": "Ошибка", + "packetCount": "Получено пакетов: {count}", + "waitingData": "Ожидание данных...", + "deviation": "Сравнение с прогнозом", + "selectForecast": "Прогноз для сравнения", + "noForecast": "— Без прогноза —", + "noData": "Нет данных телеметрии", + "altProfile": "Высотный профиль", + "selectPrediction": "Выберите прогноз для отображения отклонений", + "horizontalDev": "Горизонтальное отклонение", + "devMax": "Макс.:", + "devCurrent": "Текущее:" + }, + "forecast": { + "success": "Запрос прогноза", + "successBody": "Запрос прогноза успешно выполнен!", + "error": "Ошибка прогноза", + "errorBody": "Ошибка при получении прогноза: {error}" + } +} diff --git a/src/lib/index.js b/src/lib/index.js deleted file mode 100644 index 856f2b6..0000000 --- a/src/lib/index.js +++ /dev/null @@ -1 +0,0 @@ -// 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 new file mode 100644 index 0000000..9b9f223 --- /dev/null +++ b/src/lib/map/Map.svelte @@ -0,0 +1,77 @@ + + +
+ {#if ready && map} + {@render children?.()} + {/if} +
diff --git a/src/lib/map/context.ts b/src/lib/map/context.ts new file mode 100644 index 0000000..9c71eab --- /dev/null +++ b/src/lib/map/context.ts @@ -0,0 +1,17 @@ +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 new file mode 100644 index 0000000..37ab14b --- /dev/null +++ b/src/lib/map/core.ts @@ -0,0 +1,121 @@ +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 new file mode 100644 index 0000000..80af90b --- /dev/null +++ b/src/lib/map/index.ts @@ -0,0 +1,9 @@ +export * from './core'; +export { createMapLibreMap } from './maplibre'; +export { plotPrediction, plotTelemetry, plotAnimatedMarker, plotEndMarker } 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 new file mode 100644 index 0000000..4f79848 --- /dev/null +++ b/src/lib/map/layers.ts @@ -0,0 +1,133 @@ +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 plotEndMarker(scene: Scene, lng: number, lat: number): void { + scene.addCircle('marker-core', { + center: [lng, lat], + radiusPx: 7, + color: '#6c757d', + strokeColor: '#ffffff', + strokeWidth: 2, + }); +} + +export function plotAnimatedMarker(scene: Scene, lng: number, lat: number): void { + 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 new file mode 100644 index 0000000..5b85861 --- /dev/null +++ b/src/lib/map/maplibre.ts @@ -0,0 +1,324 @@ +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); + const existing = this.map.getSource(layerId) as maplibregl.GeoJSONSource | undefined; + if (existing) { + existing.setData({ + type: 'Feature', + properties: {}, + geometry: { type: 'Point', coordinates: options.center }, + }); + return { id: layerId, remove: () => 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 new file mode 100644 index 0000000..17a6acf --- /dev/null +++ b/src/lib/map/tools/measure.ts @@ -0,0 +1,75 @@ +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 new file mode 100644 index 0000000..e34ca9a --- /dev/null +++ b/src/lib/map/tools/selection.ts @@ -0,0 +1,23 @@ +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 new file mode 100644 index 0000000..9e93089 --- /dev/null +++ b/src/lib/state/index.ts @@ -0,0 +1,2 @@ +export { persisted } from './persisted'; +export type { Serializer, PersistedOptions } from './persisted'; diff --git a/src/lib/state/persisted.ts b/src/lib/state/persisted.ts new file mode 100644 index 0000000..6f4a037 --- /dev/null +++ b/src/lib/state/persisted.ts @@ -0,0 +1,131 @@ +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 new file mode 100644 index 0000000..651229c --- /dev/null +++ b/src/lib/ui/CollapsibleCard.svelte @@ -0,0 +1,60 @@ + + + + + + + + + {#if !collapsed} + + {@render children?.()} + + {/if} + diff --git a/src/lib/ui/ConfirmationPrompt.svelte b/src/lib/ui/ConfirmationPrompt.svelte new file mode 100644 index 0000000..31376a1 --- /dev/null +++ b/src/lib/ui/ConfirmationPrompt.svelte @@ -0,0 +1,49 @@ + + + + {title} + + {#if children}{@render children()}{/if} + + + + + + diff --git a/src/lib/ui/EditableCell.svelte b/src/lib/ui/EditableCell.svelte new file mode 100644 index 0000000..db405fc --- /dev/null +++ b/src/lib/ui/EditableCell.svelte @@ -0,0 +1,56 @@ + + + + {#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 new file mode 100644 index 0000000..811bc2b --- /dev/null +++ b/src/lib/ui/LabelGroup.svelte @@ -0,0 +1,33 @@ + + +
+ + +
+ {@render children?.()} +
+
+ + diff --git a/src/lib/ui/PanelContainer.svelte b/src/lib/ui/PanelContainer.svelte new file mode 100644 index 0000000..3087d42 --- /dev/null +++ b/src/lib/ui/PanelContainer.svelte @@ -0,0 +1,38 @@ + + + + + diff --git a/src/lib/ui/SelectSearchable.svelte b/src/lib/ui/SelectSearchable.svelte new file mode 100644 index 0000000..6056cfc --- /dev/null +++ b/src/lib/ui/SelectSearchable.svelte @@ -0,0 +1,191 @@ + + + + +
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 new file mode 100644 index 0000000..53bff0e --- /dev/null +++ b/src/lib/ui/SpoilerGroup.svelte @@ -0,0 +1,56 @@ + + +
+ + + {#if expanded} +
+ {@render children?.()} +
+ {:else} +
+ {/if} +
+ + diff --git a/src/lib/ui/TabBar.svelte b/src/lib/ui/TabBar.svelte new file mode 100644 index 0000000..74e1511 --- /dev/null +++ b/src/lib/ui/TabBar.svelte @@ -0,0 +1,48 @@ + + +
+ {#each tabs as tab (tab.id)} + + {/each} +
+ + diff --git a/src/lib/ui/Toast.svelte b/src/lib/ui/Toast.svelte new file mode 100644 index 0000000..6c4dd27 --- /dev/null +++ b/src/lib/ui/Toast.svelte @@ -0,0 +1,40 @@ + + +
+ {#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 new file mode 100644 index 0000000..6832fa4 --- /dev/null +++ b/src/lib/ui/editor/ListEditor.svelte @@ -0,0 +1,338 @@ + + + + + + + + + + (isConfirmationVisible = false)}> +

Delete "{selectedItem?.name}"?

+
diff --git a/src/lib/ui/index.ts b/src/lib/ui/index.ts new file mode 100644 index 0000000..818b54e --- /dev/null +++ b/src/lib/ui/index.ts @@ -0,0 +1,13 @@ +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 new file mode 100644 index 0000000..efb1bda --- /dev/null +++ b/src/lib/ui/toasts.ts @@ -0,0 +1,54 @@ +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 new file mode 100644 index 0000000..2d80679 --- /dev/null +++ b/src/routes/+layout.svelte @@ -0,0 +1,22 @@ + + +{#if ready} + {@render children?.()} +{/if} + diff --git a/src/routes/+layout.ts b/src/routes/+layout.ts new file mode 100644 index 0000000..eae1350 --- /dev/null +++ b/src/routes/+layout.ts @@ -0,0 +1,5 @@ +// 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 1e03cf2..9beb424 100644 --- a/src/routes/+page.svelte +++ b/src/routes/+page.svelte @@ -1,7 +1,22 @@ - -
- -
\ No newline at end of file +
+
+ Loading... +
+
diff --git a/src/routes/+page.ts b/src/routes/+page.ts deleted file mode 100644 index b2463b2..0000000 --- a/src/routes/+page.ts +++ /dev/null @@ -1 +0,0 @@ -export const ssr =false; \ No newline at end of file diff --git a/src/routes/leaflet.svelte b/src/routes/leaflet.svelte deleted file mode 100644 index 380b25b..0000000 --- a/src/routes/leaflet.svelte +++ /dev/null @@ -1,548 +0,0 @@ - - -
-
-
- 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 new file mode 100644 index 0000000..186bfed --- /dev/null +++ b/src/routes/login/+page.svelte @@ -0,0 +1,5 @@ + + + diff --git a/src/routes/predict/+page.svelte b/src/routes/predict/+page.svelte new file mode 100644 index 0000000..6ed31c6 --- /dev/null +++ b/src/routes/predict/+page.svelte @@ -0,0 +1,115 @@ + + +
+ +
+ + + + + + +
+ {#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 new file mode 100644 index 0000000..d4115b2 --- /dev/null +++ b/src/routes/track/+page.svelte @@ -0,0 +1,119 @@ + + +
+ +
+ + + + + + +
+ + +
+ +
+
+
+
diff --git a/src/routes/user/account/+page.svelte b/src/routes/user/account/+page.svelte new file mode 100644 index 0000000..1a4d9b5 --- /dev/null +++ b/src/routes/user/account/+page.svelte @@ -0,0 +1,84 @@ + + +
+ +
+
+
+ + +
+ +
{$t('nav.account')}
+ + + + + +
+ +
+
+
+
+
+
+
+
+ + (showConfirm = false)}> +

{confirmBody}

+
diff --git a/src/routes/user/flights/+page.svelte b/src/routes/user/flights/+page.svelte new file mode 100644 index 0000000..9ef900e --- /dev/null +++ b/src/routes/user/flights/+page.svelte @@ -0,0 +1,36 @@ + + +
+ +
+
+
+ +
+ +
{$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 new file mode 100644 index 0000000..6778b78 --- /dev/null +++ b/src/routes/user/predictions/+page.svelte @@ -0,0 +1,36 @@ + + +
+ +
+
+
+ +
+ +
{$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 new file mode 100644 index 0000000..80e7007 --- /dev/null +++ b/src/routes/user/templates/+page.svelte @@ -0,0 +1,145 @@ + + +
+ +
+
+
+ + +
+ +
{$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 new file mode 100644 index 0000000..b4bb94f --- /dev/null +++ b/static/css/bootstrap.min.css @@ -0,0 +1,7 @@ +@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 new file mode 100644 index 0000000..a8a41c2 --- /dev/null +++ b/static/logo-full-ru-dark.svg @@ -0,0 +1,35 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/static/logo-lg.svg b/static/logo-lg.svg new file mode 100644 index 0000000..63a2208 --- /dev/null +++ b/static/logo-lg.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/static/logo.svg b/static/logo.svg new file mode 100644 index 0000000..741bbe4 --- /dev/null +++ b/static/logo.svg @@ -0,0 +1,28 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/static/marker-sm-red.png b/static/marker-sm-red.png new file mode 100644 index 0000000..62806ab Binary files /dev/null and b/static/marker-sm-red.png differ diff --git a/static/pop-marker.png b/static/pop-marker.png new file mode 100644 index 0000000..a111bfe Binary files /dev/null and b/static/pop-marker.png differ diff --git a/static/target-blue.png b/static/target-blue.png new file mode 100644 index 0000000..6ef9e5b Binary files /dev/null and b/static/target-blue.png differ diff --git a/static/target-red.png b/static/target-red.png new file mode 100644 index 0000000..c3222b8 Binary files /dev/null and b/static/target-red.png differ diff --git a/svelte.config.js b/svelte.config.js index 10c4eeb..b591c22 100644 --- a/svelte.config.js +++ b/svelte.config.js @@ -1,13 +1,34 @@ -import adapter from '@sveltejs/adapter-auto'; +import adapter from '@sveltejs/adapter-static'; -/** @type {import('@sveltejs/kit').Config} */ +/** + * 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} + */ const config = { kit: { - // 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() - } + 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', + }, + }, }; export default config; diff --git a/test-results/.last-run.json b/test-results/.last-run.json new file mode 100644 index 0000000..cbcc1fb --- /dev/null +++ b/test-results/.last-run.json @@ -0,0 +1,4 @@ +{ + "status": "passed", + "failedTests": [] +} \ No newline at end of file diff --git a/test.json b/test.json deleted file mode 100644 index fdbd53b..0000000 --- a/test.json +++ /dev/null @@ -1,920 +0,0 @@ -{ - "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/test_client/beacon-20250406.log b/test_client/beacon-20250406.log new file mode 100644 index 0000000..91d5f27 --- /dev/null +++ b/test_client/beacon-20250406.log @@ -0,0 +1,1406 @@ +{ID 1 activated} +[1;0;0;228;43206;] +<1;1;1;228;0;0;0;0;0;0;0;17;5;22;21;0;10;50;123;0;199;0;27;16165;> +[1;0;0;226;51394;] +<1;2;1;226;0;0;0;0;0;0;0;17;5;21;21;0;6;50;123;0;199;0;27;31492;> +[1;0;0;227;22723;] +<1;3;1;227;0;0;0;0;0;0;0;17;5;22;21;0;8;50;123;0;199;0;27;50667;> +[1;61660840;129385896;227;39173;] +<1;4;1;227;102;0;97;0;0;0;9;17;5;23;21;0;9;50;122;0;199;0;27;9266;> +[1;61660860;129385912;227;29165;] +<1;5;1;227;101;0;-2;0;16;0;11;17;5;23;21;0;9;50;122;0;199;0;27;3494;> +[1;61660860;129385928;227;56286;] +<1;6;1;227;100;0;0;0;16;0;12;17;5;24;21;0;8;50;122;0;199;0;27;36504;> +[1;61660852;129385928;227;19119;] +<1;7;1;227;97;0;-2;0;16;0;12;17;5;24;21;0;8;50;122;0;199;0;27;42743;> +[1;61660836;129385944;227;58408;] +<1;8;1;227;95;0;-1;0;15;0;12;18;5;23;21;0;9;50;122;0;199;0;27;53919;> +[1;61660828;129385944;227;4349;] +<1;9;1;227;92;0;-2;0;15;0;12;18;5;23;21;0;9;50;122;0;199;0;27;46078;> +[1;61660828;129385944;227;4349;] +<1;10;1;227;92;0;0;0;14;0;12;18;5;22;21;0;10;50;122;0;199;0;27;48284;> +[1;61660824;129385944;227;5624;] +<1;11;1;227;91;0;0;0;-1;0;12;18;5;22;21;0;9;50;122;0;199;0;27;1026;> +[1;61660824;129385944;227;5624;] +<1;12;1;227;91;0;0;0;-1;0;12;18;5;22;21;0;9;50;122;0;199;0;27;13319;> +[1;61660824;129385944;227;5624;] +<1;13;1;227;91;0;0;0;-1;0;12;18;5;22;21;0;10;50;122;0;199;0;27;41867;> +[1;61660820;129385944;227;55034;] +<1;14;1;227;90;0;0;0;-1;0;12;18;5;22;21;0;9;50;122;0;199;0;27;47393;> +[1;61660820;129385944;227;55034;] +<1;15;1;227;90;0;0;0;0;0;12;18;5;22;21;0;10;50;122;0;199;0;27;5351;> +[1;61660804;129385944;231;5387;] +<1;63;1;231;87;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;1961;> +[1;61660804;129385944;230;34058;] +<1;64;1;230;87;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;971;> +[1;61660804;129385944;231;5387;] +<1;65;1;231;88;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;9295;> +[1;61660804;129385944;231;5387;] +<1;66;1;231;88;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;57614;> +[1;61660804;129385944;230;34058;] +<1;67;1;230;88;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;12269;> +[1;61660804;129385944;230;34058;] +<1;68;1;230;88;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;16301;> +[1;61660804;129385944;230;34058;] +<1;69;1;230;88;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;31890;> +[1;61660804;129385944;230;34058;] +<1;70;1;230;88;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;46672;> +[1;61660804;129385944;230;34058;] +<1;71;1;230;88;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;62831;> +[1;61660804;129385928;230;18796;] +<1;72;1;230;89;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;56643;> +[1;61660804;129385928;230;18796;] +<1;73;1;230;89;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;40572;> +[1;61660804;129385928;230;18796;] +<1;74;1;230;89;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;6082;> +[1;61660804;129385928;230;18796;] +<1;75;1;230;89;0;0;0;0;0;12;25;5;30;22;0;7;50;121;0;199;0;27;53442;> +[1;61660804;129385928;230;18796;] +<1;76;1;230;89;0;0;0;0;0;12;25;5;30;22;0;6;50;121;0;199;0;27;50366;> +[1;61660804;129385928;230;18796;] +<1;77;1;230;89;0;0;0;0;0;12;25;5;30;22;0;6;50;121;0;199;0;27;34689;> +[1;61660804;129385928;230;18796;] +<1;78;1;230;88;0;0;0;0;0;12;25;5;30;22;0;6;50;121;0;199;0;27;31404;> +[1;61660804;129385928;230;18796;] +<1;79;1;230;89;0;1;0;0;0;12;25;5;30;22;0;7;50;121;0;199;0;27;37891;> +[1;61660804;129385928;231;55661;] +<1;80;1;231;89;0;0;0;0;0;12;25;6;30;22;0;7;50;121;0;199;0;27;21196;> +[1;61660804;129385928;231;55661;] +<1;81;1;231;89;0;0;0;0;0;12;25;5;29;22;0;6;50;121;0;199;0;27;26063;> +[1;61660800;129385928;231;6767;] +<1;82;1;231;89;1;0;0;0;0;12;25;5;29;22;0;7;50;121;0;199;0;27;56610;> +[1;61660800;129385928;230;35438;] +<1;83;1;230;89;0;0;0;0;0[1;61660800;129385928;230;35438;] +<1;84;1;230;89;0;0;0;0;0;12;25;6;30;22;0;6;50;121;0;199;0;27;17106;> +[1;61660804;129386032;231;10665;] +<1;85;1;231;88;0;0;0;0;0;12;25;6;29;22;0;6;50;121;0;199;0;27;13406;> +[1;61660804;129386032;231;10665;] +<1;86;1;231;88;0;0;0;0;0;12;25;6;28;22;0;7;50;121;0;199;0;27;41692;> +[1;61660800;129386032;230;31402;] +<1;87;1;230;88;0;0;0;0;0;12;25;6;28;22;0;6;50;121;0;199;0;27;48386;> +[1;61660800;129386032;230;31402;] +<1;88;1;230;88;0;0;0;0;0;12;25;6;28;22;0;6;50;121;0;199;0;27;44354;> +[1;61660796;129386032;230;40396;] +<1;89;1;230;88;0;0;0;0;0;12;24;6;28;22;0;8;50;120;0;199;0;27;60439;> +[1;61660796;129386032;230;40396;] +<1;90;1;230;89;0;1;0;0;0;12;24;6;27;22;0;9;50;120;0;199;0;27;36933;> +[1;61660796;129386032;230;40396;] +<1;91;1;230;89;0;0;0;0;0;12;24;6;27;22;0;10;50;120;0;199;0;27;21353;> +[1;61660796;129386032;230;40396;] +<1;92;1;230;89;0;0;0;0;0;12;24;6;27;22;0;9;50;120;0;199;0;27;34811;> +[1;61660796;129386032;231;3533;] +<1;93;1;231;89;0;0;0;0;0;12;24;6;27;22;0;10;50;120;0;199;0;27;60131;> +[1;61660796;129386032;230;40396;] +<1;94;1;230;89;0;0;0;0;0;12;24;6;27;22;0;10;50;120;0;199;0;27;870;> +[1;61660796;129386016;230;55790;] +<1;95;1;230;89;0;0;0;0;0;12;24;6;27;22;0;9;50;120;0;199;0;27;3653;> +[1;61660796;129386016;230;55790;] +<1;96;1;230;89;0;0;0;0;0;12;24;6;27;22;0;10;50;120;0;199;0;27;9056;> +[1;61660796;129386016;230;55790;] +<1;97;1;230;89;-1;0;0;0;0;12;24;6;27;22;0;9;50;120;0;199;0;27;61357;> +[1;61660796;129386016;230;55790;] +<1;98;1;230;89;0;0;0;0;0[1;61660796;129386016;230;55790;] +<1;99;1;230;89;0;0;0;0;0;12;24;7;27;22;0;9;50;120;0;199;0;27;35653;> +[1;61660796;129386016;230;55790;] +<1;100;1;230;89;0;0;0;0;0;12;24;7;27;22;0;10;50;120;0;199;0;27;5122;> +[1;61660796;129386016;230;55790;] +<1;101;1;230;90;0;1;0;0;0;12;24;6;27;22;0;10;50;120;0;199;0;27;19883;> +[1;61660796;129386032;231;3533;] +<1;102;1;231;90;0;0;0;0;0;12;24;6;27;22;0;9;50;120;0;199;0;27;22991;> +[1;61660796;129386016;230;55790;] +<1;103;1;230;90;0;0;0;0;0;12;24;6;27;22;0;9;50;120;0;199;0;27;38700;> +[1;61660800;129386016;230;16008;] +<1;104;1;230;90;0;0;0;0;0;12;24;6;27;22;0;10;50;120;0;199;0;27;19765;> +[1;61660800;129386016;230;16008;] +<1;105;1;230;90;0;0;0;0;0;12;24;6;26;22;0;10;50;120;0;199;0;27;56629;> +[1;61660800;129386016;231;44681;] +<1;106;1;231;91;0;1;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;47970;> +[1;61660800;129386016;230;16008;] +<1;107;1;230;91;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;58433;> +[1;61660804;129386016;230;64906;] +<1;108;1;230;91;0;0;0;0;0;12;24;7;26;22;0;8;50;120;0;199;0;27;9532;> +[1;61660804;129386016;230;64906;] +<1;109;1;230;91;0;0;0;0;0;12;24;7;26;22;0;8;50;120;0;199;0;27;26115;> +[1;61660804;129386016;230;64906;] +<1;110;1;230;90;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;37009;> +[1;61660808;129386016;231;26766;] +<1;111;1;231;90;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;24178;> +[1;61660808;129386016;231;26766;] +<1;112;1;231;90;0;0;0;0;0;12;24;7;26;22;0;8;50;120;0;199;0;27;18958;> +[1;61660808;129386016;230;63631;] +<1;113;1;230;90;0;0;0;0;0;12;24;7;26;22;0;8;50;120;0;199;0;27;34029;> +[1;61660808;129386032;231;11436;] +<1;114;1;231;90;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;20914;> +[1;61660812;129386032;230;35706;] +<1;115;1;230;90;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;40785;> +[1;61660812;129386032;231;7035;] +<1;116;1;231;90;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;55244;> +[1;61660816;129386032;230;18552;] +<1;117;1;230;91;0;1;0;0;0;12;24;7;26;22;0;10;50;120;0;199;0;27;33103;> +[1;61660816;129386032;230;18552;] +<1;118;1;230;91;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;58370;> +[1;61660816;129386032;231;55417;] +<1;119;1;231;91;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;10977;> +[1;61660836;129386016;231;64762;] +<1;120;1;231;97;0;5;0;0;0;12;24;7;26;22;0;10;50;120;0;199;0;27;34707;> +[1;61660844;129386000;230;39640;] +<1;121;1;230;100;0;3;0;1;0;12;24;7;26;22;0;9;50;120;0;199;0;27;54728;> +[1;61660848;129386000;230;40925;] +<1;122;1;230;102;0;1;0;1;0;12;24;7;26;22;0;8;50;120;0;199;0;27;52650;> +[1;61660848;129386000;231;4060;] +<1;123;1;231;102;1;0;0;1;0;12;24;7;26;22;0;8;50;120;0;199;0;27;31092;> +[1;61660848;129386000;230;40925;] +<1;124;1;230;103;-1;1;0;1;0;12;24;7;26;22;0;9;50;120;0;199;0;27;20579;> +[1;61660848;129386000;231;4060;] +<1;125;1;231;102;0;0;0;1;0;12;24;8;26;22;0;9;50;120;0;199;0;27;38104;> +[1;61660848;129386000;231;4060;] +<1;126;1;231;103;0;0;0;1;0;12;24;8;26;22;0;9;50;120;0;199;0;27;18864;> +[1;61660852;129386000;231;14347;] +<1;127;1;231;104;0;0;0;0;0;12;24;8;26;22;0;8;50;120;0;199;0;27;19494;> +[1;61660852;129385984;231;58967;] +<1;128;1;231;104;0;0;0;0;0;12;24;7;26;22;0;7;50;120;0;199;0;27;45673;> +[1;61660856;129385984;231;9557;] +<1;129;1;231;105;0;1;0;0;0;12;24;8;26;22;0;7;50;120;0;199;0;27;11981;> +[1;61660860;129385984;230;59175;] +<1;130;1;230;105;-1;0;0;0;0;12;24;8;26;22;0;7;50;120;0;199;0;27;4034;> +[1;61660860;129385984;231;30502;] +<1;131;1;231;106;0;0;0;0;0;12;24;8;26;22;0;8;50;120;0;199;0;27;47013;> +[1;61660860;129385984;231;30502;] +<1;132;1;231;106;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;42647;> +[1;61660860;129385984;231;30502;] +<1;133;1;231;106;0;0;0;0;0;12;24;7;26;22;0;9;50;120;0;199;0;27;46740;> +[1;61660860;129385984;231;30502;] +<1;134;1;231;107;0;0;0;0;0;12;24;7;26;22;0;10;50;120;0;199;0;27;42113;> +[1;61660864;129385984;231;46116;] +<1;135;1;231;107;0;0;0;0;0;12;24;7;27;22;0;9;50;120;0;199;0;27;47373;> +[1;61660864;129385968;231;47307;] +<1;136;1;231;107;0;0;0;0;0;12;24;8;27;22;0;10;50;120;0;199;0;27;8530;> +[1;61660904;129385952;231;20298;] +<1;137;1;231;107;0;0;0;0;0;12;24;8;27;22;0;9;50;120;0;199;0;27;26881;> +[1;61660904;129385952;232;48970;] +<1;138;1;232;106;0;0;0;0;0;12;24;7;27;22;0;10;50;120;0;199;0;27;26821;> +[1;61660904;129385968;231;58745;] +<1;139;1;231;105;0;0;0;0;0;12;24;7;27;22;0;9;50;120;0;199;0;27;58304;> +[1;61660904;129385968;231;58745;] +<1;140;1;231;104;0;0;0;0;0;12;23;8;27;22;0;10;50;120;0;199;0;27;9415;> +[1;61660908;129385968;230;28797;] +<1;141;1;230;104;0;0;0;0;0;12;22;8;27;22;0;9;50;120;0;199;0;27;51584;> +[1;61660908;129385968;230;28797;] +<1;142;1;230;104;0;0;0;0;0;12;21;8;27;22;0;9;50;120;0;199;0;27;19845;> +[1;61660912;129385968;230;18346;] +<1;143;1;230;103;0;0;0;0;0;12;21;8;27;22;0;9;50;120;0;199;0;27;24451;> +[1;61660916;129385968;230;33960;] +<1;144;1;230;103;0;0;0;0;0;12;20;8;27;22;0;9;50;120;0;199;0;27;33675;> +[1;61660912;129385984;230;19269;] +<1;145;1;230;104;0;0;0;0;0;12;20;8;27;22;0;10;50;120;0;199;0;27;34299;> +[1;61661000;129385880;230;37051;] +<1;146;1;230;105;0;0;0;0;5;12;19;8;26;22;0;9;50;120;0;199;0;27;11642;> +[1;61661304;129385592;262;52616;] +<1;147;1;262;160;31;53;0;4;0;12;18;7;26;22;0;9;50;120;0;199;0;27;11908;> +[1;61661772;129385296;324;37115;] +<1;148;2;324;217;61;56;11;14;5;12;19;7;25;22;0;9;50;120;0;199;0;27;4511;> +[1;61662228;129384672;383;64250;] +<1;149;2;383;273;58;55;21;24;7;12;19;6;25;23;0;9;50;120;0;199;0;27;32619;> +[1;61662648;129384144;438;27741;] +<1;150;2;438;328;55;58;30;34;7;12;19;6;24;23;0;9;50;120;0;199;0;27;16311;> +[1;61663184;129383712;495;3526;] +<1;151;2;495;384;57;54;40;43;6;12;19;5;24;23;0;8;50;120;0;199;0;27;57466;> +[1;61663736;129383152;556;13012;] +<1;152;2;556;445;59;59;49;52;4;12;19;5;23;23;0;9;50;120;0;199;0;27;22372;> +[1;61664504;129382672;623;3180;] +<1;153;2;623;506;67;59;59;58;10;12;19;4;23;23;0;10;50;120;0;199;0;27;35023;> +[1;61665372;129382496;687;46541;] +<1;154;2;687;569;64;66;59;58;8;12;19;4;22;23;0;8;50;120;0;199;0;27;33428;> +[1;61666112;129381848;750;44176;] +<1;155;2;750;627;63;56;61;59;11;12;18;3;22;23;0;9;50;120;0;199;0;27;26666;> +[1;61667108;129381488;811;30709;] +<1;156;2;811;685;59;56;62;59;12;12;18;3;22;23;0;9;50;120;0;199;0;27;50840;> +[1;61668084;129381544;872;43164;] +<1;157;2;872;744;62;58;62;59;9;12;18;2;21;23;0;9;50;120;0;199;0;27;8917;> +[1;61668912;129382128;932;13353;] +<1;158;2;932;798;59;53;63;59;9;12;18;2;21;23;0;8;50;120;0;199;0;27;32166;> +[1;61670032;129382168;993;10464;] +<1;159;2;993;860;60;61;61;58;14;12;18;2;21;23;0;8;50;120;0;199;0;27;30092;> +[1;61671084;129382840;1057;6305;] +<1;160;2;1057;924;63;63;61;59;9;12;17;1;21;23;0;8;50;120;0;199;0;27;34453;> +[1;61672152;129383440;1123;57631;] +<1;161;2;1123;987;65;67;61;60;15;12;17;1;21;23;0;10;50;120;0;199;0;27;41772;> +[1;61673396;129384632;1185;9267;] +<1;162;2;1185;1044;64;56;62;59;12;12;17;1;21;23;0;9;50;120;0;199;0;27;37033;> +[1;61674528;129385696;1245;25420;] +<1;163;2;1245;1101;58;56;62;59;16;12;17;1;21;23;0;9;50;120;0;199;0;27;49506;> +[1;61675552;129387112;1304;21604;] +<1;164;2;1304;1162;60;60;61;60;9;12;16;1;21;23;0;8;50;120;0;199;0;27;323;> +[1;61676624;129388104;1367;64065;] +<1;165;2;1367;1218;62;54;62;59;15;12;16;2;21;23;0;8;50;120;0;199;0;27;24043;> +[1;61677484;129389528;1424;53320;] +<1;166;2;1424;1276;57;57;61;58;11;12;16;2;20;23;0;8;50;120;0;199;0;27;460;> +[1;61678436;129390720;1483;10665;] +<1;167;2;1483;1332;57;55;60;56;10;12;16;2;20;23;0;10;50;120;0;199;0;27;10802;> +[1;61679456;129391768;1543;30828;] +<1;168;2;1543;1394;61;67;59;58;10;12;16;2;20;23;0;8;50;120;0;199;0;27;60153;> +[1;61680232;129392536;1603;51938;] +<1;169;2;1603;1452;58;5[1;61681052;129393112;1664;1795;] +<1;170;2;1664;1510;63;57;60;57;12;12;16;1;20;23;0;9;50;120;0;199;0;27;6168;> +[1;61682024;129394224;1721;45781;] +<1;171;2;1721;1564;56;53;59;57;10;12;16;2;19;23;0;7;50;120;0;199;0;27;22724;> +[1;61682764;129395176;1778;27793;] +<1;172;2;1778;1622;56;56;58;57;9;12;16;1;19;23;0;8;50;120;0;199;0;27;48010;> +[1;61683564;129395872;1833;28863;] +<1;173;2;1833;1674;55;56;58;57;9;12;15;1;19;23;0;9;50;120;0;199;0;27;28497;> +[1;61684056;129397112;1887;40367;] +<1;174;2;1887;1728;54;52;58;57;6;12;15;1;19;23;0;8;50;120;0;199;0;27;52434;> +[1;61684504;129398024;1946;9737;] +<1;175;2;1946;1786;58;57;57;55;8;12;15;0;19;23;0;10;50;120;0;199;0;27;57943;> +[1;61684892;129399264;2007;12121;] +<1;176;2;2007;1846;63;59;57;55;8;12;15;0;19;23;0;10;50;120;0;199;0;27;20422;> +[1;61685116;129400560;2067;18931;] +<1;177;2;2067;1902;57;55;56;55;5;12;15;0;19;23;0;9;50;120;0;199;0;27;16764;> +[1;61685436;129401600;2121;48920;] +<1;178;2;2121;1960;58;56;57;56;5;12;15;0;19;23;0;9;50;120;0;199;0;27;58754;> +[1;61685420;129402784;2181;6451;] +<1;179;2;2181;2016;57;61;57;56;8;12;15;0;19;23;0;8;50;120;0;199;0;27;16457;> +[1;61685436;129403688;2237;14179;] +<1;180;2;2237;2072;55;55;58;57;8;12;15;0;19;23;0;8;50;120;0;199;0;27;5323;> +[1;61685544;129404696;2295;27376;] +<1;181;2;2295;2130;59;56;58;57;5;12;14;-1;19;23;0;9;50;120;0;199;0;27;25243;> +[1;61685484;129405576;2354;4874;] +<1;182;2;2354;2189;59;57;58;57;6;12;14;-1;19;23;0;8;50;120;0;199;0;27;7125;> +[1;61685516;129406464;2413;1816;] +<1;183;2;2413;2246;57;55;57;57;6;12;14;-1;18;23;0;9;50;120;0;199;0;27;65208;> +[1;61685564;129408064;2470;54971;] +<1;184;2;2470;2300;57;54;57;57;8;12;14;-1;18;22;0;8;50;120;0;199;0;27;54739;> +[1;61685660;129409712;2522;56829;] +<1;185;2;2522;2355;53;58;57;57;7;12;14;-1;18;22;0;9;50;120;0;199;0;27;48662;> +[1;61685836;129411224;2579;15087;] +<1;186;2;2579;2410;54;53;57;56;12;12;14;-1;17;22;0;8;50;120;0;199;0;27;40628;> +[1;61686096;129413024;2636;28054;] +<1;187;2;2636;2468;58;57;57;56;9;12;14;-1;18;22;0;8;50;120;0;199;0;27;53982;> +[1;61686172;129414600;2697;19188;] +<1;188;2;2697;2529;60;59;57;56;9;12;14;-2;18;22;0;8;50;120;0;199;0;27;19535;> +[1;61686308;129416424;2761;51540;] +<1;189;2;2761;2591;62;62;57;56;10;12;14;-2;17;22;0;9;50;120;0;199;0;27;52123;> +[1;61686424;129418272;2819;31196;] +<1;190;2;2819;2649;59;62;58;58;11;12;13;-2;17;22;0;9;50;120;0;199;0;27;46646;> +[1;61686768;129420256;2880;60032;] +<1;191;2;2880;2709;59;59;58;58;9;12;13;-2;17;22;0;8;50;120;0;199;0;27;4460;> +[1;61687148;129422104;2944;61697;] +<1;192;2;2944;2776;64;65;59;59;12;12;13;-3;17;22;0;8;50;120;0;199;0;27;62515;> +[1;61687292;129424072;3012;11286;] +<1;193;2;3012;2843;68;66;61;61;11;12;13;-3;17;22;0;9;50;120;0;199;0;27;20211;> +[1;61687356;129425872;3077;5759;] +<1;194;2;3077;2907;64;62;62;62;6;12;13;-4;17;22;0;9;50;120;0;199;0;27;50969;> +[1;61687504;129427704;3136;17088;] +<1;195;2;3136;2968;59;60;63;62;11;12;13;-4;18;22;0;9;50;120;0;199;0;27;11714;> +[1;61687580;129429976;3199;41066;] +<1;196;2;3199;3029;59;66;62;63;14;12;13;-5;17;22;0;8;50;120;0;199;0;27;30987;> +[1;61687664;129432496;3265;35991;] +<1;197;2;3265;3093;66;62;63;63;13;12;12;-5;17;22;0;8;50;120;0;199;0;27;25158;> +[1;61687880;129434832;3327;45804;] +<1;198;2;3327;3157;64;63;64;63;10;12;12;-5;17;22;0;9;50;120;0;199;0;27;39353;> +[1;61688056;129437040;3393;51373;] +<1;199;2;3393;3223;66;65;63;63;15;12;12;-6;17;22;0;8;50;120;0;199;0;27;7768;> +[1;61687996;129439712;3459;5510;] +<1;200;2;3459;3284;64;59;63;63;10;12;12;-6;16;22;0;7;50;120;0;199;0;27;37442;> +[1;61688172;129441864;3521;54618;] +<1;201;2;3521;3347;60;67;63;64;14;12;11;-7;16;22;0;8;50;120;0;199;0;27;7612;> +[1;61688292;129444368;3585;39153;] +<1;202;2;3585;3412;65;63;63;63;14;12;11;-7;15;22;0;7;50;120;0;199;0;27;60521;> +[1;61688340;129446552;3650;781;] +<1;203;2;3650;3474;61;60;64;63;11;12;11;-7;16;22;0;7;50;120;0;199;0;27;24345;> +[1;61688408;129449312;3714;6694;] +<1;204;2;3714;3537;64;62;64;63;15;12;11;-8;15;22;0;9;50;120;0;199;0;27;8601;> +[1;61688384;129451952;3778;19403;] +<1;205;2;3778;3599;65;60;64;63;13;12;10;-9;15;22;0;7;50;120;0;199;0;27;29387;> +[1;61688388;129454160;3841;2543;] +<1;206;2;3841;3659;62;59;63;62;12;12;10;-9;14;22;0;7;50;120;0;199;0;27;14488;> +[1;61688584;129456560;3903;54640;] +<1;207;2;3903;3720;64;65;63;62;15;12;9;-9;14;22;0;8;50;120;0;199;0;27;4224;> +[1;61688836;129458880;3967;5847;] +<1;208;2;3967;3782;61;61;63;61;10;12;9;-10;15;22;0;7;50;120;0;199;0;27;18944;> +[1;61689132;129461216;4031;51157;] +<1;209;2;4031;3845;63;61;62;61;11;12;9;-10;14;22;0;7;50;120;0;199;0;27;5838;> +[1;61689512;129463256;4094;9222;] +<1;210;2;4094;3907;64;61;62;61;14;12;9;-10;14;22;0;8;50;120;0;199;0;27;18274;> +[1;61689956;129465656;4158;42240;] +<1;211;2;4158;3971;62;62;62;61;15;12;9;-10;13;22;0;8;50;120;0;199;0;27;43526;> +[1;61690496;129468136;4222;55201;] +<1;212;2;4222;4033;64;61;63;62;17;12;8;-11;14;22;0;8;50;120;0;199;0;27;48303;> +[1;61690864;129471176;4282;54229;] +<1;213;2;4282;4094;61;65;63;62;15;12;8;-11;14;22;0;8;50;119;0;199;0;27;21145;> +[1;61691244;129473952;4346;20243;] +<1;214;2;4346;4159;62;63;63;62;15;12;8;-11;14;22;0;9;50;119;0;199;0;27;23474;> +[1;61691508;129477344;4409;62975;] +<1;215;2;4409;4215;64;54;63;62;16;12;8;-11;13;22;0;9;50;120;0;199;0;27;35706;> +[1;61691680;129480360;4468;22298;] +<1;216;2;4468;4278;57;61;62;61;15;12;8;-12;13;22;0;9;50;119;0;199;0;27;47350;> +[1;61691844;129483568;4533;36533;] +<1;217;2;4533;4341;64;6[1;61692104;129486328;4591;12404;] +<1;218;2;4591;4395;58;53;61;60;17;12;8;-12;13;22;0;9;50;120;0;199;0;27;53739;> +[1;61692472;129489608;4649;55026;] +<1;219;2;4649;4451;58;61;61;59;16;12;7;-13;12;21;0;9;50;119;0;199;0;27;40520;> +[1;61692788;129492384;4703;21858;] +<1;220;2;4703;4507;55;54;59;58;17;12;7;-13;12;21;0;8;50;120;0;199;0;27;15964;> +[1;61693316;129495272;4759;58005;] +<1;221;2;4759;4564;54;56;58;57;15;12;7;-13;13;21;0;9;50;120;0;199;0;27;57191;> +[1;61693948;129498016;4822;130;] +<1;222;2;4822;4626;62;61;59;58;16;12;7;-14;12;21;0;8;50;119;0;199;0;27;17784;> +[1;61694456;129500704;4883;40603;] +<1;223;2;4883;4689;61;62;58;57;16;12;7;-14;11;21;0;9;50;119;0;199;0;27;40513;> +[1;61694884;129503704;4948;49293;] +<1;224;2;4948;4750;62;61;59;59;18;12;6;-14;11;21;0;8;50;119;0;199;0;27;48821;> +[1;61695368;129506712;5007;28028;] +<1;225;2;5007;4811;63;65;60;59;18;12;6;-14;11;21;0;8;50;119;0;199;0;27;52542;> +[1;61695944;129509504;5071;62143;] +<1;226;2;5071;4872;60;60;60;60;14;12;6;-15;11;21;0;9;50;119;0;199;0;27;60551;> +[1;61696472;129512496;5131;38685;] +<1;227;2;5131;4936;56;62;60;61;18;12;5;-15;11;21;0;9;50;119;0;199;0;27;35580;> +[1;61696872;129516360;5198;12649;] +<1;228;2;5198;4996;69;59;62;61;22;12;5;-16;11;21;0;8;50;119;0;199;0;27;49515;> +[1;61697212;129519960;5257;38820;] +<1;229;2;5257;5058;61;60;62;61;14;12;5;-16;11;21;0;9;50;119;0;199;0;27;16252;> +[1;61697432;129523144;5317;10049;] +<1;230;2;5317;5116;59;57;61;60;21;12;5;-16;11;21;0;9;50;119;0;199;0;27;60286;> +[1;61697752;129526928;5378;57659;] +<1;231;2;5378;5176;62;64;61;60;17;12;5;-16;11;21;0;8;50;119;0;199;0;27;710;> +[1;61697948;129530168;5435;41952;] +<1;232;2;5435;5235;57;5[1;61697948;129533648;5501;31044;] +<1;233;2;5501;5297;66;60;61;60;19;12;5;-17;10;21;0;8;50;119;0;199;0;27;41467;> +[1;61698236;129537248;5565;24818;] +<1;234;2;5565;5356;62;58;62;60;16;12;4;-17;10;21;0;8;50;119;0;199;0;27;62476;> +[1;61698252;129540312;5625;44253;] +<1;235;2;5625;5419;61;62;61;59;17;12;4;-17;10;21;0;8;50;120;0;199;0;27;4124;> +[1;61698404;129543256;5685;58943;] +<1;236;2;5685;5482;59;62;61;60;14;12;4;-18;10;21;0;8;50;119;0;199;0;27;58877;> +[1;61698596;129545856;5754;62591;] +<1;237;2;5754;5546;68;69;61;61;13;12;4;-18;10;20;0;9;50;119;0;199;0;27;17313;> +[1;61698792;129548904;5817;15737;] +<1;238;2;5817;5610;61;62;62;61;19;12;3;-18;9;20;0;8;50;119;0;199;0;27;35460;> +[1;61698976;129551760;5881;1469;] +<1;239;2;5881;5673;63;61;63;62;15;12;3;-19;8;20;0;8;50;119;0;199;0;27;20546;> +[1;61699416;129554416;5944;22959;] +<1;240;2;5944;5736;65;62;63;62;14;12;3;-19;8;20;0;9;50;120;0;199;0;27;50976;> +[1;61699644;129557528;6008;23350;] +<1;241;2;6008;5798;65;61;63;63;14;12;2;-19;8;20;0;8;50;119;0;199;0;27;44104;> +[1;61700100;129559952;6072;24701;] +<1;242;2;6072;5861;61;62;63;63;18;12;2;-20;8;20;0;8;50;119;0;199;0;27;6837;> +[1;61700372;129562808;6135;52815;] +<1;243;2;6135;5925;64;67;63;63;13;12;2;-20;8;20;0;8;50;120;0;199;0;27;61246;> +[1;61700596;129565136;6200;10080;] +<1;244;2;6200;5990;64;64;63;63;14;12;2;-20;8;20;0;7;50;119;0;199;0;27;60770;> +[1;61700752;129567536;6270;50874;] +<1;245;2;6270;6058;68;66;63;63;10;12;1;-21;7;20;0;8;50;119;0;199;0;27;19459;> +[1;61700956;129569760;6334;47374;] +<1;246;2;6334;6120;62;60;64;63;11;12;1;-21;7;20;0;8;50;119;0;199;0;27;50701;> +[1;61701096;129572264;6401;3026;] +<1;247;2;6401;6186;67;65;64;63;18;12;1;-21;7;20;0;8;50;119;0;199;0;27;24160;> +[1;61701292;129575104;6462;33580;] +<1;248;2;6462;6248;61;61;65;63;12;12;1;-21;7;20;0;8;50;119;0;199;0;27;59889;> +[1;61701228;129577640;6534;37801;] +<1;249;2;6534;6317;69;73;65;65;14;12;0;-22;7;20;0;8;50;119;0;199;0;27;60868;> +[1;61701432;129580096;6604;63147;] +<1;250;2;6604;6386;69;67;66;65;15;12;0;-23;7;20;0;9;50;119;0;199;0;27;16944;> +[1;61701680;129582856;6668;22050;] +<1;251;2;6668;6450;65;63;66;65;13;12;0;-24;7;20;0;8;50;119;0;199;0;27;30617;> +[1;61701680;129585448;6736;7677;] +<1;252;2;6736;6517;66;65;66;65;15;12;0;-24;7;20;0;8;50;119;0;199;0;27;19462;> +[1;61701964;129588184;6802;2329;] +<1;253;2;6802;6578;66;61;67;65;11;12;0;-25;7;19;0;9;50;119;0;199;0;27;33424;> +[1;61702060;129590528;6865;65241;] +<1;254;2;6865;6645;63;70;66;66;18;12;0;-25;6;19;0;8;50;119;0;199;0;27;17032;> +[1;61702004;129593368;6929;55268;] +<1;255;2;6929;6707;62;60;66;65;10;12;0;-26;6;19;0;8;50;119;0;199;0;27;34984;> +[1;61702032;129595840;6993;26139;] +<1;256;2;6993;6770;65;62;65;64;17;12;0;-26;6;19;0;7;50;119;0;199;0;27;14488;> +[1;61702384;129598616;7058;28492;] +<1;257;2;7058;6832;66;61;65;64;12;12;0;-27;7;19;0;7;50;119;0;199;0;27;27467;> +[1;61702516;129601136;7122;55901;] +<1;258;2;7122;6903;62;69;64;64;14;12;-1;-27;6;19;0;8;50;119;0;199;0;27;50507;> +[1;61702692;129603680;7196;30218;] +<1;259;2;7196;6968;71;64;64;64;11;12;-1;-28;7;19;0;8;50;119;0;199;0;27;27307;> +[1;61702724;129606184;7258;51729;] +<1;260;2;7258;7031;60;67;65;63;14;12;-1;-28;7;19;0;7;50;119;0;199;0;27;28737;> +[1;61702700;129609056;7323;18330;] +<1;261;2;7323;7094;67;62;65;64;14;12;-1;-29;7;19;0;9;50;119;0;199;0;27;43071;> +[1;61703092;129611984;7389;41291;] +<1;262;2;7389;7163;68;67;65;64;17;12;-1;-29;6;19;0;8;50;119;0;199;0;27;24991;> +[1;61703188;129615536;7460;16913;] +<1;263;2;7460;7226;66;62;66;65;20;12;-2;-30;6;19;0;7;50;119;0;199;0;27;32042;> +[1;61703580;129618824;7523;6924;] +<1;264;2;7523;7290;66;62;66;65;15;12;-2;-30;5;19;0;9;50;119;0;199;0;27;62039;> +[1;61703708;129622128;7594;50570;] +<1;265;2;7594;7357;70;66;66;64;21;12;-2;-31;6;19;0;7;50;119;0;199;0;27;19893;> +[1;61704176;129625472;7660;57012;] +<1;266;2;7660;7423;68;64;67;65;13;12;-2;-31;5;19;0;8;50;119;0;199;0;27;49993;> +[1;61704476;129628360;7723;6509;] +<1;267;2;7723;7487;63;69;66;64;18;12;-2;-31;5;19;0;8;50;119;0;199;0;27;10634;> +[1;61704912;129631696;7792;27996;] +<1;268;2;7792;7551;66;62;67;65;16;12;-3;-31;4;18;0;7;50;119;0;199;0;27;56930;> +[1;61704952;129635256;7855;15279;] +<1;269;2;7855;7615;64;63;66;64;20;12;-3;-32;5;18;0;7;50;119;0;199;0;27;6765;> +[1;61705228;129638824;7925;22942;] +<1;270;2;7925;7682;69;65;66;65;17;12;-3;-32;4;18;0;9;50;119;0;199;0;27;52315;> +[1;61705484;129642072;7997;379;] +<1;271;2;7997;7755;73;71;67;66;20;12;-3;-32;4;18;0;7;50;119;0;199;0;27;24327;> +[1;61705576;129645664;8069;49802;] +<1;272;2;8069;7821;70;65;67;66;14;12;-4;-33;4;18;0;7;50;119;0;199;0;27;3986;> +[1;61705772;129648816;8133;23065;] +<1;273;2;8133;7889;67;72;68;66;19;12;-4;-33;5;18;0;9;50;119;0;199;0;27;53517;> +[1;61705732;129652096;8210;42427;] +<1;274;2;8210;7960;73;69;69;67;18;12;-4;-34;4;18;0;7;50;119;0;199;0;27;2849;> +[1;61705872;129655152;8283;44164;] +<1;275;2;8283;8031;76;69;71;69;13;12;-4;-34;3;18;0;7;50;119;0;199;0;27;48307;> +[1;61705732;129658048;8354;8695;] +<1;276;2;8354;8101;71;68;71;69;20;12;-4;-35;3;18;0;9;50;119;0;199;0;27;1086;> +[1;61705688;129660352;8423;20189;] +<1;277;2;8423;8166;67;63;71;68;12;12;-5;-36;3;18;0;7;50;119;0;199;0;27;47459;> +[1;61705660;129663192;8488;28662;] +<1;278;2;8488;8228;63;61;69;67;13;12;-5;-37;4;18;0;7;50;119;0;199;0;27;23776;> +[1;61705572;129665424;8558;29738;] +<1;279;2;8558;8296;64;72;70;67;17;12;-5;-38;4;18;0;8;50;119;0;199;0;27;55596;> +[1;61705332;129667920;8631;17081;] +<1;280;2;8631;8365;73;67;70;67;14;12;-6;-38;3;18;0;7;50;119;0;199;0;27;19869;> +[1;61705276;129669952;8696;44966;] +<1;281;2;8696;8432;67;65;68;66;8;12;-6;-39;3;17;0;7;50;119;0;199;0;27;14753;> +1;68;65;8;12;-6;-39;2;17;0;8;50;119;0;199;0;27;15842;> +[1;61705068;129673248;8827;4772;] +<1;283;2;8827;8556;58;60;66;64;13;12;-7;-39;2;17;0;7;50;119;0;199;0;27;31255;> +[1;61704692;129675320;8899;14741;] +<1;284;2;8899;8620;74;70;68;66;10;12;-7;-40;2;17;0;7;50;119;0;199;0;27;19755;> +[1;61704376;129677504;8954;50162;] +<1;285;2;8954;8682;52;61;65;64;14;12;-7;-40;1;17;0;7;50;119;0;199;0;27;833;> +[1;61704108;129680040;9029;52527;] +<1;286;2;9029;8748;75;64;66;63;10;12;-7;-41;1;17;0;8;50;119;0;199;0;27;14205;> +[1;61703932;129682480;9102;44698;] +<1;287;2;9102;8820;73;70;67;64;13;12;-7;-41;1;17;0;8;50;119;0;199;0;27;615;> +[1;61703768;129684480;9167;26309;] +<1;288;2;9167;8886;66;64;66;65;10;12;-7;-41;1;17;0;7;50;119;0;199;0;27;30468;> +[1;61703476;129686752;9232;52208;] +<1;289;2;9232;8947;64;60;67;65;13;12;-8;-41;1;17;0;9;50;119;0;199;0;27;11113;> +[1;61703344;129688600;9301;3434;] +<1;290;2;9301;9016;68;68;67;65;7;12;-8;-42;1;17;0;9;50;119;0;199;0;27;21543;> +[1;61702852;129690384;9371;17676;] +<1;291;2;9371;9081;70;70;67;65;14;12;-8;-42;1;17;0;6;50;119;0;199;0;27;45617;> +[1;61702764;129692704;9442;40235;] +<1;292;2;9442;9146;70;63;69;66;8;12;-8;-43;1;17;0;9;50;119;0;199;0;27;62409;> +[1;61702560;129695008;9515;50008;] +<1;293;2;9515;9215;68;67;68;66;15;12;-9;-44;0;16;0;6;50;119;0;199;0;27;28977;> +[1;61702336;129697648;9578;12108;] +<1;294;2;9578;9275;64;59;68;65;12;12;-9;-45;0;16;0;8;50;119;0;199;0;27;17566;> +[1;61702088;129699856;9648;62445;] +<1;295;2;9648;9346;72;70;68;65;10;12;-9;-45;0;16;0;7;50;119;0;199;0;27;22129;> +[1;61702068;129702072;9713;46744;] +<1;296;2;9713;9408;64;61;69;66;13;12;-10;-46;0;16;0;6;50;119;0;199;0;27;25408;> +[1;61701812;129704176;9786;11986;] +<1;297;2;9786;9478;73;74;68;65;11;12;-10;-46;0;16;0;7;50;119;0;199;0;27;13392;> +[1;61701800;129706144;9856;31346;] +<1;298;2;9856;9543;68;63;69;66;12;12;-10;-47;0;16;0;9;50;119;0;199;0;27;12075;> +[1;61701412;129708816;9932;53402;] +<1;299;2;9932;9607;72;63;68;66;13;12;-11;-46;0;16;0;7;50;119;0;199;0;27;43554;> +[1;61701196;129711032;9995;34755;] +<1;300;2;9995;9678;68;70;68;65;14;12;-11;-47;0;16;0;6;50;119;0;199;0;27;53298;> +[1;61700820;129713088;10078;30091;] +<1;301;2;10078;9752;78;72;69;67;11;12;-11;-48;0;16;0;6;50;119;0;199;0;27;12293;> +[1;61700376;129715288;10139;42492;] +<1;302;2;10139;9812;65;59;70;67;13;12;-11;-49;0;16;0;9;50;119;0;199;0;27;21792;> +[1;61700024;129718048;10203;24812;] +<1;303;2;10203;9882;60;73;69;67;13;12;-11;-49;0;16;0;7;50;119;0;199;0;27;31863;> +[1;61699636;129720400;10285;38955;] +<1;304;2;10285;9958;79;75;70;67;11;12;-12;-50;0;15;0;9;50;119;0;199;0;27;12447;> +[1;61699116;129723696;10370;52369;] +<1;305;2;10370;10030;86;70;72;69;21;12;-12;-50;-1;15;0;8;50;119;0;199;0;27;4865;> +[1;61698720;129726760;10428;53306;] +<1;306;2;10428;10095;61;63;73;69;11;12;-12;-51;0;15;0;8;50;119;0;199;0;27;1061;> +[1;61698132;129728896;10514;3103;] +<1;307;2;10514;10171;81;74;73;69;15;12;-13;-51;-1;15;0;8;50;119;0;199;0;27;29352;> +[1;61697524;129730880;10582;47429;] +<1;308;2;10582;10229;69;58;73;70;13;12;-13;-52;-1;15;0;9;50;119;0;199;0;27;44265;> +[1;61697052;129733352;10652;32490;] +<1;309;2;10652;10311;69;85;73;69;17;12;-13;-52;-2;15;0;7;50;119;0;199;0;27;4722;> +[1;61696920;129735760;10750;5112;] +<1;310;2;10750;10398;97;86;74;73;9;12;-14;-52;-3;15;0;9;50;119;0;199;0;27;61168;> +[1;61696740;129738544;10821;1064;] +<1;311;2;10821;10464;73;64;77;72;16;12;-14;-53;-2;15;0;9;50;119;0;199;0;27;23884;> +[1;61696572;129740664;10902;4642;] +<1;312;2;10902;10548;76;82;75;73;8;12;-14;-53;-2;15;0;7;50;119;0;199;0;27;52851;> +[1;61696264;129742768;10986;64147;] +<1;313;2;10986;10621;88;72;78;75;16;12;-14;-53;-3;15;0;9;50;119;0;199;0;27;24082;> +[1;61696108;129745224;11067;60666;] +<1;314;2;11067;10698;79;75;79;75;10;12;-14;-54;-3;14;0;9;50;119;0;199;0;27;51245;> +[1;61695864;129747664;11149;28034;] +<1;315;2;11149;10783;81;89;82;78;15;12;-15;-54;-4;14;0;9;50;119;0;199;0;27;27632;> +[1;61695372;129749880;11241;26853;] +<1;316;2;11241;10859;92;74;83;77;14;12;-15;-54;-4;14;0;9;50;119;0;199;0;27;15521;> +[1;61695124;129752880;11309;54575;] +<1;317;2;11309;10924;66;63;80;76;17;12;-15;-54;-4;14;0;9;50;119;0;199;0;27;20308;> +[1;61695080;129755368;11383;55657;] +<1;318;2;11383;10994;73;68;80;75;8;12;-15;-127;-4;14;0;9;50;119;0;199;2;27;32925;> +[1;61694988;129757296;11457;33839;] +<1;319;2;11457;11066;76;70;78;73;12;12;-16;-127;-5;14;0;9;50;119;0;199;2;27;33527;> +[1;61694344;129759808;11530;62587;] +<1;320;2;11530;11132;76;67;78;72;16;12;-16;-127;-4;14;0;9;50;119;0;199;2;27;57988;> +[1;61693836;129763120;11606;9378;] +<1;321;2;11606;11204;75;75;76;73;17;12;-16;-127;-4;14;0;9;50;119;0;199;2;27;27677;> +8;77;74;69;12;12;-16;-127;-4;14;0;9;50;119;0;199;2;27;62205;> +[1;61692872;129768160;11766;60916;] +<1;323;2;11766;11354;77;69;75;71;15;12;-16;-127;-4;13;0;9;50;119;0;199;2;27;23262;> +[1;61692680;129770464;11832;64188;] +<1;324;2;11832;11421;69;65;75;70;10;12;-16;-127;-4;13;0;9;50;119;0;199;2;27;18288;> +[1;61692512;129772112;11899;27676;] +<1;325;2;11899;11480;70;64;74;70;10;12;-16;-127;-6;13;0;9;50;119;0;199;2;27;40113;> +[1;61691960;129774400;11969;28824;] +<1;326;2;11969;11545;68;63;73;69;16;12;-16;-127;-5;13;0;9;50;119;0;199;2;27;28441;> +[1;61691772;129777296;12045;26251;] +<1;327;2;12045;11619;71;72;72;68;14;12;-16;-127;-5;13;0;8;50;119;0;199;2;27;58273;> +[1;61691672;129779600;12111;52538;] +<1;328;2;12111;11682;71;62;73;66;14;12;-16;-127;-5;13;0;8;50;119;0;199;2;27;20100;> +[1;61691020;129782576;12198;26687;] +<1;329;2;12198;11771;85;87;71;68;21;12;-16;-127;-4;13;0;8;50;119;0;199;2;27;246;> +[1;61690460;129786576;12278;47707;] +<1;330;2;12278;11830;82;61;74;69;20;12;-16;-127;-4;13;0;8;50;119;0;199;2;27;14522;> +[1;61689728;129789808;12344;8407;] +<1;331;2;12344;11910;64;82;72;70;16;12;-16;-127;-4;13;0;8;50;118;0;199;2;27;51626;> +[1;61688864;129791656;12451;762;] +<1;332;2;12451;12004;99;91;77;74;11;12;-16;-127;-4;13;0;8;50;118;0;199;2;27;44355;> +[1;61688364;129792848;12530;47197;] +<1;333;2;12530;12079;84;72;81;76;8;12;-16;-127;-4;12;0;15;50;117;0;199;3;27;4827;> +[1;61688244;129794280;12618;39893;] +<1;334;2;12618;12156;87;75;82;79;9;12;-17;-127;-5;13;0;15;50;117;0;199;3;27;48429;> +[1;61688192;129796176;12682;17331;] +<1;335;2;12682;12213;65;60;81;75;10;12;-17;-127;-5;13;0;15;50;117;0;199;3;27;48189;> +[1;61687968;129798600;12743;32058;] +<1;336;2;12743;12271;60;56;77;73;14;12;-17;-127;-4;14;0;15;50;117;0;199;3;27;30473;> +[1;61687476;129801224;12800;39541;] +<1;337;2;12800;12325;54;52;76;69;15;12;-16;-127;-3;14;0;15;50;116;0;199;3;27;44003;> +[1;61686800;129803280;12862;61877;] +<1;338;2;12862;12384;60;57;70;63;11;12;-16;-127;-2;14;0;15;50;116;0;199;3;27;38893;> +[1;61686424;129804624;12921;35577;] +<1;339;2;12921;12448;62;63;66;61;6;12;-16;-127;-2;14;0;15;50;117;0;199;3;27;25188;> +[1;61686356;129805920;12993;27677;] +<1;340;2;12993;12512;66;69;62;60;6;12;-16;-127;-3;14;0;16;50;117;0;199;3;27;53646;> +[1;61686388;129807360;13052;48567;] +<1;341;2;13052;12569;61;55;61;59;9;12;-16;-127;-1;15;0;16;50;116;0;199;3;27;49724;> +[1;61686132;129809552;13111;45898;] +<1;342;2;13111;12629;62;59;62;59;13;12;-16;-127;-2;15;0;15;50;117;0;199;3;27;64141;> +[1;61685596;129811128;13175;5248;] +<1;343;2;13175;12691;62;60;62;60;5;12;-15;-54;-1;15;0;16;50;116;0;199;1;27;36137;> +[1;61685628;129812336;13248;45014;] +<1;344;2;13248;12759;71;66;64;62;7;12;-15;-54;-2;15;0;15;50;116;0;199;1;27;17142;> +[1;61685768;129813816;13308;2614;] +<1;345;2;13308;12811;62;52;64;62;8;12;-15;-54;0;15;0;16;50;116;0;199;1;27;808;> +[1;61685944;129815432;13357;6966;] +<1;346;2;13357;12862;49;54;64;60;8;12;-15;-54;0;15;0;16;50;116;0;199;1;27;17298;> +[1;61686356;129817168;13425;38164;] +<1;347;2;13425;12930;65;66;61;58;12;12;-15;-53;-1;15;0;15;50;116;0;199;1;27;42009;> +[1;61686688;129819216;13501;8533;] +<1;348;2;13501;13015;79;83;62;60;14;12;-15;-53;-1;15;0;16;50;116;0;199;1;27;16637;> +[1;61687060;129821704;13594;29252;] +<1;349;2;13594;13083;90;67;65;64;14;12;-15;-52;-1;15;0;16;50;116;0;199;1;27;50401;> +[1;61687004;129824600;13647;18013;] +<1;350;2;13647;13147;53;63;68;64;18;12;-15;-52;-1;15;0;16;50;116;0;199;1;27;37140;> +[1;61686780;129827488;13708;21974;] +<1;351;2;13708;13198;64;56;66;64;11;12;-15;-51;0;16;0;16;50;116;0;199;1;27;28933;> +[1;61686400;129829208;13760;61732;] +<1;352;2;13760;13244;50;44;66;64;11;12;-14;-51;0;16;0;16;50;116;0;199;1;27;8433;> +[1;61686284;129831464;13803;27632;] +<1;353;2;13803;13296;41;51;66;63;11;12;-14;-50;0;16;0;16;50;116;0;199;1;27;22096;> +[1;61686368;129832824;13879;55854;] +<1;354;2;13879;13367;70;69;63;61;5;12;-13;-51;1;16;0;14;50;116;0;199;1;27;9300;> +[1;61686504;129834488;13943;31162;] +<1;355;2;13943;13432;67;64;61;58;11;12;-13;-50;1;16;0;15;50;116;0;199;1;27;12937;> +[1;61686360;129836672;14007;1637;] +<1;356;2;14007;13488;69;57;59;58;12;12;-13;-50;1;16;0;15;50;116;0;199;1;27;29611;> +[1;61686464;129838792;14063;3162;] +<1;357;2;14063;13541;55;57;58;56;12;12;-12;-50;1;16;0;14;50;116;0;199;1;27;51397;> +[1;61686204;129840272;14118;13422;] +<1;358;2;14118;13609;57;66;58;57;7;12;-12;-49;1;16;0;14;50;116;0;199;1;27;56433;> +[1;61685844;129841312;14188;1946;] +<1;359;2;14188;13665;65;55;61;61;7;12;-12;-49;1;16;0;14;50;116;0;199;1;27;61699;> +[1;61685444;129843264;14257;56034;] +<1;360;2;14257;13731;67;64;62;61;12;12;-12;-49;1;16;0;14;50;116;0;199;1;27;49194;> +[1;61685068;129844696;14323;49280;] +<1;361;2;14323;13807;70;74;62;60;6;12;-11;-49;1;16;0;15;50;116;0;199;1;27;48711;> +[1;61684828;129846712;14401;61981;] +<1;362;2;14401;13881;82;72;64;62;11;12;-11;-49;1;16;0;14;50;116;0;199;1;27;62078;> +[1;61684440;129847928;14477;12868;] +<1;363;2;14477;13945;6> +[1;61684032;129849216;14521;22115;] +<1;364;2;14521;13988;46;41;68;66;8;12;-11;-48;2;17;0;15;50;116;0;199;1;27;23194;> +[1;61683920;129850632;14573;17870;] +<1;365;2;14573;14050;52;61;64;61;5;12;-10;-48;3;17;0;7;50;118;0;199;0;27;7092;> +7;68;65;65;4;12;-10;-48;4;16;0;8;50;118;0;199;0;27;54305;> +[1;61683336;129852480;14721;55747;] +<1;367;2;14721;14198;72;76;66;64;11;12;-10;-49;4;16;0;8;50;118;0;199;0;27;6574;> +[1;61683124;129854352;14794;63010;] +<1;368;2;14794;14255;73;60;66;64;11;12;-10;-48;4;16;0;8;50;118;0;199;0;27;34968;> +[1;61682468;129856432;14855;21663;] +<1;369;2;14855;14317;59;65;63;62;14;12;-9;-48;4;15;0;8;50;118;0;199;0;27;12265;> +[1;61682056;129858280;14910;3117;] +<1;370;2;14910;14390;64;70;63;63;9;12;-9;-48;4;15;0;8;50;118;0;199;0;27;28158;> +[1;61682072;129859952;15015;55684;] +<1;371;2;15015;14490;96;97;68;70;5;12;-9;-47;3;15;0;8;50;118;0;199;0;27;43764;> +[1;61682332;129861112;15090;1582;] +<1;372;2;15090;14554;80;63;75;72;11;12;-9;-47;2;15;0;8;50;118;0;199;0;27;12187;> +[1;61682260;129862872;15146;63257;] +<1;373;2;15146;14607;55;51;72;70;6;12;-9;-47;3;14;0;8;50;118;0;199;0;27;47496;> +[1;61681788;129864072;15214;23936;] +<1;374;2;15214;14679;73;76;68;69;11;12;-9;-46;2;14;0;8;50;118;0;199;0;27;50832;> +[1;61681492;129865552;15281;38404;] +<1;375;2;15281;14749;59;68;71;70;3;12;-9;-46;1;14;0;8;50;118;0;199;0;27;52444;> +[1;61681288;129866408;15348;55220;] +<1;376;2;15348;14815;70;64;73;71;5;12;-9;-46;1;14;0;8;50;118;0;199;0;27;39683;> +[1;61681416;129867384;15429;37299;] +<1;377;2;15429;14895;80;78;71;69;8;12;-9;-45;0;14;0;8;50;118;0;199;0;27;48203;> +[1;61681884;129868680;15508;8797;] +<1;378;2;15508;14957;77;60;68;67;7;12;-9;-45;1;14;0;8;50;118;0;199;0;27;16782;> +[1;61682088;129869992;15550;20451;] +<1;379;2;15550;15005;47;50;67;67;6;12;-9;-45;2;14;0;8;50;118;0;199;0;27;6282;> +[1;61682008;129871136;15604;45370;] +<1;380;2;15604;15061;52;55;66;63;6;12;-8;-45;2;14;0;8;50;118;0;199;0;27;28286;> +[1;61682024;129872744;15652;18907;] +<1;381;2;15652;15110;53;48;64;61;6;12;-8;-45;2;14;0;8;50;118;0;199;0;27;17776;> +[1;61681924;129873992;15705;40090;] +<1;382;2;15705;15154;51;43;59;58;8;12;-8;-45;3;14;0;8;50;118;0;199;0;27;59958;> +[1;61681768;129875568;15753;18893;] +<1;383;2;15753;15208;47;53;56;54;8;12;-8;-44;3;13;0;8;50;118;0;199;0;27;35661;> +[1;61681624;129877088;15805;34437;] +<1;384;2;15805;15256;51;48;51;49;7;12;-8;-44;2;13;0;9;50;118;0;199;0;27;6743;> +[1;61681628;129878648;15849;8071;] +<1;385;2;15849;15301;42;47;49;49;6;12;-7;-44;3;13;0;8;50;118;0;199;0;27;46186;> +[1;61681684;129879824;15890;64006;] +<1;386;2;15890;15345;47;43;48;47;7;12;-7;-44;3;13;0;9;50;118;0;199;0;27;47685;> +[1;61681580;129881392;15935;56733;] +<1;387;2;15935;15392;44;46;47;47;7;12;-7;-43;3;13;0;9;50;118;0;199;0;27;29317;> +[1;61681660;129882920;15985;18463;] +<1;388;2;15985;15443;42;50;46;47;9;12;-7;-44;4;13;0;9;50;118;0;199;0;27;2302;> +[1;61681676;129884784;16030;7671;] +<1;389;2;16030;15490;51;45;46;46;8;12;-7;-43;3;13;0;9;50;118;0;199;0;27;10419;> +[1;61681744;129886752;16091;41181;] +<1;390;2;16091;15549;52;63;47;49;9;12;-7;-43;3;13;0;9;50;118;0;199;0;27;63208;> +[1;61681932;129888120;16143;13059;] +<1;391;2;16143;15599;52;49;49;49;5;12;-6;-42;3;13;0;9;50;118;0;199;0;27;27545;> +[1;61681748;129889392;16198;40509;] +<1;392;2;16198;15656;59;55;51;51;8;12;-6;-42;4;13;0;9;50;118;0;199;0;27;40880;> +[1;61681544;129890640;16252;52827;] +<1;393;2;16252;15708;53;51;51;52;2;12;-6;-42;3;13;0;9;50;118;0;199;0;27;26148;> +[1;61681636;129891416;16306;16386;] +<1;394;2;16306;15766;53;57;53;53;7;12;-6;-42;4;13;0;8;50;118;0;199;0;27;58294;> +[1;61681868;129892640;16360;7479;] +<1;395;2;16360;15821;53;54;56;55;4;12;-6;-43;4;13;0;8;50;118;0;199;0;27;9376;> +[1;61681920;129893344;16410;13532;] +<1;396;2;16410;15869;49;49;53;52;3;12;-6;-43;5;13;0;9;50;118;0;199;0;27;10404;> +[1;61681940;129894168;16462;38470;] +<1;397;2;16462;15914;45;48;52;52;4;12;-5;-43;6;13;0;8;50;118;0;199;0;27;58418;> +[1;61682044;129894672;16502;50229;] +<1;398;2;16502;15958;39;42;50;50;4;12;-5;-42;5;13;0;9;50;118;0;199;0;27;25656;> +[1;61682140;129895520;16556;40721;] +<1;399;2;16556;16013;59;54;51;50;3;12;-5;-41;4;13;0;9;50;118;0;199;0;27;41699;> +[1;61682188;129895952;16609;53234;] +<1;400;2;16609;16067;51;52;50;50;3;12;-5;-40;4;13;0;9;50;118;0;199;0;27;11144;> +[1;61682216;129896656;16652;15394;] +<1;401;2;16652;16120;42;52;47;50;2;12;-5;-40;3;12;0;16;50;116;0;199;1;27;12063;> +[1;61682356;129897552;16710;32858;] +<1;402;2;16710;16171;56;50;47;50;7;12;-5;-40;4;13;0;16;50;116;0;199;1;27;27243;> +[1;61682604;129898304;16765;16678;] +<1;403;2;16765;16230;61;64;49;50;4;12;-5;-39;4;13;0;15;50;116;0;199;1;27;27200;> +[1;61682868;129898632;16820;64193;] +<1;404;2;16820;16287;45;55;50;52;5;12;-5;-39;4;14;0;15;50;116;0;199;1;27;35026;> +[1;61683212;129899704;16882;44863;] +<1;405;2;16882;16353;67;64;53;55;4;12;-5;-39;4;14;0;15;50;116;0;199;1;27;60826;> +[1;61683404;129900144;16942;40265;] +<1;406;2;16942;16409;59;55;54;56;4;12;-4;-39;6;14;0;15;50;116;0;199;1;27;63506;> +[1;61683700;129901304;17000;38053;] +<1;407;2;17000;16464;56;54;55;56;5;12;-4;-39;5;15;0;15;50;116;0;199;1;27;1158;> +[1;61683948;129902344;17044;27841;] +<1;408;2;17044;16511;50;48;56;57;6;12;-4;-38;5;15;0;15;50;116;0;199;1;27;18739;> +[1;61684128;129903184;17094;61375;] +<1;409;2;17094;16558;42;49;55;57;4;12;-4;-38;5;15;0;15;50;116;0;199;1;27;49265;> +[1;61684300;129904392;17135;48427;] +<1;410;2;17135;16603;46;44;54;54;9;12;-3;-38;6;15;0;14;50;116;0;199;1;27;2925;> +[1;61684340;129905856;17179;42687;] +<1;411;2;17179;16651;43;47;52;52;6;12;-3;-37;6;15;0;15;50;116;0;199;1;27;29314;> +[1;61684236;129907528;17230;41389;] +<1;412;2;17230;16702;49;49;49;49;9;12;-3;-37;7;15;0;14;50;116;0;199;1;27;59117;> +[1;61684156;129908952;17281;11335;] +<1;413;2;17281;16749;50;46;48;48;6;12;-3;-37;9;16;0;14;50;116;0;199;1;27;63787;> +[1;61684000;129910232;17324;61689;] +<1;414;2;17324;16795;41;48;45;47;8;12;-2;-36;9;16;0;13;50;116;0;199;1;27;40264;> +[1;61683888;129910992;17370;35091;] +<1;415;2;17370;16859;52;62;45;47;3;12;-2;-36;9;16;0;14;50;116;0;199;1;27;32813;> +[1;61684240;129912016;17460;46721;] +<1;416;2;17460;16944;79;83;47;52;7;12;-2;-37;10;16;0;15;50;116;0;199;1;27;25794;> +[1;61684248;129912904;17522;61214;] +<1;417;2;17522;16994;62;49;54;56;6;12;-1;-37;11;16;0;13;50;116;0;199;1;27;8735;> +[1;61684304;129913592;17567;30077;] +<1;418;2;17567;17045;51;49;55;57;4;12;-1;-37;10;16;0;14;50;116;0;199;1;27;45105;> +[1;61684208;129914320;17606;41066;] +<1;419;2;17606;17089;36;47;53;56;6;12;-1;-37;11;16;0;15;50;116;0;199;1;27;17196;> +[1;61684072;129915600;17659;37277;] +<1;420;2;17659;17137;51;47;55;56;4;12;-1;-37;11;17;0;7;50;117;0;199;0;27;3605;> +[1;61683744;129916504;17706;59664;] +<1;421;2;17706;17198;53;59;54;56;8;12;0;-37;12;17;0;7;50;117;0;199;0;27;26584;> +[1;61683252;129916976;17760;48773;] +<1;422;2;17760;17244;52;45;54;54;6;12;0;-36;13;16;0;8;50;118;0;199;0;27;27017;> +[1;61682872;129918168;17807;18348;] +<1;423;2;17807;17295;46;50;48;50;5;12;0;-36;13;16;0;8;50;118;0;199;0;27;48634;> +[1;61682528;129918856;17849;49006;] +<1;424;2;17849;17341;41;45;48;49;5;12;0;-36;12;15;0;8;50;117;0;199;0;27;18641;> +[1;61682344;129920056;17891;36756;] +<1;425;2;17891;17388;49;51;47;49;7;12;0;-36;12;15;0;9;50;118;0;199;0;27;60002;> +[1;61682040;129921064;17948;14230;] +<1;426;2;17948;17443;39;54;48;50;4;12;0;-36;13;15;0;9;50;118;0;199;0;27;45146;> +[1;61681984;129921680;17995;8979;] +<1;427;2;17995;17494;53;50;48;50;2;12;0;-36;13;15;0;7;50;117;0;199;0;27;17031;> +[1;61682072;129922360;18033;8044;] +<1;428;2;18033;17540;45;45;47;48;4;12;1;-37;14;15;0;6;50;117;0;199;0;27;37150;> +[1;61682004;129922656;18084;56014;] +<1;429;2;18084;17588;50;47;46;49;2;12;1;-37;14;15;0;9;50;118;0;199;0;27;54827;> +[1;61681916;129923168;18127;25020;] +<1;430;2;18127;17632;41;43;46;48;1;12;1;-37;14;15;0;6;50;117;0;199;0;27;18105;> +[1;61681952;129923280;18185;52112;] +<1;431;2;18185;17689;47;61;47;48;1;12;1;-36;13;15;0;8;50;118;0;199;0;27;27532;> +[1;61682024;129923664;18234;52566;] +<1;432;2;18234;17736;49;46;47;49;4;12;1;-36;12;15;0;8;50;118;0;199;0;27;53642;> +[1;61682000;129924160;18279;18561;] +<1;433;2;18279;17788;52;50;48;49;1;12;2;-37;13;15;0;6;50;117;0;199;0;27;43571;> +[1;61681944;129924848;18329;62983;] +<1;434;2;18329;17841;42;52;47;49;5;12;2;-37;13;14;0;8;50;118;0;199;0;27;61441;> +[1;61681884;129925448;18383;6091;] +<1;435;2;18383;17889;59;47;48;50;1;12;2;-36;13;14;0;8;50;118;0;199;0;27;30681;> +[1;61681988;129925768;18431;11437;] +<1;436;2;18431;17945;46;55;47;51;2;12;2;-36;12;14;0;8;50;118;0;199;0;27;24360;> +[1;61681688;129926224;18480;16804;] +<1;437;2;18480;17995;48;51;49;51;4;12;2;-36;12;14;0;8;50;117;0;199;0;27;35941;> +[1;61681572;129926544;18530;63870;] +<1;438;2;18530;18042;49;49;49;50;1;12;2;-36;13;14;0;8;50;117;0;199;0;27;18302;> +[1;61681300;129926776;18579;8593;] +<1;439;2;18579;18093;47;50;48;50;3;12;3;-36;13;14;0;9;50;118;0;199;0;27;52129;> +[1;61681192;129927112;18627;29516;] +<1;440;2;18627;18147;47;53;49;50;1;12;3;-36;13;14;0;9;50;117;0;199;0;27;12386;> +[1;61681156;129927488;18678;56817;] +<1;441;2;18678;18195;50;47;51;50;4;12;3;-35;13;14;0;8;50;117;0;199;0;27;12640;> +[1;61681368;129928144;18729;44235;] +<1;442;2;18729;18244;49;48;50;49;3;12;3;-36;12;14;0;8;50;117;0;199;0;27;60792;> +[1;61681412;129928344;18771;48734;] +<1;443;2;18771;18295;41;50;49;49;3;12;3;-35;12;14;0;7;50;117;0;199;0;27;1799;> +[1;61681596;129928784;18823;40902;] +<1;444;2;18823;18352;50;60;49;51;0;12;3;-34;12;14;0;8;50;117;0;199;0;27;54021;> +[1;61681700;129929120;18876;34824;] +<1;445;2;18876;18399;53;46;50;51;3;12;3;-34;11;14;0;8;50;117;0;199;0;27;62233;> +[1;61681664;129929016;18928;25132;] +<1;446;2;18928;18455;41;54;51;51;2;12;3;-34;11;14;0;8;50;117;0;199;0;27;44930;> +;61;50;52;0;12;3;-34;11;14;0;6;50;117;0;199;0;27;29195;> +[1;61681464;129928728;19047;30638;] +<1;448;2;19047;18576;62;57;53;55;0;12;3;-34;11;14;0;7;50;117;0;199;0;27;32222;> +[1;61681732;129928144;19093;59550;] +<1;449;2;19093;18624;45;47;52;55;7;12;4;-34;11;14;0;7;50;117;0;199;0;27;57042;> +[1;61682008;129927384;19150;9082;] +<1;450;2;19150;18677;66;51;53;53;4;12;4;-34;11;14;0;7;50;117;0;199;0;27;57989;> +[1;61682204;129927488;19195;20146;] +<1;451;2;19195;18729;44;56;52;54;1;12;4;-33;11;14;0;6;50;117;0;199;0;27;47662;> +[1;61682420;129927288;19242;2417;] +<1;452;2;19242;18779;46;49;53;54;2;12;4;-33;11;14;0;6;50;117;0;199;0;27;49301;> +[1;61682512;129927416;19288;17508;] +<1;453;2;19288;18828;45;48;52;52;0;12;4;-32;11;14;0;6;50;117;0;199;0;27;25498;> +[1;61682464;129927768;19344;9375;] +<1;454;2;19344;18876;55;47;49;50;0;12;4;-32;11;14;0;6;50;117;0;199;0;27;1024;> +[1;61682528;129927600;19389;50735;] +<1;455;2;19389;18925;44;48;49;50;3;12;4;-32;12;14;0;6;50;117;0;199;0;27;63216;> +[1;61682656;129927568;19443;46241;] +<1;456;2;19443;18977;53;50;48;50;3;12;4;-33;13;14;0;6;50;117;0;199;0;27;29311;> +[1;61682808;129927504;19490;31452;] +<1;457;2;19490;19032;44;59;48;50;5;12;4;-33;13;14;0;6;50;117;0;199;0;27;23195;> +[1;61683140;129926984;19537;44259;] +<1;458;2;19537;19085;57;52;48;51;4;12;4;-32;13;14;0;6;50;117;0;199;0;27;16171;> +[1;61683528;129926744;19585;38236;] +<1;459;2;19585;19122;35;36;46;49;4;12;4;-32;13;14;0;7;50;117;0;199;0;27;15908;> +[1;61683812;129927112;19635;2108;] +<1;460;2;19635;19175;48;52;47;50;1;12;4;-32;13;14;0;6;50;117;0;199;0;27;57283;> +[1;61683824;129927080;19682;31569;] +<1;461;2;19682;19223;58;47;49;49;0;12;5;-32;13;14;0;7;50;117;0;199;0;27;16616;> +[1;61683992;129927184;19730;28171;] +<1;462;2;19730;19275;47;51;48;49;1;12;5;-31;13;14;0;8;50;117;0;199;0;27;31560;> +[1;61683988;129927136;19778;41121;] +<1;463;2;19778;19324;59;53;48;49;4;12;5;-31;13;14;0;7;50;117;0;199;0;27;30624;> +[1;61684180;129927184;19825;20561;] +<1;464;2;19825;19372;45;47;48;48;1;12;5;-31;13;14;0;8;50;117;0;199;0;27;20361;> +[1;61684144;129927368;19876;16126;] +<1;465;2;19876;19422;50;49;47;47;1;12;5;-31;12;14;0;7;50;117;0;199;0;27;12097;> +[1;61684072;129927264;19925;48935;] +<1;466;2;19925;19473;48;50;49;50;4;12;5;-30;12;14;0;7;50;117;0;199;0;27;22087;> +6;50;49;49;3;12;5;-30;13;14;0;8;50;117;0;199;0;27;61116;> +[1;61683824;129927128;20021;61571;] +<1;468;2;20021;19572;46;47;47;50;1;12;5;-30;13;14;0;8;50;117;0;199;0;27;59989;> +[1;61683764;129927352;20074;59954;] +<1;469;2;20074;19623;53;49;47;49;2;12;5;-30;13;14;0;8;50;117;0;199;0;27;63839;> +[1;61683872;129927688;20125;62307;] +<1;470;2;20125;19672;49;53;48;49;2;12;6;-30;13;14;0;7;50;117;0;199;0;27;16247;> +[1;61683672;129927720;20164;20068;] +<1;471;2;20164;19722;51;49;49;50;2;12;6;-30;13;14;0;8;50;117;0;199;0;27;50577;> +[1;61683368;129927640;20216;48508;] +<1;472;2;20216;19772;50;49;49;50;4;12;6;-30;13;14;0;7;50;117;0;199;0;27;48561;> +[1;61683084;129927688;20270;13094;] +<1;473;2;20270;19823;52;50;49;49;2;12;6;-30;13;14;0;8;50;117;0;199;0;27;19705;> +[1;61682864;129927112;20310;18951;] +<1;474;2;20310;19871;39;47;48;50;3;12;6;-30;13;14;0;8;50;117;0;199;0;27;56465;> +[1;61682636;129927096;20361;46724;] +<1;475;2;20361;19920;50;48;49;49;3;12;6;-29;13;14;0;7;50;117;0;199;0;27;39741;> +[1;61682540;129926776;20402;48013;] +<1;476;2;20402;19965;38;49;47;49;1;12;6;-29;13;14;0;8;50;117;0;199;0;27;60526;> +[1;61682432;129926632;20441;24967;] +<1;477;2;20441;20015;53;49;47;48;3;12;6;-29;13;14;0;8;50;117;0;199;0;27;54761;> +[1;61682364;129926896;20494;42642;] +<1;478;2;20494;20065;52;49;45;49;1;12;6;-29;14;14;0;8;50;117;0;199;0;27;27589;> +[1;61682236;129926912;20541;39526;] +<1;479;2;20541;20116;46;50;45;49;1;12;7;-30;15;14;0;8;50;117;0;199;0;27;22566;> +[1;61682108;129927000;20592;45005;] +<1;480;2;20592;20168;48;50;46;48;3;12;7;-29;14;14;0;8;50;117;0;199;0;27;32235;> +[1;61681992;129927232;20635;2037;] +<1;481;2;20635;20221;42;52;44;49;1;12;7;-28;15;14;0;7;50;117;0;199;0;27;22312;> +[1;61681656;129927112;20690;42041;] +<1;482;2;20690;20269;42;48;47;50;5;12;7;-28;15;14;0;7;50;117;0;199;0;27;2605;> +[1;61681356;129926776;20733;20250;] +<1;483;2;20733;20322;54;57;47;51;4;12;7;-27;14;14;0;8;50;117;0;199;0;27;43380;> +[1;61680936;129926408;20791;27756;] +<1;484;2;20791;20374;57;51;48;50;4;12;7;-27;15;14;0;8;50;117;0;199;0;27;20792;> +[1;61680496;129926176;20829;4888;] +<1;485;2;20829;20424;37;49;48;51;5;12;7;-28;16;14;0;8;50;117;0;199;0;27;24599;> +[1;61680064;129925464;20885;9641;] +<1;486;2;20885;20476;39;51;47;51;5;12;7;-27;15;14;0;8;50;117;0;199;0;27;22956;> +[1;61679652;129924776;20929;5119;] +<1;487;2;20929;20531;59;54;50;51;6;12;7;-27;15;14;0;8;50;117;0;199;0;27;15264;> +[1;61679176;129924272;20989;52291;] +<1;488;2;20989;20582;57;50;51;52;5;12;8;-27;15;14;0;8;50;117;0;199;0;27;43113;> +[1;61678896;129923432;21034;51558;] +<1;489;2;21034;20633;42;55;48;50;6;12;8;-27;15;14;0;8;50;117;0;199;0;27;78;> +[1;61678452;129922640;21065;10807;] +<1;490;2;21065;20681;47;47;49;51;6;12;8;-27;16;14;0;8;50;117;0;199;0;27;2890;> +[1;61678092;129922088;21133;7027;] +<1;491;2;21133;20736;52;54;49;51;7;12;8;-27;16;14;0;9;50;117;0;199;0;27;54923;> +[1;61677720;129921128;21171;20062;] +<1;492;2;21171;20791;50;54;49;52;2;12;8;-28;16;14;0;8;50;117;0;199;0;27;22897;> +[1;61677296;129920088;21230;15295;] +<1;493;2;21230;20844;57;51;48;52;6;12;8;-27;16;14;0;8;50;117;0;199;0;27;20837;> +[1;61677056;129919160;21281;16787;] +<1;494;2;21281;20895;50;50;49;51;8;12;8;-27;16;14;0;8;50;117;0;199;0;27;54154;> +[1;61676904;129917952;21329;26932;] +<1;495;2;21329;20949;47;53;50;52;5;12;8;-27;16;14;0;9;50;117;0;199;0;27;25621;> +[1;61676648;129916912;21387;43584;] +<1;496;2;21387;21003;57;59;50;53;4;12;9;-27;17;14;0;8;50;117;0;199;0;27;30848;> +[1;61676456;129916080;21431;61230;] +<1;497;2;21431;21054;43;50;50;53;8;12;9;-28;17;14;0;8;50;117;0;199;0;27;51319;> +[1;61676196;129915056;21491;5894;] +<1;498;2;21491;21109;59;54;49;53;6;12;9;-27;16;14;0;8;50;117;0;199;0;27;64789;> +[1;61675776;129914120;21543;55221;] +<1;499;2;21543;21162;32;52;51;52;4;12;9;-27;16;14;0;9;50;117;0;199;0;27;13478;> +[1;61675484;129913192;21593;6014;] +<1;500;2;21593;21212;67;49;48;52;8;12;9;-27;16;14;0;8;50;117;0;199;0;27;40461;> +[1;61675288;129912016;21642;16278;] +<1;501;2;21642;21267;66;53;51;53;7;12;9;-27;16;14;0;9;50;117;0;199;0;27;58604;> +[1;61675032;129910488;21693;49243;] +<1;502;2;21693;21321;49;58;51;53;7;12;9;-27;16;14;0;9;50;117;0;199;0;27;29542;> +[1;61675024;129908920;21741;46382;] +<1;503;2;21741;21373;45;51;50;53;11;12;9;-28;17;14;0;9;50;117;0;199;0;27;16943;> +[1;61674768;129907488;21799;23537;] +<1;504;2;21799;21424;55;50;52;52;4;12;10;-27;17;14;0;9;50;117;0;199;0;27;45031;> +[1;61674628;129905896;21845;63213;] +<1;505;2;21845;21473;45;48;51;52;8;12;10;-27;17;14;0;9;50;117;0;199;0;27;64228;> +[1;61674428;129904024;21899;10359;] +<1;506;2;21899;21531;53;56;52;54;11;12;10;-27;17;14;0;9;50;117;0;199;0;27;63681;> +[1;61674256;129902240;21950;28538;] +<1;507;2;21950;21581;49;46;47;52;7;12;10;-26;16;14;0;8;50;117;0;199;0;27;35156;> +[1;61674004;129900528;21986;65132;] +<1;508;2;21986;21637;35;60;50;52;8;12;10;-26;17;14;0;8;50;117;0;199;0;27;26149;> +[1;61673752;129899112;22038;56916;] +<1;509;2;22038;21687;50;49;50;52;9;12;10;-26;17;14;0;8;50;117;0;199;0;27;10977;> +[1;61673560;129897632;22091;49266;] +<1;510;2;22091;21742;52;53;52;52;9;12;10;-27;17;14;0;8;50;117;0;199;0;27;44312;> +[1;61673148;129896272;22142;47263;] +<1;511;2;22142;21792;50;49;51;52;5;12;10;-27;18;14;0;8;50;117;0;199;0;27;27088;> +[1;61672672;129894840;22194;37489;] +<1;512;2;22194;21847;51;53;48;51;10;12;10;-27;18;14;0;8;50;117;0;199;0;27;11897;> +[1;61672308;129893144;22246;9840;] +<1;513;2;22246;21901;50;58;54;54;10;12;10;-26;18;14;0;8;50;117;0;199;0;27;48533;> +[1;61671784;129891448;22302;41930;] +<1;514;2;22302;21955;54;52;48;52;10;12;10;-26;17;14;0;8;50;117;0;199;0;27;37783;> +[1;61671416;129889768;22355;18652;] +<1;515;2;22355;22008;50;51;51;53;9;12;10;-26;18;14;0;8;50;116;0;199;0;27;34177;> +[1;61671292;129887744;22393;25;] +<1;516;2;22393;22062;37;53;49;53;11;12;11;-26;17;14;0;8;50;116;0;199;0;27;44541;> +[1;61671276;129885728;22451;33490;] +<1;517;2;22451;22115;38;51;47;53;8;12;11;-27;18;14;0;8;50;117;0;199;0;27;29480;> +[1;61671320;129883760;22502;62975;] +<1;518;2;22502;22173;69;62;52;54;9;12;11;-27;18;14;0;8;50;116;0;199;0;27;239;> +[1;61671288;129882264;22553;4575;] +<1;519;2;22553;22227;50;53;50;54;7;12;11;-27;19;14;0;8;50;116;0;199;0;27;40366;> +[1;61671436;129880680;22610;13013;] +<1;520;2;22610;22281;56;52;53;54;6;12;11;-28;19;14;0;8;50;117;0;199;0;27;3835;> +[1;61671544;129879272;22662;64247;] +<1;521;2;22662;22338;51;56;50;54;7;12;11;-27;19;14;0;8;50;116;0;199;0;27;14431;> +[1;61671568;129878128;22706;61669;] +<1;522;2;22706;22393;42;53;51;55;7;12;11;-27;18;14;0;7;50;116;0;199;0;27;25380;> +[1;61671624;129876848;22771;14063;] +<1;523;2;22771;22449;62;61;51;55;3;12;11;-26;17;14;0;7;50;117;0;199;0;27;58173;> +[1;61671684;129875688;22809;19450;] +<1;524;2;22809;22506;37;56;55;56;5;12;11;-26;17;14;0;8;50;116;0;199;0;27;12266;> +[1;61671724;129874952;22872;37126;] +<1;525;2;22872;22561;61;54;50;55;6;12;11;-27;18;14;0;7;50;116;0;199;0;27;59506;> +[1;61671680;129873976;22933;58018;] +<1;526;2;22933;22616;59;53;51;55;5;12;11;-27;19;14;0;7;50;116;0;199;0;27;28417;> +[1;61671680;129872544;22972;22963;] +<1;527;2;22972;22674;38;57;52;55;8;12;11;-27;20;14;0;7;50;116;0;199;0;27;29681;> +[1;61671628;129870864;23033;52567;] +<1;528;2;23033;22725;60;50;53;55;8;12;11;-26;19;14;0;7;50;116;0;199;0;27;22411;> +[1;61671568;129869264;23079;53334;] +<1;529;2;23079;22782;44;61;53;55;9;12;11;-26;19;14;0;7;50;116;0;199;0;27;25836;> +[1;61671624;129867752;23143;65127;] +<1;530;2;23143;22835;62;51;53;55;7;12;12;-25;18;14;0;7;50;116;0;199;0;27;20491;> +[1;61671456;129865920;23184;18181;] +<1;531;2;23184;22892;40;56;54;54;8;12;12;-25;17;14;0;8;50;116;0;199;0;27;35460;> +[1;61671364;129864200;23249;41725;] +<1;532;2;23249;22947;64;53;54;55;10;12;12;-24;17;14;0;7;50;116;0;199;0;27;36590;> +[1;61671392;129862368;23311;18489;] +<1;533;2;23311;23005;61;57;55;55;9;12;12;-24;17;14;0;7;50;116;0;199;0;27;31679;> +[1;61671344;129860416;23354;9128;] +<1;534;2;23354;23059;41;54;52;55;10;12;12;-24;18;14;0;7;50;116;0;199;0;27;56283;> +[1;61671484;129858280;23416;4459;] +<1;535;2;23416;23113;38;58;55;56;11;12;12;-24;18;14;0;8;50;116;0;199;0;27;5508;> +[1;61671792;129856184;23462;17028;] +<1;536;2;23462;23166;65;51;51;55;11;12;12;-24;18;14;0;8;50;116;0;199;0;27;15597;> +[1;61671932;129854112;23499;17824;] +<1;537;2;23499;23219;36;52;52;54;13;12;12;-24;19;14;0;8;50;116;0;199;0;27;46186;> +[1;61672196;129852128;23571;12101;] +<1;538;2;23571;23276;70;55;48;54;9;12;12;-23;18;14;0;9;50;116;0;199;0;27;1932;> +[1;61672564;129850552;23616;20812;] +<1;539;2;23616;23328;66;56;53;54;7;12;12;-23;18;14;0;9;50;116;0;199;0;27;1124;> +[1;61672660;129849184;23661;37552;] +<1;540;2;23661;23381;42;51;49;54;6;12;12;-23;17;14;0;8;50;116;0;199;0;27;25898;> +[1;61672708;129847440;23724;53172;] +<1;541;2;23724;23434;63;52;49;53;11;12;12;-22;18;14;0;8;50;116;0;199;0;27;26118;> +[1;61672528;129845944;23766;34159;] +<1;542;2;23766;23485;40;50;53;53;7;12;13;-22;18;14;0;9;50;116;0;199;0;27;39337;> +[1;61672288;129843992;23814;27700;] +<1;543;2;23814;23539;47;53;53;53;10;12;13;-22;18;14;0;9;50;116;0;199;0;27;9420;> +[1;61671984;129842328;23855;46326;] +<1;544;2;23855;23592;40;52;55;53;11;12;13;-22;19;14;0;10;50;116;0;199;0;27;64364;> +[1;61671856;129840320;23902;41821;] +<1;545;2;23902;23644;47;56;46;52;7;12;13;-21;18;14;0;9;50;116;0;199;0;27;27768;> +[1;61671908;129838440;23977;5433;] +<1;546;2;23977;23699;50;54;50;53;10;12;13;-20;18;14;0;9;50;116;0;199;0;27;52379;> +[1;61671892;129836912;24028;46400;] +<1;547;2;24028;23751;73;50;53;53;8;12;13;-20;18;14;0;8;50;116;0;199;0;27;3192;> +[1;61671940;129835096;24076;18527;] +<1;548;2;24076;23803;47;50;50;52;9;12;13;-20;18;14;0;9;50;116;0;199;0;27;51814;> +[1;61672016;129833816;24124;39284;] +<1;549;2;24124;23856;47;52;51;52;8;12;13;-19;18;14;0;9;50;116;0;199;0;27;63293;> +[1;61672292;129832184;24173;16515;] +<1;550;2;24173;23908;47;51;51;52;8;12;13;-19;18;14;0;8;50;116;0;199;0;27;16743;> +[1;61672432;129830232;24226;18315;] +<1;551;2;24226;23959;50;55;57;53;10;12;13;-20;19;14;0;9;50;116;0;199;0;27;37568;> +[1;61672820;129828584;24280;4017;] +<1;552;2;24280;24013;54;52;53;52;12;12;13;-20;18;14;0;9;50;116;0;199;0;27;20267;> +[1;61672848;129826032;24334;63774;] +<1;553;2;24334;24073;53;58;50;52;13;12;13;-20;19;14;0;8;50;116;0;199;0;27;2500;> +[1;61672776;129824192;24412;25067;] +<1;554;2;24412;24127;76;53;54;54;8;12;13;-20;18;14;0;9;50;116;0;199;0;27;9782;> +[1;61672728;129822112;24432;52893;] +<1;555;2;24432;24183;19;53;54;54;10;12;13;-19;18;14;0;9;50;116;0;199;0;27;26466;> +[1;61672664;129820728;24504;2079;] +<1;556;2;24504;24239;70;60;54;55;9;12;13;-19;18;14;0;9;50;116;0;199;0;27;55554;> +[1;61672660;129819472;24551;53288;] +<1;557;2;24551;24294;45;53;53;55;5;12;14;-18;18;14;0;9;50;116;0;199;0;27;47603;> +[1;61672672;129818096;24605;43950;] +<1;558;2;24605;24351;52;55;57;56;6;12;14;-18;17;14;0;9;50;116;0;199;0;27;21464;> +[1;61672608;129817352;24660;48207;] +<1;559;2;24660;24407;53;55;53;55;6;12;14;-17;17;14;0;8;50;117;0;199;0;27;28542;> +[1;61672632;129816376;24713;32900;] +<1;560;2;24713;24460;51;53;53;54;3;12;14;-17;18;14;0;9;50;116;0;199;0;27;49358;> +[1;61672912;129815464;24761;45099;] +<1;561;2;24761;24512;48;55;54;55;5;12;14;-18;19;14;0;9;50;116;0;199;0;27;31778;> +[1;61673092;129815000;24810;23517;] +<1;562;2;24810;24568;46;54;53;55;5;12;14;-18;20;14;0;8;50;116;0;199;0;27;9765;> +[1;61673292;129814256;24866;37900;] +<1;563;2;24866;24622;53;52;50;54;2;12;14;-17;19;14;0;8;50;116;0;199;0;27;18018;> +[1;61673516;129813264;24930;29575;] +<1;564;2;24930;24681;63;58;48;54;5;12;14;-17;20;14;0;9;50;116;0;199;0;27;53886;> +[1;61673592;129812136;24968;39033;] +<1;565;2;24968;24737;37;55;54;55;8;12;14;-17;20;14;0;9;50;116;0;199;0;27;21858;> +[1;61673628;129810976;25026;15227;] +<1;566;2;25026;24791;57;57;51;55;7;12;14;-17;21;14;0;9;50;116;0;199;0;27;35000;> +[1;61673872;129809528;25074;57153;] +<1;567;2;25074;24846;46;53;50;55;5;12;14;-17;20;15;0;8;50;116;0;199;0;27;39727;> +[1;61674052;129808536;25130;33893;] +<1;568;2;25130;24901;54;53;52;55;9;12;14;-16;20;15;0;8;50;116;0;199;0;27;9570;> +[1;61674064;129807192;25156;31276;] +<1;569;2;25156;24951;26;49;47;54;5;12;14;-16;20;15;0;8;50;116;0;199;0;27;28086;> +[1;61674220;129805896;25234;55518;] +<1;570;2;25234;25007;76;54;54;54;7;12;14;-16;20;15;0;8;50;116;0;199;0;27;60408;> +[1;61674076;129804608;25291;63526;] +<1;571;2;25291;25061;54;58;53;54;6;12;14;-16;20;15;0;8;50;116;0;199;0;27;3024;> +[1;61674148;129803192;25321;14604;] +<1;572;2;25321;25117;29;54;48;54;7;12;15;-15;19;15;0;8;50;116;0;199;0;27;11824;> +[1;61674208;129801848;25411;13899;] +<1;573;2;25411;25176;5 +[1;61674216;129800296;25446;53499;] +<1;574;2;25446;25232;65;54;51;54;9;12;15;-15;19;15;0;8;50;116;0;199;0;27;44919;> +[1;61674448;129798952;25506;51610;] +<1;575;2;25506;25286;58;53;57;55;4;12;15;-15;19;15;0;8;50;116;0;199;0;27;11701;> +[1;61674612;129798016;25536;8479;] +<1;576;2;25536;25340;28;53;57;55;8;12;15;-15;19;15;0;9;50;116;0;199;0;27;64608;> +[1;61674604;129797088;25591;44603;] +<1;577;2;25591;25395;54;59;49;55;3;12;15;-15;20;15;0;8;50;116;0;199;0;27;56435;> +[1;61674904;129796096;25669;14288;] +<1;578;2;25669;25450;51;53;49;55;7;12;15;-16;21;15;0;8;50;116;0;199;0;27;49482;> +[1;61674884;129795272;25694;2520;] +<1;579;2;25694;25505;49;53;57;55;3;12;15;-16;22;15;0;9;50;116;0;199;0;27;64816;> +[1;61674908;129794648;25742;54150;] +<1;580;2;25742;25562;46;56;51;54;5;12;16;-16;22;15;0;8;50;116;0;199;0;27;29371;> +[1;61674784;129793440;25801;62262;] +<1;581;2;25801;25619;57;56;48;54;6;12;16;-16;22;15;0;8;50;116;0;199;0;27;23233;> +[1;61674868;129792408;25833;19725;] +<1;582;2;25833;25674;31;59;48;55;6;12;16;-15;21;15;0;8;50;116;0;199;0;27;56091;> +[1;61674920;129790680;25934;5772;] +<1;583;2;25934;25733;67;57;54;56;9;12;16;-16;22;15;0;8;50;116;0;199;0;27;19469;> +[1;61674884;129789416;25957;12563;] +<1;584;2;25957;25789;53;55;56;56;6;12;16;-16;22;15;0;8;50;116;0;199;0;27;46144;> +[1;61675116;129788072;26022;10811;] +<1;585;2;26022;25846;64;56;46;56;7;12;16;-16;22;15;0;8;50;116;0;199;0;27;22994;> +[1;61675368;129786496;26060;22765;] +<1;586;2;26060;25906;36;58;53;56;8;12;16;-16;22;15;0;8;50;116;0;199;0;27;55389;> +[1;61675572;129785328;26132;42791;] +<1;587;2;26132;25967;70;65;58;57;7;12;16;-17;23;15;0;8;50;116;0;199;0;27;53532;> +[1;61675968;129784040;26186;48004;] +<1;588;2;26186;26025;54;56;59;58;7;12;16;-17;23;15;0;8;50;116;0;199;0;27;49675;> +[1;61676028;129783232;26241;68;] +<1;589;2;26241;26084;53;57;52;58;5;12;16;-17;23;15;0;8;50;116;0;199;0;27;35484;> +[1;61676388;129782392;26287;53363;] +<1;590;2;26287;26139;45;53;53;57;3;12;17;-18;24;15;0;8;50;116;0;199;0;27;38768;> +[1;61676692;129781616;26355;39753;] +<1;591;2;26355;26197;68;57;59;58;6;12;17;-18;24;15;0;8;50;116;0;199;0;27;21783;> +[1;61676852;129780712;26396;56487;] +<1;592;2;26396;26256;77;56;54;58;6;12;17;-18;24;15;0;8;50;116;0;199;0;27;61520;> +[1;61677032;129779712;26474;37765;] +<1;593;2;26474;26308;76;57;55;57;3;12;17;-18;24;15;0;8;50;116;0;199;0;27;62121;> +[1;61676936;129779040;26500;17845;] +<1;594;2;26500;26346;29;37;51;56;3;12;17;-17;24;15;0;7;50;116;0;199;0;27;47352;> +[1;61677000;129778320;26159;62229;] +<1;595;2;26159;26010;-307;-332;38;31;1;12;17;-19;24;15;0;8;50;116;0;199;0;27;49027;> +[1;61676852;129777008;25796;47568;] +<1;596;4;25796;25621;-388;-381;-38;-39;1;12;17;-20;24;15;0;8;50;116;0;199;0;27;26187;> +[1;61676456;129776752;25433;23208;] +<1;597;4;25433;25241;-355;-374;-105;-113;5;12;16;-21;23;15;0;7;50;116;0;199;0;27;4423;> +[1;61676232;129776760;25117;24799;] +<1;598;4;25117;24883;-343;-349;-169;-183;6;12;16;-22;23;15;0;7;50;116;0;199;0;27;26739;> +[1;61676700;129775424;24768;51379;] +<1;599;4;24768;24526;-347;-387;-242;-257;9;12;15;-23;23;15;0;7;50;116;0;199;0;27;15034;> +[1;61677368;129774048;24460;12081;] +<1;600;4;24460;24185;-301;-334;-305;-324;11;12;15;-24;23;15;0;7;50;116;0;199;0;27;49550;> +[1;61677696;129772672;24140;36867;] +<1;601;4;24140;23855;-339;-323;-350;-363;7;12;14;-24;22;15;0;7;50;116;0;199;0;27;30009;> +[1;61677880;129771136;23816;39583;] +<1;602;4;23816;23525;-318;-325;-340;-354;4;12;14;-25;22;15;0;8;50;116;0;199;0;27;54379;> +[1;61677992;129769424;23487;14247;] +<1;603;4;23487;23200;-303;-320;-329;-344;11;12;13;-26;22;15;0;8;50;116;0;199;0;27;9087;> +[1;61678104;129767208;23167;22570;] +<1;604;4;23167;22885;-313;-339;-324;-336;10;12;13;-26;22;15;0;7;50;116;0;199;0;27;64727;> +[1;61677900;129765824;22866;48760;] +<1;605;4;22866;22579;-298;-303;-319;-327;8;12;12;-27;22;15;0;7;50;116;0;199;0;27;4683;> +[1;61677968;129764544;22617;19326;] +<1;606;4;22617;22281;-284;-290;-307;-319;13;12;11;-28;23;15;0;7;50;116;0;199;0;27;9495;> +[1;61678004;129762616;22325;21943;] +<1;607;4;22325;21989;-284;-285;-299;-313;13;12;11;-28;22;15;0;8;50;116;0;199;0;27;65411;> +[1;61677700;129760928;22046;28146;] +<1;608;4;22046;21706;-258;-280;-291;-306;10;12;10;-29;22;15;0;7;50;116;0;199;0;27;45482;> +[1;61677248;129759304;21797;49291;] +<1;609;4;21797;21432;-248;-271;-289;-298;13;12;10;-30;22;15;0;7;50;116;0;199;0;27;35083;> +[1;61677128;129757984;21544;63652;] +<1;610;4;21544;21165;-267;-287;-279;-293;14;12;9;-30;22;15;0;7;50;116;0;199;0;27[1;61676604;129755688;21312;33171;] +<1;611;4;21312;20915;-223;-244;-264;-279;12;12;8;-31;22;15;0;7;50;116;0;199;0;27;61220;> +[1;61676008;129754136;21047;15244;] +<1;612;4;21047;20666;-258;-243;-260;-271;7;12;8;-31;22;15;0;7;50;116;0;199;0;27;44792;> +[1;61675568;129753600;20829;24607;] +<1;613;4;20829;20412;-230;-250;-253;-264;6;12;7;-31;22;15;0;7;50;116;0;199;0;27;42838;> +[1;61675272;129755032;20595;8617;] +<1;614;4;20595;20178;-231;-230;-246;-257;10;12;7;-31;21;15;0;8;50;116;0;199;0;27;63973;> +[1;61675724;129755552;20367;27203;] +<1;615;4;20367;19947;-226;-228;-239;-249;6;12;7;-32;22;15;0;8;50;116;0;199;0;27;7912;> +[1;61675600;129755816;20151;58134;] +<1;616;4;20151;19713;-227;-252;-234;-242;3;12;6;-32;21;15;0;8;50;116;0;199;0;27;49401;> +[1;61675540;129756072;19926;48746;] +<1;617;4;19926;19475;-229;-232;-228;-240;4;12;6;-33;22;15;0;7;50;116;0;199;0;27;36319;> +[1;61675072;129756240;19696;8709;] +<1;618;4;19696;19248;-226;-223;-228;-237;8;12;5;-33;22;15;0;8;50;116;0;199;0;27;16367;> +[1;61675552;129756928;19477;51082;] +<1;619;4;19477;19030;-204;-213;-224;-231;10;12;5;-34;22;15;0;8;50;116;0;199;0;27;4366;> +[1;61676352;129757168;19286;24503;] +<1;620;4;19286;18825;-201;-203;-221;-226;7;12;4;-35;22;15;0;8;50;116;0;199;0;27;62904;> +[1;61677060;129756760;19101;45664;] +<1;621;4;19101;18620;-192;-201;-213;-222;8;12;3;-35;21;15;0;8;50;116;0;199;0;27;16343;> +[1;61677060;129756592;18879;3209;] +<1;622;4;18879;18405;-218;-229;-211;-218;7;12;3;-35;20;15;0;8;50;115;0;199;0;27;30431;> +[1;61677032;129757096;18672;49140;] +<1;623;4;18672;18194;-204;-207;-211;-214;7;12;2;-36;20;15;0;7;50;115;0;199;0;27;17685;> +[1;61677068;129757144;18475;44432;] +<1;624;4;18475;17984;-201;-206;-206;-210;6;12;1;-37;20;15;0;7;50;115;0;199;0;27;60315;> +[1;61676916;129757536;18269;3897;] +<1;625;4;18269;17778;-202;-202;-203;-208;5;12;0;-37;20;15;0;8;50;115;0;199;0;27;49726;> +[1;61676920;129757416;18069;57300;] +<1;626;4;18069;17573;-207;-201;-201;-207;5;12;0;-38;20;15;0;7;50;115;0;199;0;27;31080;> +[1;61676696;129758208;17873;39835;] +<1;627;4;17873;17369;-196;-222;-204;-211;11;12;-1;-38;19;15;0;8;50;115;0;199;0;27;65136;> +[1;61677020;129758880;17693;60768;] +<1;628;4;17693;17179;-170;-185;-195;-204;5;12;-2;-39;19;15;0;8;50;115;0;199;0;27;48157;> +[1;61677128;129759672;17507;63656;] +<1;629;4;17507;16992;-193;-183;-194;-200;7;12;-3;-39;19;15;0;7;50;115;0;199;0;27;33732;> +[1;61677072;129759976;17335;5306;] +<1;630;4;17335;16805;-175;-183;-189;-196;9;12;-3;-40;19;15;0;7;50;115;0;199;0;27;63025;> +[1;61676564;129760544;17155;53797;] +<1;631;4;17155;16633;-176;-168;-186;-190;8;12;-3;-40;19;15;0;7;50;115;0;199;0;27;2030;> +[1;61677000;129762240;16987;32036;] +<1;632;4;16987;16456;-166;-173;-186;-190;8;12;-4;-40;18;15;0;7;50;115;0;199;0;27;12044;> +[1;61677696;129763704;16818;56928;] +<1;633;4;16818;16286;-165;-184;-179;-186;8;12;-4;-40;18;15;0;7;50;115;0;199;0;27;56166;> +[1;61678260;129763504;16651;35389;] +<1;634;4;16651;16123;-164;-158;-174;-179;8;12;-5;-41;18;15;0;7;50;115;0;199;0;27;9336;> +[1;61678672;129763320;16506;46010;] +<1;635;4;16506;15966;-156;-154;-174;-175;4;12;-6;-41;18;15;0;7;50;115;0;199;0;27;47946;> +[1;61678092;129764360;16348;41269;] +<1;636;4;16348;15803;-155;-160;-166;-170;12;12;-7;-42;17;15;0;8;50;115;0;199;0;27;15000;> +[1;61677512;129765872;16195;53641;] +<1;637;4;16195;15647;-154;-154;-163;-166;13;12;-8;-42;18;15;0;7;50;115;0;199;0;27;14149;> +[1;61677128;129768032;16040;8708;] +<1;638;4;16040;15491;-161;-168;-159;-163;10;12;-8;-43;18;15;0;7;50;115;0;199;0;27;48267;> +[1;61677108;129768296;15886;49190;] +<1;639;4;15886;15336;-147;-150;-158;-162;2;12;-9;-44;18;15;0;8;50;115;0;199;0;27;36284;> +[1;61677224;129768464;15737;26763;] +<1;640;4;15737;15188;-146;-145;-157;-157;6;12;-9;-44;17;15;0;8;50;115;0;199;0;27;5831;> +[1;61677684;129769992;15590;25494;] +<1;641;4;15590;15043;-151;-142;-153;-155;11;12;-9;-43;16;15;0;7;50;115;0;199;0;27;55468;> +[1;61677556;129772312;15448;48469;] +<1;642;4;15448;14901;-146;-139;-152;-153;14;12;-10;-43;16;15;0;7;50;115;0;199;0;27;31189;> +[1;61676944;129774128;15295;62327;] +<1;643;4;15295;14759;-143;-152;-150;-149;10;12;-9;-43;15;15;0;7;50;115;0;199;0;27;51967;> +[1;61676220;129774952;15156;8956;] +<1;644;4;15156;14618;-141;-137;-150;-149;4;12;-9;-44;16;15;0;7;50;115;0;199;0;27;3917;> +[1;61676204;129775280;15015;13864;] +<1;645;4;15015;14482;-139;-133;-147;-144;2;12;-10;-44;15;15;0;8;50;115;0;199;0;27;39265;> +[1;61675700;129776184;14882;19333;] +<1;646;4;14882;14346;-131;-134;-144;-141;6;12;-11;-44;14;15;0;7;50;115;0;199;0;27;42692;> +[1;61675368;129777408;14729;31273;] +<1;647;4;14729;14198;-154;-145;-142;-139;5;12;-12;-45;14;15;0;8;50;115;0;199;0;27;24063;> +[1;61675256;129778608;14572;21429;] +<1;648;4;14572;14051;-151;-157;-142;-140;10;12;-12;-46;14;15;0;8;50;115;0;199;0;27;25793;> +[1;61674804;129780224;14432;62582;] +<1;649;4;14432;13908;-142;-138;-143;-143;14;12;-13;-46;13;15;0;7;50;115;0;199;0;27;613;> +[1;61674712;129781888;14288;48338;] +<1;650;4;14288;13764;-145;-140;-143;-141;12;12;-14;-47;13;15;0;8;50;115;0;199;0;27;47641;> +[1;61674832;129783552;14144;4216;] +<1;651;4;14144;13623;-145;-138;-145;-141;14;12;-16;-48;12;15;0;7;50;115;0;199;0;27;40656;> +[1;61674936;129785488;14000;45009;] +<1;652;4;14000;13481;-144;-140;-145;-142;11;12;-18;-49;11;15;0;7;50;115;0;199;0;27;35465;> +[1;61674604;129787584;13871;29757;] +<1;653;4;13871;13345;-130;-147;-144;-146;12;12;-20;-50;11;15;0;8;50;115;0;199;0;27;50958;> +[1;61674320;129789440;13725;41961;] +<1;654;4;13725;13212;-137;-130;-143;-142;12;12;-21;-50;10;15;0;8;50;115;0;199;0;27;26125;> +[1;61674532;129791928;13595;42209;] +<1;655;4;13595;13083;-136;-126;-141;-138;8;12;-21;-51;10;15;0;7;50;115;0;199;0;27;57843;> +[1;61674504;129793592;13471;24095;] +<1;656;4;13471;12956;-128;-126;-138;-136;15;12;-22;-51;10;14;0;8;50;115;0;199;0;27;33562;> +[1;61674816;129795304;13332;44794;] +<1;657;4;13332;12824;-136;-129;-134;-133;12;12;-22;-51;9;14;0;8;50;115;0;199;0;27;8499;> +[1;61674844;129796632;13204;2286;] +<1;658;4;13204;12701;-124;-134;-134;-131;9;12;-22;-52;9;14;0;8;50;115;0;199;0;27;25741;> +[1;61674548;129798832;13072;33766;] +<1;659;4;13072;12573;-126;-124;-132;-128;16;12;-23;-52;9;14;0;8;50;115;0;199;0;27;12228;> +[1;61674136;129800488;12936;50860;] +<1;660;4;12936;12447;-138;-123;-132;-127;15;12;-22;-52;9;14;0;8;50;115;0;199;0;27;28819;> +[1;61673652;129802904;12802;38056;] +<1;661;4;12802;12321;-133;-123;-130;-126;17;12;-22;-52;9;14;0;9;50;115;0;199;0;27;41263;> +[1;61673652;129805576;12694;31748;] +<1;662;4;12694;12212;-109;-107;-129;-124;10;12;-23;-53;8;14;0;8;50;115;0;199;0;27;17489;> +123;-119;-127;-123;4;12;-23;-54;8;14;0;8;50;115;0;199;0;27;12304;> +[1;61672608;129807632;12457;29938;] +<1;664;4;12457;11986;-113;-112;-123;-121;16;12;-23;-54;8;14;0;8;50;115;0;199;0;27;7598;> +[1;61671884;129809600;12334;7881;] +<1;665;4;12334;11875;-124;-108;-123;-116;22;12;-24;-127;7;14;0;8;50;115;0;199;2;27;49490;> +[1;61671408;129811664;12215;48548;] +<1;666;4;12215;11758;-115;-114;-119;-115;10;12;-24;-127;7;14;0;8;50;115;0;199;2;27;65450;> +[1;61671388;129813824;12098;3119;] +<1;667;4;12098;11649;-114;-108;-119;-112;16;12;-24;-127;7;14;0;8;50;115;0;199;2;27;12894;> +[1;61671492;129815920;11990;32650;] +<1;668;4;11990;11545;-106;-111;-117;-112;15;12;-24;-127;6;14;0;8;50;115;0;199;2;27;49013;> +[1;61671236;129818496;11871;4832;] +<1;669;4;11871;11434;-120;-109;-117;-110;17;12;-24;-127;5;13;0;8;50;115;0;199;2;27;1373;> +[1;61670752;129821288;11749;24647;] +<1;670;4;11749;11324;-122;-107;-117;-110;8;12;-25;-127;6;13;0;7;50;115;0;199;2;27;35362;> +[1;61670528;129823824;11631;48202;] +<1;671;4;11631;11210;-119;-111;-116;-110;14;12;-25;-127;5;13;0;8;50;115;0;199;2;27;7465;> +[1;61670548;129827088;11517;16266;] +<1;672;4;11517;11097;-120;-110;-117;-109;16;12;-26;-127;5;13;0;8;50;115;0;199;2;27;59631;> +[1;61670752;129829512;11415;45091;] +<1;673;4;11415;10991;-104;-114;-115;-111;12;12;-27;-127;5;13;0;8;50;115;0;199;2;27;25022;> +[1;61671052;129831816;11311;51394;] +<1;674;4;11311;10889;-98;-100;-113;-109;10;12;-28;-127;4;13;0;8;50;115;0;199;2;27;32964;> +[1;61671056;129834888;11209;5422;] +<1;675;4;11209;10791;-103;-96;-110;-107;19;12;-28;-127;4;13;0;7;50;115;0;199;2;27;12648;> +[1;61670844;129838024;11113;41684;] +<1;676;4;11113;10689;-96;-100;-108;-105;14;12;-29;-127;3;13;0;8;50;115;0;199;2;27;43656;> +[1;61670748;129839688;11008;5518;] +<1;677;4;11008;10592;-101;-96;-105;-103;9;12;-30;-127;3;13;0;8;50;115;0;199;2;27;2672;> +[1;61670620;129842328;10906;19773;] +<1;678;4;10906;10495;-102;-104;-102;-102;7;12;-30;-127;2;13;0;8;50;115;0;199;2;27;56935;> +[1;61671028;129844784;10801;64517;] +<1;679;4;10801;10398;-104;-94;-101;-98;8;12;-30;-127;2;13;0;8;50;115;0;199;2;27;34552;> +[1;61670848;129847032;10699;17124;] +<1;680;4;10699;10299;-104;-97;-102;-98;7;12;-30;-127;1;12;0;15;50;113;0;199;3;27;21323;> +[1;61670404;129849272;10598;55904;] +<1;681;4;10598;10202;-101;-95;-102;-98;9;12;-30;-127;1;13;0;15;50;113;0;199;3;27;48246;> +5;-96;-101;-97;17;12;-31;-127;1;13;0;15;50;113;0;199;3;27;9183;> +[1;61669232;129854664;10408;56030;] +<1;683;4;10408;10016;-92;-87;-100;-96;17;12;-30;-127;1;13;0;15;50;113;0;199;3;27;61180;> +[1;61668912;129858288;10309;40224;] +<1;684;4;10309;9925;-98;-89;-100;-96;19;12;-30;-127;1;14;0;14;50;113;0;199;3;27;4264;> +[1;61668452;129860840;10216;40947;] +<1;685;4;10216;9836;-93;-94;-99;-93;12;12;-30;-127;0;14;0;15;50;113;0;199;3;27;10580;> +[1;61668584;129863360;10127;32662;] +<1;686;4;10127;9746;-87;-88;-97;-93;19;12;-31;-54;0;14;0;15;50;113;0;199;1;27;58278;> +[1;61668500;129866136;10030;24424;] +<1;687;4;10030;9655;-96;-89;-94;-91;18;12;-31;-54;0;14;0;15;50;113;0;199;1;27;58276;> +[1;61668332;129868608;9933;20109;] +<1;688;4;9933;9562;-98;-91;-93;-91;14;12;-31;-53;0;14;0;15;50;113;0;199;1;27;10792;> +[1;61668312;129871400;9829;6462;] +<1;689;4;9829;9466;-100;-94;-94;-90;19;12;-31;-53;1;14;0;15;50;113;0;199;1;27;30151;> +[1;61668328;129873504;9736;16726;] +<1;690;4;9736;9373;-95;-100;-96;-91;9;12;-30;-53;1;14;0;14;50;113;0;199;1;27;3132;> +[1;61668364;129875064;9639;6793;] +<1;691;4;9639;9285;-96;-86;-94;-92;12;12;-30;-52;2;15;0;14;50;113;0;199;1;27;20388;> +[1;61668404;129876752;9553;10442;] +<1;692;4;9553;9198;-85;-85;-95;-91;11;12;-31;-52;2;15;0;14;50;113;0;199;1;27;5545;> +[1;61668528;129878600;9459;43291;] +<1;693;4;9459;9109;-94;-87;-95;-91;7;12;-31;-52;2;15;0;14;50;113;0;199;1;27;45866;> +[1;61668556;129880920;9364;35914;] +<1;694;4;9364;9017;-95;-90;-95;-90;13;12;-31;-51;2;15;0;14;50;113;0;199;1;27;43783;> +[1;61668572;129883480;9268;59781;] +<1;695;4;9268;8927;-93;-97;-94;-90;15;12;-31;-50;2;15;0;14;50;113;0;199;1;27;29419;> +[1;61668656;129884960;9178;38427;] +<1;696;4;9178;8836;-95;-88;-93;-91;6;12;-31;-50;2;15;0;14;50;113;0;199;1;27;42843;> +[1;61668904;129886488;9087;14136;] +<1;697;4;9087;8746;-89;-88;-93;-89;11;12;-31;-49;2;15;0;14;50;113;0;199;1;27;47455;> +[1;61668872;129888416;8987;7207;] +<1;698;4;8987;8655;-95;-89;-92;-89;7;12;-30;-48;2;15;0;14;50;113;0;199;1;27;20675;> +[1;61669260;129889984;8899;11215;] +<1;699;4;8899;8569;-91;-84;-94;-90;5;12;-30;-47;1;15;0;14;50;113;0;199;1;27;49587;> +[1;61669508;129891512;8811;27373;] +<1;700;4;8811;8481;-90;-94;-92;-89;14;12;-30;-46;1;15;0;14;50;113;0;199;1;27;6444;> +[1;61669560;129893080;8722;62724;] +<1;701;4;8722;8393;-88;-85;-92;-90;11;12;-30;-45;1;15;0;16;50;113;0;199;1;27;63821;> +[1;61669324;129894864;8633;30668;] +<1;702;4;8633;8311;-86;-80;-90;-87;4;12;-30;-45;2;15;0;15;50;113;0;199;1;27;56070;> +[1;61669504;129896152;8543;19897;] +<1;703;4;8543;8226;-91;-83;-90;-86;6;12;-29;-45;3;15;0;15;50;113;0;199;1;27;56516;> +[1;61669340;129898424;8460;47851;] +<1;704;4;8460;8144;-83;-80;-90;-85;14;12;-29;-44;2;15;0;16;50;113;0;199;1;27;8613;> +[1;61669412;129901184;8366;59473;] +<1;705;4;8366;8061;-88;-88;-88;-84;11;12;-29;-43;3;15;0;15;50;113;0;199;1;27;1221;> +[1;61669548;129903992;8284;57946;] +<1;706;4;8284;7977;-84;-81;-87;-85;17;12;-29;-43;3;15;0;16;50;113;0;199;1;27;25731;> +[1;61669780;129907456;8202;31161;] +<1;707;4;8202;7897;-82;-78;-87;-82;18;12;-29;-42;2;15;0;15;50;113;0;199;1;27;32168;> +[1;61669796;129911056;8117;54159;] +<1;708;4;8117;7816;-87;-79;-86;-82;20;12;-28;-41;2;15;0;15;50;113;0;199;1;27;57411;> +[1;61669700;129913984;8028;9058;] +<1;709;4;8028;7733;-87;-81;-86;-81;19;12;-28;-40;2;15;0;16;50;113;0;199;1;27;39723;> +[1;61669996;129916928;7944;1750;] +<1;710;4;7944;7649;-83;-90;-85;-81;13;12;-28;-40;2;15;0;16;50;113;0;199;1;27;52591;> +[1;61670292;129919768;7861;25632;] +<1;711;4;7861;7567;-82;-79;-84;-83;16;12;-28;-39;1;15;0;16;50;113;0;199;1;27;49002;> +[1;61670612;129922928;7772;8242;] +<1;712;4;7772;7482;-90;-83;-86;-82;18;12;-28;-39;1;15;0;16;50;113;0;199;1;27;18570;> +[1;61671084;129926408;7694;17740;] +<1;713;4;7694;7406;-76;-74;-84;-81;18;12;-28;-38;2;16;0;16;50;113;0;199;1;27;37665;> +[1;61671524;129929080;7607;14329;] +<1;714;4;7607;7325;-87;-79;-85;-81;12;12;-27;-38;2;16;0;16;50;113;0;199;1;27;42711;> +[1;61671484;129932144;7530;31016;] +<1;715;4;7530;7247;-81;-83;-83;-82;14;12;-27;-37;2;16;0;16;50;113;0;199;1;27;3813;> +[1;61671660;129935000;7444;28475;] +<1;716;4;7444;7164;-83;-81;-82;-81;18;12;-27;-36;2;16;0;16;50;113;0;199;1;27;46032;> +[1;61671848;129937880;7360;9234;] +<1;717;4;7360;7083;-83;-79;-84;-80;19;12;-27;-36;3;16;0;16;50;113;0;199;1;27;55246;> +[1;61672156;129940264;7277;23551;] +<1;718;4;7277;7004;-85;-77;-81;-79;17;12;-26;-36;3;16;0;16;50;113;0;199;1;27;32447;> +[1;61672768;129942688;7193;13662;] +<1;719;4;7193;6925;-82;-77;-83;-80;14;12;-26;-35;3;16;0;16;50;113;0;199;1;27;15397;> +[1;61673048;129945496;7116;3187;] +<1;720;4;7116;6847;-79;-83;-82;-80;11;12;-25;-34;3;16;0;15;50;113;0;199;1;27;41440;> +[1;61673520;129948576;7032;37223;] +<1;721;4;7032;6768;-82;-76;-82;-79;17;12;-25;-34;3;16;0;16;50;113;0;199;1;27;28055;> +[1;61674064;129950624;6955;28339;] +<1;722;4;6955;6691;-77;-75;-82;-79;14;12;-24;-33;3;16;0;16;50;112;0;199;1;27;48122;> +[1;61674508;129952976;6873;29178;] +<1;723;4;6873;6611;-79;3;> +[1;61675176;129956024;6793;11081;] +<1;724;4;6793;6534;-78;-76;-81;-78;17;12;-24;-31;3;16;0;16;50;113;0;199;1;27;2307;> +[1;61675600;129958800;6710;38381;] +<1;725;4;6710;6458;-82;-81;-80;-78;8;12;-23;-30;3;16;0;15;50;113;0;199;1;27;15535;> +[1;61675976;129961504;6634;37768;] +<1;726;4;6634;6379;-77;-77;-80;-77;14;12;-23;-30;3;16;0;16;50;112;0;199;1;27;39613;> +[1;61675972;129963992;6558;46973;] +<1;727;4;6558;6304;-76;-73;-79;-76;12;12;-22;-29;4;16;0;15;50;113;0;199;1;27;32395;> +[1;61676348;129966232;6477;10290;] +<1;728;4;6477;6228;-80;-74;-79;-76;13;12;-21;-29;4;16;0;16;50;113;0;199;1;27;52791;> +[1;61676684;129968416;6397;20246;] +<1;729;4;6397;6148;-79;-78;-79;-76;14;12;-21;-28;4;16;0;16;50;113;0;199;1;27;27168;> +[1;61676960;129970152;6323;2028;] +<1;730;4;6323;6072;-74;-82;-78;-78;11;12;-21;-28;4;16;0;16;50;113;0;199;1;27;11600;> +[1;61677108;129973072;6247;34763;] +<1;731;4;6247;5998;-74;-71;-77;-76;12;12;-21;-27;5;17;0;15;50;113;0;199;1;27;61207;> +[1;61677256;129975800;6172;33005;] +<1;732;4;6172;5925;-73;-71;-76;-75;16;12;-20;-26;5;16;0;9;50;114;0;199;0;27;24415;> +[1;61677392;129978056;6099;58470;] +<1;733;4;6099;5852;-73;-71;-76;-75;15;12;-19;-26;6;16;0;8;50;114;0;199;0;27;39458;> +[1;61677532;129980648;6026;54797;] +<1;734;4;6026;5780;-74;-71;-75;-74;13;12;-19;-25;6;16;0;9;50;114;0;199;0;27;61927;> +[1;61678080;129983320;5953;6773;] +<1;735;4;5953;5706;-75;-79;-75;-74;17;12;-18;-25;6;15;0;9;50;114;0;199;0;27;11018;> +[1;61678216;129985568;5877;60893;] +<1;736;4;5877;5632;-75;-72;-73;-74;15;12;-18;-24;6;15;0;9;50;114;0;199;0;27;17843;> +[1;61678508;129987888;5799;26435;] +<1;737;4;5799;5557;-78;-73;-73;-73;14;12;-17;-24;5;15;0;9;50;114;0;199;0;27;65023;> +[1;61678860;129990632;5727;54893;] +<1;738;4;5727;5485;-70;-70;-74;-73;15;12;-17;-23;5;15;0;9;50;114;0;199;0;27;39774;> +[1;61678988;129993408;5654;46086;] +<1;739;4;5654;5411;-73;-72;-74;-73;11;12;-16;-23;5;15;0;9;50;114;0;199;0;27;18122;> +[1;61679276;129996472;5580;34210;] +<1;740;4;5580;5339;-77;-71;-74;-73;14;12;-16;-23;5;14;0;9;50;114;0;199;0;27;51941;> +[1;61679448;129999800;5509;44606;] +<1;741;4;5509;5268;-69;-76;-74;-73;13;12;-16;-22;5;14;0;9;50;114;0;199;0;27;17928;> +[1;61679724;130003344;5436;51091;] +<1;742;4;5436;5198;-69;-68;-73;-72;23;12;-15;-21;4;14;0;9;50;114;0;199;0;27;65124;> +[1;61680268;130007064;5365;10045;] +<1;743;4;5365;5129;-73;-67;-73;-72;20;12;-15;-21;4;14;0;9;50;114;0;199;0;27;39435;> +[1;61680540;130010464;5291;61466;] +<1;744;4;5291;5056;-71;-71;-72;-71;20;12;-15;-20;4;14;0;10;50;114;0;199;0;27;47358;> +[1;61680948;130013552;5224;31527;] +<1;745;4;5224;4987;-68;-67;-72;-71;23;12;-15;-20;4;14;0;10;50;114;0;199;0;27;24653;> +[1;61681144;130016600;5153;25193;] +<1;746;4;5153;4917;-71;-75;-72;-70;17;12;-14;-19;3;14;0;9;50;114;0;199;0;27;7800;> +[1;61681872;130019544;5085;30294;] +<1;747;4;5085;4848;-69;-67;-71;-71;15;12;-14;-18;3;14;0;10;50;114;0;199;0;27;6396;> +[1;61682428;130022800;5012;19335;] +<1;748;4;5012;4779;-71;-67;-70;-69;19;12;-14;-18;4;14;0;10;50;114;0;199;0;27;24541;> +[1;61682952;130025224;4944;44824;] +<1;749;4;4944;4711;-67;-66;-70;-69;13;12;-13;-18;4;14;0;9;50;114;0;199;0;27;10562;> +[1;61683432;130028152;4876;10937;] +<1;750;4;4876;4642;-68;-68;-70;-69;16;12;-13;-18;4;14;0;10;50;114;0;199;0;27;48682;> +[1;61683928;130031128;4805;56686;] +<1;751;4;4805;4572;-70;-69;-69;-68;15;12;-12;-18;5;14;0;8;50;114;0;199;0;27;18353;> +[1;61684336;130033584;4735;19801;] +<1;752;4;4735;4504;-68;-72;-69;-70;15;12;-12;-17;5;14;0;9;50;114;0;199;0;27;18555;> +[1;61684824;130036648;4668;11474;] +<1;753;4;4668;4438;-66;-64;-69;-68;18;12;-11;-16;5;14;0;9;50;114;0;199;0;27;35028;> +[1;61685292;130039688;4603;63282;] +<1;754;4;4603;4374;-64;-62;-68;-67;16;12;-11;-16;5;13;0;9;50;114;0;199;0;27;16617;> +[1;61685708;130045056;4467;52279;] +<1;756;4;4467;4244;-69;-63;-68;-67;14;12;-10;-15;5;13;0;9;50;114;0;199;0;27;31841;> +[1;61685864;130047056;4398;7888;] +<1;757;4;4398;4178;-66;-68;-67;-66;13;12;-10;-15;5;13;0;8;50;114;0;199;0;27;25301;> diff --git a/test_client/client.py b/test_client/client.py new file mode 100644 index 0000000..c85637e --- /dev/null +++ b/test_client/client.py @@ -0,0 +1,372 @@ +""" +Telemetry test client for stratoflights. + +Reads a trajectory file and sends one packet every N seconds to the station +WebSocket endpoint: ws:///api/ws/station//telemetry/?token= + +Supported trajectory formats +----------------------------- +JSON array (default): + [ + {"lat": 62.1, "lon": 129.4, "alt": 500.0, "timestamp": 1716000000}, + ... + ] + +CSV (first row = header): + lat,lon,alt,timestamp + 62.1,129.4,500.0,1716000000 + +Balloon log (.log): + Supports both "beacon" and "tracking" log variants produced by the + on-board firmware. Each record has a bracketed position header, followed + by angle-bracket telemetry lines. Tracking logs may also embed a Python- + style dict with already-decoded coordinates (used as a cross-check). + + Header: [1;;;;;] + Telemetry: <1;;;;<...more fields...>;> + Optional: {'lat': 62.1, 'lon': 129.4, 'alt': 500} (tracking only) + + Coordinate decoding: + lat = lat_raw / 1_000_000 + lon = lon_raw / 1_000_000 + alt = alt_m (metres, integer) + + Records with no GPS fix (lat_raw == 0 and lon_raw == 0) are skipped. + Exact duplicate positions (same lat / lon / alt triple) are removed. + +Usage +----- + python client.py --server ws://localhost:8000 \\ + --satellite \\ + --token \\ + --file trajectory.json \\ + --interval 10 + + # Balloon log files: + python client.py --satellite --token --file beacon-20250406.log + python client.py --satellite --token --file tracking-20250406-02.log + + # Generate a sample trajectory file then run: + python client.py --generate sample.json + python client.py --satellite --token --file sample.json +""" + +import argparse +import asyncio +import csv +import json +import math +import re +import sys +import time +from pathlib import Path + +# 2025-04-06 03:00:00 UTC +BASE_TS = 1743908400 + + +# --------------------------------------------------------------------------- +# Trajectory loaders +# --------------------------------------------------------------------------- + +def load_json(path: Path) -> list[dict]: + data = json.loads(path.read_text()) + if not isinstance(data, list): + # Support Tawhiri-style nested format + if "prediction" in data: + points = [] + for stage in data["prediction"]: + for p in stage.get("trajectory", []): + points.append({ + "lat": p["latitude"], + "lon": p["longitude"], + "alt": p["altitude"], + "timestamp": p.get("time", int(time.time())), + }) + return points + raise ValueError("JSON file must be an array or a Tawhiri prediction object") + return data + + +def load_csv(path: Path) -> list[dict]: + points = [] + with path.open(newline="") as f: + reader = csv.DictReader(f) + for row in reader: + points.append({ + "lat": float(row["lat"]), + "lon": float(row["lon"]), + "alt": float(row["alt"]), + "timestamp": int(row.get("timestamp", BASE_TS + len(points) * 10)), + }) + return points + + +# Matches the position header line: [1;;;;;] +_HEADER_RE = re.compile(r'^\[1;(-?\d+);(-?\d+);(-?\d+);(\d+);]') +# Matches the start of a telemetry line to extract the sequence number +_TELEM_RE = re.compile(r'^<1;(\d+);') + + +def load_log(path: Path) -> list[dict]: + """Parse beacon or tracking .log files into a trajectory point list. + + Duplicate positions (identical lat/lon/alt) are silently dropped. + Points with no GPS fix (lat_raw == lon_raw == 0) are also dropped. + """ + base_ts = BASE_TS + points: list[dict] = [] + seen: set[tuple] = set() + + lines = path.read_text(errors="replace").splitlines() + i = 0 + while i < len(lines): + m = _HEADER_RE.match(lines[i].strip()) + if not m: + i += 1 + continue + + lat_raw = int(m.group(1)) + lon_raw = int(m.group(2)) + alt_raw = int(m.group(3)) + + # No GPS fix yet + if lat_raw == 0 and lon_raw == 0: + i += 1 + continue + + lat = round(lat_raw / 1_000_000, 6) + lon = round(lon_raw / 1_000_000, 6) + alt = float(alt_raw) + + # Drop exact-position duplicates + key = (lat, lon, alt_raw) + if key in seen: + i += 1 + continue + seen.add(key) + + # Grab the sequence number from the telemetry line that follows + seq = len(points) + for j in range(i + 1, min(i + 5, len(lines))): + tm = _TELEM_RE.match(lines[j].strip()) + if tm: + seq = int(tm.group(1)) + break + + points.append({ + "lat": lat, + "lon": lon, + "alt": alt, + "timestamp": base_ts + seq, + }) + i += 1 + + return points + + +def load_trajectory(path: Path) -> list[dict]: + suffix = path.suffix.lower() + if suffix == ".csv": + return load_csv(path) + if suffix == ".log": + return load_log(path) + return load_json(path) + + +def deduplicate(points: list[dict]) -> list[dict]: + """Remove points with identical (lat, lon, alt), preserving order.""" + seen: set[tuple] = set() + result = [] + for p in points: + key = (round(float(p.get("lat", 0)), 6), + round(float(p.get("lon", 0)), 6), + round(float(p.get("alt", 0)), 1)) + if key not in seen: + seen.add(key) + result.append(p) + return result + + +# --------------------------------------------------------------------------- +# Sample generator +# --------------------------------------------------------------------------- + +def generate_sample(path: Path, n: int = 50) -> None: + """Generate a simple ascending balloon trajectory.""" + base_lat, base_lon = 62.0, 129.5 + base_ts = BASE_TS + points = [] + for i in range(n): + angle = i * 0.05 + points.append({ + "lat": round(base_lat + math.sin(angle) * 0.02, 6), + "lon": round(base_lon + i * 0.003, 6), + "alt": round(500.0 + i * 200.0, 1), + "timestamp": base_ts + i * 10, + }) + path.write_text(json.dumps(points, indent=2)) + print(f"Sample trajectory written to {path} ({n} points)") + + +# --------------------------------------------------------------------------- +# WebSocket sender +# --------------------------------------------------------------------------- + +async def run_ws(server: str, satellite: str, token: str, + points: list[dict], interval: float, loop: bool) -> None: + try: + import websockets + except ImportError: + sys.exit("websockets package not found — run: pip install websockets") + + url = f"{server.rstrip('/')}/api/ws/station/{satellite}/telemetry/?token={token}" + print(f"Connecting to {url}") + + iteration = 0 + abs_idx = 0 # never resets across loop iterations — drives advancing timestamps + while True: + try: + async with websockets.connect(url) as ws: + print("Connected.") + for point in points: + packet = { + "lat": float(point.get("lat", 0)), + "lon": float(point.get("lon", 0)), + "alt": float(point.get("alt", 0)), + "timestamp": BASE_TS + abs_idx * int(interval), + "payload": point.get("payload", {}), + "raw_data": point.get("raw_data", {}), + } + abs_idx += 1 + await ws.send(json.dumps(packet)) + print( + f"[{abs_idx}/{len(points)}] " + f"lat={packet['lat']:.5f} " + f"lon={packet['lon']:.5f} " + f"alt={packet['alt']:.1f} m " + f"ts={packet['timestamp']}" + ) + + try: + reply = await asyncio.wait_for(ws.recv(), timeout=2.0) + data = json.loads(reply) + if "error" in data: + print(f" Server error: {data['error']}") + except asyncio.TimeoutError: + pass # server only responds on errors + + await asyncio.sleep(interval) + + print("All points sent.") + iteration += 1 + if not loop: + break + print(f"Looping (iteration {iteration+1})...") + await asyncio.sleep(interval) + + except Exception as e: + print(f"Connection error: {e}. Retrying in 5s...") + await asyncio.sleep(5) + if not loop: + break + + +# --------------------------------------------------------------------------- +# REST sender (fallback, no auth needed per stratoflights AllowAny policy) +# --------------------------------------------------------------------------- + +async def run_rest(server: str, satellite: str, + points: list[dict], interval: float, loop: bool) -> None: + try: + import aiohttp + except ImportError: + sys.exit("aiohttp package not found — run: pip install aiohttp") + + url = f"{server.rstrip('/')}/api/{satellite}/telemetry/" + print(f"Sending {len(points)} points to {url} in order, no delay...") + + async with aiohttp.ClientSession() as session: + while True: + for i, point in enumerate(points): + packet = { + "lat": float(point.get("lat", 0)), + "lon": float(point.get("lon", 0)), + "alt": float(point.get("alt", 0)), + "timestamp": int(point.get("timestamp", BASE_TS + i * int(interval))), + "payload": point.get("payload", {}), + } + async with session.post(url, json=packet) as resp: + print( + f"[{i+1}/{len(points)}] " + f"lat={packet['lat']:.5f} lon={packet['lon']:.5f} " + f"alt={packet['alt']:.1f} m → HTTP {resp.status}" + ) + print("All points sent.") + if not loop: + break + print("Looping...") + await asyncio.sleep(interval) + + +# --------------------------------------------------------------------------- +# CLI +# --------------------------------------------------------------------------- + +def main() -> None: + parser = argparse.ArgumentParser(description="Stratoflights telemetry test client") + parser.add_argument("--server", default="ws://localhost:8000", + help="Base server URL (default: ws://localhost:8000)") + parser.add_argument("--satellite", help="Satellite UUID") + parser.add_argument("--token", help="Auth token (required for WebSocket mode)") + parser.add_argument("--file", nargs='+', metavar="FILE", + help="One or more trajectory files (.json, .csv, or .log); merged in order") + parser.add_argument("--interval", type=float, default=5.0, + help="Seconds between packets (default: 10)") + parser.add_argument("--mode", choices=["ws", "rest"], default="ws", + help="Transport: ws (WebSocket, default) or rest (HTTP POST)") + parser.add_argument("--loop", action="store_true", + help="Replay the trajectory in a loop indefinitely") + parser.add_argument("--generate", metavar="FILE", + help="Generate a sample trajectory JSON and exit") + args = parser.parse_args() + + if args.generate: + generate_sample(Path(args.generate)) + return + + if not args.satellite: + parser.error("--satellite is required") + if not args.file: + parser.error("--file is required") + if args.mode == "ws" and not args.token: + parser.error("--token is required for WebSocket mode") + + merged: list[dict] = [] + for f in args.file: + path = Path(f) + if not path.exists(): + sys.exit(f"File not found: {path}") + loaded = load_trajectory(path) + print(f" {path}: {len(loaded)} points") + merged.extend(loaded) + + before = len(merged) + points = deduplicate(merged) + removed = before - len(points) + if not points: + sys.exit("No valid points found in the provided files") + print(f"Total: {before} points" + + (f", removed {removed} duplicates → {len(points)} unique" if removed else f" ({len(points)} unique)")) + + if args.mode == "ws": + # Swap http(s) → ws(s) if the user passed an HTTP URL + server = args.server.replace("http://", "ws://").replace("https://", "wss://") + asyncio.run(run_ws(server, args.satellite, args.token, points, args.interval, args.loop)) + else: + server = args.server.replace("ws://", "http://").replace("wss://", "https://") + asyncio.run(run_rest(server, args.satellite, points, args.interval, args.loop)) + + +if __name__ == "__main__": + main() diff --git a/test_client/requirements.txt b/test_client/requirements.txt new file mode 100644 index 0000000..a9a0ec9 --- /dev/null +++ b/test_client/requirements.txt @@ -0,0 +1,2 @@ +websockets>=12.0 +aiohttp>=3.9 diff --git a/test_client/tracking-20250406-02.log b/test_client/tracking-20250406-02.log new file mode 100644 index 0000000..b698542 --- /dev/null +++ b/test_client/tracking-20250406-02.log @@ -0,0 +1,2453 @@ +[1;61682072;129859952;15015;55684;] +<1;371;2;15015;14490;9 +6;97;68;70;5;12;-9;-47;3;15;0;8;50;118;0;199;0;27;43764;> + +[1;61682332;129861112;15090;1582;] +<1;372;2;15090;14554;80 +;63;75;72;11;12;-9;-47;2;15;0;8;50;118;0;199;0;27;12187;> + +[1;61682260;129862872;15146;63257;] +<1;373;2;15146;14607;5 +5;51;72;70;6;12;-9;-47;3;14;0;8;50;118;0;199;0;27;47496;> + +[1;61681788;129864072;15214;23936;] +<1;374;2;15214;14679;7 +3;76;68;69;11;12;-9;-46;2;14;0;8;50;118;0;199;0;27;50832;> + + +[1;61681492;129865552;15281;38404;] +<1;375;2;15281;14749;5 +9;68;71;70;3;12;-9;-46;1;14;0;8;50;118;0;199;0;27;52444;> + +[1;61681288;129866408;15348;55220;] +<1;376;2;15348;14815;7 +0;64;73;71;5;12;-9;-46;1;14;0;8;50;118;0;199;0;27;39683;> + +[1;61681416;129867384;15429;37299;] +<1;377;2;15429;14895;8 +0;78;71;69;8;12;-9;-45;0;14;0;8;50;118;0;199;0;27;48203;> + +[1;61681884;129868680;15508;8797;] +<1;378;2;15508;14957;77 +;60;68;67;7;12;-9;-45;1;14;0;8;50;118;0;199;0;27;16782;> + +[1;61682088;129869992;15550;20451;] +<1;379;2;15550;15005;4 +[1;61681580;129881392;15935;56733;] +<1;387;2;15935;15392;4 +{'lat': 61.68158, 'lon': 129.881392, 'alt': 15935} +4;46;47;47;7;12;-7;-43;3;13;0;9;50;118;0;199;0;27;29317;> + +[1;61681660;129882920;15985;18463;] +<1;388;2;15985;15443;4 +{'lat': 61.68166, 'lon': 129.88292, 'alt': 15985} +2;50;46;47;9;12;-7;-44;4;13;0;9;50;118;0;199;0;27;2302;> + +[1;61681676;129884784;16030;7671;] +<1;389;2;16030;15490;51 +{'lat': 61.681676, 'lon': 129.884784, 'alt': 16030} +;45;46;46;8;12;-7;-43;3;13;0;9;50;118;0;199;0;27;10419;> + +[1;61681744;129886752;16091;41181;] +<1;390;2;16091;15549;5 +{'lat': 61.681744, 'lon': 129.886752, 'alt': 16091} +2;63;47;49;9;12;-7;-43;3;13;0;9;50;118;0;199;0;27;63208;> + +[1;61681932;129888120;16143;13059;] +<1;391;2;16143;15599;5 +{'lat': 61.681932, 'lon': 129.88812, 'alt': 16143} +2;49;49;49;5;12;-6;-42;3;13;0;9;50;118;0;199;0;27;27545;> + +[1;61681748;129889392;16198;40509;] +<1;392;2;16198;15656;5 +{'lat': 61.681748, 'lon': 129.889392, 'alt': 16198} +9;55;51;51;8;12;-6;-42;4;13;0;9;50;118;0;199;0;27;40880;> + +[1;61681544;129890640;16252;52827;] +<1;393;2;16252;15708;5 +{'lat': 61.681544, 'lon': 129.89064, 'alt': 16252} +3;51;51;52;2;12;-6;-42;3;13;0;9;50;118;0;199;0;27;26148;> + +[1;61681636;129891416;16306;16386;] +<1;394;2;16306;15766;5 +{'lat': 61.681636, 'lon': 129.891416, 'alt': 16306} +3;57;53;53;7;12;-6;-42;4;13;0;8;50;118;0;199;0;27;58294;> + +[1;61681868;129892640;16360;7479;] +<1;395;2;16360;15821;53 +{'lat': 61.681868, 'lon': 129.89264, 'alt': 16360} +;54;56;55;4;12;-6;-43;4;13;0;8;50;118;0;199;0;27;9376;> + +[1;61681920;129893344;16410;13532;] +<1;396;2;16410;15869;4 +{'lat': 61.68192, 'lon': 129.893344, 'alt': 16410} +9;49;53;52;3;12;-6;-43;5;13;0;9;50;118;0;199;0;27;10404;> + +[1;61681940;129894168;16462;38470;] +<1;397;2;16462;15914;4 +{'lat': 61.68194, 'lon': 129.894168, 'alt': 16462} +5;48;52;52;4;12;-5;-43;6;13;0;8;50;118;0;199;0;27;58418;> + +[1;61682044;129894672;16502;50229;] +<1;398;2;16502;15958;3 +{'lat': 61.682044, 'lon': 129.894672, 'alt': 16502} +9;42;50;50;4;12;-5;-42;5;13;0;9;50;118;0;199;0;27;25656;> + +[1;61682140;129895520;16556;40721;] +<1;399;2;16556;16013;5 +{'lat': 61.68214, 'lon': 129.89552, 'alt': 16556} +9;54;51;50;3;12;-5;-41;4;13;0;9;50;118;0;199;0;27;41699;> + +[1;61682188;129895952;16609;53234;] +<1;400;2;16609;16067;5 +{'lat': 61.682188, 'lon': 129.895952, 'alt': 16609} +1;52;50;50;3;12;-5;-40;4;13;0;9;50;118;0;199;0;27;11144;> + +[1;61682216;129896656;16652;15394;] +<1;401;2;16652;16120;4 +{'lat': 61.682216, 'lon': 129.896656, 'alt': 16652} +2;52;47;50;2;12;-5;-40;3;12;0;16;50;116;0;199;1;27;12063;> + + +[1;61682356;129897552;16710;32858;] +<1;402;2;16710;16171;5 +{'lat': 61.682356, 'lon': 129.897552, 'alt': 16710} +6;50;47;50;7;12;-5;-40;4;13;0;16;50;116;0;199;1;27;27243;> + + +[1;61682604;129898304;16765;16678;] +<1;403;2;16765;16230;6 +{'lat': 61.682604, 'lon': 129.898304, 'alt': 16765} +1;64;49;50;4;12;-5;-39;4;13;0;15;50;116;0;199;1;27;27200;> + + +[1;61682868;129898632;16820;64193;] +<1;404;2;16820;16287;4 +{'lat': 61.682868, 'lon': 129.898632, 'alt': 16820} +5;55;50;52;5;12;-5;-39;4;14;0;15;50;116;0;199;1;27;35026;> + + +[1;61683212;129899704;16882;44863;] +<1;405;2;16882;16353;6 +{'lat': 61.683212, 'lon': 129.899704, 'alt': 16882} +7;64;53;55;4;12;-5;-39;4;14;0;15;50;116;0;199;1;27;60826;> + + +[1;61683404;129900144;16942;40265;] +<1;406;2;16942;16409;5 +{'lat': 61.683404, 'lon': 129.900144, 'alt': 16942} +9;55;54;56;4;12;-4;-39;6;14;0;15;50;116;0;199;1;27;63506;> + + +[1;61683700;129901304;17000;38053;] +<1;407;2;17000;16464;5 +{'lat': 61.6837, 'lon': 129.901304, 'alt': 17000} +6;54;55;56;5;12;-4;-39;5;15;0;15;50;116;0;199;1;27;1158;> + +[1;61683948;129902344;17044;27841;] +<1;408;2;17044;16511;5 +{'lat': 61.683948, 'lon': 129.902344, 'alt': 17044} +0;48;56;57;6;12;-4;-38;5;15;0;15;50;116;0;199;1;27;18739;> + + +[1;61684128;129903184;17094;61375;] +<1;409;2;17094;16558;4 +{'lat': 61.684128, 'lon': 129.903184, 'alt': 17094} +2;49;55;57;4;12;-4;-38;5;15;0;15;50;116;0;199;1;27;49265;> + + +[1;61684300;129904392;17135;48427;] +<1;410;2;17135;16603;4 +{'lat': 61.6843, 'lon': 129.904392, 'alt': 17135} +6;44;54;54;9;12;-3;-38;6;15;0;14;50;116;0;199;1;27;2925;> + +[1;61684340;129905856;17179;42687;] +<1;411;2;17179;16651;4 +{'lat': 61.68434, 'lon': 129.905856, 'alt': 17179} +3;47;52;52;6;12;-3;-37;6;15;0;15;50;116;0;199;1;27;29314;> + + +[1;61684236;129907528;17230;41389;] +<1;412;2;17230;16702;4 +{'lat': 61.684236, 'lon': 129.907528, 'alt': 17230} +9;49;49;49;9;12;-3;-37;7;15;0;14;50;116;0;199;1;27;59117;> + + +[1;61684156;129908952;17281;11335;] +<1;413;2;17281;16749;5 +{'lat': 61.684156, 'lon': 129.908952, 'alt': 17281} +0;46;48;48;6;12;-3;-37;9;16;0;14;50;116;0;199;1;27;63787;> + + +[1;61684000;129910232;17324;61689;] +<1;414;2;17324;16795;4 +{'lat': 61.684, 'lon': 129.910232, 'alt': 17324} +1;48;45;47;8;12;-2;-36;9;16;0;13;50;116;0;199;1;27;40264;> + + +[1;61683888;129910992;17370;35091;] +<1;415;2;17370;16859;5 +{'lat': 61.683888, 'lon': 129.910992, 'alt': 17370} +2;62;45;47;3;12;-2;-36;9;16;0;14;50;116;0;199;1;27;32813;> + + +[1;61684240;129912016;17460;46721;] +<1;416;2;17460;16944;7 +{'lat': 61.68424, 'lon': 129.912016, 'alt': 17460} +9;83;47;52;7;12;-2;-37;10;16;0;15;50;116;0;199;1;27;25794; +> + +[1;61684248;129912904;17522;61214;] +<1;417;2;17522;16994;6 +{'lat': 61.684248, 'lon': 129.912904, 'alt': 17522} +2;49;54;56;6;12;-1;-37;11;16;0;13;50;116;0;199;1;27;8735;> + + +[1;61684304;129913592;17567;30077;] +<1;418;2;17567;17045;5 +{'lat': 61.684304, 'lon': 129.913592, 'alt': 17567} +1;49;55;57;4;12;-1;-37;10;16;0;14;50;116;0;199;1;27;45105; +> + +[1;61684208;129914320;17606;41066;] +<1;419;2;17606;17089;3 +{'lat': 61.684208, 'lon': 129.91432, 'alt': 17606} +6;47;53;56;6;12;-1;-37;11;16;0;15;50;116;0;199;1;27;17196; +> + +[1;61684072;129915600;17659;37277;] +<1;420;2;17659;17137;5 +{'lat': 61.684072, 'lon': 129.9156, 'alt': 17659} +1;47;55;56;4;12;-1;-37;11;17;0;7;50;117;0;199;0;27;3605;> + +[1;61683744;129916504;17706;59664;] +<1;421;2;17706;17198;5 +{'lat': 61.683744, 'lon': 129.916504, 'alt': 17706} +3;59;54;56;8;12;0;-37;12;17;0;7;50;117;0;199;0;27;26584;> + +[1;61683252;129916976;17760;48773;] +<1;422;2;17760;17244;5 +{'lat': 61.683252, 'lon': 129.916976, 'alt': 17760} +2;45;54;54;6;12;0;-36;13;16;0;8;50;118;0;199;0;27;27017;> + +[1;61682872;129918168;17807;18348;] +<1;423;2;17807;17295;4 +{'lat': 61.682872, 'lon': 129.918168, 'alt': 17807} +6;50;48;50;5;12;0;-36;13;16;0;8;50;118;0;199;0;27;48634;> + +[1;61682528;129918856;17849;49006;] +<1;424;2;17849;17341;4 +{'lat': 61.682528, 'lon': 129.918856, 'alt': 17849} +1;45;48;49;5;12;0;-36;12;15;0;8;50;117;0;199;0;27;18641;> + +[1;61682344;129920056;17891;36756;] +<1;425;2;17891;17388;4 +{'lat': 61.682344, 'lon': 129.920056, 'alt': 17891} +9;51;47;49;7;12;0;-36;12;15;0;9;50;118;0;199;0;27;60002;> + +[1;61682040;129921064;17948;14230;] +<1;426;2;17948;17443;3 +{'lat': 61.68204, 'lon': 129.921064, 'alt': 17948} +9;54;48;50;4;12;0;-36;13;15;0;9;50;118;0;199;0;27;45146;> + +[1;61681984;129921680;17995;8979;] +<1;427;2;17995;17494;53 +{'lat': 61.681984, 'lon': 129.92168, 'alt': 17995} +;50;48;50;2;12;0;-36;13;15;0;7;50;117;0;199;0;27;17031;> + +[1;61682072;129922360;18033;8044;] +<1;428;2;18033;17540;45 +{'lat': 61.682072, 'lon': 129.92236, 'alt': 18033} +;45;47;48;4;12;1;-37;14;15;0;6;50;117;0;199;0;27;37150;> + +[1;61682004;129922656;18084;56014;] +<1;429;2;18084;17588;5 +{'lat': 61.682004, 'lon': 129.922656, 'alt': 18084} +0;47;46;49;2;12;1;-37;14;15;0;9;50;118;0;199;0;27;54827;> + +[1;61681916;129923168;18127;25020;] +<1;430;2;18127;17632;4 +{'lat': 61.681916, 'lon': 129.923168, 'alt': 18127} +1;43;46;48;1;12;1;-37;14;15;0;6;50;117;0;199;0;27;18105;> + +[1;61681952;129923280;18185;52112;] +<1;431;2;18185;17689;4 +{'lat': 61.681952, 'lon': 129.92328, 'alt': 18185} +7;61;47;48;1;12;1;-36;13;15;0;8;50;118;0;199;0;27;27532;> + +[1;61682024;129923664;18234;52566;] +<1;432;2;18234;17736;4 +{'lat': 61.682024, 'lon': 129.923664, 'alt': 18234} +9;46;47;49;4;12;1;-36;12;15;0;8;50;118;0;199;0;27;53642;> + +[1;61682000;129924160;18279;18561;] +<1;433;2;18279;17788;5 +{'lat': 61.682, 'lon': 129.92416, 'alt': 18279} +2;50;48;49;1;12;2;-37;13;15;0;6;50;117;0;199;0;27;43571;> + +[1;61681944;129924848;18329;62983;] +<1;434;2;18329;17841;4 +{'lat': 61.681944, 'lon': 129.924848, 'alt': 18329} +2;52;47;49;5;12;2;-37;13;14;0;8;50;118;0;199;0;27;61441;> + +[1;61681884;129925448;18383;6091;] +<1;435;2;18383;17889;59 +{'lat': 61.681884, 'lon': 129.925448, 'alt': 18383} +;47;48;50;1;12;2;-36;13;14;0;8;50;118;0;199;0;27;30681;> + +[1;61681988;129925768;18431;11437;] +<1;436;2;18431;17945;4 +{'lat': 61.681988, 'lon': 129.925768, 'alt': 18431} +6;55;47;51;2;12;2;-36;12;14;0;8;50;118;0;199;0;27;24360;> + +[1;61681688;129926224;18480;16804;] +<1;437;2;18480;17995;4 +{'lat': 61.681688, 'lon': 129.926224, 'alt': 18480} +8;51;49;51;4;12;2;-36;12;14;0;8;50;117;0;199;0;27;35941;> + +[1;61681572;129926544;18530;63870;] +<1;438;2;18530;18042;4 +{'lat': 61.681572, 'lon': 129.926544, 'alt': 18530} +9;49;49;50;1;12;2;-36;13;14;0;8;50;117;0;199;0;27;18302;> + +[1;61681300;129926776;18579;8593;] +<1;439;2;18579;18093;47 +{'lat': 61.6813, 'lon': 129.926776, 'alt': 18579} +;50;48;50;3;12;3;-36;13;14;0;9;50;118;0;199;0;27;52129;> + +[1;61681192;129927112;18627;29516;] +<1;440;2;18627;18147;4 +{'lat': 61.681192, 'lon': 129.927112, 'alt': 18627} +7;53;49;50;1;12;3;-36;13;14;0;9;50;117;0;199;0;27;12386;> + +[1;61681156;129927488;18678;56817;] +<1;441;2;18678;18195;5 +{'lat': 61.681156, 'lon': 129.927488, 'alt': 18678} +0;47;51;50;4;12;3;-35;13;14;0;8;50;117;0;199;0;27;12640;> + +[1;61681368;129928144;18729;44235;] +<1;442;2;18729;18244;4 +{'lat': 61.681368, 'lon': 129.928144, 'alt': 18729} +9;48;50;49;3;12;3;-36;12;14;0;8;50;117;0;199;0;27;60792;> + +[1;61681412;129928344;18771;48734;] +<1;443;2;18771;18295;4 +{'lat': 61.681412, 'lon': 129.928344, 'alt': 18771} +1;50;49;49;3;12;3;-35;12;14;0;7;50;117;0;199;0;27;1799;> + +[1;61681596;129928784;18823;40902;] +<1;444;2;18823;18352;5 +{'lat': 61.681596, 'lon': 129.928784, 'alt': 18823} +0;60;49;51;0;12;3;-34;12;14;0;8;50;117;0;199;0;27;54021;> + +[1;61681700;129929120;18876;34824;] +<1;445;2;18876;18399;5 +{'lat': 61.6817, 'lon': 129.92912, 'alt': 18876} +3;46;50;51;3;12;3;-34;11;14;0;8;50;117;0;199;0;27;62233;> + +[1;61681664;129929016;18928;25132;] +<1;446;2;18928;18455;4 +{'lat': 61.681664, 'lon': 129.929016, 'alt': 18928} +1;54;51;51;2;12;3;-34;11;14;0;8;50;117;0;199;0;27;44930;> + +;61;50;52;0;12;3;-34;11;14;0;6;50;117;0;199;0;27;29195;> + +[1;61681464;129928728;19047;30638;] +<1;448;2;19047;18576;6 +{'lat': 61.681464, 'lon': 129.928728, 'alt': 19047} +2;57;53;55;0;12;3;-34;11;14;0;7;50;117;0;199;0;27;32222;> + +[1;61681732;129928144;19093;59550;] +<1;449;2;19093;18624;4 +{'lat': 61.681732, 'lon': 129.928144, 'alt': 19093} +5;47;52;55;7;12;4;-34;11;14;0;7;50;117;0;199;0;27;57042;> + +[1;61682008;129927384;19150;9082;] +<1;450;2;19150;18677;66 +{'lat': 61.682008, 'lon': 129.927384, 'alt': 19150} +;51;53;53;4;12;4;-34;11;14;0;7;50;117;0;199;0;27;57989;> + +[1;61682204;129927488;19195;20146;] +<1;451;2;19195;18729;4 +{'lat': 61.682204, 'lon': 129.927488, 'alt': 19195} +4;56;52;54;1;12;4;-33;11;14;0;6;50;117;0;199;0;27;47662;> + +[1;61682420;129927288;19242;2417;] +<1;452;2;19242;18779;46 +{'lat': 61.68242, 'lon': 129.927288, 'alt': 19242} +;49;53;54;2;12;4;-33;11;14;0;6;50;117;0;199;0;27;49301;> + +[1;61682512;129927416;19288;17508;] +<1;453;2;19288;18828;4 +{'lat': 61.682512, 'lon': 129.927416, 'alt': 19288} +5;48;52;52;0;12;4;-32;11;14;0;6;50;117;0;199;0;27;25498;> + +[1;61682464;129927768;19344;9375;] +<1;454;2;19344;18876;55 +{'lat': 61.682464, 'lon': 129.927768, 'alt': 19344} +;47;49;50;0;12;4;-32;11;14;0;6;50;117;0;199;0;27;1024;> + +[1;61682528;129927600;19389;50735;] +<1;455;2;19389;18925;4 +{'lat': 61.682528, 'lon': 129.9276, 'alt': 19389} +4;48;49;50;3;12;4;-32;12;14;0;6;50;117;0;199;0;27;63216;> + +[1;61682656;129927568;19443;46241;] +<1;456;2;19443;18977;5 +{'lat': 61.682656, 'lon': 129.927568, 'alt': 19443} +3;50;48;50;3;12;4;-33;13;14;0;6;50;117;0;199;0;27;29311;> + +[1;61682808;129927504;19490;31452;] +<1;457;2;19490;19032;4 +{'lat': 61.682808, 'lon': 129.927504, 'alt': 19490} +4;59;48;50;5;12;4;-33;13;14;0;6;50;117;0;199;0;27;23195;> + +[1;61683140;129926984;19537;44259;] +<1;458;2;19537;19085;5 +{'lat': 61.68314, 'lon': 129.926984, 'alt': 19537} +7;52;48;51;4;12;4;-32;13;14;0;6;50;117;0;199;0;27;16171;> + +[1;61683528;129926744;19585;38236;] +<1;459;2;19585;19122;3 +{'lat': 61.683528, 'lon': 129.926744, 'alt': 19585} +5;36;46;49;4;12;4;-32;13;14;0;7;50;117;0;199;0;27;15908;> + +[1;61683812;129927112;19635;2108;] +<1;460;2;19635;19175;48 +{'lat': 61.683812, 'lon': 129.927112, 'alt': 19635} +;52;47;50;1;12;4;-32;13;14;0;6;50;117;0;199;0;27;57283;> + +[1;61683824;129927080;19682;31569;] +<1;461;2;19682;19223;5 +{'lat': 61.683824, 'lon': 129.92708, 'alt': 19682} +8;47;49;49;0;12;5;-32;13;14;0;7;50;117;0;199;0;27;16616;> + +[1;61683992;129927184;19730;28171;] +<1;462;2;19730;19275;4 +{'lat': 61.683992, 'lon': 129.927184, 'alt': 19730} +7;51;48;49;1;12;5;-31;13;14;0;8;50;117;0;199;0;27;31560;> + +[1;61683988;129927136;19778;41121;] +<1;463;2;19778;19324;5 +{'lat': 61.683988, 'lon': 129.927136, 'alt': 19778} +9;53;48;49;4;12;5;-31;13;14;0;7;50;117;0;199;0;27;30624;> + +[1;61684180;129927184;19825;20561;] +<1;464;2;19825;19372;4 +{'lat': 61.68418, 'lon': 129.927184, 'alt': 19825} +5;47;48;48;1;12;5;-31;13;14;0;8;50;117;0;199;0;27;20361;> + +[1;61684144;129927368;19876;16126;] +<1;465;2;19876;19422;5 +{'lat': 61.684144, 'lon': 129.927368, 'alt': 19876} +0;49;47;47;1;12;5;-31;12;14;0;7;50;117;0;199;0;27;12097;> + +[1;61684072;129927264;19925;48935;] +<1;466;2;19925;19473;4 +{'lat': 61.684072, 'lon': 129.927264, 'alt': 19925} +8;50;49;50;4;12;5;-30;12;14;0;7;50;117;0;199;0;27;22087;> + +6;50;49;49;3;12;5;-30;13;14;0;8;50;117;0;199;0;27;61116;> + +[1;61683824;129927128;20021;61571;] +<1;468;2;20021;19572;4 +{'lat': 61.683824, 'lon': 129.927128, 'alt': 20021} +6;47;47;50;1;12;5;-30;13;14;0;8;50;117;0;199;0;27;59989;> + +[1;61683764;129927352;20074;59954;] +<1;469;2;20074;19623;5 +{'lat': 61.683764, 'lon': 129.927352, 'alt': 20074} +3;49;47;49;2;12;5;-30;13;14;0;8;50;117;0;199;0;27;63839;> + +[1;61683872;129927688;20125;62307;] +<1;470;2;20125;19672;4 +{'lat': 61.683872, 'lon': 129.927688, 'alt': 20125} +9;53;48;49;2;12;6;-30;13;14;0;7;50;117;0;199;0;27;16247;> + +[1;61683672;129927720;20164;20068;] +<1;471;2;20164;19722;5 +{'lat': 61.683672, 'lon': 129.92772, 'alt': 20164} +1;49;49;50;2;12;6;-30;13;14;0;8;50;117;0;199;0;27;50577;> + +[1;61683368;129927640;20216;48508;] +<1;472;2;20216;19772;5 +{'lat': 61.683368, 'lon': 129.92764, 'alt': 20216} +0;49;49;50;4;12;6;-30;13;14;0;7;50;117;0;199;0;27;48561;> + +[1;61683084;129927688;20270;13094;] +<1;473;2;20270;19823;5 +{'lat': 61.683084, 'lon': 129.927688, 'alt': 20270} +2;50;49;49;2;12;6;-30;13;14;0;8;50;117;0;199;0;27;19705;> + +[1;61682864;129927112;20310;18951;] +<1;474;2;20310;19871;3 +{'lat': 61.682864, 'lon': 129.927112, 'alt': 20310} +9;47;48;50;3;12;6;-30;13;14;0;8;50;117;0;199;0;27;56465;> + +[1;61682636;129927096;20361;46724;] +<1;475;2;20361;19920;5 +{'lat': 61.682636, 'lon': 129.927096, 'alt': 20361} +0;48;49;49;3;12;6;-29;13;14;0;7;50;117;0;199;0;27;39741;> + +[1;61682540;129926776;20402;48013;] +<1;476;2;20402;19965;3 +{'lat': 61.68254, 'lon': 129.926776, 'alt': 20402} +8;49;47;49;1;12;6;-29;13;14;0;8;50;117;0;199;0;27;60526;> + +[1;61682432;129926632;20441;24967;] +<1;477;2;20441;20015;5 +{'lat': 61.682432, 'lon': 129.926632, 'alt': 20441} +3;49;47;48;3;12;6;-29;13;14;0;8;50;117;0;199;0;27;54761;> + +[1;61682364;129926896;20494;42642;] +<1;478;2;20494;20065;5 +{'lat': 61.682364, 'lon': 129.926896, 'alt': 20494} +2;49;45;49;1;12;6;-29;14;14;0;8;50;117;0;199;0;27;27589;> + +[1;61682236;129926912;20541;39526;] +<1;479;2;20541;20116;4 +{'lat': 61.682236, 'lon': 129.926912, 'alt': 20541} +6;50;45;49;1;12;7;-30;15;14;0;8;50;117;0;199;0;27;22566;> + +[1;61682108;129927000;20592;45005;] +<1;480;2;20592;20168;4 +{'lat': 61.682108, 'lon': 129.927, 'alt': 20592} +8;50;46;48;3;12;7;-29;14;14;0;8;50;117;0;199;0;27;32235;> + +[1;61681992;129927232;20635;2037;] +<1;481;2;20635;20221;42 +{'lat': 61.681992, 'lon': 129.927232, 'alt': 20635} +;52;44;49;1;12;7;-28;15;14;0;7;50;117;0;199;0;27;22312;> + +[1;61681656;129927112;20690;42041;] +<1;482;2;20690;20269;4 +{'lat': 61.681656, 'lon': 129.927112, 'alt': 20690} +2;48;47;50;5;12;7;-28;15;14;0;7;50;117;0;199;0;27;2605;> + +[1;61681356;129926776;20733;20250;] +<1;483;2;20733;20322;5 +{'lat': 61.681356, 'lon': 129.926776, 'alt': 20733} +4;57;47;51;4;12;7;-27;14;14;0;8;50;117;0;199;0;27;43380;> + +[1;61680936;129926408;20791;27756;] +<1;484;2;20791;20374;5 +{'lat': 61.680936, 'lon': 129.926408, 'alt': 20791} +7;51;48;50;4;12;7;-27;15;14;0;8;50;117;0;199;0;27;20792;> + +[1;61680496;129926176;20829;4888;] +<1;485;2;20829;20424;37 +{'lat': 61.680496, 'lon': 129.926176, 'alt': 20829} +;49;48;51;5;12;7;-28;16;14;0;8;50;117;0;199;0;27;24599;> + +[1;61680064;129925464;20885;9641;] +<1;486;2;20885;20476;39 +{'lat': 61.680064, 'lon': 129.925464, 'alt': 20885} +;51;47;51;5;12;7;-27;15;14;0;8;50;117;0;199;0;27;22956;> + +[1;61679652;129924776;20929;5119;] +<1;487;2;20929;20531;59 +{'lat': 61.679652, 'lon': 129.924776, 'alt': 20929} +;54;50;51;6;12;7;-27;15;14;0;8;50;117;0;199;0;27;15264;> + +[1;61679176;129924272;20989;52291;] +<1;488;2;20989;20582;5 +{'lat': 61.679176, 'lon': 129.924272, 'alt': 20989} +7;50;51;52;5;12;8;-27;15;14;0;8;50;117;0;199;0;27;43113;> + +[1;61678896;129923432;21034;51558;] +<1;489;2;21034;20633;4 +{'lat': 61.678896, 'lon': 129.923432, 'alt': 21034} +2;55;48;50;6;12;8;-27;15;14;0;8;50;117;0;199;0;27;78;> + +[1;61678452;129922640;21065;10807;] +<1;490;2;21065;20681;4 +{'lat': 61.678452, 'lon': 129.92264, 'alt': 21065} +7;47;49;51;6;12;8;-27;16;14;0;8;50;117;0;199;0;27;2890;> + +[1;61678092;129922088;21133;7027;] +<1;491;2;21133;20736;52 +{'lat': 61.678092, 'lon': 129.922088, 'alt': 21133} +;54;49;51;7;12;8;-27;16;14;0;9;50;117;0;199;0;27;54923;> + +[1;61677720;129921128;21171;20062;] +<1;492;2;21171;20791;5 +{'lat': 61.67772, 'lon': 129.921128, 'alt': 21171} +0;54;49;52;2;12;8;-28;16;14;0;8;50;117;0;199;0;27;22897;> + +[1;61677296;129920088;21230;15295;] +<1;493;2;21230;20844;5 +{'lat': 61.677296, 'lon': 129.920088, 'alt': 21230} +7;51;48;52;6;12;8;-27;16;14;0;8;50;117;0;199;0;27;20837;> + +[1;61677056;129919160;21281;16787;] +<1;494;2;21281;20895;5 +{'lat': 61.677056, 'lon': 129.91916, 'alt': 21281} +0;50;49;51;8;12;8;-27;16;14;0;8;50;117;0;199;0;27;54154;> + +[1;61676904;129917952;21329;26932;] +<1;495;2;21329;20949;4 +{'lat': 61.676904, 'lon': 129.917952, 'alt': 21329} +7;53;50;52;5;12;8;-27;16;14;0;9;50;117;0;199;0;27;25621;> + +[1;61676648;129916912;21387;43584;] +<1;496;2;21387;21003;5 +{'lat': 61.676648, 'lon': 129.916912, 'alt': 21387} +7;59;50;53;4;12;9;-27;17;14;0;8;50;117;0;199;0;27;30848;> + +[1;61676456;129916080;21431;61230;] +<1;497;2;21431;21054;4 +{'lat': 61.676456, 'lon': 129.91608, 'alt': 21431} +3;50;50;53;8;12;9;-28;17;14;0;8;50;117;0;199;0;27;51319;> + +[1;61676196;129915056;21491;5894;] +<1;498;2;21491;21109;59 +{'lat': 61.676196, 'lon': 129.915056, 'alt': 21491} +;54;49;53;6;12;9;-27;16;14;0;8;50;117;0;199;0;27;64789;> + +[1;61675776;129914120;21543;55221;] +<1;499;2;21543;21162;3 +{'lat': 61.675776, 'lon': 129.91412, 'alt': 21543} +2;52;51;52;4;12;9;-27;16;14;0;9;50;117;0;199;0;27;13478;> + +[1;61675484;129913192;21593;6014;] +<1;500;2;21593;21212;67 +{'lat': 61.675484, 'lon': 129.913192, 'alt': 21593} +;49;48;52;8;12;9;-27;16;14;0;8;50;117;0;199;0;27;40461;> + +[1;61675288;129912016;21642;16278;] +<1;501;2;21642;21267;6 +{'lat': 61.675288, 'lon': 129.912016, 'alt': 21642} +6;53;51;53;7;12;9;-27;16;14;0;9;50;117;0;199;0;27;58604;> + +[1;61675032;129910488;21693;49243;] +<1;502;2;21693;21321;4 +{'lat': 61.675032, 'lon': 129.910488, 'alt': 21693} +9;58;51;53;7;12;9;-27;16;14;0;9;50;117;0;199;0;27;29542;> + +[1;61675024;129908920;21741;46382;] +<1;503;2;21741;21373;4 +{'lat': 61.675024, 'lon': 129.90892, 'alt': 21741} +5;51;50;53;11;12;9;-28;17;14;0;9;50;117;0;199;0;27;16943;> + + +[1;61674768;129907488;21799;23537;] +<1;504;2;21799;21424;5 +{'lat': 61.674768, 'lon': 129.907488, 'alt': 21799} +5;50;52;52;4;12;10;-27;17;14;0;9;50;117;0;199;0;27;45031;> + + +[1;61674628;129905896;21845;63213;] +<1;505;2;21845;21473;4 +{'lat': 61.674628, 'lon': 129.905896, 'alt': 21845} +5;48;51;52;8;12;10;-27;17;14;0;9;50;117;0;199;0;27;64228;> + + +[1;61674428;129904024;21899;10359;] +<1;506;2;21899;21531;5 +{'lat': 61.674428, 'lon': 129.904024, 'alt': 21899} +3;56;52;54;11;12;10;-27;17;14;0;9;50;117;0;199;0;27;63681; +> + +[1;61674256;129902240;21950;28538;] +<1;507;2;21950;21581;4 +{'lat': 61.674256, 'lon': 129.90224, 'alt': 21950} +9;46;47;52;7;12;10;-26;16;14;0;8;50;117;0;199;0;27;35156;> + + +[1;61674004;129900528;21986;65132;] +<1;508;2;21986;21637;3 +{'lat': 61.674004, 'lon': 129.900528, 'alt': 21986} +5;60;50;52;8;12;10;-26;17;14;0;8;50;117;0;199;0;27;26149;> + + +[1;61673752;129899112;22038;56916;] +<1;509;2;22038;21687;5 +{'lat': 61.673752, 'lon': 129.899112, 'alt': 22038} +0;49;50;52;9;12;10;-26;17;14;0;8;50;117;0;199;0;27;10977;> + + +[1;61673560;129897632;22091;49266;] +<1;510;2;22091;21742;5 +{'lat': 61.67356, 'lon': 129.897632, 'alt': 22091} +2;53;52;52;9;12;10;-27;17;14;0;8;50;117;0;199;0;27;44312;> + + +[1;61673148;129896272;22142;47263;] +<1;511;2;22142;21792;5 +{'lat': 61.673148, 'lon': 129.896272, 'alt': 22142} +0;49;51;52;5;12;10;-27;18;14;0;8;50;117;0;199;0;27;27088;> + + +[1;61672672;129894840;22194;37489;] +<1;512;2;22194;21847;5 +{'lat': 61.672672, 'lon': 129.89484, 'alt': 22194} +1;53;48;51;10;12;10;-27;18;14;0;8;50;117;0;199;0;27;11897; +> + +[1;61672308;129893144;22246;9840;] +<1;513;2;22246;21901;50 +{'lat': 61.672308, 'lon': 129.893144, 'alt': 22246} +;58;54;54;10;12;10;-26;18;14;0;8;50;117;0;199;0;27;48533;> + + +[1;61671784;129891448;22302;41930;] +<1;514;2;22302;21955;5 +{'lat': 61.671784, 'lon': 129.891448, 'alt': 22302} +4;52;48;52;10;12;10;-26;17;14;0;8;50;117;0;199;0;27;37783; +> + +[1;61671416;129889768;22355;18652;] +<1;515;2;22355;22008;5 +{'lat': 61.671416, 'lon': 129.889768, 'alt': 22355} +0;51;51;53;9;12;10;-26;18;14;0;8;50;116;0;199;0;27;34177;> + + +[1;61671292;129887744;22393;25;] +<1;516;2;22393;22062;37;5 +{'lat': 61.671292, 'lon': 129.887744, 'alt': 22393} +3;49;53;11;12;11;-26;17;14;0;8;50;116;0;199;0;27;44541;> + +[1;61671276;129885728;22451;33490;] +<1;517;2;22451;22115;3 +{'lat': 61.671276, 'lon': 129.885728, 'alt': 22451} +8;51;47;53;8;12;11;-27;18;14;0;8;50;117;0;199;0;27;29480;> + + +[1;61671320;129883760;22502;62975;] +<1;518;2;22502;22173;6 +{'lat': 61.67132, 'lon': 129.88376, 'alt': 22502} +9;62;52;54;9;12;11;-27;18;14;0;8;50;116;0;199;0;27;239;> + +[1;61671288;129882264;22553;4575;] +<1;519;2;22553;22227;50 +{'lat': 61.671288, 'lon': 129.882264, 'alt': 22553} +;53;50;54;7;12;11;-27;19;14;0;8;50;116;0;199;0;27;40366;> + +[1;61671436;129880680;22610;13013;] +<1;520;2;22610;22281;5 +{'lat': 61.671436, 'lon': 129.88068, 'alt': 22610} +6;52;53;54;6;12;11;-28;19;14;0;8;50;117;0;199;0;27;3835;> + +[1;61671544;129879272;22662;64247;] +<1;521;2;22662;22338;5 +{'lat': 61.671544, 'lon': 129.879272, 'alt': 22662} +1;56;50;54;7;12;11;-27;19;14;0;8;50;116;0;199;0;27;14431;> + + +[1;61671568;129878128;22706;61669;] +<1;522;2;22706;22393;4 +{'lat': 61.671568, 'lon': 129.878128, 'alt': 22706} +2;53;51;55;7;12;11;-27;18;14;0;7;50;116;0;199;0;27;25380;> + + +[1;61671624;129876848;22771;14063;] +<1;523;2;22771;22449;6 +{'lat': 61.671624, 'lon': 129.876848, 'alt': 22771} +2;61;51;55;3;12;11;-26;17;14;0;7;50;117;0;199;0;27;58173;> + + +[1;61671684;129875688;22809;19450;] +<1;524;2;22809;22506;3 +{'lat': 61.671684, 'lon': 129.875688, 'alt': 22809} +7;56;55;56;5;12;11;-26;17;14;0;8;50;116;0;199;0;27;12266;> + + +[1;61671724;129874952;22872;37126;] +<1;525;2;22872;22561;6 +{'lat': 61.671724, 'lon': 129.874952, 'alt': 22872} +1;54;50;55;6;12;11;-27;18;14;0;7;50;116;0;199;0;27;59506;> + + +[1;61671680;129873976;22933;58018;] +<1;526;2;22933;22616;5 +{'lat': 61.67168, 'lon': 129.873976, 'alt': 22933} +9;53;51;55;5;12;11;-27;19;14;0;7;50;116;0;199;0;27;28417;> + + +[1;61671680;129872544;22972;22963;] +<1;527;2;22972;22674;3 +{'lat': 61.67168, 'lon': 129.872544, 'alt': 22972} +8;57;52;55;8;12;11;-27;20;14;0;7;50;116;0;199;0;27;29681;> + + +[1;61671628;129870864;23033;52567;] +<1;528;2;23033;22725;6 +{'lat': 61.671628, 'lon': 129.870864, 'alt': 23033} +0;50;53;55;8;12;11;-26;19;14;0;7;50;116;0;199;0;27;22411;> + + +[1;61671568;129869264;23079;53334;] +<1;529;2;23079;22782;4 +{'lat': 61.671568, 'lon': 129.869264, 'alt': 23079} +4;61;53;55;9;12;11;-26;19;14;0;7;50;116;0;199;0;27;25836;> + + +[1;61671624;129867752;23143;65127;] +<1;530;2;23143;22835;6 +{'lat': 61.671624, 'lon': 129.867752, 'alt': 23143} +2;51;53;55;7;12;12;-25;18;14;0;7;50;116;0;199;0;27;20491;> + + +[1;61671456;129865920;23184;18181;] +<1;531;2;23184;22892;4 +{'lat': 61.671456, 'lon': 129.86592, 'alt': 23184} +0;56;54;54;8;12;12;-25;17;14;0;8;50;116;0;199;0;27;35460;> + + +[1;61671364;129864200;23249;41725;] +<1;532;2;23249;22947;6 +{'lat': 61.671364, 'lon': 129.8642, 'alt': 23249} +4;53;54;55;10;12;12;-24;17;14;0;7;50;116;0;199;0;27;36590; +> + +[1;61671392;129862368;23311;18489;] +<1;533;2;23311;23005;6 +{'lat': 61.671392, 'lon': 129.862368, 'alt': 23311} +1;57;55;55;9;12;12;-24;17;14;0;7;50;116;0;199;0;27;31679;> + + +[1;61671344;129860416;23354;9128;] +<1;534;2;23354;23059;41 +{'lat': 61.671344, 'lon': 129.860416, 'alt': 23354} +;54;52;55;10;12;12;-24;18;14;0;7;50;116;0;199;0;27;56283;> + + +[1;61671484;129858280;23416;4459;] +<1;535;2;23416;23113;38 +{'lat': 61.671484, 'lon': 129.85828, 'alt': 23416} +;58;55;56;11;12;12;-24;18;14;0;8;50;116;0;199;0;27;5508;> + +[1;61671792;129856184;23462;17028;] +<1;536;2;23462;23166;6 +{'lat': 61.671792, 'lon': 129.856184, 'alt': 23462} +5;51;51;55;11;12;12;-24;18;14;0;8;50;116;0;199;0;27;15597; +> + +[1;61671932;129854112;23499;17824;] +<1;537;2;23499;23219;3 +{'lat': 61.671932, 'lon': 129.854112, 'alt': 23499} +6;52;52;54;13;12;12;-24;19;14;0;8;50;116;0;199;0;27;46186; +> + +[1;61672196;129852128;23571;12101;] +<1;538;2;23571;23276;7 +{'lat': 61.672196, 'lon': 129.852128, 'alt': 23571} +0;55;48;54;9;12;12;-23;18;14;0;9;50;116;0;199;0;27;1932;> + +[1;61672564;129850552;23616;20812;] +<1;539;2;23616;23328;6 +{'lat': 61.672564, 'lon': 129.850552, 'alt': 23616} +6;56;53;54;7;12;12;-23;18;14;0;9;50;116;0;199;0;27;1124;> + +[1;61672660;129849184;23661;37552;] +<1;540;2;23661;23381;4 +{'lat': 61.67266, 'lon': 129.849184, 'alt': 23661} +2;51;49;54;6;12;12;-23;17;14;0;8;50;116;0;199;0;27;25898;> + + +[1;61672708;129847440;23724;53172;] +<1;541;2;23724;23434;6 +{'lat': 61.672708, 'lon': 129.84744, 'alt': 23724} +3;52;49;53;11;12;12;-22;18;14;0;8;50;116;0;199;0;27;26118; +> + +[1;61672528;129845944;23766;34159;] +<1;542;2;23766;23485;4 +{'lat': 61.672528, 'lon': 129.845944, 'alt': 23766} +0;50;53;53;7;12;13;-22;18;14;0;9;50;116;0;199;0;27;39337;> + + +[1;61672288;129843992;23814;27700;] +<1;543;2;23814;23539;4 +{'lat': 61.672288, 'lon': 129.843992, 'alt': 23814} +7;53;53;53;10;12;13;-22;18;14;0;9;50;116;0;199;0;27;9420;> + + +[1;61671984;129842328;23855;46326;] +<1;544;2;23855;23592;4 +{'lat': 61.671984, 'lon': 129.842328, 'alt': 23855} +0;52;55;53;11;12;13;-22;19;14;0;10;50;116;0;199;0;27;64364 +;> + +[1;61671856;129840320;23902;41821;] +<1;545;2;23902;23644;4 +{'lat': 61.671856, 'lon': 129.84032, 'alt': 23902} +7;56;46;52;7;12;13;-21;18;14;0;9;50;116;0;199;0;27;27768;> + + +[1;61671908;129838440;23977;5433;] +<1;546;2;23977;23699;50 +{'lat': 61.671908, 'lon': 129.83844, 'alt': 23977} +;54;50;53;10;12;13;-20;18;14;0;9;50;116;0;199;0;27;52379;> + + +[1;61671892;129836912;24028;46400;] +<1;547;2;24028;23751;7 +{'lat': 61.671892, 'lon': 129.836912, 'alt': 24028} +3;50;53;53;8;12;13;-20;18;14;0;8;50;116;0;199;0;27;3192;> + +[1;61671940;129835096;24076;18527;] +<1;548;2;24076;23803;4 +{'lat': 61.67194, 'lon': 129.835096, 'alt': 24076} +7;50;50;52;9;12;13;-20;18;14;0;9;50;116;0;199;0;27;51814;> + + +[1;61672016;129833816;24124;39284;] +<1;549;2;24124;23856;4 +{'lat': 61.672016, 'lon': 129.833816, 'alt': 24124} +7;52;51;52;8;12;13;-19;18;14;0;9;50;116;0;199;0;27;63293;> + + +[1;61672292;129832184;24173;16515;] +<1;550;2;24173;23908;4 +{'lat': 61.672292, 'lon': 129.832184, 'alt': 24173} +7;51;51;52;8;12;13;-19;18;14;0;8;50;116;0;199;0;27;16743;> + + +[1;61672432;129830232;24226;18315;] +<1;551;2;24226;23959;5 +{'lat': 61.672432, 'lon': 129.830232, 'alt': 24226} +0;55;57;53;10;12;13;-20;19;14;0;9;50;116;0;199;0;27;37568; +> + +[1;61672820;129828584;24280;4017;] +<1;552;2;24280;24013;54 +{'lat': 61.67282, 'lon': 129.828584, 'alt': 24280} +;52;53;52;12;12;13;-20;18;14;0;9;50;116;0;199;0;27;20267;> + + +[1;61672848;129826032;24334;63774;] +<1;553;2;24334;24073;5 +{'lat': 61.672848, 'lon': 129.826032, 'alt': 24334} +3;58;50;52;13;12;13;-20;19;14;0;8;50;116;0;199;0;27;2500;> + + +[1;61672776;129824192;24412;25067;] +<1;554;2;24412;24127;7 +{'lat': 61.672776, 'lon': 129.824192, 'alt': 24412} +6;53;54;54;8;12;13;-20;18;14;0;9;50;116;0;199;0;27;9782;> + +[1;61672728;129822112;24432;52893;] +<1;555;2;24432;24183;1 +{'lat': 61.672728, 'lon': 129.822112, 'alt': 24432} +9;53;54;54;10;12;13;-19;18;14;0;9;50;116;0;199;0;27;26466; +> + +[1;61672664;129820728;24504;2079;] +<1;556;2;24504;24239;70 +{'lat': 61.672664, 'lon': 129.820728, 'alt': 24504} +;60;54;55;9;12;13;-19;18;14;0;9;50;116;0;199;0;27;55554;> + +[1;61672660;129819472;24551;53288;] +<1;557;2;24551;24294;4 +{'lat': 61.67266, 'lon': 129.819472, 'alt': 24551} +5;53;53;55;5;12;14;-18;18;14;0;9;50;116;0;199;0;27;47603;> + + +[1;61672672;129818096;24605;43950;] +<1;558;2;24605;24351;5 +{'lat': 61.672672, 'lon': 129.818096, 'alt': 24605} +2;55;57;56;6;12;14;-18;17;14;0;9;50;116;0;199;0;27;21464;> + + +[1;61672608;129817352;24660;48207;] +<1;559;2;24660;24407;5 +{'lat': 61.672608, 'lon': 129.817352, 'alt': 24660} +3;55;53;55;6;12;14;-17;17;14;0;8;50;117;0;199;0;27;28542;> + + +[1;61672632;129816376;24713;32900;] +<1;560;2;24713;24460;5 +{'lat': 61.672632, 'lon': 129.816376, 'alt': 24713} +1;53;53;54;3;12;14;-17;18;14;0;9;50;116;0;199;0;27;49358;> + + +[1;61672912;129815464;24761;45099;] +<1;561;2;24761;24512;4 +{'lat': 61.672912, 'lon': 129.815464, 'alt': 24761} +8;55;54;55;5;12;14;-18;19;14;0;9;50;116;0;199;0;27;31778;> + + +[1;61673092;129815000;24810;23517;] +<1;562;2;24810;24568;4 +{'lat': 61.673092, 'lon': 129.815, 'alt': 24810} +6;54;53;55;5;12;14;-18;20;14;0;8;50;116;0;199;0;27;9765;> + +[1;61673292;129814256;24866;37900;] +<1;563;2;24866;24622;5 +{'lat': 61.673292, 'lon': 129.814256, 'alt': 24866} +3;52;50;54;2;12;14;-17;19;14;0;8;50;116;0;199;0;27;18018;> + + +[1;61673516;129813264;24930;29575;] +<1;564;2;24930;24681;6 +{'lat': 61.673516, 'lon': 129.813264, 'alt': 24930} +3;58;48;54;5;12;14;-17;20;14;0;9;50;116;0;199;0;27;53886;> + + +[1;61673592;129812136;24968;39033;] +<1;565;2;24968;24737;3 +{'lat': 61.673592, 'lon': 129.812136, 'alt': 24968} +7;55;54;55;8;12;14;-17;20;14;0;9;50;116;0;199;0;27;21858;> + + +[1;61673628;129810976;25026;15227;] +<1;566;2;25026;24791;5 +{'lat': 61.673628, 'lon': 129.810976, 'alt': 25026} +7;57;51;55;7;12;14;-17;21;14;0;9;50;116;0;199;0;27;35000;> + + +[1;61673872;129809528;25074;57153;] +<1;567;2;25074;24846;4 +{'lat': 61.673872, 'lon': 129.809528, 'alt': 25074} +6;53;50;55;5;12;14;-17;20;15;0;8;50;116;0;199;0;27;39727;> + + +[1;61674052;129808536;25130;33893;] +<1;568;2;25130;24901;5 +{'lat': 61.674052, 'lon': 129.808536, 'alt': 25130} +4;53;52;55;9;12;14;-16;20;15;0;8;50;116;0;199;0;27;9570;> + +[1;61674064;129807192;25156;31276;] +<1;569;2;25156;24951;2 +{'lat': 61.674064, 'lon': 129.807192, 'alt': 25156} +6;49;47;54;5;12;14;-16;20;15;0;8;50;116;0;199;0;27;28086;> + + +[1;61674220;129805896;25234;55518;] +<1;570;2;25234;25007;7 +{'lat': 61.67422, 'lon': 129.805896, 'alt': 25234} +6;54;54;54;7;12;14;-16;20;15;0;8;50;116;0;199;0;27;60408;> + + +[1;61674076;129804608;25291;63526;] +<1;571;2;25291;25061;5 +{'lat': 61.674076, 'lon': 129.804608, 'alt': 25291} +4;58;53;54;6;12;14;-16;20;15;0;8;50;116;0;199;0;27;3024;> + +[1;61674148;129803192;25321;14604;] +<1;572;2;25321;25117;2 +{'lat': 61.674148, 'lon': 129.803192, 'alt': 25321} +9;54;48;54;7;12;15;-15;19;15;0;8;50;116;0;199;0;27;11824;> + + +[1;61674208;129801848;25411;13899;] +<1;573;2;25411;25176;5 +{'lat': 61.674208, 'lon': 129.801848, 'alt': 25411} + + +[1;61674216;129800296;25446;53499;] +<1;574;2;25446;25232;6 +{'lat': 61.674216, 'lon': 129.800296, 'alt': 25446} +5;54;51;54;9;12;15;-15;19;15;0;8;50;116;0;199;0;27;44919;> + + +[1;61674448;129798952;25506;51610;] +<1;575;2;25506;25286;5 +{'lat': 61.674448, 'lon': 129.798952, 'alt': 25506} +8;53;57;55;4;12;15;-15;19;15;0;8;50;116;0;199;0;27;11701;> + + +[1;61674612;129798016;25536;8479;] +<1;576;2;25536;25340;28 +{'lat': 61.674612, 'lon': 129.798016, 'alt': 25536} +;53;57;55;8;12;15;-15;19;15;0;9;50;116;0;199;0;27;64608;> + +[1;61674604;129797088;25591;44603;] +<1;577;2;25591;25395;5 +{'lat': 61.674604, 'lon': 129.797088, 'alt': 25591} +4;59;49;55;3;12;15;-15;20;15;0;8;50;116;0;199;0;27;56435;> + + +[1;61674904;129796096;25669;14288;] +<1;578;2;25669;25450;5 +{'lat': 61.674904, 'lon': 129.796096, 'alt': 25669} +1;53;49;55;7;12;15;-16;21;15;0;8;50;116;0;199;0;27;49482;> + + +[1;61674884;129795272;25694;2520;] +<1;579;2;25694;25505;49 +{'lat': 61.674884, 'lon': 129.795272, 'alt': 25694} +;53;57;55;3;12;15;-16;22;15;0;9;50;116;0;199;0;27;64816;> + +[1;61674908;129794648;25742;54150;] +<1;580;2;25742;25562;4 +{'lat': 61.674908, 'lon': 129.794648, 'alt': 25742} +6;56;51;54;5;12;16;-16;22;15;0;8;50;116;0;199;0;27;29371;> + + +[1;61674784;129793440;25801;62262;] +<1;581;2;25801;25619;5 +{'lat': 61.674784, 'lon': 129.79344, 'alt': 25801} +7;56;48;54;6;12;16;-16;22;15;0;8;50;116;0;199;0;27;23233;> + + +[1;61674868;129792408;25833;19725;] +<1;582;2;25833;25674;3 +{'lat': 61.674868, 'lon': 129.792408, 'alt': 25833} +1;59;48;55;6;12;16;-15;21;15;0;8;50;116;0;199;0;27;56091;> + + +[1;61674920;129790680;25934;5772;] +<1;583;2;25934;25733;67 +{'lat': 61.67492, 'lon': 129.79068, 'alt': 25934} +;57;54;56;9;12;16;-16;22;15;0;8;50;116;0;199;0;27;19469;> + +[1;61674884;129789416;25957;12563;] +<1;584;2;25957;25789;5 +{'lat': 61.674884, 'lon': 129.789416, 'alt': 25957} +3;55;56;56;6;12;16;-16;22;15;0;8;50;116;0;199;0;27;46144;> + + +[1;61675116;129788072;26022;10811;] +<1;585;2;26022;25846;6 +{'lat': 61.675116, 'lon': 129.788072, 'alt': 26022} +4;56;46;56;7;12;16;-16;22;15;0;8;50;116;0;199;0;27;22994;> + + +[1;61675368;129786496;26060;22765;] +<1;586;2;26060;25906;3 +{'lat': 61.675368, 'lon': 129.786496, 'alt': 26060} +6;58;53;56;8;12;16;-16;22;15;0;8;50;116;0;199;0;27;55389;> + + +[1;61675572;129785328;26132;42791;] +<1;587;2;26132;25967;7 +{'lat': 61.675572, 'lon': 129.785328, 'alt': 26132} +0;65;58;57;7;12;16;-17;23;15;0;8;50;116;0;199;0;27;53532;> + + +[1;61675968;129784040;26186;48004;] +<1;588;2;26186;26025;5 +{'lat': 61.675968, 'lon': 129.78404, 'alt': 26186} +4;56;59;58;7;12;16;-17;23;15;0;8;50;116;0;199;0;27;49675;> + + +[1;61676028;129783232;26241;68;] +<1;589;2;26241;26084;53;5 +{'lat': 61.676028, 'lon': 129.783232, 'alt': 26241} +7;52;58;5;12;16;-17;23;15;0;8;50;116;0;199;0;27;35484;> + +[1;61676388;129782392;26287;53363;] +<1;590;2;26287;26139;4 +{'lat': 61.676388, 'lon': 129.782392, 'alt': 26287} +5;53;53;57;3;12;17;-18;24;15;0;8;50;116;0;199;0;27;38768;> + + +[1;61676692;129781616;26355;39753;] +<1;591;2;26355;26197;6 +{'lat': 61.676692, 'lon': 129.781616, 'alt': 26355} +8;57;59;58;6;12;17;-18;24;15;0;8;50;116;0;199;0;27;21783;> + + +[1;61676852;129780712;26396;56487;] +<1;592;2;26396;26256;7 +{'lat': 61.676852, 'lon': 129.780712, 'alt': 26396} +7;56;54;58;6;12;17;-18;24;15;0;8;50;116;0;199;0;27;61520;> + + +[1;61677032;129779712;26474;37765;] +<1;593;2;26474;26308;7 +{'lat': 61.677032, 'lon': 129.779712, 'alt': 26474} +6;57;55;57;3;12;17;-18;24;15;0;8;50;116;0;199;0;27;62121;> + + +[1;61676936;129779040;26500;17845;] +<1;594;2;26500;26346;2 +{'lat': 61.676936, 'lon': 129.77904, 'alt': 26500} +9;37;51;56;3;12;17;-17;24;15;0;7;50;116;0;199;0;27;47352;> + + +[1;61677000;129778320;26159;62229;] +<1;595;2;26159;26010;- +{'lat': 61.677, 'lon': 129.77832, 'alt': 26159} +307;-332;38;31;1;12;17;-19;24;15;0;8;50;116;0;199;0;27;490 +27;> + +[1;61676852;129777008;25796;47568;] +<1;596;4;25796;25621;- +{'lat': 61.676852, 'lon': 129.777008, 'alt': 25796} +388;-381;-38;-39;1;12;17;-20;24;15;0;8;50;116;0;199;0;27;2 +6187;> + +[1;61676456;129776752;25433;23208;] +<1;597;4;25433;25241;- +{'lat': 61.676456, 'lon': 129.776752, 'alt': 25433} +355;-374;-105;-113;5;12;16;-21;23;15;0;7;50;116;0;199;0;27 +;4423;> + +[1;61676232;129776760;25117;24799;] +<1;598;4;25117;24883;- +{'lat': 61.676232, 'lon': 129.77676, 'alt': 25117} +343;-349;-169;-183;6;12;16;-22;23;15;0;7;50;116;0;199;0;27 +;26739;> + +[1;61676700;129775424;24768;51379;] +<1;599;4;24768;24526;- +{'lat': 61.6767, 'lon': 129.775424, 'alt': 24768} +347;-387;-242;-257;9;12;15;-23;23;15;0;7;50;116;0;199;0;27 +;15034;> + +[1;61677368;129774048;24460;12081;] +<1;600;4;24460;24185;- +{'lat': 61.677368, 'lon': 129.774048, 'alt': 24460} +301;-334;-305;-324;11;12;15;-24;23;15;0;7;50;116;0;199;0;2 +7;49550;> + +[1;61677696;129772672;24140;36867;] +<1;601;4;24140;23855;- +{'lat': 61.677696, 'lon': 129.772672, 'alt': 24140} +339;-323;-350;-363;7;12;14;-24;22;15;0;7;50;116;0;199;0;27 +;30009;> + +[1;61677880;129771136;23816;39583;] +<1;602;4;23816;23525;- +{'lat': 61.67788, 'lon': 129.771136, 'alt': 23816} +318;-325;-340;-354;4;12;14;-25;22;15;0;8;50;116;0;199;0;27 +;54379;> + +[1;61677992;129769424;23487;14247;] +<1;603;4;23487;23200;- +{'lat': 61.677992, 'lon': 129.769424, 'alt': 23487} +303;-320;-329;-344;11;12;13;-26;22;15;0;8;50;116;0;199;0;2 +7;9087;> + +[1;61678104;129767208;23167;22570;] +<1;604;4;23167;22885;- +{'lat': 61.678104, 'lon': 129.767208, 'alt': 23167} +313;-339;-324;-336;10;12;13;-26;22;15;0;7;50;116;0;199;0;2 +7;64727;> + +[1;61677900;129765824;22866;48760;] +<1;605;4;22866;22579;- +{'lat': 61.6779, 'lon': 129.765824, 'alt': 22866} +298;-303;-319;-327;8;12;12;-27;22;15;0;7;50;116;0;199;0;27 +;4683;> + +[1;61677968;129764544;22617;19326;] +<1;606;4;22617;22281;- +{'lat': 61.677968, 'lon': 129.764544, 'alt': 22617} +284;-290;-307;-319;13;12;11;-28;23;15;0;7;50;116;0;199;0;2 +7;9495;> + +[1;61678004;129762616;22325;21943;] +<1;607;4;22325;21989;- +{'lat': 61.678004, 'lon': 129.762616, 'alt': 22325} +284;-285;-299;-313;13;12;11;-28;22;15;0;8;50;116;0;199;0;2 +7;65411;> + +[1;61677700;129760928;22046;28146;] +<1;608;4;22046;21706;- +{'lat': 61.6777, 'lon': 129.760928, 'alt': 22046} +258;-280;-291;-306;10;12;10;-29;22;15;0;7;50;116;0;199;0;2 +7;45482;> + +[1;61677248;129759304;21797;49291;] +<1;609;4;21797;21432;- +{'lat': 61.677248, 'lon': 129.759304, 'alt': 21797} +248;-271;-289;-298;13;12;10;-30;22;15;0;7;50;116;0;199;0;2 +7;35083;> + +[1;61677128;129757984;21544;63652;] +<1;610;4;21544;21165;- +{'lat': 61.677128, 'lon': 129.757984, 'alt': 21544} +267;-287;-279;-293;14;12;9;-30;22;15;0;7;50;116;0;199;0;27 +[1;61676604;129755688;21312;33171;] +<1;611;4;21312;20915;- +{'lat': 61.676604, 'lon': 129.755688, 'alt': 21312} +223;-244;-264;-279;12;12;8;-31;22;15;0;7;50;116;0;199;0;27 +;61220;> + +[1;61676008;129754136;21047;15244;] +<1;612;4;21047;20666;- +{'lat': 61.676008, 'lon': 129.754136, 'alt': 21047} +258;-243;-260;-271;7;12;8;-31;22;15;0;7;50;116;0;199;0;27; +44792;> + +[1;61675568;129753600;20829;24607;] +<1;613;4;20829;20412;- +{'lat': 61.675568, 'lon': 129.7536, 'alt': 20829} +230;-250;-253;-264;6;12;7;-31;22;15;0;7;50;116;0;199;0;27; +42838;> + +[1;61675272;129755032;20595;8617;] +<1;614;4;20595;20178;-2 +{'lat': 61.675272, 'lon': 129.755032, 'alt': 20595} +31;-230;-246;-257;10;12;7;-31;21;15;0;8;50;116;0;199;0;27; +63973;> + +[1;61675724;129755552;20367;27203;] +<1;615;4;20367;19947;- +{'lat': 61.675724, 'lon': 129.755552, 'alt': 20367} +226;-228;-239;-249;6;12;7;-32;22;15;0;8;50;116;0;199;0;27; +7912;> + +[1;61675600;129755816;20151;58134;] +<1;616;4;20151;19713;- +{'lat': 61.6756, 'lon': 129.755816, 'alt': 20151} +227;-252;-234;-242;3;12;6;-32;21;15;0;8;50;116;0;199;0;27; +49401;> + +[1;61675540;129756072;19926;48746;] +<1;617;4;19926;19475;- +{'lat': 61.67554, 'lon': 129.756072, 'alt': 19926} +229;-232;-228;-240;4;12;6;-33;22;15;0;7;50;116;0;199;0;27; +36319;> + +[1;61675072;129756240;19696;8709;] +<1;618;4;19696;19248;-2 +{'lat': 61.675072, 'lon': 129.75624, 'alt': 19696} +26;-223;-228;-237;8;12;5;-33;22;15;0;8;50;116;0;199;0;27;1 +6367;> + +[1;61675552;129756928;19477;51082;] +<1;619;4;19477;19030;- +{'lat': 61.675552, 'lon': 129.756928, 'alt': 19477} +204;-213;-224;-231;10;12;5;-34;22;15;0;8;50;116;0;199;0;27 +;4366;> + +[1;61676352;129757168;19286;24503;] +<1;620;4;19286;18825;- +{'lat': 61.676352, 'lon': 129.757168, 'alt': 19286} +201;-203;-221;-226;7;12;4;-35;22;15;0;8;50;116;0;199;0;27; +62904;> + +[1;61677060;129756760;19101;45664;] +<1;621;4;19101;18620;- +{'lat': 61.67706, 'lon': 129.75676, 'alt': 19101} +192;-201;-213;-222;8;12;3;-35;21;15;0;8;50;116;0;199;0;27; +16343;> + +[1;61677060;129756592;18879;3209;] +<1;622;4;18879;18405;-2 +{'lat': 61.67706, 'lon': 129.756592, 'alt': 18879} +18;-229;-211;-218;7;12;3;-35;20;15;0;8;50;115;0;199;0;27;3 +0431;> + +[1;61677032;129757096;18672;49140;] +<1;623;4;18672;18194;- +{'lat': 61.677032, 'lon': 129.757096, 'alt': 18672} +204;-207;-211;-214;7;12;2;-36;20;15;0;7;50;115;0;199;0;27; +17685;> + +[1;61677068;129757144;18475;44432;] +<1;624;4;18475;17984;- +{'lat': 61.677068, 'lon': 129.757144, 'alt': 18475} +201;-206;-206;-210;6;12;1;-37;20;15;0;7;50;115;0;199;0;27; +60315;> + +[1;61676916;129757536;18269;3897;] +<1;625;4;18269;17778;-2 +{'lat': 61.676916, 'lon': 129.757536, 'alt': 18269} +02;-202;-203;-208;5;12;0;-37;20;15;0;8;50;115;0;199;0;27;4 +9726;> + +[1;61676920;129757416;18069;57300;] +<1;626;4;18069;17573;- +{'lat': 61.67692, 'lon': 129.757416, 'alt': 18069} +207;-201;-201;-207;5;12;0;-38;20;15;0;7;50;115;0;199;0;27; +31080;> + +[1;61676696;129758208;17873;39835;] +<1;627;4;17873;17369;- +{'lat': 61.676696, 'lon': 129.758208, 'alt': 17873} +196;-222;-204;-211;11;12;-1;-38;19;15;0;8;50;115;0;199;0;2 +7;65136;> + +[1;61677020;129758880;17693;60768;] +<1;628;4;17693;17179;- +{'lat': 61.67702, 'lon': 129.75888, 'alt': 17693} +170;-185;-195;-204;5;12;-2;-39;19;15;0;8;50;115;0;199;0;27 +;48157;> + +[1;61677128;129759672;17507;63656;] +<1;629;4;17507;16992;- +{'lat': 61.677128, 'lon': 129.759672, 'alt': 17507} +193;-183;-194;-200;7;12;-3;-39;19;15;0;7;50;115;0;199;0;27 +;33732;> + +[1;61677072;129759976;17335;5306;] +<1;630;4;17335;16805;-1 +{'lat': 61.677072, 'lon': 129.759976, 'alt': 17335} +75;-183;-189;-196;9;12;-3;-40;19;15;0;7;50;115;0;199;0;27; +63025;> + +[1;61676564;129760544;17155;53797;] +<1;631;4;17155;16633;- +{'lat': 61.676564, 'lon': 129.760544, 'alt': 17155} +176;-168;-186;-190;8;12;-3;-40;19;15;0;7;50;115;0;199;0;27 +;2030;> + +[1;61677000;129762240;16987;32036;] +<1;632;4;16987;16456;- +{'lat': 61.677, 'lon': 129.76224, 'alt': 16987} +166;-173;-186;-190;8;12;-4;-40;18;15;0;7;50;115;0;199;0;27 +;12044;> + +[1;61677696;129763704;16818;56928;] +<1;633;4;16818;16286;- +{'lat': 61.677696, 'lon': 129.763704, 'alt': 16818} +165;-184;-179;-186;8;12;-4;-40;18;15;0;7;50;115;0;199;0;27 +;56166;> + +[1;61678260;129763504;16651;35389;] +<1;634;4;16651;16123;- +{'lat': 61.67826, 'lon': 129.763504, 'alt': 16651} +164;-158;-174;-179;8;12;-5;-41;18;15;0;7;50;115;0;199;0;27 +;9336;> + +[1;61678672;129763320;16506;46010;] +<1;635;4;16506;15966;- +{'lat': 61.678672, 'lon': 129.76332, 'alt': 16506} +156;-154;-174;-175;4;12;-6;-41;18;15;0;7;50;115;0;199;0;27 +;47946;> + +[1;61678092;129764360;16348;41269;] +<1;636;4;16348;15803;- +{'lat': 61.678092, 'lon': 129.76436, 'alt': 16348} +155;-160;-166;-170;12;12;-7;-42;17;15;0;8;50;115;0;199;0;2 +7;15000;> + +[1;61677512;129765872;16195;53641;] +<1;637;4;16195;15647;- +{'lat': 61.677512, 'lon': 129.765872, 'alt': 16195} +154;-154;-163;-166;13;12;-8;-42;18;15;0;7;50;115;0;199;0;2 +7;14149;> + +[1;61677128;129768032;16040;8708;] +<1;638;4;16040;15491;-1 +{'lat': 61.677128, 'lon': 129.768032, 'alt': 16040} +61;-168;-159;-163;10;12;-8;-43;18;15;0;7;50;115;0;199;0;27 +;48267;> + +[1;61677108;129768296;15886;49190;] +<1;639;4;15886;15336;- +{'lat': 61.677108, 'lon': 129.768296, 'alt': 15886} +147;-150;-158;-162;2;12;-9;-44;18;15;0;8;50;115;0;199;0;27 +;36284;> + +[1;61677224;129768464;15737;26763;] +<1;640;4;15737;15188;- +{'lat': 61.677224, 'lon': 129.768464, 'alt': 15737} +146;-145;-157;-157;6;12;-9;-44;17;15;0;8;50;115;0;199;0;27 +;5831;> + +[1;61677684;129769992;15590;25494;] +<1;641;4;15590;15043;- +{'lat': 61.677684, 'lon': 129.769992, 'alt': 15590} +151;-142;-153;-155;11;12;-9;-43;16;15;0;7;50;115;0;199;0;2 +7;55468;> + +[1;61677556;129772312;15448;48469;] +<1;642;4;15448;14901;- +{'lat': 61.677556, 'lon': 129.772312, 'alt': 15448} +146;-139;-152;-153;14;12;-10;-43;16;15;0;7;50;115;0;199;0; +27;31189;> + +[1;61676944;129774128;15295;62327;] +<1;643;4;15295;14759;- +{'lat': 61.676944, 'lon': 129.774128, 'alt': 15295} +143;-152;-150;-149;10;12;-9;-43;15;15;0;7;50;115;0;199;0;2 +7;51967;> + +[1;61676220;129774952;15156;8956;] +<1;644;4;15156;14618;-1 +{'lat': 61.67622, 'lon': 129.774952, 'alt': 15156} +41;-137;-150;-149;4;12;-9;-44;16;15;0;7;50;115;0;199;0;27; +3917;> + +[1;61676204;129775280;15015;13864;] +<1;645;4;15015;14482;- +{'lat': 61.676204, 'lon': 129.77528, 'alt': 15015} +139;-133;-147;-144;2;12;-10;-44;15;15;0;8;50;115;0;199;0;2 +7;39265;> + +[1;61675700;129776184;14882;19333;] +<1;646;4;14882;14346;- +{'lat': 61.6757, 'lon': 129.776184, 'alt': 14882} +131;-134;-144;-141;6;12;-11;-44;14;15;0;7;50;115;0;199;0;2 +7;42692;> + +[1;61675368;129777408;14729;31273;] +<1;647;4;14729;14198;- +{'lat': 61.675368, 'lon': 129.777408, 'alt': 14729} +154;-145;-142;-139;5;12;-12;-45;14;15;0;8;50;115;0;199;0;2 +7;24063;> + +[1;61675256;129778608;14572;21429;] +<1;648;4;14572;14051;- +{'lat': 61.675256, 'lon': 129.778608, 'alt': 14572} +151;-157;-142;-140;10;12;-12;-46;14;15;0;8;50;115;0;199;0; +27;25793;> + +[1;61674804;129780224;14432;62582;] +<1;649;4;14432;13908;- +{'lat': 61.674804, 'lon': 129.780224, 'alt': 14432} +142;-138;-143;-143;14;12;-13;-46;13;15;0;7;50;115;0;199;0; +27;613;> + +[1;61674712;129781888;14288;48338;] +<1;650;4;14288;13764;- +{'lat': 61.674712, 'lon': 129.781888, 'alt': 14288} +145;-140;-143;-141;12;12;-14;-47;13;15;0;8;50;115;0;199;0; +27;47641;> + +[1;61674832;129783552;14144;4216;] +<1;651;4;14144;13623;-1 +{'lat': 61.674832, 'lon': 129.783552, 'alt': 14144} +45;-138;-145;-141;14;12;-16;-48;12;15;0;7;50;115;0;199;0;2 +7;40656;> + +[1;61674936;129785488;14000;45009;] +<1;652;4;14000;13481;- +{'lat': 61.674936, 'lon': 129.785488, 'alt': 14000} +144;-140;-145;-142;11;12;-18;-49;11;15;0;7;50;115;0;199;0; +27;35465;> + +[1;61674604;129787584;13871;29757;] +<1;653;4;13871;13345;- +{'lat': 61.674604, 'lon': 129.787584, 'alt': 13871} +130;-147;-144;-146;12;12;-20;-50;11;15;0;8;50;115;0;199;0; +27;50958;> + +[1;61674320;129789440;13725;41961;] +<1;654;4;13725;13212;- +{'lat': 61.67432, 'lon': 129.78944, 'alt': 13725} +137;-130;-143;-142;12;12;-21;-50;10;15;0;8;50;115;0;199;0; +27;26125;> + +[1;61674532;129791928;13595;42209;] +<1;655;4;13595;13083;- +{'lat': 61.674532, 'lon': 129.791928, 'alt': 13595} +136;-126;-141;-138;8;12;-21;-51;10;15;0;7;50;115;0;199;0;2 +7;57843;> + +[1;61674504;129793592;13471;24095;] +<1;656;4;13471;12956;- +{'lat': 61.674504, 'lon': 129.793592, 'alt': 13471} +128;-126;-138;-136;15;12;-22;-51;10;14;0;8;50;115;0;199;0; +27;33562;> + +[1;61674816;129795304;13332;44794;] +<1;657;4;13332;12824;- +{'lat': 61.674816, 'lon': 129.795304, 'alt': 13332} +136;-129;-134;-133;12;12;-22;-51;9;14;0;8;50;115;0;199;0;2 +7;8499;> + +[1;61674844;129796632;13204;2286;] +<1;658;4;13204;12701;-1 +{'lat': 61.674844, 'lon': 129.796632, 'alt': 13204} +24;-134;-134;-131;9;12;-22;-52;9;14;0;8;50;115;0;199;0;27; +25741;> + +[1;61674548;129798832;13072;33766;] +<1;659;4;13072;12573;- +{'lat': 61.674548, 'lon': 129.798832, 'alt': 13072} +126;-124;-132;-128;16;12;-23;-52;9;14;0;8;50;115;0;199;0;2 +7;12228;> + +[1;61674136;129800488;12936;50860;] +<1;660;4;12936;12447;- +{'lat': 61.674136, 'lon': 129.800488, 'alt': 12936} +138;-123;-132;-127;15;12;-22;-52;9;14;0;8;50;115;0;199;0;2 +7;28819;> + +[1;61673652;129802904;12802;38056;] +<1;661;4;12802;12321;- +{'lat': 61.673652, 'lon': 129.802904, 'alt': 12802} +133;-123;-130;-126;17;12;-22;-52;9;14;0;9;50;115;0;199;0;2 +7;41263;> + +[1;61673652;129805576;12694;31748;] +<1;662;4;12694;12212;- +{'lat': 61.673652, 'lon': 129.805576, 'alt': 12694} +109;-107;-129;-124;10;12;-23;-53;8;14;0;8;50;115;0;199;0;2 +7;17489;> + +123;-119;-127;-123;4;12;-23;-54;8;14;0;8;50;115;0;199;0;27 +;12304;> + +[1;61672608;129807632;12457;29938;] +<1;664;4;12457;11986;- +{'lat': 61.672608, 'lon': 129.807632, 'alt': 12457} +113;-112;-123;-121;16;12;-23;-54;8;14;0;8;50;115;0;199;0;2 +7;7598;> + +[1;61671884;129809600;12334;7881;] +<1;665;4;12334;11875;-1 +{'lat': 61.671884, 'lon': 129.8096, 'alt': 12334} +24;-108;-123;-116;22;12;-24;-127;7;14;0;8;50;115;0;199;2;2 +7;49490;> + +[1;61671408;129811664;12215;48548;] +<1;666;4;12215;11758;- +{'lat': 61.671408, 'lon': 129.811664, 'alt': 12215} +115;-114;-119;-115;10;12;-24;-127;7;14;0;8;50;115;0;199;2; +27;65450;> + +[1;61671388;129813824;12098;3119;] +<1;667;4;12098;11649;-1 +{'lat': 61.671388, 'lon': 129.813824, 'alt': 12098} +14;-108;-119;-112;16;12;-24;-127;7;14;0;8;50;115;0;199;2;2 +7;12894;> + +[1;61671492;129815920;11990;32650;] +<1;668;4;11990;11545;- +{'lat': 61.671492, 'lon': 129.81592, 'alt': 11990} +106;-111;-117;-112;15;12;-24;-127;6;14;0;8;50;115;0;199;2; +27;49013;> + +[1;61671236;129818496;11871;4832;] +<1;669;4;11871;11434;-1 +{'lat': 61.671236, 'lon': 129.818496, 'alt': 11871} +20;-109;-117;-110;17;12;-24;-127;5;13;0;8;50;115;0;199;2;2 +7;1373;> + +[1;61670752;129821288;11749;24647;] +<1;670;4;11749;11324;- +{'lat': 61.670752, 'lon': 129.821288, 'alt': 11749} +122;-107;-117;-110;8;12;-25;-127;6;13;0;7;50;115;0;199;2;2 +7;35362;> + +[1;61670528;129823824;11631;48202;] +<1;671;4;11631;11210;- +{'lat': 61.670528, 'lon': 129.823824, 'alt': 11631} +119;-111;-116;-110;14;12;-25;-127;5;13;0;8;50;115;0;199;2; +27;7465;> + +[1;61670548;129827088;11517;16266;] +<1;672;4;11517;11097;- +{'lat': 61.670548, 'lon': 129.827088, 'alt': 11517} +120;-110;-117;-109;16;12;-26;-127;5;13;0;8;50;115;0;199;2; +27;59631;> + +[1;61670752;129829512;11415;45091;] +<1;673;4;11415;10991;- +{'lat': 61.670752, 'lon': 129.829512, 'alt': 11415} +104;-114;-115;-111;12;12;-27;-127;5;13;0;8;50;115;0;199;2; +27;25022;> + +[1;61671052;129831816;11311;51394;] +<1;674;4;11311;10889;- +{'lat': 61.671052, 'lon': 129.831816, 'alt': 11311} +98;-100;-113;-109;10;12;-28;-127;4;13;0;8;50;115;0;199;2;2 +7;32964;> + +[1;61671056;129834888;11209;5422;] +<1;675;4;11209;10791;-1 +{'lat': 61.671056, 'lon': 129.834888, 'alt': 11209} +03;-96;-110;-107;19;12;-28;-127;4;13;0;7;50;115;0;199;2;27 +;12648;> + +[1;61670844;129838024;11113;41684;] +<1;676;4;11113;10689;- +{'lat': 61.670844, 'lon': 129.838024, 'alt': 11113} +96;-100;-108;-105;14;12;-29;-127;3;13;0;8;50;115;0;199;2;2 +7;43656;> + +[1;61670748;129839688;11008;5518;] +<1;677;4;11008;10592;-1 +{'lat': 61.670748, 'lon': 129.839688, 'alt': 11008} +01;-96;-105;-103;9;12;-30;-127;3;13;0;8;50;115;0;199;2;27; +2672;> + +[1;61670620;129842328;10906;19773;] +<1;678;4;10906;10495;- +{'lat': 61.67062, 'lon': 129.842328, 'alt': 10906} +102;-104;-102;-102;7;12;-30;-127;2;13;0;8;50;115;0;199;2;2 +7;56935;> + +[1;61671028;129844784;10801;64517;] +<1;679;4;10801;10398;- +{'lat': 61.671028, 'lon': 129.844784, 'alt': 10801} +104;-94;-101;-98;8;12;-30;-127;2;13;0;8;50;115;0;199;2;27; +34552;> + +[1;61670848;129847032;10699;17124;] +<1;680;4;10699;10299;- +{'lat': 61.670848, 'lon': 129.847032, 'alt': 10699} +104;-97;-102;-98;7;12;-30;-127;1;12;0;15;50;113;0;199;3;27 +;21323;> + +[1;61670404;129849272;10598;55904;] +<1;681;4;10598;10202;- +{'lat': 61.670404, 'lon': 129.849272, 'alt': 10598} +101;-95;-102;-98;9;12;-30;-127;1;13;0;15;50;113;0;199;3;27 +;48246;> + +5;-96;-101;-97;17;12;-31;-127;1;13;0;15;50;113;0;199;3;27; +9183;> + +[1;61669232;129854664;10408;56030;] +<1;683;4;10408;10016;- +{'lat': 61.669232, 'lon': 129.854664, 'alt': 10408} +92;-87;-100;-96;17;12;-30;-127;1;13;0;15;50;113;0;199;3;27 +;61180;> + +[1;61668912;129858288;10309;40224;] +<1;684;4;10309;9925;-9 +{'lat': 61.668912, 'lon': 129.858288, 'alt': 10309} +8;-89;-100;-96;19;12;-30;-127;1;14;0;14;50;113;0;199;3;27; +4264;> + +[1;61668452;129860840;10216;40947;] +<1;685;4;10216;9836;-9 +{'lat': 61.668452, 'lon': 129.86084, 'alt': 10216} +3;-94;-99;-93;12;12;-30;-127;0;14;0;15;50;113;0;199;3;27;1 +0580;> + +[1;61668584;129863360;10127;32662;] +<1;686;4;10127;9746;-8 +{'lat': 61.668584, 'lon': 129.86336, 'alt': 10127} +7;-88;-97;-93;19;12;-31;-54;0;14;0;15;50;113;0;199;1;27;58 +278;> + +[1;61668500;129866136;10030;24424;] +<1;687;4;10030;9655;-9 +{'lat': 61.6685, 'lon': 129.866136, 'alt': 10030} +6;-89;-94;-91;18;12;-31;-54;0;14;0;15;50;113;0;199;1;27;58 +276;> + +[1;61668332;129868608;9933;20109;] +<1;688;4;9933;9562;-98; +{'lat': 61.668332, 'lon': 129.868608, 'alt': 9933} +-91;-93;-91;14;12;-31;-53;0;14;0;15;50;113;0;199;1;27;1079 +2;> + +[1;61668312;129871400;9829;6462;] +<1;689;4;9829;9466;-100; +{'lat': 61.668312, 'lon': 129.8714, 'alt': 9829} +-94;-94;-90;19;12;-31;-53;1;14;0;15;50;113;0;199;1;27;3015 +1;> + +[1;61668328;129873504;9736;16726;] +<1;690;4;9736;9373;-95; +{'lat': 61.668328, 'lon': 129.873504, 'alt': 9736} +-100;-96;-91;9;12;-30;-53;1;14;0;14;50;113;0;199;1;27;3132 +;> + +[1;61668364;129875064;9639;6793;] +<1;691;4;9639;9285;-96;- +{'lat': 61.668364, 'lon': 129.875064, 'alt': 9639} +86;-94;-92;12;12;-30;-52;2;15;0;14;50;113;0;199;1;27;20388 +;> + +[1;61668404;129876752;9553;10442;] +<1;692;4;9553;9198;-85; +{'lat': 61.668404, 'lon': 129.876752, 'alt': 9553} +-85;-95;-91;11;12;-31;-52;2;15;0;14;50;113;0;199;1;27;5545 +;> + +[1;61668528;129878600;9459;43291;] +<1;693;4;9459;9109;-94; +{'lat': 61.668528, 'lon': 129.8786, 'alt': 9459} +-87;-95;-91;7;12;-31;-52;2;15;0;14;50;113;0;199;1;27;45866 +;> + +[1;61668556;129880920;9364;35914;] +<1;694;4;9364;9017;-95; +{'lat': 61.668556, 'lon': 129.88092, 'alt': 9364} +-90;-95;-90;13;12;-31;-51;2;15;0;14;50;113;0;199;1;27;4378 +3;> + +[1;61668572;129883480;9268;59781;] +<1;695;4;9268;8927;-93; +{'lat': 61.668572, 'lon': 129.88348, 'alt': 9268} +-97;-94;-90;15;12;-31;-50;2;15;0;14;50;113;0;199;1;27;2941 +9;> + +[1;61668656;129884960;9178;38427;] +<1;696;4;9178;8836;-95; +{'lat': 61.668656, 'lon': 129.88496, 'alt': 9178} +-88;-93;-91;6;12;-31;-50;2;15;0;14;50;113;0;199;1;27;42843 +;> + +[1;61668904;129886488;9087;14136;] +<1;697;4;9087;8746;-89; +{'lat': 61.668904, 'lon': 129.886488, 'alt': 9087} +-88;-93;-89;11;12;-31;-49;2;15;0;14;50;113;0;199;1;27;4745 +5;> + +[1;61668872;129888416;8987;7207;] +<1;698;4;8987;8655;-95;- +{'lat': 61.668872, 'lon': 129.888416, 'alt': 8987} +89;-92;-89;7;12;-30;-48;2;15;0;14;50;113;0;199;1;27;20675; +> + +[1;61669260;129889984;8899;11215;] +<1;699;4;8899;8569;-91; +{'lat': 61.66926, 'lon': 129.889984, 'alt': 8899} +-84;-94;-90;5;12;-30;-47;1;15;0;14;50;113;0;199;1;27;49587 +;> + +[1;61669508;129891512;8811;27373;] +<1;700;4;8811;8481;-90; +{'lat': 61.669508, 'lon': 129.891512, 'alt': 8811} +-94;-92;-89;14;12;-30;-46;1;15;0;14;50;113;0;199;1;27;6444 +;> + +[1;61669560;129893080;8722;62724;] +<1;701;4;8722;8393;-88; +{'lat': 61.66956, 'lon': 129.89308, 'alt': 8722} +-85;-92;-90;11;12;-30;-45;1;15;0;16;50;113;0;199;1;27;6382 +1;> + +[1;61669324;129894864;8633;30668;] +<1;702;4;8633;8311;-86; +{'lat': 61.669324, 'lon': 129.894864, 'alt': 8633} +-80;-90;-87;4;12;-30;-45;2;15;0;15;50;113;0;199;1;27;56070 +;> + +[1;61669504;129896152;8543;19897;] +<1;703;4;8543;8226;-91; +{'lat': 61.669504, 'lon': 129.896152, 'alt': 8543} +-83;-90;-86;6;12;-29;-45;3;15;0;15;50;113;0;199;1;27;56516 +;> + +[1;61669340;129898424;8460;47851;] +<1;704;4;8460;8144;-83; +{'lat': 61.66934, 'lon': 129.898424, 'alt': 8460} +-80;-90;-85;14;12;-29;-44;2;15;0;16;50;113;0;199;1;27;8613 +;> + +[1;61669412;129901184;8366;59473;] +<1;705;4;8366;8061;-88; +{'lat': 61.669412, 'lon': 129.901184, 'alt': 8366} +-88;-88;-84;11;12;-29;-43;3;15;0;15;50;113;0;199;1;27;1221 +;> + +[1;61669548;129903992;8284;57946;] +<1;706;4;8284;7977;-84; +{'lat': 61.669548, 'lon': 129.903992, 'alt': 8284} +-81;-87;-85;17;12;-29;-43;3;15;0;16;50;113;0;199;1;27;2573 +1;> + +[1;61669780;129907456;8202;31161;] +<1;707;4;8202;7897;-82; +{'lat': 61.66978, 'lon': 129.907456, 'alt': 8202} +-78;-87;-82;18;12;-29;-42;2;15;0;15;50;113;0;199;1;27;3216 +8;> + +[1;61669796;129911056;8117;54159;] +<1;708;4;8117;7816;-87; +{'lat': 61.669796, 'lon': 129.911056, 'alt': 8117} +-79;-86;-82;20;12;-28;-41;2;15;0;15;50;113;0;199;1;27;5741 +1;> + +[1;61669700;129913984;8028;9058;] +<1;709;4;8028;7733;-87;- +{'lat': 61.6697, 'lon': 129.913984, 'alt': 8028} +81;-86;-81;19;12;-28;-40;2;15;0;16;50;113;0;199;1;27;39723 +;> + +[1;61669996;129916928;7944;1750;] +<1;710;4;7944;7649;-83;- +{'lat': 61.669996, 'lon': 129.916928, 'alt': 7944} +90;-85;-81;13;12;-28;-40;2;15;0;16;50;113;0;199;1;27;52591 +;> + +[1;61670292;129919768;7861;25632;] +<1;711;4;7861;7567;-82; +{'lat': 61.670292, 'lon': 129.919768, 'alt': 7861} +-79;-84;-83;16;12;-28;-39;1;15;0;16;50;113;0;199;1;27;4900 +2;> + +[1;61670612;129922928;7772;8242;] +<1;712;4;7772;7482;-90;- +{'lat': 61.670612, 'lon': 129.922928, 'alt': 7772} +83;-86;-82;18;12;-28;-39;1;15;0;16;50;113;0;199;1;27;18570 +;> + +[1;61671084;129926408;7694;17740;] +<1;713;4;7694;7406;-76; +{'lat': 61.671084, 'lon': 129.926408, 'alt': 7694} +-74;-84;-81;18;12;-28;-38;2;16;0;16;50;113;0;199;1;27;3766 +5;> + +[1;61671524;129929080;7607;14329;] +<1;714;4;7607;7325;-87; +{'lat': 61.671524, 'lon': 129.92908, 'alt': 7607} +-79;-85;-81;12;12;-27;-38;2;16;0;16;50;113;0;199;1;27;4271 +1;> + +[1;61671484;129932144;7530;31016;] +<1;715;4;7530;7247;-81; +{'lat': 61.671484, 'lon': 129.932144, 'alt': 7530} +-83;-83;-82;14;12;-27;-37;2;16;0;16;50;113;0;199;1;27;3813 +;> + +[1;61671660;129935000;7444;28475;] +<1;716;4;7444;7164;-83; +{'lat': 61.67166, 'lon': 129.935, 'alt': 7444} +-81;-82;-81;18;12;-27;-36;2;16;0;16;50;113;0;199;1;27;4603 +2;> + +[1;61671848;129937880;7360;9234;] +<1;717;4;7360;7083;-83;- +{'lat': 61.671848, 'lon': 129.93788, 'alt': 7360} +79;-84;-80;19;12;-27;-36;3;16;0;16;50;113;0;199;1;27;55246 +;> + +[1;61672156;129940264;7277;23551;] +<1;718;4;7277;7004;-85; +{'lat': 61.672156, 'lon': 129.940264, 'alt': 7277} +-77;-81;-79;17;12;-26;-36;3;16;0;16;50;113;0;199;1;27;3244 +7;> + +[1;61672768;129942688;7193;13662;] +<1;719;4;7193;6925;-82; +{'lat': 61.672768, 'lon': 129.942688, 'alt': 7193} +-77;-83;-80;14;12;-26;-35;3;16;0;16;50;113;0;199;1;27;1539 +7;> + +[1;61673048;129945496;7116;3187;] +<1;720;4;7116;6847;-79;- +{'lat': 61.673048, 'lon': 129.945496, 'alt': 7116} +83;-82;-80;11;12;-25;-34;3;16;0;15;50;113;0;199;1;27;41440 +;> + +[1;61673520;129948576;7032;37223;] +<1;721;4;7032;6768;-82; +{'lat': 61.67352, 'lon': 129.948576, 'alt': 7032} +-76;-82;-79;17;12;-25;-34;3;16;0;16;50;113;0;199;1;27;2805 +5;> + +[1;61674064;129950624;6955;28339;] +<1;722;4;6955;6691;-77; +{'lat': 61.674064, 'lon': 129.950624, 'alt': 6955} +-75;-82;-79;14;12;-24;-33;3;16;0;16;50;112;0;199;1;27;4812 +2;> + +[1;61674508;129952976;6873;29178;] +<1;723;4;6873;6611;-79; +{'lat': 61.674508, 'lon': 129.952976, 'alt': 6873} +3;> + +[1;61675176;129956024;6793;11081;] +<1;724;4;6793;6534;-78; +{'lat': 61.675176, 'lon': 129.956024, 'alt': 6793} +-76;-81;-78;17;12;-24;-31;3;16;0;16;50;113;0;199;1;27;2307 +;> + +[1;61675600;129958800;6710;38381;] +<1;725;4;6710;6458;-82; +{'lat': 61.6756, 'lon': 129.9588, 'alt': 6710} +-81;-80;-78;8;12;-23;-30;3;16;0;15;50;113;0;199;1;27;15535 +;> + +[1;61675976;129961504;6634;37768;] +<1;726;4;6634;6379;-77; +{'lat': 61.675976, 'lon': 129.961504, 'alt': 6634} +-77;-80;-77;14;12;-23;-30;3;16;0;16;50;112;0;199;1;27;3961 +3;> + +[1;61675972;129963992;6558;46973;] +<1;727;4;6558;6304;-76; +{'lat': 61.675972, 'lon': 129.963992, 'alt': 6558} +-73;-79;-76;12;12;-22;-29;4;16;0;15;50;113;0;199;1;27;3239 +5;> + +[1;61676348;129966232;6477;10290;] +<1;728;4;6477;6228;-80; +{'lat': 61.676348, 'lon': 129.966232, 'alt': 6477} +-74;-79;-76;13;12;-21;-29;4;16;0;16;50;113;0;199;1;27;5279 +1;> + +[1;61676684;129968416;6397;20246;] +<1;729;4;6397;6148;-79; +{'lat': 61.676684, 'lon': 129.968416, 'alt': 6397} +-78;-79;-76;14;12;-21;-28;4;16;0;16;50;113;0;199;1;27;2716 +8;> + +[1;61676960;129970152;6323;2028;] +<1;730;4;6323;6072;-74;- +{'lat': 61.67696, 'lon': 129.970152, 'alt': 6323} +82;-78;-78;11;12;-21;-28;4;16;0;16;50;113;0;199;1;27;11600 +;> + +[1;61677108;129973072;6247;34763;] +<1;731;4;6247;5998;-74; +{'lat': 61.677108, 'lon': 129.973072, 'alt': 6247} +-71;-77;-76;12;12;-21;-27;5;17;0;15;50;113;0;199;1;27;6120 +7;> + +[1;61677256;129975800;6172;33005;] +<1;732;4;6172;5925;-73; +{'lat': 61.677256, 'lon': 129.9758, 'alt': 6172} +-71;-76;-75;16;12;-20;-26;5;16;0;9;50;114;0;199;0;27;24415 +;> + +[1;61677392;129978056;6099;58470;] +<1;733;4;6099;5852;-73; +{'lat': 61.677392, 'lon': 129.978056, 'alt': 6099} +-71;-76;-75;15;12;-19;-26;6;16;0;8;50;114;0;199;0;27;39458 +;> + +[1;61677532;129980648;6026;54797;] +<1;734;4;6026;5780;-74; +{'lat': 61.677532, 'lon': 129.980648, 'alt': 6026} +-71;-75;-74;13;12;-19;-25;6;16;0;9;50;114;0;199;0;27;61927 +;> + +[1;61678080;129983320;5953;6773;] +<1;735;4;5953;5706;-75;- +{'lat': 61.67808, 'lon': 129.98332, 'alt': 5953} +79;-75;-74;17;12;-18;-25;6;15;0;9;50;114;0;199;0;27;11018; +> + +[1;61678216;129985568;5877;60893;] +<1;736;4;5877;5632;-75; +{'lat': 61.678216, 'lon': 129.985568, 'alt': 5877} +-72;-73;-74;15;12;-18;-24;6;15;0;9;50;114;0;199;0;27;17843 +;> + +[1;61678508;129987888;5799;26435;] +<1;737;4;5799;5557;-78; +{'lat': 61.678508, 'lon': 129.987888, 'alt': 5799} +-73;-73;-73;14;12;-17;-24;5;15;0;9;50;114;0;199;0;27;65023 +;> + +[1;61678860;129990632;5727;54893;] +<1;738;4;5727;5485;-70; +{'lat': 61.67886, 'lon': 129.990632, 'alt': 5727} +-70;-74;-73;15;12;-17;-23;5;15;0;9;50;114;0;199;0;27;39774 +;> + +[1;61678988;129993408;5654;46086;] +<1;739;4;5654;5411;-73; +{'lat': 61.678988, 'lon': 129.993408, 'alt': 5654} +-72;-74;-73;11;12;-16;-23;5;15;0;9;50;114;0;199;0;27;18122 +;> + +[1;61679276;129996472;5580;34210;] +<1;740;4;5580;5339;-77; +{'lat': 61.679276, 'lon': 129.996472, 'alt': 5580} +-71;-74;-73;14;12;-16;-23;5;14;0;9;50;114;0;199;0;27;51941 +;> + +[1;61679448;129999800;5509;44606;] +<1;741;4;5509;5268;-69; +{'lat': 61.679448, 'lon': 129.9998, 'alt': 5509} +-76;-74;-73;13;12;-16;-22;5;14;0;9;50;114;0;199;0;27;17928 +;> + +[1;61679724;130003344;5436;51091;] +<1;742;4;5436;5198;-69; +{'lat': 61.679724, 'lon': 130.003344, 'alt': 5436} +-68;-73;-72;23;12;-15;-21;4;14;0;9;50;114;0;199;0;27;65124 +;> + +[1;61680268;130007064;5365;10045;] +<1;743;4;5365;5129;-73; +{'lat': 61.680268, 'lon': 130.007064, 'alt': 5365} +-67;-73;-72;20;12;-15;-21;4;14;0;9;50;114;0;199;0;27;39435 +;> + +[1;61680540;130010464;5291;61466;] +<1;744;4;5291;5056;-71; +{'lat': 61.68054, 'lon': 130.010464, 'alt': 5291} +-71;-72;-71;20;12;-15;-20;4;14;0;10;50;114;0;199;0;27;4735 +8;> + +[1;61680948;130013552;5224;31527;] +<1;745;4;5224;4987;-68; +{'lat': 61.680948, 'lon': 130.013552, 'alt': 5224} +-67;-72;-71;23;12;-15;-20;4;14;0;10;50;114;0;199;0;27;2465 +3;> + +[1;61681144;130016600;5153;25193;] +<1;746;4;5153;4917;-71; +{'lat': 61.681144, 'lon': 130.0166, 'alt': 5153} +-75;-72;-70;17;12;-14;-19;3;14;0;9;50;114;0;199;0;27;7800; +> + +[1;61681872;130019544;5085;30294;] +<1;747;4;5085;4848;-69; +{'lat': 61.681872, 'lon': 130.019544, 'alt': 5085} +-67;-71;-71;15;12;-14;-18;3;14;0;10;50;114;0;199;0;27;6396 +;> + +[1;61682428;130022800;5012;19335;] +<1;748;4;5012;4779;-71; +{'lat': 61.682428, 'lon': 130.0228, 'alt': 5012} +-67;-70;-69;19;12;-14;-18;4;14;0;10;50;114;0;199;0;27;2454 +1;> + +[1;61682952;130025224;4944;44824;] +<1;749;4;4944;4711;-67; +{'lat': 61.682952, 'lon': 130.025224, 'alt': 4944} +-66;-70;-69;13;12;-13;-18;4;14;0;9;50;114;0;199;0;27;10562 +;> + +[1;61683432;130028152;4876;10937;] +<1;750;4;4876;4642;-68; +{'lat': 61.683432, 'lon': 130.028152, 'alt': 4876} +-68;-70;-69;16;12;-13;-18;4;14;0;10;50;114;0;199;0;27;4868 +2;> + +[1;61683928;130031128;4805;56686;] +<1;751;4;4805;4572;-70; +{'lat': 61.683928, 'lon': 130.031128, 'alt': 4805} +-69;-69;-68;15;12;-12;-18;5;14;0;8;50;114;0;199;0;27;18353 +;> + +[1;61684336;130033584;4735;19801;] +<1;752;4;4735;4504;-68; +{'lat': 61.684336, 'lon': 130.033584, 'alt': 4735} +-72;-69;-70;15;12;-12;-17;5;14;0;9;50;114;0;199;0;27;18555 +;> + +[1;61684824;130036648;4668;11474;] +<1;753;4;4668;4438;-66; +{'lat': 61.684824, 'lon': 130.036648, 'alt': 4668} +-64;-69;-68;18;12;-11;-16;5;14;0;9;50;114;0;199;0;27;35028 +;> + +[1;61685292;130039688;4603;63282;] +<1;754;4;4603;4374;-64; +{'lat': 61.685292, 'lon': 130.039688, 'alt': 4603} +-62;-68;-67;16;12;-11;-16;5;13;0;9;50;114;0;199;0;27;16617 +;> + +[1;61685620;130042600;4535;37376;] +<1;755;4;4535;4307;-66; +{'lat': 61.68562, 'lon': 130.0426, 'alt': 4535} +-65;-68;-67;15;12;-11;-15;5;13;0;9;50;114;0;199;0;27;64010 +;> + +[1;61685708;130045056;4467;52279;] +<1;756;4;4467;4244;-69; +{'lat': 61.685708, 'lon': 130.045056, 'alt': 4467} +-63;-68;-67;14;12;-10;-15;5;13;0;9;50;114;0;199;0;27;31841 +;> + +[1;61685864;130047056;4398;7888;] +<1;757;4;4398;4178;-66;- +{'lat': 61.685864, 'lon': 130.047056, 'alt': 4398} +68;-67;-66;13;12;-10;-15;5;13;0;8;50;114;0;199;0;27;25301; +> + +[1;61685760;130054704;4137;48336;] +<1;761;4;4137;3925;-69; +{'lat': 61.68576, 'lon': 130.054704, 'alt': 4137} +-64;-66;-63;10;12;-9;-14;7;13;0;8;50;114;0;199;0;27;3513;> + + +[1;61685812;130057024;4072;14211;] +<1;762;4;4072;3858;-64; +{'lat': 61.685812, 'lon': 130.057024, 'alt': 4072} +-72;-65;-63;7;12;-9;-14;7;13;0;8;50;114;0;199;0;27;2908;> + +[1;61685616;130059112;4006;28242;] +<1;763;4;4006;3795;-65; +{'lat': 61.685616, 'lon': 130.059112, 'alt': 4006} +> + +[1;61685444;130060824;3943;39165;] +<1;764;4;3943;3733;-64; +{'lat': 61.685444, 'lon': 130.060824, 'alt': 3943} +-61;-65;-63;11;12;-9;-13;6;13;0;8;50;114;0;199;0;27;36808; +> + +[1;61685352;130062696;3879;13442;] +<1;765;4;3879;3671;-63; +{'lat': 61.685352, 'lon': 130.062696, 'alt': 3879} +-60;-64;-63;12;12;-8;-13;7;13;0;7;50;114;0;199;0;27;32252; +> + +[1;61685388;130064944;3811;20114;] +<1;766;4;3811;3607;-66; +{'lat': 61.685388, 'lon': 130.064944, 'alt': 3811} +-62;-65;-63;7;12;-8;-12;7;13;0;8;50;114;0;199;0;27;9322;> + +[1;61685616;130066984;3748;42333;] +<1;767;4;3748;3544;-63; +{'lat': 61.685616, 'lon': 130.066984, 'alt': 3748} +-67;-65;-63;7;12;-8;-12;7;13;0;8;50;114;0;199;0;27;50622;> + + +[1;61685416;130069064;3684;3643;] +<1;768;4;3684;3481;-65;- +{'lat': 61.685416, 'lon': 130.069064, 'alt': 3684} +62;-64;-64;7;12;-8;-12;7;13;0;7;50;114;0;199;0;27;4798;> + +[1;61685400;130071760;3619;10171;] +<1;769;4;3619;3418;-65; +{'lat': 61.6854, 'lon': 130.07176, 'alt': 3619} +> + +[1;61685136;130074176;3557;38845;] +<1;770;4;3557;3356;-61; +{'lat': 61.685136, 'lon': 130.074176, 'alt': 3557} +-60;-63;-62;12;12;-7;-11;8;13;0;7;50;114;0;199;0;27;21338; +> + +[1;61684892;130076184;3495;5436;] +<1;771;4;3495;3293;-61;- +{'lat': 61.684892, 'lon': 130.076184, 'alt': 3495} +61;-63;-62;9;12;-7;-11;7;13;0;7;50;114;0;199;0;27;43627;> + +-61;-63;-62;7;12;-7;-10;7;13;0;8;50;114;0;199;0;27;41195;> +[1;61684916;130081512;3305;22533;] +<1;774;4;3305;3104;-61; +{'lat': 61.684916, 'lon': 130.081512, 'alt': 3305} +-60;-63;-62;7;12;-7;-9;7;13;0;7;50;114;0;199;0;27;61022;> + +[1;61684912;130083752;3246;3518;] +<1;775;4;3246;3044;-58;- +{'lat': 61.684912, 'lon': 130.083752, 'alt': 3246} +59;-61;-62;11;12;-6;-9;7;13;0;7;50;114;0;199;0;27;46183;> + +[1;61684896;130085864;3182;12038;] +<1;776;4;3182;2982;-64; +{'lat': 61.684896, 'lon': 130.085864, 'alt': 3182} +-60;-62;-62;6;12;-6;-8;7;13;0;8;50;114;0;199;0;27;22151;> + +[1;61684992;130087280;3123;42563;] +<1;777;4;3123;2922;-60; +{'lat': 61.684992, 'lon': 130.08728, 'alt': 3123} +-58;-62;-61;7;12;-6;-8;7;13;0;8;50;114;0;199;0;27;62599;> + +[1;61685128;130088880;3063;16326;] +<1;778;4;3063;2861;-59; +{'lat': 61.685128, 'lon': 130.08888, 'alt': 3063} +-60;-61;-61;9;12;-5;-7;7;13;0;9;50;113;0;199;0;27;32233;> + +[1;61685356;130091000;2999;56272;] +<1;779;4;299;2800;-63; +{'lat': 61.685356, 'lon': 130.091, 'alt': 2999} +-64;-61;-61;6;12;-5;-7;7;13;0;9;50;114;0;199;0;27;12079;> + +[1;61685800;130094240;2878;48876;] +<1;781;4;2878;2679;-64; +{'lat': 61.6858, 'lon': 130.09424, 'alt': 2878} +-61;-61;-60;7;12;-4;-6;7;13;0;16;50;112;0;199;1;27;56485;> + + +[1;61685844;130095536;2819;52318;] +<1;782;4;2819;2622;-58; +{'lat': 61.685844, 'lon': 130.095536, 'alt': 2819} +-55;-60;-59;10;12;-4;-6;8;13;0;16;50;112;0;199;1;27;56066; +> + +[1;61686096;130097624;2759;45790;] +<1;783;4;2759;2563;-59; +{'lat': 61.686096, 'lon': 130.097624, 'alt': 2759} +-58;-60;-59;11;12;-4;-6;8;13;0;16;50;112;0;199;1;27;65434; +> + +[1;61686176;130098848;2697;16500;] +<1;784;4;2697;2503;-60; +{'lat': 61.686176, 'lon': 130.098848, 'alt': 2697} +-59;-60;-59;7;12;-3;-5;9;14;0;16;50;112;0;199;1;27;35277;> + + +[1;61686068;130100360;2641;56519;] +<1;785;4;2641;2445;-57; +{'lat': 61.686068, 'lon': 130.10036, 'alt': 2641} +-62;-59;-59;7;12;-3;-5;9;14;0;16;50;112;0;199;1;27;35062;> + + +[1;61686132;130101592;2581;60693;] +<1;786;4;2581;2387;-59; +{'lat': 61.686132, 'lon': 130.101592, 'alt': 2581} +-56;-59;-59;4;12;-3;-5;11;14;0;16;50;112;0;199;1;27;7950;> + + +[1;61686124;130103072;2523;50329;] +<1;787;4;2523;2329;-59; +{'lat': 61.686124, 'lon': 130.103072, 'alt': 2523} +-56;-59;-58;6;12;-2;-4;11;15;0;16;50;112;0;199;1;27;36621; +> + +[1;61686360;130104168;2462;42869;] +<1;788;4;2462;2271;-60; +{'lat': 61.68636, 'lon': 130.104168, 'alt': 2462} +-57;-59;-58;8;12;-2;-4;11;15;0;15;50;112;0;199;1;27;9244;> + + +[1;61686396;130104600;2404;27241;] +<1;789;4;2404;2213;-58; +{'lat': 61.686396, 'lon': 130.1046, 'alt': 2404} +-57;-59;-58;1;12;-1;-3;11;15;0;16;50;112;0;199;1;27;2311;> + + +[1;61686816;130105192;2344;51097;] +<1;790;4;2344;2154;-59; +{'lat': 61.686816, 'lon': 130.105192, 'alt': 2344} +-62;-59;-58;2;12;-1;-3;12;15;0;16;50;112;0;199;1;27;52395; +> + +[1;61686928;130106152;2287;18689;] +<1;791;4;2287;2098;-56; +{'lat': 61.686928, 'lon': 130.106152, 'alt': 2287} +-55;-58;-58;7;12;-1;-2;12;15;0;16;50;112;0;199;1;27;39283; +> + +[1;61687316;130107408;2230;15940;] +<1;792;4;2230;2041;-58; +{'lat': 61.687316, 'lon': 130.107408, 'alt': 2230} +-56;-58;-57;4;12;0;-2;13;16;0;16;50;112;0;199;1;27;28405;> + + +[1;61687576;130108368;2174;20729;] +<1;793;4;2174;1985;-55; +{'lat': 61.687576, 'lon': 130.108368, 'alt': 2174} +-55;-58;-57;11;12;0;-2;13;16;0;16;50;112;0;199;1;27;51855; +> + +[1;61688280;130109312;2120;36448;] +<1;794;4;2120;1932;-54; +{'lat': 61.68828, 'lon': 130.109312, 'alt': 2120} +-52;-57;-57;10;12;0;-2;13;16;0;16;50;112;0;199;1;27;10461; +> + +[1;61688664;130110792;2066;48149;] +<1;795;4;2066;1880;-52; +{'lat': 61.688664, 'lon': 130.110792, 'alt': 2066} +-56;-57;-56;10;12;0;-2;14;16;0;16;50;112;0;199;1;27;25044; +> + +[1;61689296;130111904;2014;17472;] +<1;796;4;2014;1827;-53; +{'lat': 61.689296, 'lon': 130.111904, 'alt': 2014} +-52;-56;-56;6;12;0;-1;15;16;0;16;50;112;0;199;1;27;56914;> + + +[1;61689900;130112896;1959;42698;] +<1;797;4;1959;1773;-55; +{'lat': 61.6899, 'lon': 130.112896, 'alt': 1959} +-52;-55;-54;11;12;1;-1;15;16;0;16;50;112;0;199;1;27;32811; +> + +[1;61690828;130114368;1907;42805;] +<1;798;4;1907;1723;-50; +{'lat': 61.690828, 'lon': 130.114368, 'alt': 1907} +-49;-54;-53;13;12;1;-1;16;16;0;16;50;112;0;199;1;27;14221; +> + +[1;61691516;130115584;1854;6292;] +<1;799;4;1854;1671;-53;- +{'lat': 61.691516, 'lon': 130.115584, 'alt': 1854} +51;-53;-52;12;12;1;-1;16;17;0;9;50;113;0;199;0;27;11006;> + +[1;61692336;130117144;1801;5497;] +<1;800;4;1801;1619;-53;- +{'lat': 61.692336, 'lon': 130.117144, 'alt': 1801} +50;-53;-52;9;12;2;-1;16;17;0;9;50;113;0;199;0;27;13164;> + +[1;61692968;130118176;1747;62084;] +<1;801;4;1747;1566;-52; +{'lat': 61.692968, 'lon': 130.118176, 'alt': 1747} +-57;-53;-52;12;12;2;0;17;16;0;9;50;113;0;199;0;27;5589;> + +[1;61693828;130118976;1692;16689;] +<1;802;4;1692;1513;-55; +{'lat': 61.693828, 'lon': 130.118976, 'alt': 1692} +-51;-53;-52;12;12;2;0;16;16;0;9;50;113;0;199;0;27;41540;> + +[1;61694640;130120256;1639;8658;] +<1;803;4;1639;1462;-52;- +{'lat': 61.69464, 'lon': 130.120256, 'alt': 1639} +49;-53;-51;11;12;3;0;16;16;0;9;50;113;0;199;0;27;56032;> + +-52;-53;-51;12;12;3;0;16;16;0;8;50;113;0;199;0;27;51228;> + +[1;61696168;130123352;1531;5767;] +<1;805;4;1531;1356;-55;- +{'lat': 61.696168, 'lon': 130.123352, 'alt': 1531} +[1;61697028;130125064;1472;15700;] +<1;806;4;1472;1297;-59; +{'lat': 61.697028, 'lon': 130.125064, 'alt': 1472} +-62;-53;-52;13;12;3;0;16;15;0;8;50;113;0;199;0;27;65141;> + +[1;61698104;130126512;1404;23555;] +<1;807;4;1404;!232;-68; +{'lat': 61.698104, 'lon': 130.126512, 'alt': 1404} +-63;-55;-54;14;12;4;0;16;15;0;7;50;113;0;199;0;27;35062;> + +[1;61699272;130128048;1324;13244;] +<1;808;4;1324;1157;-77; +{'lat': 61.699272, 'lon': 130.128048, 'alt': 1324} +-73;-58;-56;12;12;4;1;16;15;0;7;50;113;0;199;0;27;2131;> + +[1;61700468;130128448;1249;34049;] +<1;809;4;1249;1083;-75; +{'lat': 61.700468, 'lon': 130.128448, 'alt': 1249} +-72;-62;-60;12;12;4;1;16;15;0;7;50;113;0;199;0;27;44728;> + +[1;61701488;130128952;1180;62635;] +<1;810;4;1180;1014;-69; +{'lat': 61.701488, 'lon': 130.128952, 'alt': 1180} +-68;-65;-63;16;12;4;0;16;15;0;8;50;113;0;199;0;27;58454;> + +[1;61702472;130128832;1116;27300;] +<1;811;4;1116;954;-65;- +{'lat': 61.702472, 'lon': 130.128832, 'alt': 1116} +59;-67;-65;8;12;4;0;16;15;0;8;50;113;0;199;0;27;43848;> + +[1;61703352;130128680;1053;2528;] +<1;812;4;1053;890;-63;-6 +{'lat': 61.703352, 'lon': 130.12868, 'alt': 1053} +8;-69;-68;5;12;4;0;16;15;0;8;50;113;0;199;0;27;8092;> + +[1;61704364;130128448;994;41854;] +<1;813;4;994;833;-57;-55 +{'lat': 61.704364, 'lon': 130.128448, 'alt': 994} +;-69;-67;12;12;5;0;16;15;0;10;50;113;0;199;0;27;24006;> + +[1;61705388;130128216;942;24164;] +<1;814;4;942;783;-53;-49 +{'lat': 61.705388, 'lon': 130.128216, 'alt': 942} +;-66;-64;11;12;5;1;16;15;0;9;50;113;0;199;0;27;8719;> + +[1;61706496;130127688;887;6494;] +<1;815;4;887;729;-55;-53; +{'lat': 61.706496, 'lon': 130.127688, 'alt': 887} +-62;-60;12;12;5;1;16;15;0;8;50;113;0;199;0;27;22006;> + +[1;61707368;130127304;825;22027;] +<1;816;4;825;670;-61;-58 +{'lat': 61.707368, 'lon': 130.127304, 'alt': 825} +;-60;-58;12;12;5;1;16;15;0;7;50;113;0;199;0;27;29454;> + +[1;61708096;130126208;766;9619;] +<1;817;4;766;612;-58;-62; +{'lat': 61.708096, 'lon': 130.126208, 'alt': 766} +-59;-57;10;12;5;1;17;15;0;8;50;113;0;199;0;27;61843;> + +[1;61708944;130125184;713;12602;] +<1;818;4;713;560;-54;-50 +{'lat': 61.708944, 'lon': 130.125184, 'alt': 713} +;-57;-55;10;12;5;2;17;15;0;7;50;113;0;199;0;27;39020;> + +[1;61709480;130124600;673;28023;] +<1;819;4;673;521;-40;-38 +{'lat': 61.70948, 'lon': 130.1246, 'alt': 673} +;-54;-52;9;12;6;2;17;15;0;8;50;113;0;199;0;27;43471;> + +[1;61710156;130123520;617;63797;] +<1;820;4;617;469;-53;-50 +{'lat': 61.710156, 'lon': 130.12352, 'alt': 617} +;-53;-51;12;12;6;3;17;15;0;7;50;113;0;199;0;27;2781;> + +[1;61710992;130122744;552;17369;] +<1;821;4;552;406;-65;-61 +{'lat': 61.710992, 'lon': 130.122744, 'alt': 552} +;-54;-53;6;12;6;3;17;15;0;7;50;113;0;199;0;27;24309;> + +[1;61711956;130122224;478;56355;] +<1;822;4;478;335;-74;-70 +{'lat': 61.711956, 'lon': 130.122224, 'alt': 478} +;-57;-55;9;12;7;4;17;15;0;8;50;113;0;199;0;27;30459;> + +[1;61712632;130121720;405;63445;] +<1;823;4;405;264;-71;-74 +{'lat': 61.712632, 'lon': 130.12172, 'alt': 405} +;-58;-57;8;12;7;4;16;15;0;7;50;113;0;199;0;27;36923;> + +[1;61713296;130121264;342;42315;] +<1;824;4;342;198;-67;-65 +{'lat': 61.713296, 'lon': 130.121264, 'alt': 342} +;-61;-59;7;12;7;5;17;15;0;8;50;113;0;199;0;27;32722;> + diff --git a/tests/e2e/auth.spec.ts b/tests/e2e/auth.spec.ts new file mode 100644 index 0000000..49777af --- /dev/null +++ b/tests/e2e/auth.spec.ts @@ -0,0 +1,27 @@ +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 new file mode 100644 index 0000000..16cebcc --- /dev/null +++ b/tests/e2e/fake_tawhiri.py @@ -0,0 +1,104 @@ +""" +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 new file mode 100644 index 0000000..ec60836 --- /dev/null +++ b/tests/e2e/fixtures.ts @@ -0,0 +1,74 @@ +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 new file mode 100644 index 0000000..05af2fd --- /dev/null +++ b/tests/e2e/saved-points.spec.ts @@ -0,0 +1,26 @@ +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 new file mode 100644 index 0000000..b358a7c --- /dev/null +++ b/tests/e2e/settings.spec.ts @@ -0,0 +1,24 @@ +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 new file mode 100644 index 0000000..80345c3 --- /dev/null +++ b/tests/e2e/smoke.spec.ts @@ -0,0 +1,31 @@ +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 new file mode 100644 index 0000000..8bc8ac1 --- /dev/null +++ b/tests/e2e/workspaces.spec.ts @@ -0,0 +1,68 @@ +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 bbf8c7d..3c2539f 100644 --- a/vite.config.js +++ b/vite.config.js @@ -1,6 +1,42 @@ import { sveltekit } from '@sveltejs/kit/vite'; -import { defineConfig } from 'vite'; +import { defineConfig, loadEnv } from 'vite'; +import { mockApiPlugin } from './mocks/vitePlugin'; -export default defineConfig({ - plugins: [sveltekit()] +/** + * 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, + ws: true, + }, + } + : undefined, + }, + }; });